FastDateParser.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.lang3.time;

  18. import java.io.IOException;
  19. import java.io.ObjectInputStream;
  20. import java.io.Serializable;
  21. import java.text.DateFormatSymbols;
  22. import java.text.ParseException;
  23. import java.text.ParsePosition;
  24. import java.text.SimpleDateFormat;
  25. import java.util.ArrayList;
  26. import java.util.Calendar;
  27. import java.util.Comparator;
  28. import java.util.Date;
  29. import java.util.HashMap;
  30. import java.util.List;
  31. import java.util.ListIterator;
  32. import java.util.Locale;
  33. import java.util.Map;
  34. import java.util.Objects;
  35. import java.util.Set;
  36. import java.util.TimeZone;
  37. import java.util.TreeSet;
  38. import java.util.concurrent.ConcurrentHashMap;
  39. import java.util.concurrent.ConcurrentMap;
  40. import java.util.regex.Matcher;
  41. import java.util.regex.Pattern;

  42. import org.apache.commons.lang3.LocaleUtils;

  43. /**
  44.  * FastDateParser is a fast and thread-safe version of {@link java.text.SimpleDateFormat}.
  45.  *
  46.  * <p>
  47.  * To obtain a proxy to a FastDateParser, use {@link FastDateFormat#getInstance(String, TimeZone, Locale)} or another variation of the factory methods of
  48.  * {@link FastDateFormat}.
  49.  * </p>
  50.  *
  51.  * <p>
  52.  * Since FastDateParser is thread safe, you can use a static member instance:
  53.  * </p>
  54.  * {@code
  55.  *     private static final DateParser DATE_PARSER = FastDateFormat.getInstance("yyyy-MM-dd");
  56.  * }
  57.  *
  58.  * <p>
  59.  * This class can be used as a direct replacement for {@link SimpleDateFormat} in most parsing situations. This class is especially useful in multi-threaded
  60.  * server environments. {@link SimpleDateFormat} is not thread-safe in any JDK version, nor will it be as Sun has closed the
  61.  * <a href="https://bugs.openjdk.org/browse/JDK-4228335">bug</a>/RFE.
  62.  * </p>
  63.  *
  64.  * <p>
  65.  * Only parsing is supported by this class, but all patterns are compatible with SimpleDateFormat.
  66.  * </p>
  67.  *
  68.  * <p>
  69.  * The class operates in lenient mode, so for example a time of 90 minutes is treated as 1 hour 30 minutes.
  70.  * </p>
  71.  *
  72.  * <p>
  73.  * Timing tests indicate this class is as about as fast as SimpleDateFormat in single thread applications and about 25% faster in multi-thread applications.
  74.  * </p>
  75.  *
  76.  * @since 3.2
  77.  * @see FastDatePrinter
  78.  */
  79. public class FastDateParser implements DateParser, Serializable {

  80.     /**
  81.      * A strategy that handles a text field in the parsing pattern
  82.      */
  83.     private static final class CaseInsensitiveTextStrategy extends PatternStrategy {
  84.         private final int field;
  85.         final Locale locale;
  86.         private final Map<String, Integer> lKeyValues;

  87.         /**
  88.          * Constructs a Strategy that parses a Text field
  89.          *
  90.          * @param field            The Calendar field
  91.          * @param definingCalendar The Calendar to use
  92.          * @param locale           The Locale to use
  93.          */
  94.         CaseInsensitiveTextStrategy(final int field, final Calendar definingCalendar, final Locale locale) {
  95.             this.field = field;
  96.             this.locale = LocaleUtils.toLocale(locale);

  97.             final StringBuilder regex = new StringBuilder();
  98.             regex.append("((?iu)");
  99.             lKeyValues = appendDisplayNames(definingCalendar, locale, field, regex);
  100.             regex.setLength(regex.length() - 1);
  101.             regex.append(")");
  102.             createPattern(regex);
  103.         }

  104.         /**
  105.          * {@inheritDoc}
  106.          */
  107.         @Override
  108.         void setCalendar(final FastDateParser parser, final Calendar calendar, final String value) {
  109.             final String lowerCase = value.toLowerCase(locale);
  110.             Integer iVal = lKeyValues.get(lowerCase);
  111.             if (iVal == null) {
  112.                 // match missing the optional trailing period
  113.                 iVal = lKeyValues.get(lowerCase + '.');
  114.             }
  115.             // LANG-1669: Mimic fix done in OpenJDK 17 to resolve issue with parsing newly supported day periods added in OpenJDK 16
  116.             if (Calendar.AM_PM != this.field || iVal <= 1) {
  117.                 calendar.set(field, iVal.intValue());
  118.             }
  119.         }

  120.         /**
  121.          * Converts this instance to a handy debug string.
  122.          *
  123.          * @since 3.12.0
  124.          */
  125.         @Override
  126.         public String toString() {
  127.             return "CaseInsensitiveTextStrategy [field=" + field + ", locale=" + locale + ", lKeyValues=" + lKeyValues + ", pattern=" + pattern + "]";
  128.         }
  129.     }

  130.     /**
  131.      * A strategy that copies the static or quoted field in the parsing pattern
  132.      */
  133.     private static final class CopyQuotedStrategy extends Strategy {

  134.         private final String formatField;

  135.         /**
  136.          * Constructs a Strategy that ensures the formatField has literal text
  137.          *
  138.          * @param formatField The literal text to match
  139.          */
  140.         CopyQuotedStrategy(final String formatField) {
  141.             this.formatField = formatField;
  142.         }

  143.         /**
  144.          * {@inheritDoc}
  145.          */
  146.         @Override
  147.         boolean isNumber() {
  148.             return false;
  149.         }

  150.         @Override
  151.         boolean parse(final FastDateParser parser, final Calendar calendar, final String source, final ParsePosition pos, final int maxWidth) {
  152.             for (int idx = 0; idx < formatField.length(); ++idx) {
  153.                 final int sIdx = idx + pos.getIndex();
  154.                 if (sIdx == source.length()) {
  155.                     pos.setErrorIndex(sIdx);
  156.                     return false;
  157.                 }
  158.                 if (formatField.charAt(idx) != source.charAt(sIdx)) {
  159.                     pos.setErrorIndex(sIdx);
  160.                     return false;
  161.                 }
  162.             }
  163.             pos.setIndex(formatField.length() + pos.getIndex());
  164.             return true;
  165.         }

  166.         /**
  167.          * Converts this instance to a handy debug string.
  168.          *
  169.          * @since 3.12.0
  170.          */
  171.         @Override
  172.         public String toString() {
  173.             return "CopyQuotedStrategy [formatField=" + formatField + "]";
  174.         }
  175.     }

  176.     private static final class ISO8601TimeZoneStrategy extends PatternStrategy {
  177.         // Z, +hh, -hh, +hhmm, -hhmm, +hh:mm or -hh:mm

  178.         private static final Strategy ISO_8601_1_STRATEGY = new ISO8601TimeZoneStrategy("(Z|(?:[+-]\\d{2}))");

  179.         private static final Strategy ISO_8601_2_STRATEGY = new ISO8601TimeZoneStrategy("(Z|(?:[+-]\\d{2}\\d{2}))");

  180.         private static final Strategy ISO_8601_3_STRATEGY = new ISO8601TimeZoneStrategy("(Z|(?:[+-]\\d{2}(?::)\\d{2}))");
  181.         /**
  182.          * Factory method for ISO8601TimeZoneStrategies.
  183.          *
  184.          * @param tokenLen a token indicating the length of the TimeZone String to be formatted.
  185.          * @return a ISO8601TimeZoneStrategy that can format TimeZone String of length {@code tokenLen}. If no such strategy exists, an IllegalArgumentException
  186.          *         will be thrown.
  187.          */
  188.         static Strategy getStrategy(final int tokenLen) {
  189.             switch (tokenLen) {
  190.             case 1:
  191.                 return ISO_8601_1_STRATEGY;
  192.             case 2:
  193.                 return ISO_8601_2_STRATEGY;
  194.             case 3:
  195.                 return ISO_8601_3_STRATEGY;
  196.             default:
  197.                 throw new IllegalArgumentException("invalid number of X");
  198.             }
  199.         }
  200.         /**
  201.          * Constructs a Strategy that parses a TimeZone
  202.          *
  203.          * @param pattern The Pattern
  204.          */
  205.         ISO8601TimeZoneStrategy(final String pattern) {
  206.             createPattern(pattern);
  207.         }

  208.         /**
  209.          * {@inheritDoc}
  210.          */
  211.         @Override
  212.         void setCalendar(final FastDateParser parser, final Calendar calendar, final String value) {
  213.             calendar.setTimeZone(FastTimeZone.getGmtTimeZone(value));
  214.         }
  215.     }

  216.     /**
  217.      * A strategy that handles a number field in the parsing pattern
  218.      */
  219.     private static class NumberStrategy extends Strategy {

  220.         private final int field;

  221.         /**
  222.          * Constructs a Strategy that parses a Number field
  223.          *
  224.          * @param field The Calendar field
  225.          */
  226.         NumberStrategy(final int field) {
  227.             this.field = field;
  228.         }

  229.         /**
  230.          * {@inheritDoc}
  231.          */
  232.         @Override
  233.         boolean isNumber() {
  234.             return true;
  235.         }

  236.         /**
  237.          * Make any modifications to parsed integer
  238.          *
  239.          * @param parser The parser
  240.          * @param iValue The parsed integer
  241.          * @return The modified value
  242.          */
  243.         int modify(final FastDateParser parser, final int iValue) {
  244.             return iValue;
  245.         }

  246.         @Override
  247.         boolean parse(final FastDateParser parser, final Calendar calendar, final String source, final ParsePosition pos, final int maxWidth) {
  248.             int idx = pos.getIndex();
  249.             int last = source.length();

  250.             if (maxWidth == 0) {
  251.                 // if no maxWidth, strip leading white space
  252.                 for (; idx < last; ++idx) {
  253.                     final char c = source.charAt(idx);
  254.                     if (!Character.isWhitespace(c)) {
  255.                         break;
  256.                     }
  257.                 }
  258.                 pos.setIndex(idx);
  259.             } else {
  260.                 final int end = idx + maxWidth;
  261.                 if (last > end) {
  262.                     last = end;
  263.                 }
  264.             }

  265.             for (; idx < last; ++idx) {
  266.                 final char c = source.charAt(idx);
  267.                 if (!Character.isDigit(c)) {
  268.                     break;
  269.                 }
  270.             }

  271.             if (pos.getIndex() == idx) {
  272.                 pos.setErrorIndex(idx);
  273.                 return false;
  274.             }

  275.             final int value = Integer.parseInt(source.substring(pos.getIndex(), idx));
  276.             pos.setIndex(idx);

  277.             calendar.set(field, modify(parser, value));
  278.             return true;
  279.         }

  280.         /**
  281.          * Converts this instance to a handy debug string.
  282.          *
  283.          * @since 3.12.0
  284.          */
  285.         @Override
  286.         public String toString() {
  287.             return "NumberStrategy [field=" + field + "]";
  288.         }
  289.     }

  290.     /**
  291.      * A strategy to parse a single field from the parsing pattern
  292.      */
  293.     private abstract static class PatternStrategy extends Strategy {

  294.         Pattern pattern;

  295.         void createPattern(final String regex) {
  296.             this.pattern = Pattern.compile(regex);
  297.         }

  298.         void createPattern(final StringBuilder regex) {
  299.             createPattern(regex.toString());
  300.         }

  301.         /**
  302.          * Is this field a number? The default implementation returns false.
  303.          *
  304.          * @return true, if field is a number
  305.          */
  306.         @Override
  307.         boolean isNumber() {
  308.             return false;
  309.         }

  310.         @Override
  311.         boolean parse(final FastDateParser parser, final Calendar calendar, final String source, final ParsePosition pos, final int maxWidth) {
  312.             final Matcher matcher = pattern.matcher(source.substring(pos.getIndex()));
  313.             if (!matcher.lookingAt()) {
  314.                 pos.setErrorIndex(pos.getIndex());
  315.                 return false;
  316.             }
  317.             pos.setIndex(pos.getIndex() + matcher.end(1));
  318.             setCalendar(parser, calendar, matcher.group(1));
  319.             return true;
  320.         }

  321.         abstract void setCalendar(FastDateParser parser, Calendar calendar, String value);

  322.         /**
  323.          * Converts this instance to a handy debug string.
  324.          *
  325.          * @since 3.12.0
  326.          */
  327.         @Override
  328.         public String toString() {
  329.             return getClass().getSimpleName() + " [pattern=" + pattern + "]";
  330.         }

  331.     }

  332.     /**
  333.      * A strategy to parse a single field from the parsing pattern
  334.      */
  335.     private abstract static class Strategy {

  336.         /**
  337.          * Is this field a number? The default implementation returns false.
  338.          *
  339.          * @return true, if field is a number
  340.          */
  341.         boolean isNumber() {
  342.             return false;
  343.         }

  344.         abstract boolean parse(FastDateParser parser, Calendar calendar, String source, ParsePosition pos, int maxWidth);
  345.     }

  346.     /**
  347.      * Holds strategy and field width
  348.      */
  349.     private static final class StrategyAndWidth {

  350.         final Strategy strategy;
  351.         final int width;

  352.         StrategyAndWidth(final Strategy strategy, final int width) {
  353.             this.strategy = Objects.requireNonNull(strategy, "strategy");
  354.             this.width = width;
  355.         }

  356.         int getMaxWidth(final ListIterator<StrategyAndWidth> lt) {
  357.             if (!strategy.isNumber() || !lt.hasNext()) {
  358.                 return 0;
  359.             }
  360.             final Strategy nextStrategy = lt.next().strategy;
  361.             lt.previous();
  362.             return nextStrategy.isNumber() ? width : 0;
  363.         }

  364.         @Override
  365.         public String toString() {
  366.             return "StrategyAndWidth [strategy=" + strategy + ", width=" + width + "]";
  367.         }
  368.     }

  369.     /**
  370.      * Parse format into Strategies
  371.      */
  372.     private final class StrategyParser {
  373.         private final Calendar definingCalendar;
  374.         private int currentIdx;

  375.         StrategyParser(final Calendar definingCalendar) {
  376.             this.definingCalendar = Objects.requireNonNull(definingCalendar, "definingCalendar");
  377.         }

  378.         StrategyAndWidth getNextStrategy() {
  379.             if (currentIdx >= pattern.length()) {
  380.                 return null;
  381.             }

  382.             final char c = pattern.charAt(currentIdx);
  383.             if (isFormatLetter(c)) {
  384.                 return letterPattern(c);
  385.             }
  386.             return literal();
  387.         }

  388.         private StrategyAndWidth letterPattern(final char c) {
  389.             final int begin = currentIdx;
  390.             while (++currentIdx < pattern.length()) {
  391.                 if (pattern.charAt(currentIdx) != c) {
  392.                     break;
  393.                 }
  394.             }

  395.             final int width = currentIdx - begin;
  396.             return new StrategyAndWidth(getStrategy(c, width, definingCalendar), width);
  397.         }

  398.         private StrategyAndWidth literal() {
  399.             boolean activeQuote = false;

  400.             final StringBuilder sb = new StringBuilder();
  401.             while (currentIdx < pattern.length()) {
  402.                 final char c = pattern.charAt(currentIdx);
  403.                 if (!activeQuote && isFormatLetter(c)) {
  404.                     break;
  405.                 }
  406.                 if (c == '\'' && (++currentIdx == pattern.length() || pattern.charAt(currentIdx) != '\'')) {
  407.                     activeQuote = !activeQuote;
  408.                     continue;
  409.                 }
  410.                 ++currentIdx;
  411.                 sb.append(c);
  412.             }

  413.             if (activeQuote) {
  414.                 throw new IllegalArgumentException("Unterminated quote");
  415.             }

  416.             final String formatField = sb.toString();
  417.             return new StrategyAndWidth(new CopyQuotedStrategy(formatField), formatField.length());
  418.         }
  419.     }

  420.     /**
  421.      * A strategy that handles a time zone field in the parsing pattern
  422.      */
  423.     static class TimeZoneStrategy extends PatternStrategy {
  424.         private static final class TzInfo {
  425.             final TimeZone zone;
  426.             final int dstOffset;

  427.             TzInfo(final TimeZone tz, final boolean useDst) {
  428.                 zone = tz;
  429.                 dstOffset = useDst ? tz.getDSTSavings() : 0;
  430.             }

  431.             @Override
  432.             public String toString() {
  433.                 return "TzInfo [zone=" + zone + ", dstOffset=" + dstOffset + "]";
  434.             }
  435.         }
  436.         private static final String RFC_822_TIME_ZONE = "[+-]\\d{4}";

  437.         private static final String GMT_OPTION = TimeZones.GMT_ID + "[+-]\\d{1,2}:\\d{2}";
  438.         /**
  439.          * Index of zone id
  440.          */
  441.         private static final int ID = 0;

  442.         private final Locale locale;

  443.         private final Map<String, TzInfo> tzNames = new HashMap<>();

  444.         /**
  445.          * Constructs a Strategy that parses a TimeZone
  446.          *
  447.          * @param locale The Locale
  448.          */
  449.         TimeZoneStrategy(final Locale locale) {
  450.             this.locale = LocaleUtils.toLocale(locale);

  451.             final StringBuilder sb = new StringBuilder();
  452.             sb.append("((?iu)" + RFC_822_TIME_ZONE + "|" + GMT_OPTION);

  453.             final Set<String> sorted = new TreeSet<>(LONGER_FIRST_LOWERCASE);

  454.             // Order is undefined.
  455.             // TODO Use of getZoneStrings() is discouraged per its Javadoc.
  456.             final String[][] zones = DateFormatSymbols.getInstance(locale).getZoneStrings();
  457.             for (final String[] zoneNames : zones) {
  458.                 // offset 0 is the time zone ID and is not localized
  459.                 final String tzId = zoneNames[ID];
  460.                 if (tzId.equalsIgnoreCase(TimeZones.GMT_ID)) {
  461.                     continue;
  462.                 }
  463.                 final TimeZone tz = TimeZone.getTimeZone(tzId);
  464.                 // offset 1 is long standard name
  465.                 // offset 2 is short standard name
  466.                 final TzInfo standard = new TzInfo(tz, false);
  467.                 TzInfo tzInfo = standard;
  468.                 for (int i = 1; i < zoneNames.length; ++i) {
  469.                     switch (i) {
  470.                     case 3: // offset 3 is long daylight savings (or summertime) name
  471.                             // offset 4 is the short summertime name
  472.                         tzInfo = new TzInfo(tz, true);
  473.                         break;
  474.                     case 5: // offset 5 starts additional names, probably standard time
  475.                         tzInfo = standard;
  476.                         break;
  477.                     default:
  478.                         break;
  479.                     }
  480.                     final String zoneName = zoneNames[i];
  481.                     if (zoneName != null) {
  482.                         final String key = zoneName.toLowerCase(locale);
  483.                         // ignore the data associated with duplicates supplied in
  484.                         // the additional names
  485.                         if (sorted.add(key)) {
  486.                             tzNames.put(key, tzInfo);
  487.                         }
  488.                     }
  489.                 }
  490.             }

  491.             // Order is undefined.
  492.             for (final String tzId : TimeZone.getAvailableIDs()) {
  493.                 if (tzId.equalsIgnoreCase(TimeZones.GMT_ID)) {
  494.                     continue;
  495.                 }
  496.                 final TimeZone tz = TimeZone.getTimeZone(tzId);
  497.                 final String zoneName = tz.getDisplayName(locale);
  498.                 final String key = zoneName.toLowerCase(locale);
  499.                 if (sorted.add(key)) {
  500.                     tzNames.put(key, new TzInfo(tz, tz.observesDaylightTime()));
  501.                 }
  502.             }

  503.             // order the regex alternatives with longer strings first, greedy
  504.             // match will ensure the longest string will be consumed
  505.             sorted.forEach(zoneName -> simpleQuote(sb.append('|'), zoneName));
  506.             sb.append(")");
  507.             createPattern(sb);
  508.         }

  509.         /**
  510.          * {@inheritDoc}
  511.          */
  512.         @Override
  513.         void setCalendar(final FastDateParser parser, final Calendar calendar, final String timeZone) {
  514.             final TimeZone tz = FastTimeZone.getGmtTimeZone(timeZone);
  515.             if (tz != null) {
  516.                 calendar.setTimeZone(tz);
  517.             } else {
  518.                 final String lowerCase = timeZone.toLowerCase(locale);
  519.                 TzInfo tzInfo = tzNames.get(lowerCase);
  520.                 if (tzInfo == null) {
  521.                     // match missing the optional trailing period
  522.                     tzInfo = tzNames.get(lowerCase + '.');
  523.                 }
  524.                 calendar.set(Calendar.DST_OFFSET, tzInfo.dstOffset);
  525.                 calendar.set(Calendar.ZONE_OFFSET, tzInfo.zone.getRawOffset());
  526.             }
  527.         }

  528.         /**
  529.          * Converts this instance to a handy debug string.
  530.          *
  531.          * @since 3.12.0
  532.          */
  533.         @Override
  534.         public String toString() {
  535.             return "TimeZoneStrategy [locale=" + locale + ", tzNames=" + tzNames + ", pattern=" + pattern + "]";
  536.         }

  537.     }

  538.     /**
  539.      * Required for serialization support.
  540.      *
  541.      * @see java.io.Serializable
  542.      */
  543.     private static final long serialVersionUID = 3L;

  544.     static final Locale JAPANESE_IMPERIAL = new Locale("ja", "JP", "JP");

  545.     /**
  546.      * comparator used to sort regex alternatives. Alternatives should be ordered longer first, and shorter last. ('february' before 'feb'). All entries must be
  547.      * lower-case by locale.
  548.      */
  549.     private static final Comparator<String> LONGER_FIRST_LOWERCASE = Comparator.reverseOrder();

  550.     // helper classes to parse the format string

  551.     @SuppressWarnings("unchecked") // OK because we are creating an array with no entries
  552.     private static final ConcurrentMap<Locale, Strategy>[] caches = new ConcurrentMap[Calendar.FIELD_COUNT];

  553.     private static final Strategy ABBREVIATED_YEAR_STRATEGY = new NumberStrategy(Calendar.YEAR) {
  554.         /**
  555.          * {@inheritDoc}
  556.          */
  557.         @Override
  558.         int modify(final FastDateParser parser, final int iValue) {
  559.             return iValue < 100 ? parser.adjustYear(iValue) : iValue;
  560.         }
  561.     };

  562.     private static final Strategy NUMBER_MONTH_STRATEGY = new NumberStrategy(Calendar.MONTH) {
  563.         @Override
  564.         int modify(final FastDateParser parser, final int iValue) {
  565.             return iValue - 1;
  566.         }
  567.     };

  568.     private static final Strategy LITERAL_YEAR_STRATEGY = new NumberStrategy(Calendar.YEAR);

  569.     private static final Strategy WEEK_OF_YEAR_STRATEGY = new NumberStrategy(Calendar.WEEK_OF_YEAR);

  570.     private static final Strategy WEEK_OF_MONTH_STRATEGY = new NumberStrategy(Calendar.WEEK_OF_MONTH);

  571.     private static final Strategy DAY_OF_YEAR_STRATEGY = new NumberStrategy(Calendar.DAY_OF_YEAR);

  572.     private static final Strategy DAY_OF_MONTH_STRATEGY = new NumberStrategy(Calendar.DAY_OF_MONTH);

  573.     private static final Strategy DAY_OF_WEEK_STRATEGY = new NumberStrategy(Calendar.DAY_OF_WEEK) {
  574.         @Override
  575.         int modify(final FastDateParser parser, final int iValue) {
  576.             return iValue == 7 ? Calendar.SUNDAY : iValue + 1;
  577.         }
  578.     };

  579.     private static final Strategy DAY_OF_WEEK_IN_MONTH_STRATEGY = new NumberStrategy(Calendar.DAY_OF_WEEK_IN_MONTH);

  580.     private static final Strategy HOUR_OF_DAY_STRATEGY = new NumberStrategy(Calendar.HOUR_OF_DAY);

  581.     private static final Strategy HOUR24_OF_DAY_STRATEGY = new NumberStrategy(Calendar.HOUR_OF_DAY) {
  582.         @Override
  583.         int modify(final FastDateParser parser, final int iValue) {
  584.             return iValue == 24 ? 0 : iValue;
  585.         }
  586.     };

  587.     private static final Strategy HOUR12_STRATEGY = new NumberStrategy(Calendar.HOUR) {
  588.         @Override
  589.         int modify(final FastDateParser parser, final int iValue) {
  590.             return iValue == 12 ? 0 : iValue;
  591.         }
  592.     };

  593.     private static final Strategy HOUR_STRATEGY = new NumberStrategy(Calendar.HOUR);

  594.     private static final Strategy MINUTE_STRATEGY = new NumberStrategy(Calendar.MINUTE);

  595.     private static final Strategy SECOND_STRATEGY = new NumberStrategy(Calendar.SECOND);

  596.     // Support for strategies

  597.     private static final Strategy MILLISECOND_STRATEGY = new NumberStrategy(Calendar.MILLISECOND);

  598.     /**
  599.      * Gets the short and long values displayed for a field
  600.      *
  601.      * @param calendar The calendar to obtain the short and long values
  602.      * @param locale   The locale of display names
  603.      * @param field    The field of interest
  604.      * @param regex    The regular expression to build
  605.      * @return The map of string display names to field values
  606.      */
  607.     private static Map<String, Integer> appendDisplayNames(final Calendar calendar, final Locale locale, final int field, final StringBuilder regex) {
  608.         Objects.requireNonNull(calendar, "calendar");
  609.         final Map<String, Integer> values = new HashMap<>();
  610.         final Locale actualLocale = LocaleUtils.toLocale(locale);
  611.         final Map<String, Integer> displayNames = calendar.getDisplayNames(field, Calendar.ALL_STYLES, actualLocale);
  612.         final TreeSet<String> sorted = new TreeSet<>(LONGER_FIRST_LOWERCASE);
  613.         displayNames.forEach((k, v) -> {
  614.             final String keyLc = k.toLowerCase(actualLocale);
  615.             if (sorted.add(keyLc)) {
  616.                 values.put(keyLc, v);
  617.             }
  618.         });
  619.         sorted.forEach(symbol -> simpleQuote(regex, symbol).append('|'));
  620.         return values;
  621.     }

  622.     /**
  623.      * Gets a cache of Strategies for a particular field
  624.      *
  625.      * @param field The Calendar field
  626.      * @return a cache of Locale to Strategy
  627.      */
  628.     private static ConcurrentMap<Locale, Strategy> getCache(final int field) {
  629.         synchronized (caches) {
  630.             if (caches[field] == null) {
  631.                 caches[field] = new ConcurrentHashMap<>(3);
  632.             }
  633.             return caches[field];
  634.         }
  635.     }

  636.     private static boolean isFormatLetter(final char c) {
  637.         return c >= 'A' && c <= 'Z' || c >= 'a' && c <= 'z';
  638.     }

  639.     private static StringBuilder simpleQuote(final StringBuilder sb, final String value) {
  640.         for (int i = 0; i < value.length(); ++i) {
  641.             final char c = value.charAt(i);
  642.             switch (c) {
  643.             case '\\':
  644.             case '^':
  645.             case '$':
  646.             case '.':
  647.             case '|':
  648.             case '?':
  649.             case '*':
  650.             case '+':
  651.             case '(':
  652.             case ')':
  653.             case '[':
  654.             case '{':
  655.                 sb.append('\\');
  656.             default:
  657.                 sb.append(c);
  658.             }
  659.         }
  660.         if (sb.charAt(sb.length() - 1) == '.') {
  661.             // trailing '.' is optional
  662.             sb.append('?');
  663.         }
  664.         return sb;
  665.     }

  666.     /** Input pattern. */
  667.     private final String pattern;

  668.     /** Input TimeZone. */
  669.     private final TimeZone timeZone;

  670.     /** Input Locale. */
  671.     private final Locale locale;

  672.     /**
  673.      * Century from Date.
  674.      */
  675.     private final int century;

  676.     /**
  677.      * Start year from Date.
  678.      */
  679.     private final int startYear;

  680.     /** Initialized from Calendar. */
  681.     private transient List<StrategyAndWidth> patterns;

  682.     /**
  683.      * Constructs a new FastDateParser.
  684.      *
  685.      * Use {@link FastDateFormat#getInstance(String, TimeZone, Locale)} or another variation of the factory methods of {@link FastDateFormat} to get a cached
  686.      * FastDateParser instance.
  687.      *
  688.      * @param pattern  non-null {@link java.text.SimpleDateFormat} compatible pattern
  689.      * @param timeZone non-null time zone to use
  690.      * @param locale   non-null locale
  691.      */
  692.     protected FastDateParser(final String pattern, final TimeZone timeZone, final Locale locale) {
  693.         this(pattern, timeZone, locale, null);
  694.     }

  695.     /**
  696.      * Constructs a new FastDateParser.
  697.      *
  698.      * @param pattern      non-null {@link java.text.SimpleDateFormat} compatible pattern
  699.      * @param timeZone     non-null time zone to use
  700.      * @param locale       locale, null maps to the default Locale.
  701.      * @param centuryStart The start of the century for 2 digit year parsing
  702.      *
  703.      * @since 3.5
  704.      */
  705.     protected FastDateParser(final String pattern, final TimeZone timeZone, final Locale locale, final Date centuryStart) {
  706.         this.pattern = Objects.requireNonNull(pattern, "pattern");
  707.         this.timeZone = Objects.requireNonNull(timeZone, "timeZone");
  708.         this.locale = LocaleUtils.toLocale(locale);

  709.         final Calendar definingCalendar = Calendar.getInstance(timeZone, this.locale);

  710.         final int centuryStartYear;
  711.         if (centuryStart != null) {
  712.             definingCalendar.setTime(centuryStart);
  713.             centuryStartYear = definingCalendar.get(Calendar.YEAR);
  714.         } else if (this.locale.equals(JAPANESE_IMPERIAL)) {
  715.             centuryStartYear = 0;
  716.         } else {
  717.             // from 80 years ago to 20 years from now
  718.             definingCalendar.setTime(new Date());
  719.             centuryStartYear = definingCalendar.get(Calendar.YEAR) - 80;
  720.         }
  721.         century = centuryStartYear / 100 * 100;
  722.         startYear = centuryStartYear - century;

  723.         init(definingCalendar);
  724.     }

  725.     /**
  726.      * Adjusts dates to be within appropriate century
  727.      *
  728.      * @param twoDigitYear The year to adjust
  729.      * @return A value between centuryStart(inclusive) to centuryStart+100(exclusive)
  730.      */
  731.     private int adjustYear(final int twoDigitYear) {
  732.         final int trial = century + twoDigitYear;
  733.         return twoDigitYear >= startYear ? trial : trial + 100;
  734.     }

  735.     // Basics
  736.     /**
  737.      * Compares another object for equality with this object.
  738.      *
  739.      * @param obj the object to compare to
  740.      * @return {@code true}if equal to this instance
  741.      */
  742.     @Override
  743.     public boolean equals(final Object obj) {
  744.         if (!(obj instanceof FastDateParser)) {
  745.             return false;
  746.         }
  747.         final FastDateParser other = (FastDateParser) obj;
  748.         return pattern.equals(other.pattern) && timeZone.equals(other.timeZone) && locale.equals(other.locale);
  749.     }

  750.     /*
  751.      * (non-Javadoc)
  752.      *
  753.      * @see org.apache.commons.lang3.time.DateParser#getLocale()
  754.      */
  755.     @Override
  756.     public Locale getLocale() {
  757.         return locale;
  758.     }

  759.     /**
  760.      * Constructs a Strategy that parses a Text field
  761.      *
  762.      * @param field            The Calendar field
  763.      * @param definingCalendar The calendar to obtain the short and long values
  764.      * @return a TextStrategy for the field and Locale
  765.      */
  766.     private Strategy getLocaleSpecificStrategy(final int field, final Calendar definingCalendar) {
  767.         final ConcurrentMap<Locale, Strategy> cache = getCache(field);
  768.         return cache.computeIfAbsent(locale,
  769.                 k -> field == Calendar.ZONE_OFFSET ? new TimeZoneStrategy(locale) : new CaseInsensitiveTextStrategy(field, definingCalendar, locale));
  770.     }
  771.     // Accessors
  772.     /*
  773.      * (non-Javadoc)
  774.      *
  775.      * @see org.apache.commons.lang3.time.DateParser#getPattern()
  776.      */
  777.     @Override
  778.     public String getPattern() {
  779.         return pattern;
  780.     }
  781.     /**
  782.      * Gets a Strategy given a field from a SimpleDateFormat pattern
  783.      *
  784.      * @param f                A sub-sequence of the SimpleDateFormat pattern
  785.      * @param width            formatting width
  786.      * @param definingCalendar The calendar to obtain the short and long values
  787.      * @return The Strategy that will handle parsing for the field
  788.      */
  789.     private Strategy getStrategy(final char f, final int width, final Calendar definingCalendar) {
  790.         switch (f) {
  791.         default:
  792.             throw new IllegalArgumentException("Format '" + f + "' not supported");
  793.         case 'D':
  794.             return DAY_OF_YEAR_STRATEGY;
  795.         case 'E':
  796.             return getLocaleSpecificStrategy(Calendar.DAY_OF_WEEK, definingCalendar);
  797.         case 'F':
  798.             return DAY_OF_WEEK_IN_MONTH_STRATEGY;
  799.         case 'G':
  800.             return getLocaleSpecificStrategy(Calendar.ERA, definingCalendar);
  801.         case 'H': // Hour in day (0-23)
  802.             return HOUR_OF_DAY_STRATEGY;
  803.         case 'K': // Hour in am/pm (0-11)
  804.             return HOUR_STRATEGY;
  805.         case 'M':
  806.         case 'L':
  807.             return width >= 3 ? getLocaleSpecificStrategy(Calendar.MONTH, definingCalendar) : NUMBER_MONTH_STRATEGY;
  808.         case 'S':
  809.             return MILLISECOND_STRATEGY;
  810.         case 'W':
  811.             return WEEK_OF_MONTH_STRATEGY;
  812.         case 'a':
  813.             return getLocaleSpecificStrategy(Calendar.AM_PM, definingCalendar);
  814.         case 'd':
  815.             return DAY_OF_MONTH_STRATEGY;
  816.         case 'h': // Hour in am/pm (1-12), i.e. midday/midnight is 12, not 0
  817.             return HOUR12_STRATEGY;
  818.         case 'k': // Hour in day (1-24), i.e. midnight is 24, not 0
  819.             return HOUR24_OF_DAY_STRATEGY;
  820.         case 'm':
  821.             return MINUTE_STRATEGY;
  822.         case 's':
  823.             return SECOND_STRATEGY;
  824.         case 'u':
  825.             return DAY_OF_WEEK_STRATEGY;
  826.         case 'w':
  827.             return WEEK_OF_YEAR_STRATEGY;
  828.         case 'y':
  829.         case 'Y':
  830.             return width > 2 ? LITERAL_YEAR_STRATEGY : ABBREVIATED_YEAR_STRATEGY;
  831.         case 'X':
  832.             return ISO8601TimeZoneStrategy.getStrategy(width);
  833.         case 'Z':
  834.             if (width == 2) {
  835.                 return ISO8601TimeZoneStrategy.ISO_8601_3_STRATEGY;
  836.             }
  837.             //$FALL-THROUGH$
  838.         case 'z':
  839.             return getLocaleSpecificStrategy(Calendar.ZONE_OFFSET, definingCalendar);
  840.         }
  841.     }
  842.     /*
  843.      * (non-Javadoc)
  844.      *
  845.      * @see org.apache.commons.lang3.time.DateParser#getTimeZone()
  846.      */
  847.     @Override
  848.     public TimeZone getTimeZone() {
  849.         return timeZone;
  850.     }
  851.     /**
  852.      * Returns a hash code compatible with equals.
  853.      *
  854.      * @return a hash code compatible with equals
  855.      */
  856.     @Override
  857.     public int hashCode() {
  858.         return pattern.hashCode() + 13 * (timeZone.hashCode() + 13 * locale.hashCode());
  859.     }
  860.     /**
  861.      * Initializes derived fields from defining fields. This is called from constructor and from readObject (de-serialization)
  862.      *
  863.      * @param definingCalendar the {@link java.util.Calendar} instance used to initialize this FastDateParser
  864.      */
  865.     private void init(final Calendar definingCalendar) {
  866.         patterns = new ArrayList<>();

  867.         final StrategyParser strategyParser = new StrategyParser(definingCalendar);
  868.         for (;;) {
  869.             final StrategyAndWidth field = strategyParser.getNextStrategy();
  870.             if (field == null) {
  871.                 break;
  872.             }
  873.             patterns.add(field);
  874.         }
  875.     }

  876.     /*
  877.      * (non-Javadoc)
  878.      *
  879.      * @see org.apache.commons.lang3.time.DateParser#parse(String)
  880.      */
  881.     @Override
  882.     public Date parse(final String source) throws ParseException {
  883.         final ParsePosition pp = new ParsePosition(0);
  884.         final Date date = parse(source, pp);
  885.         if (date == null) {
  886.             // Add a note regarding supported date range
  887.             if (locale.equals(JAPANESE_IMPERIAL)) {
  888.                 throw new ParseException("(The " + locale + " locale does not support dates before 1868 AD)\nUnparseable date: \"" + source,
  889.                         pp.getErrorIndex());
  890.             }
  891.             throw new ParseException("Unparseable date: " + source, pp.getErrorIndex());
  892.         }
  893.         return date;
  894.     }
  895.     /**
  896.      * This implementation updates the ParsePosition if the parse succeeds. However, it sets the error index to the position before the failed field unlike the
  897.      * method {@link java.text.SimpleDateFormat#parse(String, ParsePosition)} which sets the error index to after the failed field.
  898.      * <p>
  899.      * To determine if the parse has succeeded, the caller must check if the current parse position given by {@link ParsePosition#getIndex()} has been updated.
  900.      * If the input buffer has been fully parsed, then the index will point to just after the end of the input buffer.
  901.      *
  902.      * @see org.apache.commons.lang3.time.DateParser#parse(String, java.text.ParsePosition)
  903.      */
  904.     @Override
  905.     public Date parse(final String source, final ParsePosition pos) {
  906.         // timing tests indicate getting new instance is 19% faster than cloning
  907.         final Calendar cal = Calendar.getInstance(timeZone, locale);
  908.         cal.clear();

  909.         return parse(source, pos, cal) ? cal.getTime() : null;
  910.     }
  911.     /**
  912.      * Parses a formatted date string according to the format. Updates the Calendar with parsed fields. Upon success, the ParsePosition index is updated to
  913.      * indicate how much of the source text was consumed. Not all source text needs to be consumed. Upon parse failure, ParsePosition error index is updated to
  914.      * the offset of the source text which does not match the supplied format.
  915.      *
  916.      * @param source   The text to parse.
  917.      * @param pos      On input, the position in the source to start parsing, on output, updated position.
  918.      * @param calendar The calendar into which to set parsed fields.
  919.      * @return true, if source has been parsed (pos parsePosition is updated); otherwise false (and pos errorIndex is updated)
  920.      * @throws IllegalArgumentException when Calendar has been set to be not lenient, and a parsed field is out of range.
  921.      */
  922.     @Override
  923.     public boolean parse(final String source, final ParsePosition pos, final Calendar calendar) {
  924.         final ListIterator<StrategyAndWidth> lt = patterns.listIterator();
  925.         while (lt.hasNext()) {
  926.             final StrategyAndWidth strategyAndWidth = lt.next();
  927.             final int maxWidth = strategyAndWidth.getMaxWidth(lt);
  928.             if (!strategyAndWidth.strategy.parse(this, calendar, source, pos, maxWidth)) {
  929.                 return false;
  930.             }
  931.         }
  932.         return true;
  933.     }

  934.     /*
  935.      * (non-Javadoc)
  936.      *
  937.      * @see org.apache.commons.lang3.time.DateParser#parseObject(String)
  938.      */
  939.     @Override
  940.     public Object parseObject(final String source) throws ParseException {
  941.         return parse(source);
  942.     }

  943.     /*
  944.      * (non-Javadoc)
  945.      *
  946.      * @see org.apache.commons.lang3.time.DateParser#parseObject(String, java.text.ParsePosition)
  947.      */
  948.     @Override
  949.     public Object parseObject(final String source, final ParsePosition pos) {
  950.         return parse(source, pos);
  951.     }
  952.     // Serializing
  953.     /**
  954.      * Creates the object after serialization. This implementation reinitializes the transient properties.
  955.      *
  956.      * @param in ObjectInputStream from which the object is being deserialized.
  957.      * @throws IOException            if there is an IO issue.
  958.      * @throws ClassNotFoundException if a class cannot be found.
  959.      */
  960.     private void readObject(final ObjectInputStream in) throws IOException, ClassNotFoundException {
  961.         in.defaultReadObject();

  962.         final Calendar definingCalendar = Calendar.getInstance(timeZone, locale);
  963.         init(definingCalendar);
  964.     }
  965.     /**
  966.      * Gets a string version of this formatter.
  967.      *
  968.      * @return a debugging string
  969.      */
  970.     @Override
  971.     public String toString() {
  972.         return "FastDateParser[" + pattern + ", " + locale + ", " + timeZone.getID() + "]";
  973.     }
  974.     /**
  975.      * Converts all state of this instance to a String handy for debugging.
  976.      *
  977.      * @return a string.
  978.      * @since 3.12.0
  979.      */
  980.     public String toStringAll() {
  981.         return "FastDateParser [pattern=" + pattern + ", timeZone=" + timeZone + ", locale=" + locale + ", century=" + century + ", startYear=" + startYear
  982.                 + ", patterns=" + patterns + "]";
  983.     }
  984. }