LANG-1153

Implement ParsePosition api for FastDateParser
This commit is contained in:
Chas Honton 2015-07-07 20:20:19 -07:00
commit 40134ecdb3
7 changed files with 463 additions and 314 deletions

View File

@ -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>

View File

@ -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);
} }

View File

@ -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})?))");
} }

View File

@ -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) {

View File

@ -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);
}
} }
} }
} }

View File

@ -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;

View File

@ -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());
}
}