commit
40134ecdb3
|
@ -22,6 +22,7 @@
|
||||||
<body>
|
<body>
|
||||||
|
|
||||||
<release version="3.5" date="tba" description="tba">
|
<release version="3.5" date="tba" description="tba">
|
||||||
|
<action issue="LANG-1153" type="add" dev="chas">Implement ParsePosition api for FastDateParser</action>
|
||||||
<action issue="LANG-1141" type="fix" dev="oheger">StrLookup.systemPropertiesLookup() no longer reacts on changes on system properties</action>
|
<action issue="LANG-1141" type="fix" dev="oheger">StrLookup.systemPropertiesLookup() no longer reacts on changes on system properties</action>
|
||||||
<action issue="LANG-1147" type="fix" dev="sebb" due-to="Loic Guibert">EnumUtils *BitVector issue with more than 32 values Enum</action>
|
<action issue="LANG-1147" type="fix" dev="sebb" due-to="Loic Guibert">EnumUtils *BitVector issue with more than 32 values Enum</action>
|
||||||
<action issue="LANG-1059" type="fix" dev="sebb" due-to="Colin Casey">Capitalize javadoc is incorrect</action>
|
<action issue="LANG-1059" type="fix" dev="sebb" due-to="Colin Casey">Capitalize javadoc is incorrect</action>
|
||||||
|
|
|
@ -368,19 +368,22 @@ public class DateUtils {
|
||||||
final TimeZone tz = TimeZone.getDefault();
|
final TimeZone tz = TimeZone.getDefault();
|
||||||
final Locale lcl = locale==null ?Locale.getDefault() : locale;
|
final Locale lcl = locale==null ?Locale.getDefault() : locale;
|
||||||
final ParsePosition pos = new ParsePosition(0);
|
final ParsePosition pos = new ParsePosition(0);
|
||||||
|
final Calendar calendar = Calendar.getInstance(tz, lcl);
|
||||||
|
calendar.setLenient(lenient);
|
||||||
|
|
||||||
for (final String parsePattern : parsePatterns) {
|
for (final String parsePattern : parsePatterns) {
|
||||||
FastDateParser fdp = new FastDateParser(parsePattern, tz, lcl, null, lenient);
|
FastDateParser fdp = new FastDateParser(parsePattern, tz, lcl);
|
||||||
|
calendar.clear();
|
||||||
try {
|
try {
|
||||||
Date date = fdp.parse(str, pos);
|
if (fdp.parse(str, pos, calendar) && pos.getIndex()==str.length()) {
|
||||||
if (pos.getIndex() == str.length()) {
|
return calendar.getTime();
|
||||||
return date;
|
}
|
||||||
|
}
|
||||||
|
catch(IllegalArgumentException ignore) {
|
||||||
|
// leniency is preventing calendar from being set
|
||||||
}
|
}
|
||||||
pos.setIndex(0);
|
pos.setIndex(0);
|
||||||
}
|
}
|
||||||
catch(IllegalArgumentException iae) {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
throw new ParseException("Unable to parse the date: " + str, -1);
|
throw new ParseException("Unable to parse the date: " + str, -1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -24,12 +24,17 @@ import java.text.ParseException;
|
||||||
import java.text.ParsePosition;
|
import java.text.ParsePosition;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Calendar;
|
import java.util.Calendar;
|
||||||
|
import java.util.Comparator;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.ListIterator;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.Map.Entry;
|
||||||
|
import java.util.Set;
|
||||||
import java.util.TimeZone;
|
import java.util.TimeZone;
|
||||||
|
import java.util.TreeSet;
|
||||||
import java.util.concurrent.ConcurrentHashMap;
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
import java.util.concurrent.ConcurrentMap;
|
import java.util.concurrent.ConcurrentMap;
|
||||||
import java.util.regex.Matcher;
|
import java.util.regex.Matcher;
|
||||||
|
@ -67,6 +72,7 @@ import java.util.regex.Pattern;
|
||||||
* @see FastDatePrinter
|
* @see FastDatePrinter
|
||||||
*/
|
*/
|
||||||
public class FastDateParser implements DateParser, Serializable {
|
public class FastDateParser implements DateParser, Serializable {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Required for serialization support.
|
* Required for serialization support.
|
||||||
*
|
*
|
||||||
|
@ -82,15 +88,10 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
private final Locale locale;
|
private final Locale locale;
|
||||||
private final int century;
|
private final int century;
|
||||||
private final int startYear;
|
private final int startYear;
|
||||||
private final boolean lenient;
|
|
||||||
|
|
||||||
// derived fields
|
// derived fields
|
||||||
private transient Pattern parsePattern;
|
private transient List<StrategyAndWidth> patterns;
|
||||||
private transient Strategy[] strategies;
|
|
||||||
|
|
||||||
// dynamic fields to communicate with Strategy
|
|
||||||
private transient String currentFormatField;
|
|
||||||
private transient Strategy nextStrategy;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* <p>Constructs a new FastDateParser.</p>
|
* <p>Constructs a new FastDateParser.</p>
|
||||||
|
@ -104,22 +105,7 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
* @param locale non-null locale
|
* @param locale non-null locale
|
||||||
*/
|
*/
|
||||||
protected FastDateParser(final String pattern, final TimeZone timeZone, final Locale locale) {
|
protected FastDateParser(final String pattern, final TimeZone timeZone, final Locale locale) {
|
||||||
this(pattern, timeZone, locale, null, true);
|
this(pattern, timeZone, locale, null);
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* <p>Constructs a new FastDateParser.</p>
|
|
||||||
*
|
|
||||||
* @param pattern non-null {@link java.text.SimpleDateFormat} compatible
|
|
||||||
* pattern
|
|
||||||
* @param timeZone non-null time zone to use
|
|
||||||
* @param locale non-null locale
|
|
||||||
* @param centuryStart The start of the century for 2 digit year parsing
|
|
||||||
*
|
|
||||||
* @since 3.3
|
|
||||||
*/
|
|
||||||
protected FastDateParser(final String pattern, final TimeZone timeZone, final Locale locale, final Date centuryStart) {
|
|
||||||
this(pattern, timeZone, locale, centuryStart, true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -135,12 +121,10 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
*
|
*
|
||||||
* @since 3.5
|
* @since 3.5
|
||||||
*/
|
*/
|
||||||
protected FastDateParser(final String pattern, final TimeZone timeZone, final Locale locale,
|
protected FastDateParser(final String pattern, final TimeZone timeZone, final Locale locale, final Date centuryStart) {
|
||||||
final Date centuryStart, final boolean lenient) {
|
|
||||||
this.pattern = pattern;
|
this.pattern = pattern;
|
||||||
this.timeZone = timeZone;
|
this.timeZone = timeZone;
|
||||||
this.locale = locale;
|
this.locale = locale;
|
||||||
this.lenient = lenient;
|
|
||||||
|
|
||||||
final Calendar definingCalendar = Calendar.getInstance(timeZone, locale);
|
final Calendar definingCalendar = Calendar.getInstance(timeZone, locale);
|
||||||
|
|
||||||
|
@ -170,41 +154,112 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
* @param definingCalendar the {@link java.util.Calendar} instance used to initialize this FastDateParser
|
* @param definingCalendar the {@link java.util.Calendar} instance used to initialize this FastDateParser
|
||||||
*/
|
*/
|
||||||
private void init(final Calendar definingCalendar) {
|
private void init(final Calendar definingCalendar) {
|
||||||
|
patterns = new ArrayList<StrategyAndWidth>();
|
||||||
|
|
||||||
final StringBuilder regex= new StringBuilder();
|
StrategyParser fm = new StrategyParser(pattern, definingCalendar);
|
||||||
final List<Strategy> collector = new ArrayList<Strategy>();
|
|
||||||
|
|
||||||
final Matcher patternMatcher= formatPattern.matcher(pattern);
|
|
||||||
if(!patternMatcher.lookingAt()) {
|
|
||||||
throw new IllegalArgumentException(
|
|
||||||
"Illegal pattern character '" + pattern.charAt(patternMatcher.regionStart()) + "'");
|
|
||||||
}
|
|
||||||
|
|
||||||
currentFormatField= patternMatcher.group();
|
|
||||||
Strategy currentStrategy= getStrategy(currentFormatField, definingCalendar);
|
|
||||||
for(;;) {
|
for(;;) {
|
||||||
patternMatcher.region(patternMatcher.end(), patternMatcher.regionEnd());
|
StrategyAndWidth field = fm.getNextStrategy();
|
||||||
if(!patternMatcher.lookingAt()) {
|
if(field==null) {
|
||||||
nextStrategy = null;
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
final String nextFormatField= patternMatcher.group();
|
patterns.add(field);
|
||||||
nextStrategy = getStrategy(nextFormatField, definingCalendar);
|
|
||||||
if(currentStrategy.addRegex(this, regex)) {
|
|
||||||
collector.add(currentStrategy);
|
|
||||||
}
|
}
|
||||||
currentFormatField= nextFormatField;
|
|
||||||
currentStrategy= nextStrategy;
|
|
||||||
}
|
}
|
||||||
if (patternMatcher.regionStart() != patternMatcher.regionEnd()) {
|
|
||||||
throw new IllegalArgumentException("Failed to parse \""+pattern+"\" ; gave up at index "+patternMatcher.regionStart());
|
// helper classes to parse the format string
|
||||||
|
//-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Struct to hold strategy and filed width
|
||||||
|
*/
|
||||||
|
private static class StrategyAndWidth {
|
||||||
|
final Strategy strategy;
|
||||||
|
final int width;
|
||||||
|
|
||||||
|
StrategyAndWidth(Strategy strategy, int width) {
|
||||||
|
this.strategy = strategy;
|
||||||
|
this.width = width;
|
||||||
}
|
}
|
||||||
if(currentStrategy.addRegex(this, regex)) {
|
|
||||||
collector.add(currentStrategy);
|
int getMaxWidth(ListIterator<StrategyAndWidth> lt) {
|
||||||
|
if(!strategy.isNumber() || !lt.hasNext()) {
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
currentFormatField= null;
|
Strategy nextStrategy = lt.next().strategy;
|
||||||
strategies= collector.toArray(new Strategy[collector.size()]);
|
lt.previous();
|
||||||
parsePattern= Pattern.compile(regex.toString());
|
return nextStrategy.isNumber() ?width :0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parse format into Strategies
|
||||||
|
*/
|
||||||
|
private class StrategyParser {
|
||||||
|
final private String pattern;
|
||||||
|
final private Calendar definingCalendar;
|
||||||
|
private int currentIdx;
|
||||||
|
|
||||||
|
StrategyParser(String pattern, Calendar definingCalendar) {
|
||||||
|
this.pattern = pattern;
|
||||||
|
this.definingCalendar = definingCalendar;
|
||||||
|
}
|
||||||
|
|
||||||
|
StrategyAndWidth getNextStrategy() {
|
||||||
|
if(currentIdx >= pattern.length()) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
char c = pattern.charAt(currentIdx);
|
||||||
|
if( isFormatLetter(c)) {
|
||||||
|
return letterPattern(c);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return literal();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private StrategyAndWidth letterPattern(char c) {
|
||||||
|
int begin = currentIdx;
|
||||||
|
while( ++currentIdx<pattern.length() ) {
|
||||||
|
if(pattern.charAt(currentIdx) != c) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int width = currentIdx - begin;
|
||||||
|
return new StrategyAndWidth(getStrategy(c, width, definingCalendar), width);
|
||||||
|
}
|
||||||
|
|
||||||
|
private StrategyAndWidth literal() {
|
||||||
|
boolean activeQuote = false;
|
||||||
|
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
|
while( currentIdx<pattern.length() ) {
|
||||||
|
char c= pattern.charAt(currentIdx);
|
||||||
|
if( !activeQuote && isFormatLetter( c ) ) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
else if( c=='\'' ) {
|
||||||
|
if(++currentIdx==pattern.length() || pattern.charAt(currentIdx)!='\'') {
|
||||||
|
activeQuote = !activeQuote;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
++currentIdx;
|
||||||
|
sb.append(c);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(activeQuote) {
|
||||||
|
throw new IllegalArgumentException("Unterminated quote");
|
||||||
|
}
|
||||||
|
|
||||||
|
String formatField = sb.toString();
|
||||||
|
return new StrategyAndWidth(new CopyQuotedStrategy(formatField), formatField.length());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static boolean isFormatLetter(char c) {
|
||||||
|
return c>='A' && c<='Z' || c>='a' && c<='z';
|
||||||
}
|
}
|
||||||
|
|
||||||
// Accessors
|
// Accessors
|
||||||
|
@ -233,14 +288,6 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
return locale;
|
return locale;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the generated pattern (for testing purposes).
|
|
||||||
*
|
|
||||||
* @return the generated pattern
|
|
||||||
*/
|
|
||||||
Pattern getParsePattern() {
|
|
||||||
return parsePattern;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Basics
|
// Basics
|
||||||
//-----------------------------------------------------------------------
|
//-----------------------------------------------------------------------
|
||||||
|
@ -311,15 +358,16 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public Date parse(final String source) throws ParseException {
|
public Date parse(final String source) throws ParseException {
|
||||||
final Date date= parse(source, new ParsePosition(0));
|
ParsePosition pp = new ParsePosition(0);
|
||||||
|
final Date date= parse(source, pp);
|
||||||
if(date==null) {
|
if(date==null) {
|
||||||
// Add a note re supported date range
|
// Add a note re supported date range
|
||||||
if (locale.equals(JAPANESE_IMPERIAL)) {
|
if (locale.equals(JAPANESE_IMPERIAL)) {
|
||||||
throw new ParseException(
|
throw new ParseException(
|
||||||
"(The " +locale + " locale does not support dates before 1868 AD)\n" +
|
"(The " +locale + " locale does not support dates before 1868 AD)\n" +
|
||||||
"Unparseable date: \""+source+"\" does not match "+parsePattern.pattern(), 0);
|
"Unparseable date: \""+source, pp.getErrorIndex());
|
||||||
}
|
}
|
||||||
throw new ParseException("Unparseable date: \""+source+"\" does not match "+parsePattern.pattern(), 0);
|
throw new ParseException("Unparseable date: "+source, pp.getErrorIndex());
|
||||||
}
|
}
|
||||||
return date;
|
return date;
|
||||||
}
|
}
|
||||||
|
@ -333,9 +381,10 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This implementation updates the ParsePosition if the parse succeeeds.
|
* This implementation updates the ParsePosition if the parse succeeds.
|
||||||
* However, unlike the method {@link java.text.SimpleDateFormat#parse(String, ParsePosition)}
|
* However, it sets the error index to the position before the failed field unlike
|
||||||
* it is not able to set the error Index - i.e. {@link ParsePosition#getErrorIndex()} - if the parse fails.
|
* the method {@link java.text.SimpleDateFormat#parse(String, ParsePosition)} which sets
|
||||||
|
* the error index to after the failed field.
|
||||||
* <p>
|
* <p>
|
||||||
* To determine if the parse has succeeded, the caller must check if the current parse position
|
* To determine if the parse has succeeded, the caller must check if the current parse position
|
||||||
* given by {@link ParsePosition#getIndex()} has been updated. If the input buffer has been fully
|
* given by {@link ParsePosition#getIndex()} has been updated. If the input buffer has been fully
|
||||||
|
@ -346,22 +395,36 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public Date parse(final String source, final ParsePosition pos) {
|
public Date parse(final String source, final ParsePosition pos) {
|
||||||
final int offset= pos.getIndex();
|
|
||||||
final Matcher matcher= parsePattern.matcher(source.substring(offset));
|
|
||||||
if(!matcher.lookingAt()) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
// timing tests indicate getting new instance is 19% faster than cloning
|
// timing tests indicate getting new instance is 19% faster than cloning
|
||||||
final Calendar cal= Calendar.getInstance(timeZone, locale);
|
final Calendar cal= Calendar.getInstance(timeZone, locale);
|
||||||
cal.clear();
|
cal.clear();
|
||||||
cal.setLenient(lenient);
|
|
||||||
|
|
||||||
for(int i=0; i<strategies.length;) {
|
return parse(source, pos, cal) ?cal.getTime() :null;
|
||||||
final Strategy strategy= strategies[i++];
|
|
||||||
strategy.setCalendar(this, cal, matcher.group(i));
|
|
||||||
}
|
}
|
||||||
pos.setIndex(offset+matcher.end());
|
|
||||||
return cal.getTime();
|
/**
|
||||||
|
* Parse a formatted date string according to the format. Updates the Calendar with parsed fields.
|
||||||
|
* Upon success, the ParsePosition index is updated to 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
|
||||||
|
* the offset of the source text which does not match the supplied format.
|
||||||
|
*
|
||||||
|
* @param source The text to parse.
|
||||||
|
* @param pos On input, the position in the source to start parsing, on output, updated position.
|
||||||
|
* @param calendar The calendar into which to set parsed fields.
|
||||||
|
* @return true, if source has been parsed (pos parsePosition is updated); otherwise false (and pos errorIndex is updated)
|
||||||
|
* @throws IllegalArgumentException when Calendar has been set to be not lenient, and a parsed field is
|
||||||
|
* out of range.
|
||||||
|
*/
|
||||||
|
public boolean parse(final String source, final ParsePosition pos, final Calendar calendar) {
|
||||||
|
ListIterator<StrategyAndWidth> lt = patterns.listIterator();
|
||||||
|
while(lt.hasNext()) {
|
||||||
|
StrategyAndWidth pattern = lt.next();
|
||||||
|
int maxWidth = pattern.getMaxWidth(lt);
|
||||||
|
if(!pattern.strategy.parse(this, calendar, source, pos, maxWidth)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Support for strategies
|
// Support for strategies
|
||||||
|
@ -392,62 +455,42 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Escape constant fields into regular expression
|
* alternatives should be ordered longer first, and shorter last. comparisons should be case insensitive.
|
||||||
* @param regex The destination regex
|
|
||||||
* @param value The source field
|
|
||||||
* @param unquote If true, replace two success quotes ('') with single quote (')
|
|
||||||
* @return The <code>StringBuilder</code>
|
|
||||||
*/
|
*/
|
||||||
private static StringBuilder escapeRegex(final StringBuilder regex, final String value, final boolean unquote) {
|
private static final Comparator<Map.Entry<String, Integer>> ALTERNATIVES_ORDERING = new Comparator<Map.Entry<String, Integer>>() {
|
||||||
regex.append("\\Q");
|
@Override
|
||||||
for(int i= 0; i<value.length(); ++i) {
|
public int compare(Map.Entry<String, Integer> left, Map.Entry<String, Integer> right) {
|
||||||
char c= value.charAt(i);
|
int v = left.getValue() - right.getValue();
|
||||||
switch(c) {
|
if(v!=0) {
|
||||||
case '\'':
|
return v;
|
||||||
if(unquote) {
|
|
||||||
if(++i==value.length()) {
|
|
||||||
return regex;
|
|
||||||
}
|
}
|
||||||
c= value.charAt(i);
|
return right.getKey().compareToIgnoreCase(left.getKey());
|
||||||
}
|
}
|
||||||
break;
|
};
|
||||||
case '\\':
|
|
||||||
if(++i==value.length()) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
/*
|
|
||||||
* If we have found \E, we replace it with \E\\E\Q, i.e. we stop the quoting,
|
|
||||||
* quote the \ in \E, then restart the quoting.
|
|
||||||
*
|
|
||||||
* Otherwise we just output the two characters.
|
|
||||||
* In each case the initial \ needs to be output and the final char is done at the end
|
|
||||||
*/
|
|
||||||
regex.append(c); // we always want the original \
|
|
||||||
c = value.charAt(i); // Is it followed by E ?
|
|
||||||
if (c == 'E') { // \E detected
|
|
||||||
regex.append("E\\\\E\\"); // see comment above
|
|
||||||
c = 'Q'; // appended below
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
regex.append(c);
|
|
||||||
}
|
|
||||||
regex.append("\\E");
|
|
||||||
return regex;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the short and long values displayed for a field
|
* Get the short and long values displayed for a field
|
||||||
* @param field The field of interest
|
* @param cal The calendar to obtain the short and long values
|
||||||
* @param definingCalendar The calendar to obtain the short and long values
|
|
||||||
* @param locale The locale of display names
|
* @param locale The locale of display names
|
||||||
* @return A Map of the field key / value pairs
|
* @param field The field of interest
|
||||||
|
* @param regex The regular expression to build
|
||||||
|
* @param vales The map to fill
|
||||||
*/
|
*/
|
||||||
private static Map<String, Integer> getDisplayNames(final int field, final Calendar definingCalendar, final Locale locale) {
|
private static void appendDisplayNames(Calendar cal, Locale locale, int field,
|
||||||
return definingCalendar.getDisplayNames(field, Calendar.ALL_STYLES, locale);
|
StringBuilder regex, Map<String, Integer> values) {
|
||||||
|
|
||||||
|
Set<Entry<String, Integer>> displayNames = cal.getDisplayNames(field, Calendar.ALL_STYLES, locale).entrySet();
|
||||||
|
TreeSet<Map.Entry<String, Integer>> sort = new TreeSet<Map.Entry<String, Integer>>(ALTERNATIVES_ORDERING);
|
||||||
|
sort.addAll(displayNames);
|
||||||
|
|
||||||
|
for (Map.Entry<String, Integer> entry : sort) {
|
||||||
|
String symbol = entry.getKey();
|
||||||
|
if (symbol.length() > 0) {
|
||||||
|
if (values.put(symbol.toLowerCase(locale), entry.getValue()) == null) {
|
||||||
|
simpleQuote(regex, symbol).append('|');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -460,27 +503,10 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
return twoDigitYear>=startYear ?trial :trial+100;
|
return twoDigitYear>=startYear ?trial :trial+100;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Is the next field a number?
|
|
||||||
* @return true, if next field will be a number
|
|
||||||
*/
|
|
||||||
boolean isNextNumber() {
|
|
||||||
return nextStrategy!=null && nextStrategy.isNumber();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* What is the width of the current field?
|
|
||||||
* @return The number of characters in the current format field
|
|
||||||
*/
|
|
||||||
int getFieldWidth() {
|
|
||||||
return currentFormatField.length();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A strategy to parse a single field from the parsing pattern
|
* A strategy to parse a single field from the parsing pattern
|
||||||
*/
|
*/
|
||||||
private static abstract class Strategy {
|
private static abstract class Strategy {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Is this field a number?
|
* Is this field a number?
|
||||||
* The default implementation returns false.
|
* The default implementation returns false.
|
||||||
|
@ -491,36 +517,49 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
abstract boolean parse(FastDateParser parser, Calendar calendar, String source, ParsePosition pos, int maxWidth);
|
||||||
* Set the Calendar with the parsed field.
|
|
||||||
*
|
|
||||||
* The default implementation does nothing.
|
|
||||||
*
|
|
||||||
* @param parser The parser calling this strategy
|
|
||||||
* @param cal The <code>Calendar</code> to set
|
|
||||||
* @param value The parsed field to translate and set in cal
|
|
||||||
*/
|
|
||||||
void setCalendar(final FastDateParser parser, final Calendar cal, final String value) {
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate a <code>Pattern</code> regular expression to the <code>StringBuilder</code>
|
* A strategy to parse a single field from the parsing pattern
|
||||||
* which will accept this field
|
|
||||||
* @param parser The parser calling this strategy
|
|
||||||
* @param regex The <code>StringBuilder</code> to append to
|
|
||||||
* @return true, if this field will set the calendar;
|
|
||||||
* false, if this field is a constant value
|
|
||||||
*/
|
*/
|
||||||
abstract boolean addRegex(FastDateParser parser, StringBuilder regex);
|
private static abstract class PatternStrategy extends Strategy {
|
||||||
|
|
||||||
|
private Pattern pattern;
|
||||||
|
|
||||||
|
void createPattern(StringBuilder regex) {
|
||||||
|
createPattern(regex.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
void createPattern(String regex) {
|
||||||
|
this.pattern = Pattern.compile(regex);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A <code>Pattern</code> to parse the user supplied SimpleDateFormat pattern
|
* Is this field a number?
|
||||||
|
* The default implementation returns false.
|
||||||
|
*
|
||||||
|
* @return true, if field is a number
|
||||||
*/
|
*/
|
||||||
private static final Pattern formatPattern= Pattern.compile(
|
@Override
|
||||||
"D+|E+|F+|G+|H+|K+|M+|S+|W+|X+|Z+|a+|d+|h+|k+|m+|s+|w+|y+|z+|''|'[^']++(''[^']*+)*+'|[^'A-Za-z]++");
|
boolean isNumber() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
boolean parse(FastDateParser parser, Calendar calendar, String source, ParsePosition pos, int maxWidth) {
|
||||||
|
Matcher matcher = pattern.matcher(source.substring(pos.getIndex()));
|
||||||
|
if(!matcher.lookingAt()) {
|
||||||
|
pos.setErrorIndex(pos.getIndex());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
pos.setIndex(pos.getIndex() + matcher.end(1));
|
||||||
|
setCalendar(parser, calendar, matcher.group(1));
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
abstract void setCalendar(FastDateParser parser, Calendar cal, String value);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Obtain a Strategy given a field from a SimpleDateFormat pattern
|
* Obtain a Strategy given a field from a SimpleDateFormat pattern
|
||||||
|
@ -528,15 +567,10 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
* @param definingCalendar The calendar to obtain the short and long values
|
* @param definingCalendar The calendar to obtain the short and long values
|
||||||
* @return The Strategy that will handle parsing for the field
|
* @return The Strategy that will handle parsing for the field
|
||||||
*/
|
*/
|
||||||
private Strategy getStrategy(final String formatField, final Calendar definingCalendar) {
|
private Strategy getStrategy(char f, int width, final Calendar definingCalendar) {
|
||||||
switch(formatField.charAt(0)) {
|
switch(f) {
|
||||||
case '\'':
|
|
||||||
if(formatField.length()>2) {
|
|
||||||
return new CopyQuotedStrategy(formatField.substring(1, formatField.length()-1));
|
|
||||||
}
|
|
||||||
//$FALL-THROUGH$
|
|
||||||
default:
|
default:
|
||||||
return new CopyQuotedStrategy(formatField);
|
throw new IllegalArgumentException("Format '"+f+"' not supported");
|
||||||
case 'D':
|
case 'D':
|
||||||
return DAY_OF_YEAR_STRATEGY;
|
return DAY_OF_YEAR_STRATEGY;
|
||||||
case 'E':
|
case 'E':
|
||||||
|
@ -550,7 +584,7 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
case 'K': // Hour in am/pm (0-11)
|
case 'K': // Hour in am/pm (0-11)
|
||||||
return HOUR_STRATEGY;
|
return HOUR_STRATEGY;
|
||||||
case 'M':
|
case 'M':
|
||||||
return formatField.length()>=3 ?getLocaleSpecificStrategy(Calendar.MONTH, definingCalendar) :NUMBER_MONTH_STRATEGY;
|
return width>=3 ?getLocaleSpecificStrategy(Calendar.MONTH, definingCalendar) :NUMBER_MONTH_STRATEGY;
|
||||||
case 'S':
|
case 'S':
|
||||||
return MILLISECOND_STRATEGY;
|
return MILLISECOND_STRATEGY;
|
||||||
case 'W':
|
case 'W':
|
||||||
|
@ -570,12 +604,12 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
case 'w':
|
case 'w':
|
||||||
return WEEK_OF_YEAR_STRATEGY;
|
return WEEK_OF_YEAR_STRATEGY;
|
||||||
case 'y':
|
case 'y':
|
||||||
return formatField.length()>2 ?LITERAL_YEAR_STRATEGY :ABBREVIATED_YEAR_STRATEGY;
|
return width>2 ?LITERAL_YEAR_STRATEGY :ABBREVIATED_YEAR_STRATEGY;
|
||||||
case 'X':
|
case 'X':
|
||||||
return ISO8601TimeZoneStrategy.getStrategy(formatField.length());
|
return ISO8601TimeZoneStrategy.getStrategy(width);
|
||||||
case 'Z':
|
case 'Z':
|
||||||
if (formatField.equals("ZZ")) {
|
if (width==2) {
|
||||||
return ISO_8601_STRATEGY;
|
return ISO8601TimeZoneStrategy.ISO_8601_3_STRATEGY;
|
||||||
}
|
}
|
||||||
//$FALL-THROUGH$
|
//$FALL-THROUGH$
|
||||||
case 'z':
|
case 'z':
|
||||||
|
@ -611,7 +645,7 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
Strategy strategy= cache.get(locale);
|
Strategy strategy= cache.get(locale);
|
||||||
if(strategy==null) {
|
if(strategy==null) {
|
||||||
strategy= field==Calendar.ZONE_OFFSET
|
strategy= field==Calendar.ZONE_OFFSET
|
||||||
? new TimeZoneStrategy(locale)
|
? new TimeZoneStrategy(definingCalendar, locale)
|
||||||
: new CaseInsensitiveTextStrategy(field, definingCalendar, locale);
|
: new CaseInsensitiveTextStrategy(field, definingCalendar, locale);
|
||||||
final Strategy inCache= cache.putIfAbsent(locale, strategy);
|
final Strategy inCache= cache.putIfAbsent(locale, strategy);
|
||||||
if(inCache!=null) {
|
if(inCache!=null) {
|
||||||
|
@ -625,14 +659,15 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
* A strategy that copies the static or quoted field in the parsing pattern
|
* A strategy that copies the static or quoted field in the parsing pattern
|
||||||
*/
|
*/
|
||||||
private static class CopyQuotedStrategy extends Strategy {
|
private static class CopyQuotedStrategy extends Strategy {
|
||||||
private final String formatField;
|
|
||||||
|
final private String formatField;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a Strategy that ensures the formatField has literal text
|
* Construct a Strategy that ensures the formatField has literal text
|
||||||
* @param formatField The literal text to match
|
* @param formatField The literal text to match
|
||||||
*/
|
*/
|
||||||
CopyQuotedStrategy(final String formatField) {
|
CopyQuotedStrategy(final String formatField) {
|
||||||
this.formatField= formatField;
|
this.formatField = formatField;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -640,30 +675,34 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
boolean isNumber() {
|
boolean isNumber() {
|
||||||
char c= formatField.charAt(0);
|
return false;
|
||||||
if(c=='\'') {
|
|
||||||
c= formatField.charAt(1);
|
|
||||||
}
|
|
||||||
return Character.isDigit(c);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* {@inheritDoc}
|
|
||||||
*/
|
|
||||||
@Override
|
@Override
|
||||||
boolean addRegex(final FastDateParser parser, final StringBuilder regex) {
|
boolean parse(FastDateParser parser, Calendar calendar, String source, ParsePosition pos, int maxWidth) {
|
||||||
escapeRegex(regex, formatField, true);
|
for (int idx = 0; idx < formatField.length(); ++idx) {
|
||||||
|
int sIdx = idx + pos.getIndex();
|
||||||
|
if (sIdx == source.length()) {
|
||||||
|
pos.setErrorIndex(sIdx);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
if (formatField.charAt(idx) != source.charAt(sIdx)) {
|
||||||
|
pos.setErrorIndex(sIdx);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pos.setIndex(formatField.length() + pos.getIndex());
|
||||||
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A strategy that handles a text field in the parsing pattern
|
* A strategy that handles a text field in the parsing pattern
|
||||||
*/
|
*/
|
||||||
private static class CaseInsensitiveTextStrategy extends Strategy {
|
private static class CaseInsensitiveTextStrategy extends PatternStrategy {
|
||||||
private final int field;
|
private final int field;
|
||||||
private final Locale locale;
|
final Locale locale;
|
||||||
private final Map<String, Integer> lKeyValues;
|
private final Map<String, Integer> lKeyValues = new HashMap<String,Integer>();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a Strategy that parses a Text field
|
* Construct a Strategy that parses a Text field
|
||||||
|
@ -672,44 +711,23 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
* @param locale The Locale to use
|
* @param locale The Locale to use
|
||||||
*/
|
*/
|
||||||
CaseInsensitiveTextStrategy(final int field, final Calendar definingCalendar, final Locale locale) {
|
CaseInsensitiveTextStrategy(final int field, final Calendar definingCalendar, final Locale locale) {
|
||||||
this.field= field;
|
this.field = field;
|
||||||
this.locale= locale;
|
this.locale = locale;
|
||||||
final Map<String, Integer> keyValues = getDisplayNames(field, definingCalendar, locale);
|
|
||||||
this.lKeyValues= new HashMap<String,Integer>();
|
|
||||||
|
|
||||||
for(final Map.Entry<String, Integer> entry : keyValues.entrySet()) {
|
StringBuilder regex = new StringBuilder();
|
||||||
lKeyValues.put(entry.getKey().toLowerCase(locale), entry.getValue());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* {@inheritDoc}
|
|
||||||
*/
|
|
||||||
@Override
|
|
||||||
boolean addRegex(final FastDateParser parser, final StringBuilder regex) {
|
|
||||||
regex.append("((?iu)");
|
regex.append("((?iu)");
|
||||||
for(final String textKeyValue : lKeyValues.keySet()) {
|
appendDisplayNames(definingCalendar, locale, field, regex, lKeyValues);
|
||||||
simpleQuote(regex, textKeyValue).append('|');
|
regex.setLength(regex.length()-1);
|
||||||
}
|
regex.append(")");
|
||||||
regex.setCharAt(regex.length()-1, ')');
|
createPattern(regex);
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@inheritDoc}
|
* {@inheritDoc}
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
void setCalendar(final FastDateParser parser, final Calendar cal, final String value) {
|
void setCalendar(FastDateParser parser, Calendar cal, String value) {
|
||||||
final Integer iVal = lKeyValues.get(value.toLowerCase(locale));
|
final Integer iVal = lKeyValues.get(value.toLowerCase(locale));
|
||||||
if(iVal == null) {
|
|
||||||
final StringBuilder sb= new StringBuilder(value);
|
|
||||||
sb.append(" not in (");
|
|
||||||
for(final String textKeyValue : lKeyValues.keySet()) {
|
|
||||||
sb.append(textKeyValue).append(' ');
|
|
||||||
}
|
|
||||||
sb.setCharAt(sb.length()-1, ')');
|
|
||||||
throw new IllegalArgumentException(sb.toString());
|
|
||||||
}
|
|
||||||
cal.set(field, iVal.intValue());
|
cal.set(field, iVal.intValue());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -737,37 +755,56 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* {@inheritDoc}
|
|
||||||
*/
|
|
||||||
@Override
|
@Override
|
||||||
boolean addRegex(final FastDateParser parser, final StringBuilder regex) {
|
boolean parse(FastDateParser parser, Calendar calendar, String source, ParsePosition pos, int maxWidth) {
|
||||||
// See LANG-954: We use {Nd} rather than {IsNd} because Android does not support the Is prefix
|
int idx = pos.getIndex();
|
||||||
if(parser.isNextNumber()) {
|
int last = source.length();
|
||||||
regex.append("(\\p{Nd}{").append(parser.getFieldWidth()).append("}+)");
|
|
||||||
|
if (maxWidth == 0) {
|
||||||
|
// if no maxWidth, strip leading white space
|
||||||
|
for (; idx < last; ++idx) {
|
||||||
|
char c = source.charAt(idx);
|
||||||
|
if (!Character.isWhitespace(c)) {
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
else {
|
|
||||||
regex.append("(\\p{Nd}++)");
|
|
||||||
}
|
}
|
||||||
|
pos.setIndex(idx);
|
||||||
|
} else {
|
||||||
|
int end = idx + maxWidth;
|
||||||
|
if (last > end) {
|
||||||
|
last = end;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (; idx < last; ++idx) {
|
||||||
|
char c = source.charAt(idx);
|
||||||
|
if (!Character.isDigit(c)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pos.getIndex() == idx) {
|
||||||
|
pos.setErrorIndex(idx);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
int value = Integer.parseInt(source.substring(pos.getIndex(), idx));
|
||||||
|
pos.setIndex(idx);
|
||||||
|
|
||||||
|
calendar.set(field, modify(parser, value));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* {@inheritDoc}
|
|
||||||
*/
|
|
||||||
@Override
|
|
||||||
void setCalendar(final FastDateParser parser, final Calendar cal, final String value) {
|
|
||||||
cal.set(field, modify(Integer.parseInt(value)));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Make any modifications to parsed integer
|
* Make any modifications to parsed integer
|
||||||
|
* @param parser The parser
|
||||||
* @param iValue The parsed integer
|
* @param iValue The parsed integer
|
||||||
* @return The modified value
|
* @return The modified value
|
||||||
*/
|
*/
|
||||||
int modify(final int iValue) {
|
int modify(FastDateParser parser, final int iValue) {
|
||||||
return iValue;
|
return iValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static final Strategy ABBREVIATED_YEAR_STRATEGY = new NumberStrategy(Calendar.YEAR) {
|
private static final Strategy ABBREVIATED_YEAR_STRATEGY = new NumberStrategy(Calendar.YEAR) {
|
||||||
|
@ -775,25 +812,20 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
* {@inheritDoc}
|
* {@inheritDoc}
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
void setCalendar(final FastDateParser parser, final Calendar cal, final String value) {
|
int modify(FastDateParser parser, final int iValue) {
|
||||||
int iValue= Integer.parseInt(value);
|
return iValue<100 ?parser.adjustYear(iValue) :iValue;
|
||||||
if(iValue<100) {
|
|
||||||
iValue= parser.adjustYear(iValue);
|
|
||||||
}
|
|
||||||
cal.set(Calendar.YEAR, iValue);
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A strategy that handles a timezone field in the parsing pattern
|
* A strategy that handles a timezone field in the parsing pattern
|
||||||
*/
|
*/
|
||||||
static class TimeZoneStrategy extends Strategy {
|
static class TimeZoneStrategy extends PatternStrategy {
|
||||||
private static final String RFC_822_TIME_ZONE = "[+-]\\d{4}";
|
private static final String RFC_822_TIME_ZONE = "[+-]\\d{4}";
|
||||||
private static final String GMT_OPTION= "GMT[+-]\\d{1,2}:\\d{2}";
|
private static final String GMT_OPTION= "GMT[+-]\\d{1,2}:\\d{2}";
|
||||||
|
|
||||||
private final Locale locale;
|
private final Locale locale;
|
||||||
private final Map<String, TimeZone> tzNames= new HashMap<String, TimeZone>();
|
private final Map<String, TimeZone> tzNames= new HashMap<String, TimeZone>();
|
||||||
private final String validTimeZoneChars;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Index of zone id
|
* Index of zone id
|
||||||
|
@ -802,9 +834,11 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a Strategy that parses a TimeZone
|
* Construct a Strategy that parses a TimeZone
|
||||||
|
* @param cal TODO
|
||||||
* @param locale The Locale
|
* @param locale The Locale
|
||||||
*/
|
*/
|
||||||
TimeZoneStrategy(final Locale locale) {
|
TimeZoneStrategy(Calendar cal, final Locale locale) {
|
||||||
|
|
||||||
this.locale = locale;
|
this.locale = locale;
|
||||||
|
|
||||||
final StringBuilder sb = new StringBuilder();
|
final StringBuilder sb = new StringBuilder();
|
||||||
|
@ -818,25 +852,15 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
}
|
}
|
||||||
final TimeZone tz = TimeZone.getTimeZone(tzId);
|
final TimeZone tz = TimeZone.getTimeZone(tzId);
|
||||||
for(int i= 1; i<zoneNames.length; ++i) {
|
for(int i= 1; i<zoneNames.length; ++i) {
|
||||||
String zoneName = zoneNames[i].toLowerCase(locale);
|
String zoneName = zoneNames[i];
|
||||||
if (!tzNames.containsKey(zoneName)){
|
if (tzNames.put(zoneName.toLowerCase(locale), tz) == null) {
|
||||||
tzNames.put(zoneName, tz);
|
|
||||||
simpleQuote(sb.append('|'), zoneName);
|
simpleQuote(sb.append('|'), zoneName);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
sb.append(')');
|
sb.append(")");
|
||||||
validTimeZoneChars = sb.toString();
|
createPattern(sb);
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* {@inheritDoc}
|
|
||||||
*/
|
|
||||||
@Override
|
|
||||||
boolean addRegex(final FastDateParser parser, final StringBuilder regex) {
|
|
||||||
regex.append(validTimeZoneChars);
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -853,33 +877,20 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
tz= tzNames.get(value.toLowerCase(locale));
|
tz= tzNames.get(value.toLowerCase(locale));
|
||||||
if(tz==null) {
|
|
||||||
throw new IllegalArgumentException(value + " is not a supported timezone name");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
cal.setTimeZone(tz);
|
cal.setTimeZone(tz);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class ISO8601TimeZoneStrategy extends Strategy {
|
private static class ISO8601TimeZoneStrategy extends PatternStrategy {
|
||||||
// Z, +hh, -hh, +hhmm, -hhmm, +hh:mm or -hh:mm
|
// Z, +hh, -hh, +hhmm, -hhmm, +hh:mm or -hh:mm
|
||||||
private final String pattern;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Construct a Strategy that parses a TimeZone
|
* Construct a Strategy that parses a TimeZone
|
||||||
* @param pattern The Pattern
|
* @param pattern The Pattern
|
||||||
*/
|
*/
|
||||||
ISO8601TimeZoneStrategy(String pattern) {
|
ISO8601TimeZoneStrategy(String pattern) {
|
||||||
this.pattern = pattern;
|
createPattern(pattern);
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* {@inheritDoc}
|
|
||||||
*/
|
|
||||||
@Override
|
|
||||||
boolean addRegex(FastDateParser parser, StringBuilder regex) {
|
|
||||||
regex.append(pattern);
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -921,7 +932,7 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
|
|
||||||
private static final Strategy NUMBER_MONTH_STRATEGY = new NumberStrategy(Calendar.MONTH) {
|
private static final Strategy NUMBER_MONTH_STRATEGY = new NumberStrategy(Calendar.MONTH) {
|
||||||
@Override
|
@Override
|
||||||
int modify(final int iValue) {
|
int modify(FastDateParser parser, final int iValue) {
|
||||||
return iValue-1;
|
return iValue-1;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -934,13 +945,13 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
private static final Strategy HOUR_OF_DAY_STRATEGY = new NumberStrategy(Calendar.HOUR_OF_DAY);
|
private static final Strategy HOUR_OF_DAY_STRATEGY = new NumberStrategy(Calendar.HOUR_OF_DAY);
|
||||||
private static final Strategy HOUR24_OF_DAY_STRATEGY = new NumberStrategy(Calendar.HOUR_OF_DAY) {
|
private static final Strategy HOUR24_OF_DAY_STRATEGY = new NumberStrategy(Calendar.HOUR_OF_DAY) {
|
||||||
@Override
|
@Override
|
||||||
int modify(final int iValue) {
|
int modify(FastDateParser parser, final int iValue) {
|
||||||
return iValue == 24 ? 0 : iValue;
|
return iValue == 24 ? 0 : iValue;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
private static final Strategy HOUR12_STRATEGY = new NumberStrategy(Calendar.HOUR) {
|
private static final Strategy HOUR12_STRATEGY = new NumberStrategy(Calendar.HOUR) {
|
||||||
@Override
|
@Override
|
||||||
int modify(final int iValue) {
|
int modify(FastDateParser parser, final int iValue) {
|
||||||
return iValue == 12 ? 0 : iValue;
|
return iValue == 12 ? 0 : iValue;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -948,7 +959,4 @@ public class FastDateParser implements DateParser, Serializable {
|
||||||
private static final Strategy MINUTE_STRATEGY = new NumberStrategy(Calendar.MINUTE);
|
private static final Strategy MINUTE_STRATEGY = new NumberStrategy(Calendar.MINUTE);
|
||||||
private static final Strategy SECOND_STRATEGY = new NumberStrategy(Calendar.SECOND);
|
private static final Strategy SECOND_STRATEGY = new NumberStrategy(Calendar.SECOND);
|
||||||
private static final Strategy MILLISECOND_STRATEGY = new NumberStrategy(Calendar.MILLISECOND);
|
private static final Strategy MILLISECOND_STRATEGY = new NumberStrategy(Calendar.MILLISECOND);
|
||||||
private static final Strategy ISO_8601_STRATEGY = new ISO8601TimeZoneStrategy("(Z|(?:[+-]\\d{2}(?::?\\d{2})?))");
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,8 +35,8 @@ import java.util.concurrent.TimeUnit;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
import java.util.concurrent.atomic.AtomicLong;
|
import java.util.concurrent.atomic.AtomicLong;
|
||||||
|
|
||||||
import org.apache.commons.lang3.test.SystemDefaultsSwitch;
|
|
||||||
import org.apache.commons.lang3.test.SystemDefaults;
|
import org.apache.commons.lang3.test.SystemDefaults;
|
||||||
|
import org.apache.commons.lang3.test.SystemDefaultsSwitch;
|
||||||
import org.junit.Rule;
|
import org.junit.Rule;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
|
@ -230,7 +230,7 @@ public class FastDateFormatTest {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testParseSync() throws InterruptedException {
|
public void testParseSync() throws InterruptedException {
|
||||||
final String pattern = "yyyy-MM-dd'T'HH:mm:ss.SSS Z";
|
final String pattern = "yyyy-MM-dd'T'HH:mm:ss.SSS";
|
||||||
final FastDateFormat formatter= FastDateFormat.getInstance(pattern);
|
final FastDateFormat formatter= FastDateFormat.getInstance(pattern);
|
||||||
|
|
||||||
final long sdfTime= measureTime(formatter, new SimpleDateFormat(pattern) {
|
final long sdfTime= measureTime(formatter, new SimpleDateFormat(pattern) {
|
||||||
|
|
|
@ -16,7 +16,10 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.commons.lang3.time;
|
package org.apache.commons.lang3.time;
|
||||||
|
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertNotEquals;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.junit.Assert.fail;
|
||||||
|
|
||||||
import java.text.ParseException;
|
import java.text.ParseException;
|
||||||
import java.text.ParsePosition;
|
import java.text.ParsePosition;
|
||||||
|
@ -193,8 +196,9 @@ public class FastDateParserSDFTest {
|
||||||
|
|
||||||
ParsePosition sdfP = new ParsePosition(0);
|
ParsePosition sdfP = new ParsePosition(0);
|
||||||
Date expectedTime = sdf.parse(formattedDate, sdfP);
|
Date expectedTime = sdf.parse(formattedDate, sdfP);
|
||||||
|
final int sdferrorIndex = sdfP.getErrorIndex();
|
||||||
if (valid) {
|
if (valid) {
|
||||||
assertEquals("Expected SDF error index -1 ", -1, sdfP.getErrorIndex());
|
assertEquals("Expected SDF error index -1 ", -1, sdferrorIndex);
|
||||||
final int endIndex = sdfP.getIndex();
|
final int endIndex = sdfP.getIndex();
|
||||||
final int length = formattedDate.length();
|
final int length = formattedDate.length();
|
||||||
if (endIndex != length) {
|
if (endIndex != length) {
|
||||||
|
@ -216,15 +220,11 @@ public class FastDateParserSDFTest {
|
||||||
final int endIndex = fdfP.getIndex();
|
final int endIndex = fdfP.getIndex();
|
||||||
final int length = formattedDate.length();
|
final int length = formattedDate.length();
|
||||||
assertEquals("Expected FDF to parse full string " + fdfP, length, endIndex);
|
assertEquals("Expected FDF to parse full string " + fdfP, length, endIndex);
|
||||||
assertEquals(locale.toString()+" "+formattedDate +"\n",expectedTime, actualTime);
|
assertEquals(locale.toString()+" "+formattedDate +"\n", expectedTime, actualTime);
|
||||||
} else {
|
} else {
|
||||||
final int endIndex = fdfP.getIndex();
|
assertNotEquals("Test data error: expected FDF parse to fail, but got " + actualTime, -1, fdferrorIndex);
|
||||||
if (endIndex != -0) {
|
assertTrue("FDF error index ("+ fdferrorIndex + ") should approxiamate SDF index (" + sdferrorIndex + ")",
|
||||||
fail("Expected FDF parse to fail, but got " + fdfP);
|
sdferrorIndex - fdferrorIndex <= 4);
|
||||||
}
|
|
||||||
if (fdferrorIndex != -1) {
|
|
||||||
assertEquals("FDF error index should match SDF index (if it is set)", sdfP.getErrorIndex(), fdferrorIndex);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,6 +31,7 @@ import java.util.Locale;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.TimeZone;
|
import java.util.TimeZone;
|
||||||
|
|
||||||
|
import org.apache.commons.lang3.LocaleUtils;
|
||||||
import org.apache.commons.lang3.SerializationUtils;
|
import org.apache.commons.lang3.SerializationUtils;
|
||||||
import org.junit.Assert;
|
import org.junit.Assert;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
@ -325,7 +326,8 @@ public class FastDateParserTest {
|
||||||
if (eraBC) {
|
if (eraBC) {
|
||||||
cal.set(Calendar.ERA, GregorianCalendar.BC);
|
cal.set(Calendar.ERA, GregorianCalendar.BC);
|
||||||
}
|
}
|
||||||
for(final Locale locale : Locale.getAvailableLocales()) {
|
|
||||||
|
for(final Locale locale : Locale.getAvailableLocales() ) {
|
||||||
// ja_JP_JP cannot handle dates before 1868 properly
|
// ja_JP_JP cannot handle dates before 1868 properly
|
||||||
if (eraBC && locale.equals(FastDateParser.JAPANESE_IMPERIAL)) {
|
if (eraBC && locale.equals(FastDateParser.JAPANESE_IMPERIAL)) {
|
||||||
continue;
|
continue;
|
||||||
|
@ -341,6 +343,28 @@ public class FastDateParserTest {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testJpLocales() {
|
||||||
|
|
||||||
|
final Calendar cal= Calendar.getInstance(GMT);
|
||||||
|
cal.clear();
|
||||||
|
cal.set(2003, Calendar.FEBRUARY, 10);
|
||||||
|
cal.set(Calendar.ERA, GregorianCalendar.BC);
|
||||||
|
|
||||||
|
final Locale locale = LocaleUtils.toLocale("zh"); {
|
||||||
|
// ja_JP_JP cannot handle dates before 1868 properly
|
||||||
|
|
||||||
|
final SimpleDateFormat sdf = new SimpleDateFormat(LONG_FORMAT, locale);
|
||||||
|
final DateParser fdf = getInstance(LONG_FORMAT, locale);
|
||||||
|
|
||||||
|
try {
|
||||||
|
checkParse(locale, cal, sdf, fdf);
|
||||||
|
} catch(final ParseException ex) {
|
||||||
|
Assert.fail("Locale "+locale+ " failed with "+LONG_FORMAT+"\n" + trimMessage(ex.toString()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private String trimMessage(final String msg) {
|
private String trimMessage(final String msg) {
|
||||||
if (msg.length() < 100) {
|
if (msg.length() < 100) {
|
||||||
return msg;
|
return msg;
|
||||||
|
@ -441,7 +465,7 @@ public class FastDateParserTest {
|
||||||
final DateParser fdp = getInstance(format, NEW_YORK, Locale.US);
|
final DateParser fdp = getInstance(format, NEW_YORK, Locale.US);
|
||||||
dfdp = fdp.parse(date);
|
dfdp = fdp.parse(date);
|
||||||
if (shouldFail) {
|
if (shouldFail) {
|
||||||
Assert.fail("Expected FDF failure, but got " + dfdp + " for ["+format+","+date+"] using "+((FastDateParser)fdp).getParsePattern());
|
Assert.fail("Expected FDF failure, but got " + dfdp + " for ["+format+","+date+"]");
|
||||||
}
|
}
|
||||||
} catch (final Exception e) {
|
} catch (final Exception e) {
|
||||||
f = e;
|
f = e;
|
||||||
|
|
|
@ -0,0 +1,113 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.apache.commons.lang3.time;
|
||||||
|
|
||||||
|
import java.text.ParseException;
|
||||||
|
import java.text.ParsePosition;
|
||||||
|
import java.util.Calendar;
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.Locale;
|
||||||
|
import java.util.TimeZone;
|
||||||
|
|
||||||
|
import org.junit.Assert;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
public class FastDateParser_MoreOrLessTest {
|
||||||
|
|
||||||
|
private static final TimeZone NEW_YORK = TimeZone.getTimeZone("America/New_York");
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInputHasPrecedingCharacters() throws ParseException {
|
||||||
|
FastDateParser parser = new FastDateParser("MM/dd", TimeZone.getDefault(), Locale.getDefault());
|
||||||
|
ParsePosition parsePosition = new ParsePosition(0);
|
||||||
|
Date date = parser.parse("A 3/23/61", parsePosition);
|
||||||
|
Assert.assertNull(date);
|
||||||
|
Assert.assertEquals(0, parsePosition.getIndex());
|
||||||
|
Assert.assertEquals(0, parsePosition.getErrorIndex());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInputHasWhitespace() throws ParseException {
|
||||||
|
FastDateParser parser = new FastDateParser("M/d/y", TimeZone.getDefault(), Locale.getDefault());
|
||||||
|
//SimpleDateFormat parser = new SimpleDateFormat("M/d/y");
|
||||||
|
ParsePosition parsePosition = new ParsePosition(0);
|
||||||
|
Date date = parser.parse(" 3/ 23/ 1961", parsePosition);
|
||||||
|
Assert.assertEquals(12, parsePosition.getIndex());
|
||||||
|
|
||||||
|
Calendar calendar = Calendar.getInstance();
|
||||||
|
calendar.setTime(date);
|
||||||
|
Assert.assertEquals(1961, calendar.get(Calendar.YEAR));
|
||||||
|
Assert.assertEquals(2, calendar.get(Calendar.MONTH));
|
||||||
|
Assert.assertEquals(23, calendar.get(Calendar.DATE));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInputHasMoreCharacters() throws ParseException {
|
||||||
|
FastDateParser parser = new FastDateParser("MM/dd", TimeZone.getDefault(), Locale.getDefault());
|
||||||
|
ParsePosition parsePosition = new ParsePosition(0);
|
||||||
|
Date date = parser.parse("3/23/61", parsePosition);
|
||||||
|
Assert.assertEquals(4, parsePosition.getIndex());
|
||||||
|
|
||||||
|
Calendar calendar = Calendar.getInstance();
|
||||||
|
calendar.setTime(date);
|
||||||
|
Assert.assertEquals(2, calendar.get(Calendar.MONTH));
|
||||||
|
Assert.assertEquals(23, calendar.get(Calendar.DATE));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInputHasWrongCharacters() {
|
||||||
|
FastDateParser parser = new FastDateParser("MM-dd-yyy", TimeZone.getDefault(), Locale.getDefault());
|
||||||
|
ParsePosition parsePosition = new ParsePosition(0);
|
||||||
|
Assert.assertNull(parser.parse("03/23/1961", parsePosition));
|
||||||
|
Assert.assertEquals(2, parsePosition.getErrorIndex());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInputHasLessCharacters() {
|
||||||
|
FastDateParser parser = new FastDateParser("MM/dd/yyy", TimeZone.getDefault(), Locale.getDefault());
|
||||||
|
ParsePosition parsePosition = new ParsePosition(0);
|
||||||
|
Assert.assertNull(parser.parse("03/23", parsePosition));
|
||||||
|
Assert.assertEquals(5, parsePosition.getErrorIndex());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInputHasWrongTimeZone() {
|
||||||
|
FastDateParser parser = new FastDateParser("mm:ss z", NEW_YORK, Locale.US);
|
||||||
|
|
||||||
|
String input = "11:23 Pacific Standard Time";
|
||||||
|
ParsePosition parsePosition = new ParsePosition(0);
|
||||||
|
Assert.assertNotNull(parser.parse(input, parsePosition));
|
||||||
|
Assert.assertEquals(input.length(), parsePosition.getIndex());
|
||||||
|
|
||||||
|
parsePosition.setIndex(0);
|
||||||
|
Assert.assertNull(parser.parse( "11:23 Pacific Standard ", parsePosition));
|
||||||
|
Assert.assertEquals(6, parsePosition.getErrorIndex());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInputHasWrongDay() throws ParseException {
|
||||||
|
FastDateParser parser = new FastDateParser("EEEE, MM/dd/yyy", NEW_YORK, Locale.US);
|
||||||
|
String input = "Thursday, 03/23/61";
|
||||||
|
ParsePosition parsePosition = new ParsePosition(0);
|
||||||
|
Assert.assertNotNull(parser.parse(input, parsePosition));
|
||||||
|
Assert.assertEquals(input.length(), parsePosition.getIndex());
|
||||||
|
|
||||||
|
parsePosition.setIndex(0);
|
||||||
|
Assert.assertNull(parser.parse( "Thorsday, 03/23/61", parsePosition));
|
||||||
|
Assert.assertEquals(0, parsePosition.getErrorIndex());
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue