[ In Java/Spring, how to gracefully handle missing translation values? ]

I'm using Java + Spring for an international website.

The 2 languages are ZH and EN (Chinese and English).

I have 2 files: messages.properties (for English key/value pairs) and messages_zh.properties.

The site is coded in English using #springMessage tags. I then use Poeditor.com to have translators provide Chinese values for each English phrase. So, although the English messages.properties file is always complete, and although the messages_zh.properties file always has all of the keys, the messages_zh.properties file will sometimes have blank values because I receive translations from the translators several days later.

I need my system to display the equivalent English value (on my website) whenever the Chinese value is missing.

How can I tell Spring to "fall back" to use English whenever a Chinese value isn't available? I need this to happen on a value-by-value basis.

Right now, I see blank button labels on my site wherever a Chinese value is missing. I'd rather that English (the default language) be used whenever the Chinese is blank.

Answer 1


You could create your own Custom MessageSource for this purpose.

something like

public class SpecialMessageSource extends ReloadableResourceBundleMessageSource {

      @Override
      protected MessageFormat resolveCode(String code, Locale locale) {
         MessageFormat result = super.resolveCode(code, locale);
         if (result.getPattern().isEmpty() && locale == Locale.CHINESE) {
            return super.resolveCode(code, Locale.ENGLISH);
         }
         return result;
      }

      @Override
      protected String resolveCodeWithoutArguments(String code, Locale locale) {
         String result= super.resolveCodeWithoutArguments(code, locale);
         if ((result.isEmpty() || result == null) && locale == Locale.CHINESE) {
            return super.resolveCodeWithoutArguments(code, Locale.ENGLISH);
         }
         return result;
      }
   }

and configure this messageSource bean in spring xml as

<bean id="messageSource" class="SpecialMessageSource">
.....
</bean>

Now to get resolved Label you will be invoking MessageSource's either of the below methods

String getMessage(String code, Object[] args, Locale locale);
String getMessage(String code, Object[] args, String defaultMessage, Locale locale);


resolveCode() will be called when your message label has arguments and you pass those arguments via args parameter like below
invalid.number= {0} is Invalid
and you invoke messageSource.getMessage("INVALID_NUMBER", new Object[]{2d}, locale)

resolveCodeWithoutArguments() will be called when your message label does not have arguments and you pass args parameter as null
validation.success = Validation Success
and you invoke messageSource.getMessage("INVALID_NUMBER", null, locale)

Answer 2


@harrybvp's answer got me on the right track. This is the code that seems to work for me (and is unit-testable when you mock the methods that call "super"):

    public class GracefulMessageSource extends org.springframework.context.support.ReloadableResourceBundleMessageSource {

        final static private Locale defaultLocale = Locale.ENGLISH;

        @Override protected MessageFormat resolveCode(final String code, final Locale locale) {
            MessageFormat result = superResolveCode(code, locale);

            if (result.toPattern().isEmpty() && !locale.equals(this.defaultLocale)) {
                result = superResolveCode(code, this.defaultLocale);
            }

            return result;
        }

        @Override protected String resolveCodeWithoutArguments(final String code, final Locale locale) {
            String result = superResolveCodeWithoutArguments(code, locale);

            if (result.isEmpty() && !locale.equals(this.defaultLocale)) {
                result = superResolveCodeWithoutArguments(code, this.defaultLocale);
            }

            return result;
        }

        protected MessageFormat superResolveCode(final String code, final Locale locale) {
            return super.resolveCode(code, locale);

        }

        protected String superResolveCodeWithoutArguments(final String code, final Locale locale) {
            return super.resolveCodeWithoutArguments(code, locale);

        }

    }

and in my webmvc-config.xml:

<bean id="messageSource" class="com.mycode.fe.web.GracefulMessageSource">
    <property name="basename">
        <value>${content.path.config}/WEB-INF/messages</value>
    </property>
    <property name="defaultEncoding" value="UTF-8" />
    <property name="cacheSeconds" value="2"/>
    <property name="fallbackToSystemLocale" value="true"/>
</bean>

Then in my Velocity view templates:

For a normal key with no arguments: #springMessage('menu_item1')

For a key that accepts arguments: #springMessageText('male_to_female_ratio' [3, 2])

Then in the messages.properties (English translations): male_to_female_ratio = There is a {0}:{1} male-to-female ratio.