StrSubstitutor.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.util.ArrayList;
  19. import java.util.Enumeration;
  20. import java.util.HashMap;
  21. import java.util.List;
  22. import java.util.Map;
  23. import java.util.Properties;

  24. /**
  25.  * Substitutes variables within a string by values.
  26.  * <p>
  27.  * This class takes a piece of text and substitutes all the variables within it.
  28.  * The default definition of a variable is <code>${variableName}</code>.
  29.  * The prefix and suffix can be changed via constructors and set methods.
  30.  * <p>
  31.  * Variable values are typically resolved from a map, but could also be resolved
  32.  * from system properties, or by supplying a custom variable resolver.
  33.  * <p>
  34.  * The simplest example is to use this class to replace Java System properties. For example:
  35.  * <pre>
  36.  * StrSubstitutor.replaceSystemProperties(
  37.  *      "You are running with java.version = ${java.version} and os.name = ${os.name}.");
  38.  * </pre>
  39.  * <p>
  40.  * Typical usage of this class follows the following pattern: First an instance is created
  41.  * and initialized with the map that contains the values for the available variables.
  42.  * If a prefix and/or suffix for variables should be used other than the default ones,
  43.  * the appropriate settings can be performed. After that the <code>replace()</code>
  44.  * method can be called passing in the source text for interpolation. In the returned
  45.  * text all variable references (as long as their values are known) will be resolved.
  46.  * The following example demonstrates this:
  47.  * <pre>
  48.  * Map valuesMap = HashMap();
  49.  * valuesMap.put(&quot;animal&quot;, &quot;quick brown fox&quot;);
  50.  * valuesMap.put(&quot;target&quot;, &quot;lazy dog&quot;);
  51.  * String templateString = &quot;The ${animal} jumped over the ${target}.&quot;;
  52.  * StrSubstitutor sub = new StrSubstitutor(valuesMap);
  53.  * String resolvedString = sub.replace(templateString);
  54.  * </pre>
  55.  * yielding:
  56.  * <pre>
  57.  *      The quick brown fox jumped over the lazy dog.
  58.  * </pre>
  59.  * <p>
  60.  * Also, this class allows to set a default value for unresolved variables.
  61.  * The default value for a variable can be appended to the variable name after the variable
  62.  * default value delimiter. The default value of the variable default value delimiter is ':-',
  63.  * as in bash and other *nix shells, as those are arguably where the default ${} delimiter set originated.
  64.  * The variable default value delimiter can be manually set by calling {@link #setValueDelimiterMatcher(StrMatcher)},
  65.  * {@link #setValueDelimiter(char)} or {@link #setValueDelimiter(String)}.
  66.  * The following shows an example with variable default value settings:
  67.  * <pre>
  68.  * Map valuesMap = HashMap();
  69.  * valuesMap.put(&quot;animal&quot;, &quot;quick brown fox&quot;);
  70.  * valuesMap.put(&quot;target&quot;, &quot;lazy dog&quot;);
  71.  * String templateString = &quot;The ${animal} jumped over the ${target}. ${undefined.number:-1234567890}.&quot;;
  72.  * StrSubstitutor sub = new StrSubstitutor(valuesMap);
  73.  * String resolvedString = sub.replace(templateString);
  74.  * </pre>
  75.  * yielding:
  76.  * <pre>
  77.  *      The quick brown fox jumped over the lazy dog. 1234567890.
  78.  * </pre>
  79.  * <p>
  80.  * In addition to this usage pattern there are some static convenience methods that
  81.  * cover the most common use cases. These methods can be used without the need of
  82.  * manually creating an instance. However if multiple replace operations are to be
  83.  * performed, creating and reusing an instance of this class will be more efficient.
  84.  * <p>
  85.  * Variable replacement works in a recursive way. Thus, if a variable value contains
  86.  * a variable then that variable will also be replaced. Cyclic replacements are
  87.  * detected and will cause an exception to be thrown.
  88.  * <p>
  89.  * Sometimes the interpolation's result must contain a variable prefix. As an example
  90.  * take the following source text:
  91.  * <pre>
  92.  *   The variable ${${name}} must be used.
  93.  * </pre>
  94.  * Here only the variable's name referred to in the text should be replaced resulting
  95.  * in the text (assuming that the value of the <code>name</code> variable is <code>x</code>):
  96.  * <pre>
  97.  *   The variable ${x} must be used.
  98.  * </pre>
  99.  * To achieve this effect there are two possibilities: Either set a different prefix
  100.  * and suffix for variables which do not conflict with the result text you want to
  101.  * produce. The other possibility is to use the escape character, by default '$'.
  102.  * If this character is placed before a variable reference, this reference is ignored
  103.  * and won't be replaced. For example:
  104.  * <pre>
  105.  *   The variable $${${name}} must be used.
  106.  * </pre>
  107.  * <p>
  108.  * In some complex scenarios you might even want to perform substitution in the
  109.  * names of variables, for instance
  110.  * <pre>
  111.  * ${jre-${java.specification.version}}
  112.  * </pre>
  113.  * <code>StrSubstitutor</code> supports this recursive substitution in variable
  114.  * names, but it has to be enabled explicitly by setting the
  115.  * {@link #setEnableSubstitutionInVariables(boolean) enableSubstitutionInVariables}
  116.  * property to <b>true</b>.
  117.  * <p>This class is <b>not</b> thread safe.</p>
  118.  *
  119.  * @since 1.0
  120.  */
  121. public class StrSubstitutor {

  122.     /**
  123.      * Constant for the default escape character.
  124.      */
  125.     public static final char DEFAULT_ESCAPE = '$';
  126.     /**
  127.      * Constant for the default variable prefix.
  128.      */
  129.     public static final StrMatcher DEFAULT_PREFIX = StrMatcher.stringMatcher("${");
  130.     /**
  131.      * Constant for the default variable suffix.
  132.      */
  133.     public static final StrMatcher DEFAULT_SUFFIX = StrMatcher.stringMatcher("}");
  134.     /**
  135.      * Constant for the default value delimiter of a variable.
  136.      */
  137.     public static final StrMatcher DEFAULT_VALUE_DELIMITER = StrMatcher.stringMatcher(":-");

  138.     /**
  139.      * Stores the escape character.
  140.      */
  141.     private char escapeChar;
  142.     /**
  143.      * Stores the variable prefix.
  144.      */
  145.     private StrMatcher prefixMatcher;
  146.     /**
  147.      * Stores the variable suffix.
  148.      */
  149.     private StrMatcher suffixMatcher;
  150.     /**
  151.      * Stores the default variable value delimiter
  152.      */
  153.     private StrMatcher valueDelimiterMatcher;
  154.     /**
  155.      * Variable resolution is delegated to an implementor of VariableResolver.
  156.      */
  157.     private StrLookup<?> variableResolver;
  158.     /**
  159.      * The flag whether substitution in variable names is enabled.
  160.      */
  161.     private boolean enableSubstitutionInVariables;
  162.     /**
  163.      * Whether escapes should be preserved.  Default is false;
  164.      */
  165.     private boolean preserveEscapes = false;

  166.     //-----------------------------------------------------------------------
  167.     /**
  168.      * Replaces all the occurrences of variables in the given source object with
  169.      * their matching values from the map.
  170.      *
  171.      * @param <V> the type of the values in the map
  172.      * @param source  the source text containing the variables to substitute, null returns null
  173.      * @param valueMap  the map with the values, may be null
  174.      * @return the result of the replace operation
  175.      */
  176.     public static <V> String replace(final Object source, final Map<String, V> valueMap) {
  177.         return new StrSubstitutor(valueMap).replace(source);
  178.     }

  179.     /**
  180.      * Replaces all the occurrences of variables in the given source object with
  181.      * their matching values from the map. This method allows to specifiy a
  182.      * custom variable prefix and suffix
  183.      *
  184.      * @param <V> the type of the values in the map
  185.      * @param source  the source text containing the variables to substitute, null returns null
  186.      * @param valueMap  the map with the values, may be null
  187.      * @param prefix  the prefix of variables, not null
  188.      * @param suffix  the suffix of variables, not null
  189.      * @return the result of the replace operation
  190.      * @throws IllegalArgumentException if the prefix or suffix is null
  191.      */
  192.     public static <V> String replace(final Object source, final Map<String, V> valueMap, final String prefix, final String suffix) {
  193.         return new StrSubstitutor(valueMap, prefix, suffix).replace(source);
  194.     }

  195.     /**
  196.      * Replaces all the occurrences of variables in the given source object with their matching
  197.      * values from the properties.
  198.      *
  199.      * @param source the source text containing the variables to substitute, null returns null
  200.      * @param valueProperties the properties with values, may be null
  201.      * @return the result of the replace operation
  202.      */
  203.     public static String replace(final Object source, final Properties valueProperties) {
  204.         if (valueProperties == null) {
  205.             return source.toString();
  206.         }
  207.         final Map<String,String> valueMap = new HashMap<>();
  208.         final Enumeration<?> propNames = valueProperties.propertyNames();
  209.         while (propNames.hasMoreElements()) {
  210.             final String propName = (String)propNames.nextElement();
  211.             final String propValue = valueProperties.getProperty(propName);
  212.             valueMap.put(propName, propValue);
  213.         }
  214.         return StrSubstitutor.replace(source, valueMap);
  215.     }

  216.     /**
  217.      * Replaces all the occurrences of variables in the given source object with
  218.      * their matching values from the system properties.
  219.      *
  220.      * @param source  the source text containing the variables to substitute, null returns null
  221.      * @return the result of the replace operation
  222.      */
  223.     public static String replaceSystemProperties(final Object source) {
  224.         return new StrSubstitutor(StrLookup.systemPropertiesLookup()).replace(source);
  225.     }

  226.     //-----------------------------------------------------------------------
  227.     /**
  228.      * Creates a new instance with defaults for variable prefix and suffix
  229.      * and the escaping character.
  230.      */
  231.     public StrSubstitutor() {
  232.         this((StrLookup<?>) null, DEFAULT_PREFIX, DEFAULT_SUFFIX, DEFAULT_ESCAPE);
  233.     }

  234.     /**
  235.      * Creates a new instance and initializes it. Uses defaults for variable
  236.      * prefix and suffix and the escaping character.
  237.      *
  238.      * @param <V> the type of the values in the map
  239.      * @param valueMap  the map with the variables' values, may be null
  240.      */
  241.     public <V> StrSubstitutor(final Map<String, V> valueMap) {
  242.         this(StrLookup.mapLookup(valueMap), DEFAULT_PREFIX, DEFAULT_SUFFIX, DEFAULT_ESCAPE);
  243.     }

  244.     /**
  245.      * Creates a new instance and initializes it. Uses a default escaping character.
  246.      *
  247.      * @param <V> the type of the values in the map
  248.      * @param valueMap  the map with the variables' values, may be null
  249.      * @param prefix  the prefix for variables, not null
  250.      * @param suffix  the suffix for variables, not null
  251.      * @throws IllegalArgumentException if the prefix or suffix is null
  252.      */
  253.     public <V> StrSubstitutor(final Map<String, V> valueMap, final String prefix, final String suffix) {
  254.         this(StrLookup.mapLookup(valueMap), prefix, suffix, DEFAULT_ESCAPE);
  255.     }

  256.     /**
  257.      * Creates a new instance and initializes it.
  258.      *
  259.      * @param <V> the type of the values in the map
  260.      * @param valueMap  the map with the variables' values, may be null
  261.      * @param prefix  the prefix for variables, not null
  262.      * @param suffix  the suffix for variables, not null
  263.      * @param escape  the escape character
  264.      * @throws IllegalArgumentException if the prefix or suffix is null
  265.      */
  266.     public <V> StrSubstitutor(final Map<String, V> valueMap, final String prefix, final String suffix,
  267.                               final char escape) {
  268.         this(StrLookup.mapLookup(valueMap), prefix, suffix, escape);
  269.     }

  270.     /**
  271.      * Creates a new instance and initializes it.
  272.      *
  273.      * @param <V> the type of the values in the map
  274.      * @param valueMap  the map with the variables' values, may be null
  275.      * @param prefix  the prefix for variables, not null
  276.      * @param suffix  the suffix for variables, not null
  277.      * @param escape  the escape character
  278.      * @param valueDelimiter  the variable default value delimiter, may be null
  279.      * @throws IllegalArgumentException if the prefix or suffix is null
  280.      */
  281.     public <V> StrSubstitutor(final Map<String, V> valueMap, final String prefix, final String suffix,
  282.                               final char escape, final String valueDelimiter) {
  283.         this(StrLookup.mapLookup(valueMap), prefix, suffix, escape, valueDelimiter);
  284.     }

  285.     /**
  286.      * Creates a new instance and initializes it.
  287.      *
  288.      * @param variableResolver  the variable resolver, may be null
  289.      */
  290.     public StrSubstitutor(final StrLookup<?> variableResolver) {
  291.         this(variableResolver, DEFAULT_PREFIX, DEFAULT_SUFFIX, DEFAULT_ESCAPE);
  292.     }

  293.     /**
  294.      * Creates a new instance and initializes it.
  295.      *
  296.      * @param variableResolver  the variable resolver, may be null
  297.      * @param prefix  the prefix for variables, not null
  298.      * @param suffix  the suffix for variables, not null
  299.      * @param escape  the escape character
  300.      * @throws IllegalArgumentException if the prefix or suffix is null
  301.      */
  302.     public StrSubstitutor(final StrLookup<?> variableResolver, final String prefix, final String suffix,
  303.                           final char escape) {
  304.         this.setVariableResolver(variableResolver);
  305.         this.setVariablePrefix(prefix);
  306.         this.setVariableSuffix(suffix);
  307.         this.setEscapeChar(escape);
  308.         this.setValueDelimiterMatcher(DEFAULT_VALUE_DELIMITER);
  309.     }

  310.     /**
  311.      * Creates a new instance and initializes it.
  312.      *
  313.      * @param variableResolver  the variable resolver, may be null
  314.      * @param prefix  the prefix for variables, not null
  315.      * @param suffix  the suffix for variables, not null
  316.      * @param escape  the escape character
  317.      * @param valueDelimiter  the variable default value delimiter string, may be null
  318.      * @throws IllegalArgumentException if the prefix or suffix is null
  319.      */
  320.     public StrSubstitutor(final StrLookup<?> variableResolver, final String prefix, final String suffix,
  321.                           final char escape, final String valueDelimiter) {
  322.         this.setVariableResolver(variableResolver);
  323.         this.setVariablePrefix(prefix);
  324.         this.setVariableSuffix(suffix);
  325.         this.setEscapeChar(escape);
  326.         this.setValueDelimiter(valueDelimiter);
  327.     }

  328.     /**
  329.      * Creates a new instance and initializes it.
  330.      *
  331.      * @param variableResolver  the variable resolver, may be null
  332.      * @param prefixMatcher  the prefix for variables, not null
  333.      * @param suffixMatcher  the suffix for variables, not null
  334.      * @param escape  the escape character
  335.      * @throws IllegalArgumentException if the prefix or suffix is null
  336.      */
  337.     public StrSubstitutor(
  338.             final StrLookup<?> variableResolver, final StrMatcher prefixMatcher, final StrMatcher suffixMatcher,
  339.             final char escape) {
  340.         this(variableResolver, prefixMatcher, suffixMatcher, escape, DEFAULT_VALUE_DELIMITER);
  341.     }

  342.     /**
  343.      * Creates a new instance and initializes it.
  344.      *
  345.      * @param variableResolver  the variable resolver, may be null
  346.      * @param prefixMatcher  the prefix for variables, not null
  347.      * @param suffixMatcher  the suffix for variables, not null
  348.      * @param escape  the escape character
  349.      * @param valueDelimiterMatcher  the variable default value delimiter matcher, may be null
  350.      * @throws IllegalArgumentException if the prefix or suffix is null
  351.      */
  352.     public StrSubstitutor(
  353.             final StrLookup<?> variableResolver, final StrMatcher prefixMatcher, final StrMatcher suffixMatcher,
  354.             final char escape, final StrMatcher valueDelimiterMatcher) {
  355.         this.setVariableResolver(variableResolver);
  356.         this.setVariablePrefixMatcher(prefixMatcher);
  357.         this.setVariableSuffixMatcher(suffixMatcher);
  358.         this.setEscapeChar(escape);
  359.         this.setValueDelimiterMatcher(valueDelimiterMatcher);
  360.     }

  361.     //-----------------------------------------------------------------------
  362.     /**
  363.      * Replaces all the occurrences of variables with their matching values
  364.      * from the resolver using the given source string as a template.
  365.      *
  366.      * @param source  the string to replace in, null returns null
  367.      * @return the result of the replace operation
  368.      */
  369.     public String replace(final String source) {
  370.         if (source == null) {
  371.             return null;
  372.         }
  373.         final StrBuilder buf = new StrBuilder(source);
  374.         if (substitute(buf, 0, source.length()) == false) {
  375.             return source;
  376.         }
  377.         return buf.toString();
  378.     }

  379.     /**
  380.      * Replaces all the occurrences of variables with their matching values
  381.      * from the resolver using the given source string as a template.
  382.      * <p>
  383.      * Only the specified portion of the string will be processed.
  384.      * The rest of the string is not processed, and is not returned.
  385.      *
  386.      * @param source  the string to replace in, null returns null
  387.      * @param offset  the start offset within the array, must be valid
  388.      * @param length  the length within the array to be processed, must be valid
  389.      * @return the result of the replace operation
  390.      */
  391.     public String replace(final String source, final int offset, final int length) {
  392.         if (source == null) {
  393.             return null;
  394.         }
  395.         final StrBuilder buf = new StrBuilder(length).append(source, offset, length);
  396.         if (substitute(buf, 0, length) == false) {
  397.             return source.substring(offset, offset + length);
  398.         }
  399.         return buf.toString();
  400.     }

  401.     //-----------------------------------------------------------------------
  402.     /**
  403.      * Replaces all the occurrences of variables with their matching values
  404.      * from the resolver using the given source array as a template.
  405.      * The array is not altered by this method.
  406.      *
  407.      * @param source  the character array to replace in, not altered, null returns null
  408.      * @return the result of the replace operation
  409.      */
  410.     public String replace(final char[] source) {
  411.         if (source == null) {
  412.             return null;
  413.         }
  414.         final StrBuilder buf = new StrBuilder(source.length).append(source);
  415.         substitute(buf, 0, source.length);
  416.         return buf.toString();
  417.     }

  418.     /**
  419.      * Replaces all the occurrences of variables with their matching values
  420.      * from the resolver using the given source array as a template.
  421.      * The array is not altered by this method.
  422.      * <p>
  423.      * Only the specified portion of the array will be processed.
  424.      * The rest of the array is not processed, and is not returned.
  425.      *
  426.      * @param source  the character array to replace in, not altered, null returns null
  427.      * @param offset  the start offset within the array, must be valid
  428.      * @param length  the length within the array to be processed, must be valid
  429.      * @return the result of the replace operation
  430.      */
  431.     public String replace(final char[] source, final int offset, final int length) {
  432.         if (source == null) {
  433.             return null;
  434.         }
  435.         final StrBuilder buf = new StrBuilder(length).append(source, offset, length);
  436.         substitute(buf, 0, length);
  437.         return buf.toString();
  438.     }

  439.     //-----------------------------------------------------------------------
  440.     /**
  441.      * Replaces all the occurrences of variables with their matching values
  442.      * from the resolver using the given source buffer as a template.
  443.      * The buffer is not altered by this method.
  444.      *
  445.      * @param source  the buffer to use as a template, not changed, null returns null
  446.      * @return the result of the replace operation
  447.      */
  448.     public String replace(final StringBuffer source) {
  449.         if (source == null) {
  450.             return null;
  451.         }
  452.         final StrBuilder buf = new StrBuilder(source.length()).append(source);
  453.         substitute(buf, 0, buf.length());
  454.         return buf.toString();
  455.     }

  456.     /**
  457.      * Replaces all the occurrences of variables with their matching values
  458.      * from the resolver using the given source buffer as a template.
  459.      * The buffer is not altered by this method.
  460.      * <p>
  461.      * Only the specified portion of the buffer will be processed.
  462.      * The rest of the buffer is not processed, and is not returned.
  463.      *
  464.      * @param source  the buffer to use as a template, not changed, null returns null
  465.      * @param offset  the start offset within the array, must be valid
  466.      * @param length  the length within the array to be processed, must be valid
  467.      * @return the result of the replace operation
  468.      */
  469.     public String replace(final StringBuffer source, final int offset, final int length) {
  470.         if (source == null) {
  471.             return null;
  472.         }
  473.         final StrBuilder buf = new StrBuilder(length).append(source, offset, length);
  474.         substitute(buf, 0, length);
  475.         return buf.toString();
  476.     }

  477.     /**
  478.      * Replaces all the occurrences of variables with their matching values
  479.      * from the resolver using the given source as a template.
  480.      * The source is not altered by this method.
  481.      *
  482.      * @param source  the buffer to use as a template, not changed, null returns null
  483.      * @return the result of the replace operation
  484.      */
  485.     public String replace(final CharSequence source) {
  486.         if (source == null) {
  487.             return null;
  488.         }
  489.         return replace(source, 0, source.length());
  490.     }

  491.     /**
  492.      * Replaces all the occurrences of variables with their matching values
  493.      * from the resolver using the given source as a template.
  494.      * The source is not altered by this method.
  495.      * <p>
  496.      * Only the specified portion of the buffer will be processed.
  497.      * The rest of the buffer is not processed, and is not returned.
  498.      *
  499.      * @param source  the buffer to use as a template, not changed, null returns null
  500.      * @param offset  the start offset within the array, must be valid
  501.      * @param length  the length within the array to be processed, must be valid
  502.      * @return the result of the replace operation
  503.      */
  504.     public String replace(final CharSequence source, final int offset, final int length) {
  505.         if (source == null) {
  506.             return null;
  507.         }
  508.         final StrBuilder buf = new StrBuilder(length).append(source, offset, length);
  509.         substitute(buf, 0, length);
  510.         return buf.toString();
  511.     }

  512.     //-----------------------------------------------------------------------
  513.     /**
  514.      * Replaces all the occurrences of variables with their matching values
  515.      * from the resolver using the given source builder as a template.
  516.      * The builder is not altered by this method.
  517.      *
  518.      * @param source  the builder to use as a template, not changed, null returns null
  519.      * @return the result of the replace operation
  520.      */
  521.     public String replace(final StrBuilder source) {
  522.         if (source == null) {
  523.             return null;
  524.         }
  525.         final StrBuilder buf = new StrBuilder(source.length()).append(source);
  526.         substitute(buf, 0, buf.length());
  527.         return buf.toString();
  528.     }

  529.     /**
  530.      * Replaces all the occurrences of variables with their matching values
  531.      * from the resolver using the given source builder as a template.
  532.      * The builder is not altered by this method.
  533.      * <p>
  534.      * Only the specified portion of the builder will be processed.
  535.      * The rest of the builder is not processed, and is not returned.
  536.      *
  537.      * @param source  the builder to use as a template, not changed, null returns null
  538.      * @param offset  the start offset within the array, must be valid
  539.      * @param length  the length within the array to be processed, must be valid
  540.      * @return the result of the replace operation
  541.      */
  542.     public String replace(final StrBuilder source, final int offset, final int length) {
  543.         if (source == null) {
  544.             return null;
  545.         }
  546.         final StrBuilder buf = new StrBuilder(length).append(source, offset, length);
  547.         substitute(buf, 0, length);
  548.         return buf.toString();
  549.     }

  550.     //-----------------------------------------------------------------------
  551.     /**
  552.      * Replaces all the occurrences of variables in the given source object with
  553.      * their matching values from the resolver. The input source object is
  554.      * converted to a string using <code>toString</code> and is not altered.
  555.      *
  556.      * @param source  the source to replace in, null returns null
  557.      * @return the result of the replace operation
  558.      */
  559.     public String replace(final Object source) {
  560.         if (source == null) {
  561.             return null;
  562.         }
  563.         final StrBuilder buf = new StrBuilder().append(source);
  564.         substitute(buf, 0, buf.length());
  565.         return buf.toString();
  566.     }

  567.     //-----------------------------------------------------------------------
  568.     /**
  569.      * Replaces all the occurrences of variables within the given source buffer
  570.      * with their matching values from the resolver.
  571.      * The buffer is updated with the result.
  572.      *
  573.      * @param source  the buffer to replace in, updated, null returns zero
  574.      * @return true if altered
  575.      */
  576.     public boolean replaceIn(final StringBuffer source) {
  577.         if (source == null) {
  578.             return false;
  579.         }
  580.         return replaceIn(source, 0, source.length());
  581.     }

  582.     /**
  583.      * Replaces all the occurrences of variables within the given source buffer
  584.      * with their matching values from the resolver.
  585.      * The buffer is updated with the result.
  586.      * <p>
  587.      * Only the specified portion of the buffer will be processed.
  588.      * The rest of the buffer is not processed, but it is not deleted.
  589.      *
  590.      * @param source  the buffer to replace in, updated, null returns zero
  591.      * @param offset  the start offset within the array, must be valid
  592.      * @param length  the length within the buffer to be processed, must be valid
  593.      * @return true if altered
  594.      */
  595.     public boolean replaceIn(final StringBuffer source, final int offset, final int length) {
  596.         if (source == null) {
  597.             return false;
  598.         }
  599.         final StrBuilder buf = new StrBuilder(length).append(source, offset, length);
  600.         if (substitute(buf, 0, length) == false) {
  601.             return false;
  602.         }
  603.         source.replace(offset, offset + length, buf.toString());
  604.         return true;
  605.     }

  606.   //-----------------------------------------------------------------------
  607.     /**
  608.      * Replaces all the occurrences of variables within the given source buffer
  609.      * with their matching values from the resolver.
  610.      * The buffer is updated with the result.
  611.      *
  612.      * @param source  the buffer to replace in, updated, null returns zero
  613.      * @return true if altered
  614.      */
  615.     public boolean replaceIn(final StringBuilder source) {
  616.         if (source == null) {
  617.             return false;
  618.         }
  619.         return replaceIn(source, 0, source.length());
  620.     }

  621.     /**
  622.      * Replaces all the occurrences of variables within the given source builder
  623.      * with their matching values from the resolver.
  624.      * The builder is updated with the result.
  625.      * <p>
  626.      * Only the specified portion of the buffer will be processed.
  627.      * The rest of the buffer is not processed, but it is not deleted.
  628.      *
  629.      * @param source  the buffer to replace in, updated, null returns zero
  630.      * @param offset  the start offset within the array, must be valid
  631.      * @param length  the length within the buffer to be processed, must be valid
  632.      * @return true if altered
  633.      */
  634.     public boolean replaceIn(final StringBuilder source, final int offset, final int length) {
  635.         if (source == null) {
  636.             return false;
  637.         }
  638.         final StrBuilder buf = new StrBuilder(length).append(source, offset, length);
  639.         if (substitute(buf, 0, length) == false) {
  640.             return false;
  641.         }
  642.         source.replace(offset, offset + length, buf.toString());
  643.         return true;
  644.     }

  645.     //-----------------------------------------------------------------------
  646.     /**
  647.      * Replaces all the occurrences of variables within the given source
  648.      * builder with their matching values from the resolver.
  649.      *
  650.      * @param source  the builder to replace in, updated, null returns zero
  651.      * @return true if altered
  652.      */
  653.     public boolean replaceIn(final StrBuilder source) {
  654.         if (source == null) {
  655.             return false;
  656.         }
  657.         return substitute(source, 0, source.length());
  658.     }

  659.     /**
  660.      * Replaces all the occurrences of variables within the given source
  661.      * builder with their matching values from the resolver.
  662.      * <p>
  663.      * Only the specified portion of the builder will be processed.
  664.      * The rest of the builder is not processed, but it is not deleted.
  665.      *
  666.      * @param source  the builder to replace in, null returns zero
  667.      * @param offset  the start offset within the array, must be valid
  668.      * @param length  the length within the builder to be processed, must be valid
  669.      * @return true if altered
  670.      */
  671.     public boolean replaceIn(final StrBuilder source, final int offset, final int length) {
  672.         if (source == null) {
  673.             return false;
  674.         }
  675.         return substitute(source, offset, length);
  676.     }

  677.     //-----------------------------------------------------------------------
  678.     /**
  679.      * Internal method that substitutes the variables.
  680.      * <p>
  681.      * Most users of this class do not need to call this method. This method will
  682.      * be called automatically by another (public) method.
  683.      * <p>
  684.      * Writers of subclasses can override this method if they need access to
  685.      * the substitution process at the start or end.
  686.      *
  687.      * @param buf  the string builder to substitute into, not null
  688.      * @param offset  the start offset within the builder, must be valid
  689.      * @param length  the length within the builder to be processed, must be valid
  690.      * @return true if altered
  691.      */
  692.     protected boolean substitute(final StrBuilder buf, final int offset, final int length) {
  693.         return substitute(buf, offset, length, null) > 0;
  694.     }

  695.     /**
  696.      * Recursive handler for multiple levels of interpolation. This is the main
  697.      * interpolation method, which resolves the values of all variable references
  698.      * contained in the passed in text.
  699.      *
  700.      * @param buf  the string builder to substitute into, not null
  701.      * @param offset  the start offset within the builder, must be valid
  702.      * @param length  the length within the builder to be processed, must be valid
  703.      * @param priorVariables  the stack keeping track of the replaced variables, may be null
  704.      * @return the length change that occurs, unless priorVariables is null when the int
  705.      *  represents a boolean flag as to whether any change occurred.
  706.      */
  707.     private int substitute(final StrBuilder buf, final int offset, final int length, List<String> priorVariables) {
  708.         final StrMatcher pfxMatcher = getVariablePrefixMatcher();
  709.         final StrMatcher suffMatcher = getVariableSuffixMatcher();
  710.         final char escape = getEscapeChar();
  711.         final StrMatcher valueDelimMatcher = getValueDelimiterMatcher();
  712.         final boolean substitutionInVariablesEnabled = isEnableSubstitutionInVariables();

  713.         final boolean top = priorVariables == null;
  714.         boolean altered = false;
  715.         int lengthChange = 0;
  716.         char[] chars = buf.buffer;
  717.         int bufEnd = offset + length;
  718.         int pos = offset;
  719.         while (pos < bufEnd) {
  720.             final int startMatchLen = pfxMatcher.isMatch(chars, pos, offset,
  721.                     bufEnd);
  722.             if (startMatchLen == 0) {
  723.                 pos++;
  724.             } else {
  725.                 // found variable start marker
  726.                 if (pos > offset && chars[pos - 1] == escape) {
  727.                     // escaped
  728.                     if (preserveEscapes) {
  729.                         pos++;
  730.                         continue;
  731.                     }
  732.                     buf.deleteCharAt(pos - 1);
  733.                     chars = buf.buffer; // in case buffer was altered
  734.                     lengthChange--;
  735.                     altered = true;
  736.                     bufEnd--;
  737.                 } else {
  738.                     // find suffix
  739.                     final int startPos = pos;
  740.                     pos += startMatchLen;
  741.                     int endMatchLen = 0;
  742.                     int nestedVarCount = 0;
  743.                     while (pos < bufEnd) {
  744.                         if (substitutionInVariablesEnabled
  745.                                 && (endMatchLen = pfxMatcher.isMatch(chars,
  746.                                         pos, offset, bufEnd)) != 0) {
  747.                             // found a nested variable start
  748.                             nestedVarCount++;
  749.                             pos += endMatchLen;
  750.                             continue;
  751.                         }

  752.                         endMatchLen = suffMatcher.isMatch(chars, pos, offset,
  753.                                 bufEnd);
  754.                         if (endMatchLen == 0) {
  755.                             pos++;
  756.                         } else {
  757.                             // found variable end marker
  758.                             if (nestedVarCount == 0) {
  759.                                 String varNameExpr = new String(chars, startPos
  760.                                         + startMatchLen, pos - startPos
  761.                                         - startMatchLen);
  762.                                 if (substitutionInVariablesEnabled) {
  763.                                     final StrBuilder bufName = new StrBuilder(varNameExpr);
  764.                                     substitute(bufName, 0, bufName.length());
  765.                                     varNameExpr = bufName.toString();
  766.                                 }
  767.                                 pos += endMatchLen;
  768.                                 final int endPos = pos;

  769.                                 String varName = varNameExpr;
  770.                                 String varDefaultValue = null;

  771.                                 if (valueDelimMatcher != null) {
  772.                                     final char [] varNameExprChars = varNameExpr.toCharArray();
  773.                                     int valueDelimiterMatchLen = 0;
  774.                                     for (int i = 0; i < varNameExprChars.length; i++) {
  775.                                         // if there's any nested variable when nested variable substitution disabled, then stop resolving name and default value.
  776.                                         if (!substitutionInVariablesEnabled
  777.                                                 && pfxMatcher.isMatch(varNameExprChars, i, i, varNameExprChars.length) != 0) {
  778.                                             break;
  779.                                         }
  780.                                         if ((valueDelimiterMatchLen = valueDelimMatcher.isMatch(varNameExprChars, i)) != 0) {
  781.                                             varName = varNameExpr.substring(0, i);
  782.                                             varDefaultValue = varNameExpr.substring(i + valueDelimiterMatchLen);
  783.                                             break;
  784.                                         }
  785.                                     }
  786.                                 }

  787.                                 // on the first call initialize priorVariables
  788.                                 if (priorVariables == null) {
  789.                                     priorVariables = new ArrayList<>();
  790.                                     priorVariables.add(new String(chars,
  791.                                             offset, length));
  792.                                 }

  793.                                 // handle cyclic substitution
  794.                                 checkCyclicSubstitution(varName, priorVariables);
  795.                                 priorVariables.add(varName);

  796.                                 // resolve the variable
  797.                                 String varValue = resolveVariable(varName, buf,
  798.                                         startPos, endPos);
  799.                                 if (varValue == null) {
  800.                                     varValue = varDefaultValue;
  801.                                 }
  802.                                 if (varValue != null) {
  803.                                     // recursive replace
  804.                                     final int varLen = varValue.length();
  805.                                     buf.replace(startPos, endPos, varValue);
  806.                                     altered = true;
  807.                                     int change = substitute(buf, startPos,
  808.                                             varLen, priorVariables);
  809.                                     change = change
  810.                                             + varLen - (endPos - startPos);
  811.                                     pos += change;
  812.                                     bufEnd += change;
  813.                                     lengthChange += change;
  814.                                     chars = buf.buffer; // in case buffer was
  815.                                                         // altered
  816.                                 }

  817.                                 // remove variable from the cyclic stack
  818.                                 priorVariables
  819.                                         .remove(priorVariables.size() - 1);
  820.                                 break;
  821.                             }
  822.                             nestedVarCount--;
  823.                             pos += endMatchLen;
  824.                         }
  825.                     }
  826.                 }
  827.             }
  828.         }
  829.         if (top) {
  830.             return altered ? 1 : 0;
  831.         }
  832.         return lengthChange;
  833.     }

  834.     /**
  835.      * Checks if the specified variable is already in the stack (list) of variables.
  836.      *
  837.      * @param varName  the variable name to check
  838.      * @param priorVariables  the list of prior variables
  839.      */
  840.     private void checkCyclicSubstitution(final String varName, final List<String> priorVariables) {
  841.         if (priorVariables.contains(varName) == false) {
  842.             return;
  843.         }
  844.         final StrBuilder buf = new StrBuilder(256);
  845.         buf.append("Infinite loop in property interpolation of ");
  846.         buf.append(priorVariables.remove(0));
  847.         buf.append(": ");
  848.         buf.appendWithSeparators(priorVariables, "->");
  849.         throw new IllegalStateException(buf.toString());
  850.     }

  851.     /**
  852.      * Internal method that resolves the value of a variable.
  853.      * <p>
  854.      * Most users of this class do not need to call this method. This method is
  855.      * called automatically by the substitution process.
  856.      * <p>
  857.      * Writers of subclasses can override this method if they need to alter
  858.      * how each substitution occurs. The method is passed the variable's name
  859.      * and must return the corresponding value. This implementation uses the
  860.      * {@link #getVariableResolver()} with the variable's name as the key.
  861.      *
  862.      * @param variableName  the name of the variable, not null
  863.      * @param buf  the buffer where the substitution is occurring, not null
  864.      * @param startPos  the start position of the variable including the prefix, valid
  865.      * @param endPos  the end position of the variable including the suffix, valid
  866.      * @return the variable's value or <b>null</b> if the variable is unknown
  867.      */
  868.     protected String resolveVariable(final String variableName, final StrBuilder buf, final int startPos, final int endPos) {
  869.         final StrLookup<?> resolver = getVariableResolver();
  870.         if (resolver == null) {
  871.             return null;
  872.         }
  873.         return resolver.lookup(variableName);
  874.     }

  875.     // Escape
  876.     //-----------------------------------------------------------------------
  877.     /**
  878.      * Returns the escape character.
  879.      *
  880.      * @return the character used for escaping variable references
  881.      */
  882.     public char getEscapeChar() {
  883.         return this.escapeChar;
  884.     }

  885.     /**
  886.      * Sets the escape character.
  887.      * If this character is placed before a variable reference in the source
  888.      * text, this variable will be ignored.
  889.      *
  890.      * @param escapeCharacter  the escape character (0 for disabling escaping)
  891.      */
  892.     public void setEscapeChar(final char escapeCharacter) {
  893.         this.escapeChar = escapeCharacter;
  894.     }

  895.     // Prefix
  896.     //-----------------------------------------------------------------------
  897.     /**
  898.      * Gets the variable prefix matcher currently in use.
  899.      * <p>
  900.      * The variable prefix is the characer or characters that identify the
  901.      * start of a variable. This prefix is expressed in terms of a matcher
  902.      * allowing advanced prefix matches.
  903.      *
  904.      * @return the prefix matcher in use
  905.      */
  906.     public StrMatcher getVariablePrefixMatcher() {
  907.         return prefixMatcher;
  908.     }

  909.     /**
  910.      * Sets the variable prefix matcher currently in use.
  911.      * <p>
  912.      * The variable prefix is the characer or characters that identify the
  913.      * start of a variable. This prefix is expressed in terms of a matcher
  914.      * allowing advanced prefix matches.
  915.      *
  916.      * @param prefixMatcher  the prefix matcher to use, null ignored
  917.      * @return this, to enable chaining
  918.      * @throws IllegalArgumentException if the prefix matcher is null
  919.      */
  920.     public StrSubstitutor setVariablePrefixMatcher(final StrMatcher prefixMatcher) {
  921.         if (prefixMatcher == null) {
  922.             throw new IllegalArgumentException("Variable prefix matcher must not be null!");
  923.         }
  924.         this.prefixMatcher = prefixMatcher;
  925.         return this;
  926.     }

  927.     /**
  928.      * Sets the variable prefix to use.
  929.      * <p>
  930.      * The variable prefix is the character or characters that identify the
  931.      * start of a variable. This method allows a single character prefix to
  932.      * be easily set.
  933.      *
  934.      * @param prefix  the prefix character to use
  935.      * @return this, to enable chaining
  936.      */
  937.     public StrSubstitutor setVariablePrefix(final char prefix) {
  938.         return setVariablePrefixMatcher(StrMatcher.charMatcher(prefix));
  939.     }

  940.     /**
  941.      * Sets the variable prefix to use.
  942.      * <p>
  943.      * The variable prefix is the characer or characters that identify the
  944.      * start of a variable. This method allows a string prefix to be easily set.
  945.      *
  946.      * @param prefix  the prefix for variables, not null
  947.      * @return this, to enable chaining
  948.      * @throws IllegalArgumentException if the prefix is null
  949.      */
  950.     public StrSubstitutor setVariablePrefix(final String prefix) {
  951.        if (prefix == null) {
  952.             throw new IllegalArgumentException("Variable prefix must not be null!");
  953.         }
  954.         return setVariablePrefixMatcher(StrMatcher.stringMatcher(prefix));
  955.     }

  956.     // Suffix
  957.     //-----------------------------------------------------------------------
  958.     /**
  959.      * Gets the variable suffix matcher currently in use.
  960.      * <p>
  961.      * The variable suffix is the characer or characters that identify the
  962.      * end of a variable. This suffix is expressed in terms of a matcher
  963.      * allowing advanced suffix matches.
  964.      *
  965.      * @return the suffix matcher in use
  966.      */
  967.     public StrMatcher getVariableSuffixMatcher() {
  968.         return suffixMatcher;
  969.     }

  970.     /**
  971.      * Sets the variable suffix matcher currently in use.
  972.      * <p>
  973.      * The variable suffix is the characer or characters that identify the
  974.      * end of a variable. This suffix is expressed in terms of a matcher
  975.      * allowing advanced suffix matches.
  976.      *
  977.      * @param suffixMatcher  the suffix matcher to use, null ignored
  978.      * @return this, to enable chaining
  979.      * @throws IllegalArgumentException if the suffix matcher is null
  980.      */
  981.     public StrSubstitutor setVariableSuffixMatcher(final StrMatcher suffixMatcher) {
  982.         if (suffixMatcher == null) {
  983.             throw new IllegalArgumentException("Variable suffix matcher must not be null!");
  984.         }
  985.         this.suffixMatcher = suffixMatcher;
  986.         return this;
  987.     }

  988.     /**
  989.      * Sets the variable suffix to use.
  990.      * <p>
  991.      * The variable suffix is the characer or characters that identify the
  992.      * end of a variable. This method allows a single character suffix to
  993.      * be easily set.
  994.      *
  995.      * @param suffix  the suffix character to use
  996.      * @return this, to enable chaining
  997.      */
  998.     public StrSubstitutor setVariableSuffix(final char suffix) {
  999.         return setVariableSuffixMatcher(StrMatcher.charMatcher(suffix));
  1000.     }

  1001.     /**
  1002.      * Sets the variable suffix to use.
  1003.      * <p>
  1004.      * The variable suffix is the character or characters that identify the
  1005.      * end of a variable. This method allows a string suffix to be easily set.
  1006.      *
  1007.      * @param suffix  the suffix for variables, not null
  1008.      * @return this, to enable chaining
  1009.      * @throws IllegalArgumentException if the suffix is null
  1010.      */
  1011.     public StrSubstitutor setVariableSuffix(final String suffix) {
  1012.        if (suffix == null) {
  1013.             throw new IllegalArgumentException("Variable suffix must not be null!");
  1014.         }
  1015.         return setVariableSuffixMatcher(StrMatcher.stringMatcher(suffix));
  1016.     }

  1017.     // Variable Default Value Delimiter
  1018.     //-----------------------------------------------------------------------
  1019.     /**
  1020.      * Gets the variable default value delimiter matcher currently in use.
  1021.      * <p>
  1022.      * The variable default value delimiter is the characer or characters that delimite the
  1023.      * variable name and the variable default value. This delimiter is expressed in terms of a matcher
  1024.      * allowing advanced variable default value delimiter matches.
  1025.      * <p>
  1026.      * If it returns null, then the variable default value resolution is disabled.
  1027.      *
  1028.      * @return the variable default value delimiter matcher in use, may be null
  1029.      */
  1030.     public StrMatcher getValueDelimiterMatcher() {
  1031.         return valueDelimiterMatcher;
  1032.     }

  1033.     /**
  1034.      * Sets the variable default value delimiter matcher to use.
  1035.      * <p>
  1036.      * The variable default value delimiter is the characer or characters that delimite the
  1037.      * variable name and the variable default value. This delimiter is expressed in terms of a matcher
  1038.      * allowing advanced variable default value delimiter matches.
  1039.      * <p>
  1040.      * If the <code>valueDelimiterMatcher</code> is null, then the variable default value resolution
  1041.      * becomes disabled.
  1042.      *
  1043.      * @param valueDelimiterMatcher  variable default value delimiter matcher to use, may be null
  1044.      * @return this, to enable chaining
  1045.      */
  1046.     public StrSubstitutor setValueDelimiterMatcher(final StrMatcher valueDelimiterMatcher) {
  1047.         this.valueDelimiterMatcher = valueDelimiterMatcher;
  1048.         return this;
  1049.     }

  1050.     /**
  1051.      * Sets the variable default value delimiter to use.
  1052.      * <p>
  1053.      * The variable default value delimiter is the characer or characters that delimite the
  1054.      * variable name and the variable default value. This method allows a single character
  1055.      * variable default value delimiter to be easily set.
  1056.      *
  1057.      * @param valueDelimiter  the variable default value delimiter character to use
  1058.      * @return this, to enable chaining
  1059.      */
  1060.     public StrSubstitutor setValueDelimiter(final char valueDelimiter) {
  1061.         return setValueDelimiterMatcher(StrMatcher.charMatcher(valueDelimiter));
  1062.     }

  1063.     /**
  1064.      * Sets the variable default value delimiter to use.
  1065.      * <p>
  1066.      * The variable default value delimiter is the characer or characters that delimite the
  1067.      * variable name and the variable default value. This method allows a string
  1068.      * variable default value delimiter to be easily set.
  1069.      * <p>
  1070.      * If the <code>valueDelimiter</code> is null or empty string, then the variable default
  1071.      * value resolution becomes disabled.
  1072.      *
  1073.      * @param valueDelimiter  the variable default value delimiter string to use, may be null or empty
  1074.      * @return this, to enable chaining
  1075.      */
  1076.     public StrSubstitutor setValueDelimiter(final String valueDelimiter) {
  1077.         if (valueDelimiter == null || valueDelimiter.length() == 0) {
  1078.             setValueDelimiterMatcher(null);
  1079.             return this;
  1080.         }
  1081.         return setValueDelimiterMatcher(StrMatcher.stringMatcher(valueDelimiter));
  1082.     }

  1083.     // Resolver
  1084.     //-----------------------------------------------------------------------
  1085.     /**
  1086.      * Gets the VariableResolver that is used to lookup variables.
  1087.      *
  1088.      * @return the VariableResolver
  1089.      */
  1090.     public StrLookup<?> getVariableResolver() {
  1091.         return this.variableResolver;
  1092.     }

  1093.     /**
  1094.      * Sets the VariableResolver that is used to lookup variables.
  1095.      *
  1096.      * @param variableResolver  the VariableResolver
  1097.      */
  1098.     public void setVariableResolver(final StrLookup<?> variableResolver) {
  1099.         this.variableResolver = variableResolver;
  1100.     }

  1101.     // Substitution support in variable names
  1102.     //-----------------------------------------------------------------------
  1103.     /**
  1104.      * Returns a flag whether substitution is done in variable names.
  1105.      *
  1106.      * @return the substitution in variable names flag
  1107.      */
  1108.     public boolean isEnableSubstitutionInVariables() {
  1109.         return enableSubstitutionInVariables;
  1110.     }

  1111.     /**
  1112.      * Sets a flag whether substitution is done in variable names. If set to
  1113.      * <b>true</b>, the names of variables can contain other variables which are
  1114.      * processed first before the original variable is evaluated, e.g.
  1115.      * <code>${jre-${java.version}}</code>. The default value is <b>false</b>.
  1116.      *
  1117.      * @param enableSubstitutionInVariables the new value of the flag
  1118.      */
  1119.     public void setEnableSubstitutionInVariables(
  1120.             final boolean enableSubstitutionInVariables) {
  1121.         this.enableSubstitutionInVariables = enableSubstitutionInVariables;
  1122.     }

  1123.     /**
  1124.      * Returns the flag controlling whether escapes are preserved during
  1125.      * substitution.
  1126.      *
  1127.      * @return the preserve escape flag
  1128.      */
  1129.     public boolean isPreserveEscapes() {
  1130.         return preserveEscapes;
  1131.     }

  1132.     /**
  1133.      * Sets a flag controlling whether escapes are preserved during
  1134.      * substitution.  If set to <b>true</b>, the escape character is retained
  1135.      * during substitution (e.g. <code>$${this-is-escaped}</code> remains
  1136.      * <code>$${this-is-escaped}</code>).  If set to <b>false</b>, the escape
  1137.      * character is removed during substitution (e.g.
  1138.      * <code>$${this-is-escaped}</code> becomes
  1139.      * <code>${this-is-escaped}</code>).  The default value is <b>false</b>
  1140.      *
  1141.      * @param preserveEscapes true if escapes are to be preserved
  1142.      */
  1143.     public void setPreserveEscapes(final boolean preserveEscapes) {
  1144.         this.preserveEscapes = preserveEscapes;
  1145.     }
  1146. }