Added support for RFC 3339 dates.

This commit is contained in:
moandji.ezana 2014-12-15 15:44:03 +02:00
parent 6267ac6c9d
commit 12e73fb314
3 changed files with 51 additions and 18 deletions

View file

@ -183,6 +183,10 @@ Long tableD = toml.getLong("table.d"); // returns null, not 5
Long arrayD = toml.getLong("array[0].d"); // returns 3 Long arrayD = toml.getLong("array[0].d"); // returns 3
```` ````
### Limitations
Date precision is limited to milliseconds.
## License ## License
toml4j is copyright of Moandji Ezana and is licensed under the [MIT License](LICENSE) toml4j is copyright of Moandji Ezana and is licensed under the [MIT License](LICENSE)

View file

@ -2,6 +2,7 @@ package com.moandjiezana.toml;
import static com.moandjiezana.toml.ValueConverterUtils.INVALID; import static com.moandjiezana.toml.ValueConverterUtils.INVALID;
import java.text.ParseException;
import java.text.SimpleDateFormat; import java.text.SimpleDateFormat;
import java.util.regex.Matcher; import java.util.regex.Matcher;
import java.util.regex.Pattern; import java.util.regex.Pattern;
@ -9,23 +10,36 @@ import java.util.regex.Pattern;
class DateConverter implements ValueConverter { class DateConverter implements ValueConverter {
static final DateConverter DATE_PARSER = new DateConverter(); static final DateConverter DATE_PARSER = new DateConverter();
private static final Pattern DATE_REGEX = Pattern.compile("(\\d{4}-[0-1][0-9]-[0-3][0-9]T[0-2][0-9]:[0-5][0-9]:[0-5][0-9]Z)(.*)"); private static final Pattern DATE_REGEX = Pattern.compile("(\\d{4}-[0-1][0-9]-[0-3][0-9]T[0-2][0-9]:[0-5][0-9]:[0-5][0-9])(\\.\\d*)?(Z|(?:[+\\-]\\d{2}:\\d{2}))(.*)");
@Override @Override
public boolean canConvert(String s) { public boolean canConvert(String s) {
Matcher matcher = DATE_REGEX.matcher(s); Matcher matcher = DATE_REGEX.matcher(s);
return matcher.matches() && ValueConverterUtils.isComment(matcher.group(2)); return matcher.matches() && ValueConverterUtils.isComment(matcher.group(4));
} }
@Override @Override
public Object convert(String s) { public Object convert(String s) {
Matcher matcher = DATE_REGEX.matcher(s); Matcher matcher = DATE_REGEX.matcher(s);
matcher.matches(); matcher.matches();
s = matcher.group(1).replace("Z", "+00:00"); s = matcher.group(1);
String zone = matcher.group(3);
String fractionalSeconds = matcher.group(2);
String format = "yyyy-MM-dd'T'HH:mm:ss";
if (fractionalSeconds != null && !fractionalSeconds.isEmpty()) {
format += ".SSS";
s += fractionalSeconds;
}
format += "Z";
if ("Z".equals(zone)) {
s += "+0000";
// s = s.substring(0, 22) + s.substring(23);
} else if (zone.contains(":")) {
s += zone.replace(":", "");
}
try { try {
s = s.substring(0, 22) + s.substring(23); SimpleDateFormat dateFormat = new SimpleDateFormat(format);
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ");
dateFormat.setLenient(false); dateFormat.setLenient(false);
return dateFormat.parse(s); return dateFormat.parse(s);
} catch (Exception e) { } catch (Exception e) {
@ -34,4 +48,13 @@ class DateConverter implements ValueConverter {
} }
private DateConverter() {} private DateConverter() {}
public static void main(String[] args) throws ParseException {
Pattern DATE_REGEX = Pattern.compile("(\\d{4}-[0-1][0-9]-[0-3][0-9]T[0-2][0-9]:[0-5][0-9]:[0-5][0-9](?:\\.\\d*)?)(Z|(?:[+\\-]\\d{2}:\\d{2}))(.*)");
Pattern DATE_REGEX2 = Pattern.compile("(\\d{4}-[0-1][0-9]-[0-3][0-9]T[0-2][0-9]:[0-5][0-9]:[0-5][0-9])(Z|(?:[+\\-]\\d{2}:\\d{2}))");
System.out.println(DATE_REGEX.matcher("2011-11-11T12:32:00.999-07:00").matches());
System.out.println(new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSSZ").parse("1979-05-27T00:32:00+0000"));
}
} }

View file

@ -9,30 +9,25 @@ import org.junit.Test;
public class DateTest { public class DateTest {
private static final TimeZone UTC = TimeZone.getTimeZone("UTC");
@Test @Test
public void should_get_date() throws Exception { public void should_get_date() throws Exception {
Toml toml = new Toml().parse("a_date = 2011-11-10T13:12:00Z"); Toml toml = new Toml().parse("a_date = 2011-11-10T13:12:00Z");
Calendar calendar = Calendar.getInstance(); Calendar calendar = Calendar.getInstance(UTC);
calendar.set(2011, Calendar.NOVEMBER, 10, 13, 12, 00); calendar.set(2011, Calendar.NOVEMBER, 10, 13, 12, 00);
calendar.set(Calendar.MILLISECOND, 0); calendar.set(Calendar.MILLISECOND, 0);
calendar.setTimeZone(TimeZone.getTimeZone("UTC"));
assertEquals(calendar.getTime(), toml.getDate("a_date")); assertEquals(calendar.getTime(), toml.getDate("a_date"));
} }
public static void main(String[] args) {
for (String tz : TimeZone.getAvailableIDs(-7 * 60 * 60 * 1000)) {
System.out.println(tz);
}
}
@Test @Test
public void should_get_date_with_timezone_offset() throws Exception { public void should_get_date_with_offset() throws Exception {
Toml toml = new Toml().parse("a_date = 1979-05-27T00:32:00-07:00"); Toml toml = new Toml().parse("a_date = 1979-05-27T00:32:00-07:00");
Calendar calendar = Calendar.getInstance(TimeZone.getTimeZone("Etc/GMT-7")); Calendar calendar = Calendar.getInstance(UTC);
calendar.set(1979, Calendar.MAY, 27, 0, 32, 00); calendar.set(1979, Calendar.MAY, 27, 7, 32, 00);
calendar.set(Calendar.MILLISECOND, 0); calendar.set(Calendar.MILLISECOND, 0);
assertEquals(calendar.getTime(), toml.getDate("a_date")); assertEquals(calendar.getTime(), toml.getDate("a_date"));
@ -40,10 +35,21 @@ public class DateTest {
@Test @Test
public void should_get_date_with_fractional_seconds() throws Exception { public void should_get_date_with_fractional_seconds() throws Exception {
Toml toml = new Toml().parse("a_date = 1979-05-27T00:32:00.999Z");
Calendar calendar = Calendar.getInstance(UTC);
calendar.set(1979, Calendar.MAY, 27, 0, 32, 00);
calendar.set(Calendar.MILLISECOND, 999);
assertEquals(calendar.getTime(), toml.getDate("a_date"));
}
@Test
public void should_get_date_with_fractional_seconds_and_offset() throws Exception {
Toml toml = new Toml().parse("a_date = 1979-05-27T00:32:00.999-07:00"); Toml toml = new Toml().parse("a_date = 1979-05-27T00:32:00.999-07:00");
Calendar calendar = Calendar.getInstance(TimeZone.getTimeZone("Etc/GMT-7")); Calendar calendar = Calendar.getInstance(UTC);
calendar.set(1979, Calendar.MAY, 27, 0, 32, 00); calendar.set(1979, Calendar.MAY, 27, 7, 32, 00);
calendar.set(Calendar.MILLISECOND, 999); calendar.set(Calendar.MILLISECOND, 999);
assertEquals(calendar.getTime(), toml.getDate("a_date")); assertEquals(calendar.getTime(), toml.getDate("a_date"));