ExtendedMessageFormat.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.apache.commons.text;

  18. import java.text.Format;
  19. import java.text.MessageFormat;
  20. import java.text.ParsePosition;
  21. import java.util.ArrayList;
  22. import java.util.Collection;
  23. import java.util.Iterator;
  24. import java.util.Locale;
  25. import java.util.Map;
  26. import java.util.Objects;

  27. /**
  28.  * Extends <code>java.text.MessageFormat</code> to allow pluggable/additional formatting
  29.  * options for embedded format elements.  Client code should specify a registry
  30.  * of <code>FormatFactory</code> instances associated with <code>String</code>
  31.  * format names.  This registry will be consulted when the format elements are
  32.  * parsed from the message pattern.  In this way custom patterns can be specified,
  33.  * and the formats supported by <code>java.text.MessageFormat</code> can be overridden
  34.  * at the format and/or format style level (see MessageFormat).  A "format element"
  35.  * embedded in the message pattern is specified (<b>()?</b> signifies optionality):<br>
  36.  * <code>{</code><i>argument-number</i><b>(</b><code>,</code><i>format-name</i><b>
  37.  * (</b><code>,</code><i>format-style</i><b>)?)?</b><code>}</code>
  38.  *
  39.  * <p>
  40.  * <i>format-name</i> and <i>format-style</i> values are trimmed of surrounding whitespace
  41.  * in the manner of <code>java.text.MessageFormat</code>.  If <i>format-name</i> denotes
  42.  * <code>FormatFactory formatFactoryInstance</code> in <code>registry</code>, a <code>Format</code>
  43.  * matching <i>format-name</i> and <i>format-style</i> is requested from
  44.  * <code>formatFactoryInstance</code>.  If this is successful, the <code>Format</code>
  45.  * found is used for this format element.
  46.  * </p>
  47.  *
  48.  * <p><b>NOTICE:</b> The various subformat mutator methods are considered unnecessary; they exist on the parent
  49.  * class to allow the type of customization which it is the job of this class to provide in
  50.  * a configurable fashion.  These methods have thus been disabled and will throw
  51.  * <code>UnsupportedOperationException</code> if called.
  52.  * </p>
  53.  *
  54.  * <p>Limitations inherited from <code>java.text.MessageFormat</code>:</p>
  55.  * <ul>
  56.  * <li>When using "choice" subformats, support for nested formatting instructions is limited
  57.  *     to that provided by the base class.</li>
  58.  * <li>Thread-safety of <code>Format</code>s, including <code>MessageFormat</code> and thus
  59.  *     <code>ExtendedMessageFormat</code>, is not guaranteed.</li>
  60.  * </ul>
  61.  *
  62.  * @since 1.0
  63.  */
  64. public class ExtendedMessageFormat extends MessageFormat {
  65.     private static final long serialVersionUID = -2362048321261811743L;
  66.     private static final int HASH_SEED = 31;

  67.     private static final String DUMMY_PATTERN = "";
  68.     private static final char START_FMT = ',';
  69.     private static final char END_FE = '}';
  70.     private static final char START_FE = '{';
  71.     private static final char QUOTE = '\'';

  72.     private String toPattern;
  73.     private final Map<String, ? extends FormatFactory> registry;

  74.     /**
  75.      * Create a new ExtendedMessageFormat for the default locale.
  76.      *
  77.      * @param pattern  the pattern to use, not null
  78.      * @throws IllegalArgumentException in case of a bad pattern.
  79.      */
  80.     public ExtendedMessageFormat(final String pattern) {
  81.         this(pattern, Locale.getDefault());
  82.     }

  83.     /**
  84.      * Create a new ExtendedMessageFormat.
  85.      *
  86.      * @param pattern  the pattern to use, not null
  87.      * @param locale  the locale to use, not null
  88.      * @throws IllegalArgumentException in case of a bad pattern.
  89.      */
  90.     public ExtendedMessageFormat(final String pattern, final Locale locale) {
  91.         this(pattern, locale, null);
  92.     }

  93.     /**
  94.      * Create a new ExtendedMessageFormat for the default locale.
  95.      *
  96.      * @param pattern  the pattern to use, not null
  97.      * @param registry  the registry of format factories, may be null
  98.      * @throws IllegalArgumentException in case of a bad pattern.
  99.      */
  100.     public ExtendedMessageFormat(final String pattern, final Map<String, ? extends FormatFactory> registry) {
  101.         this(pattern, Locale.getDefault(), registry);
  102.     }

  103.     /**
  104.      * Create a new ExtendedMessageFormat.
  105.      *
  106.      * @param pattern  the pattern to use, not null
  107.      * @param locale  the locale to use, not null
  108.      * @param registry  the registry of format factories, may be null
  109.      * @throws IllegalArgumentException in case of a bad pattern.
  110.      */
  111.     public ExtendedMessageFormat(final String pattern, final Locale locale, final Map<String, ? extends FormatFactory> registry) {
  112.         super(DUMMY_PATTERN);
  113.         setLocale(locale);
  114.         this.registry = registry;
  115.         applyPattern(pattern);
  116.     }

  117.     /**
  118.      * {@inheritDoc}
  119.      */
  120.     @Override
  121.     public String toPattern() {
  122.         return toPattern;
  123.     }

  124.     /**
  125.      * Apply the specified pattern.
  126.      *
  127.      * @param pattern String
  128.      */
  129.     @Override
  130.     public final void applyPattern(final String pattern) {
  131.         if (registry == null) {
  132.             super.applyPattern(pattern);
  133.             toPattern = super.toPattern();
  134.             return;
  135.         }
  136.         final ArrayList<Format> foundFormats = new ArrayList<>();
  137.         final ArrayList<String> foundDescriptions = new ArrayList<>();
  138.         final StringBuilder stripCustom = new StringBuilder(pattern.length());

  139.         final ParsePosition pos = new ParsePosition(0);
  140.         final char[] c = pattern.toCharArray();
  141.         int fmtCount = 0;
  142.         while (pos.getIndex() < pattern.length()) {
  143.             switch (c[pos.getIndex()]) {
  144.             case QUOTE:
  145.                 appendQuotedString(pattern, pos, stripCustom);
  146.                 break;
  147.             case START_FE:
  148.                 fmtCount++;
  149.                 seekNonWs(pattern, pos);
  150.                 final int start = pos.getIndex();
  151.                 final int index = readArgumentIndex(pattern, next(pos));
  152.                 stripCustom.append(START_FE).append(index);
  153.                 seekNonWs(pattern, pos);
  154.                 Format format = null;
  155.                 String formatDescription = null;
  156.                 if (c[pos.getIndex()] == START_FMT) {
  157.                     formatDescription = parseFormatDescription(pattern,
  158.                             next(pos));
  159.                     format = getFormat(formatDescription);
  160.                     if (format == null) {
  161.                         stripCustom.append(START_FMT).append(formatDescription);
  162.                     }
  163.                 }
  164.                 foundFormats.add(format);
  165.                 foundDescriptions.add(format == null ? null : formatDescription);
  166.                 if(foundFormats.size() != fmtCount) {
  167.                     throw new IllegalArgumentException("The validated expression is false");
  168.                 }
  169.                 if (foundDescriptions.size() != fmtCount) {
  170.                     throw new IllegalArgumentException("The validated expression is false");
  171.                 }
  172.                 if (c[pos.getIndex()] != END_FE) {
  173.                     throw new IllegalArgumentException(
  174.                             "Unreadable format element at position " + start);
  175.                 }
  176.                 //$FALL-THROUGH$
  177.             default:
  178.                 stripCustom.append(c[pos.getIndex()]);
  179.                 next(pos);
  180.             }
  181.         }
  182.         super.applyPattern(stripCustom.toString());
  183.         toPattern = insertFormats(super.toPattern(), foundDescriptions);
  184.         if (containsElements(foundFormats)) {
  185.             final Format[] origFormats = getFormats();
  186.             // only loop over what we know we have, as MessageFormat on Java 1.3
  187.             // seems to provide an extra format element:
  188.             int i = 0;
  189.             for (final Iterator<Format> it = foundFormats.iterator(); it.hasNext(); i++) {
  190.                 final Format f = it.next();
  191.                 if (f != null) {
  192.                     origFormats[i] = f;
  193.                 }
  194.             }
  195.             super.setFormats(origFormats);
  196.         }
  197.     }

  198.     /**
  199.      * Throws UnsupportedOperationException - see class Javadoc for details.
  200.      *
  201.      * @param formatElementIndex format element index
  202.      * @param newFormat the new format
  203.      * @throws UnsupportedOperationException always thrown since this isn't supported by ExtendMessageFormat
  204.      */
  205.     @Override
  206.     public void setFormat(final int formatElementIndex, final Format newFormat) {
  207.         throw new UnsupportedOperationException();
  208.     }

  209.     /**
  210.      * Throws UnsupportedOperationException - see class Javadoc for details.
  211.      *
  212.      * @param argumentIndex argument index
  213.      * @param newFormat the new format
  214.      * @throws UnsupportedOperationException always thrown since this isn't supported by ExtendMessageFormat
  215.      */
  216.     @Override
  217.     public void setFormatByArgumentIndex(final int argumentIndex, final Format newFormat) {
  218.         throw new UnsupportedOperationException();
  219.     }

  220.     /**
  221.      * Throws UnsupportedOperationException - see class Javadoc for details.
  222.      *
  223.      * @param newFormats new formats
  224.      * @throws UnsupportedOperationException always thrown since this isn't supported by ExtendMessageFormat
  225.      */
  226.     @Override
  227.     public void setFormats(final Format[] newFormats) {
  228.         throw new UnsupportedOperationException();
  229.     }

  230.     /**
  231.      * Throws UnsupportedOperationException - see class Javadoc for details.
  232.      *
  233.      * @param newFormats new formats
  234.      * @throws UnsupportedOperationException always thrown since this isn't supported by ExtendMessageFormat
  235.      */
  236.     @Override
  237.     public void setFormatsByArgumentIndex(final Format[] newFormats) {
  238.         throw new UnsupportedOperationException();
  239.     }

  240.     /**
  241.      * Check if this extended message format is equal to another object.
  242.      *
  243.      * @param obj the object to compare to
  244.      * @return true if this object equals the other, otherwise false
  245.      */
  246.     @Override
  247.     public boolean equals(final Object obj) {
  248.         if (obj == this) {
  249.             return true;
  250.         }
  251.         if (obj == null) {
  252.             return false;
  253.         }
  254.         if (!super.equals(obj)) {
  255.             return false;
  256.         }
  257.         if (!Objects.equals(getClass(), obj.getClass())) {
  258.           return false;
  259.         }
  260.         final ExtendedMessageFormat rhs = (ExtendedMessageFormat)obj;
  261.         if (!Objects.equals(toPattern, rhs.toPattern)) {
  262.             return false;
  263.         }
  264.         if (!Objects.equals(registry, rhs.registry)) {
  265.             return false;
  266.         }
  267.         return true;
  268.     }

  269.     /**
  270.      * {@inheritDoc}
  271.      */
  272.     @Override
  273.     public int hashCode() {
  274.         int result = super.hashCode();
  275.         result = HASH_SEED * result + Objects.hashCode(registry);
  276.         result = HASH_SEED * result + Objects.hashCode(toPattern);
  277.         return result;
  278.     }

  279.     /**
  280.      * Get a custom format from a format description.
  281.      *
  282.      * @param desc String
  283.      * @return Format
  284.      */
  285.     private Format getFormat(final String desc) {
  286.         if (registry != null) {
  287.             String name = desc;
  288.             String args = null;
  289.             final int i = desc.indexOf(START_FMT);
  290.             if (i > 0) {
  291.                 name = desc.substring(0, i).trim();
  292.                 args = desc.substring(i + 1).trim();
  293.             }
  294.             final FormatFactory factory = registry.get(name);
  295.             if (factory != null) {
  296.                 return factory.getFormat(name, args, getLocale());
  297.             }
  298.         }
  299.         return null;
  300.     }

  301.     /**
  302.      * Read the argument index from the current format element
  303.      *
  304.      * @param pattern pattern to parse
  305.      * @param pos current parse position
  306.      * @return argument index
  307.      */
  308.     private int readArgumentIndex(final String pattern, final ParsePosition pos) {
  309.         final int start = pos.getIndex();
  310.         seekNonWs(pattern, pos);
  311.         final StringBuilder result = new StringBuilder();
  312.         boolean error = false;
  313.         for (; !error && pos.getIndex() < pattern.length(); next(pos)) {
  314.             char c = pattern.charAt(pos.getIndex());
  315.             if (Character.isWhitespace(c)) {
  316.                 seekNonWs(pattern, pos);
  317.                 c = pattern.charAt(pos.getIndex());
  318.                 if (c != START_FMT && c != END_FE) {
  319.                     error = true;
  320.                     continue;
  321.                 }
  322.             }
  323.             if ((c == START_FMT || c == END_FE) && result.length() > 0) {
  324.                 try {
  325.                     return Integer.parseInt(result.toString());
  326.                 } catch (final NumberFormatException e) { // NOPMD
  327.                     // we've already ensured only digits, so unless something
  328.                     // outlandishly large was specified we should be okay.
  329.                 }
  330.             }
  331.             error = !Character.isDigit(c);
  332.             result.append(c);
  333.         }
  334.         if (error) {
  335.             throw new IllegalArgumentException(
  336.                     "Invalid format argument index at position " + start + ": "
  337.                             + pattern.substring(start, pos.getIndex()));
  338.         }
  339.         throw new IllegalArgumentException(
  340.                 "Unterminated format element at position " + start);
  341.     }

  342.     /**
  343.      * Parse the format component of a format element.
  344.      *
  345.      * @param pattern string to parse
  346.      * @param pos current parse position
  347.      * @return Format description String
  348.      */
  349.     private String parseFormatDescription(final String pattern, final ParsePosition pos) {
  350.         final int start = pos.getIndex();
  351.         seekNonWs(pattern, pos);
  352.         final int text = pos.getIndex();
  353.         int depth = 1;
  354.         for (; pos.getIndex() < pattern.length(); next(pos)) {
  355.             switch (pattern.charAt(pos.getIndex())) {
  356.             case START_FE:
  357.                 depth++;
  358.                 break;
  359.             case END_FE:
  360.                 depth--;
  361.                 if (depth == 0) {
  362.                     return pattern.substring(text, pos.getIndex());
  363.                 }
  364.                 break;
  365.             case QUOTE:
  366.                 getQuotedString(pattern, pos);
  367.                 break;
  368.             default:
  369.                 break;
  370.             }
  371.         }
  372.         throw new IllegalArgumentException(
  373.                 "Unterminated format element at position " + start);
  374.     }

  375.     /**
  376.      * Insert formats back into the pattern for toPattern() support.
  377.      *
  378.      * @param pattern source
  379.      * @param customPatterns The custom patterns to re-insert, if any
  380.      * @return full pattern
  381.      */
  382.     private String insertFormats(final String pattern, final ArrayList<String> customPatterns) {
  383.         if (!containsElements(customPatterns)) {
  384.             return pattern;
  385.         }
  386.         final StringBuilder sb = new StringBuilder(pattern.length() * 2);
  387.         final ParsePosition pos = new ParsePosition(0);
  388.         int fe = -1;
  389.         int depth = 0;
  390.         while (pos.getIndex() < pattern.length()) {
  391.             final char c = pattern.charAt(pos.getIndex());
  392.             switch (c) {
  393.             case QUOTE:
  394.                 appendQuotedString(pattern, pos, sb);
  395.                 break;
  396.             case START_FE:
  397.                 depth++;
  398.                 sb.append(START_FE).append(readArgumentIndex(pattern, next(pos)));
  399.                 // do not look for custom patterns when they are embedded, e.g. in a choice
  400.                 if (depth == 1) {
  401.                     fe++;
  402.                     final String customPattern = customPatterns.get(fe);
  403.                     if (customPattern != null) {
  404.                         sb.append(START_FMT).append(customPattern);
  405.                     }
  406.                 }
  407.                 break;
  408.             case END_FE:
  409.                 depth--;
  410.                 //$FALL-THROUGH$
  411.             default:
  412.                 sb.append(c);
  413.                 next(pos);
  414.             }
  415.         }
  416.         return sb.toString();
  417.     }

  418.     /**
  419.      * Consume whitespace from the current parse position.
  420.      *
  421.      * @param pattern String to read
  422.      * @param pos current position
  423.      */
  424.     private void seekNonWs(final String pattern, final ParsePosition pos) {
  425.         int len = 0;
  426.         final char[] buffer = pattern.toCharArray();
  427.         do {
  428.             len = StrMatcher.splitMatcher().isMatch(buffer, pos.getIndex());
  429.             pos.setIndex(pos.getIndex() + len);
  430.         } while (len > 0 && pos.getIndex() < pattern.length());
  431.     }

  432.     /**
  433.      * Convenience method to advance parse position by 1
  434.      *
  435.      * @param pos ParsePosition
  436.      * @return <code>pos</code>
  437.      */
  438.     private ParsePosition next(final ParsePosition pos) {
  439.         pos.setIndex(pos.getIndex() + 1);
  440.         return pos;
  441.     }

  442.     /**
  443.      * Consume a quoted string, adding it to <code>appendTo</code> if
  444.      * specified.
  445.      *
  446.      * @param pattern pattern to parse
  447.      * @param pos current parse position
  448.      * @param appendTo optional StringBuilder to append
  449.      * @return <code>appendTo</code>
  450.      */
  451.     private StringBuilder appendQuotedString(final String pattern, final ParsePosition pos,
  452.             final StringBuilder appendTo) {
  453.         assert pattern.toCharArray()[pos.getIndex()] == QUOTE :
  454.             "Quoted string must start with quote character";

  455.         // handle quote character at the beginning of the string
  456.         if(appendTo != null) {
  457.             appendTo.append(QUOTE);
  458.         }
  459.         next(pos);

  460.         final int start = pos.getIndex();
  461.         final char[] c = pattern.toCharArray();
  462.         final int lastHold = start;
  463.         for (int i = pos.getIndex(); i < pattern.length(); i++) {
  464.             switch (c[pos.getIndex()]) {
  465.             case QUOTE:
  466.                 next(pos);
  467.                 return appendTo == null ? null : appendTo.append(c, lastHold,
  468.                         pos.getIndex() - lastHold);
  469.             default:
  470.                 next(pos);
  471.             }
  472.         }
  473.         throw new IllegalArgumentException(
  474.                 "Unterminated quoted string at position " + start);
  475.     }

  476.     /**
  477.      * Consume quoted string only
  478.      *
  479.      * @param pattern pattern to parse
  480.      * @param pos current parse position
  481.      */
  482.     private void getQuotedString(final String pattern, final ParsePosition pos) {
  483.         appendQuotedString(pattern, pos, null);
  484.     }

  485.     /**
  486.      * Learn whether the specified Collection contains non-null elements.
  487.      * @param coll to check
  488.      * @return <code>true</code> if some Object was found, <code>false</code> otherwise.
  489.      */
  490.     private boolean containsElements(final Collection<?> coll) {
  491.         if (coll == null || coll.isEmpty()) {
  492.             return false;
  493.         }
  494.         for (final Object name : coll) {
  495.             if (name != null) {
  496.                 return true;
  497.             }
  498.         }
  499.         return false;
  500.     }
  501. }