/*******************************************************************************
* Copyright (c) quickfixengine.org All rights reserved.
*
* This file is part of the QuickFIX FIX Engine
*
* This file may be distributed under the terms of the quickfixengine.org
* license as defined by quickfixengine.org and appearing in the file
* LICENSE included in the packaging of this file.
*
* This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING
* THE WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A
* PARTICULAR PURPOSE.
*
* See http://www.quickfixengine.org/LICENSE for licensing information.
*
* Contact ask@quickfixengine.org if any conditions of this licensing
* are not clear to you.
******************************************************************************/
package quickfix.field.converter;
import java.text.DateFormat;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.HashMap;
import quickfix.FieldConvertError;
import quickfix.SystemTime;
/**
* Convert between a timestamp and a String. A timestamp includes both a date
* and a time.
*/
public class UtcTimestampConverter extends AbstractDateTimeConverter {
private static ThreadLocal<UtcTimestampConverter> utcTimestampConverter = new ThreadLocal<UtcTimestampConverter>();
private final DateFormat utcTimestampFormat = createDateFormat("yyyyMMdd-HH:mm:ss");
private final DateFormat utcTimestampFormatMillis = createDateFormat("yyyyMMdd-HH:mm:ss.SSS");
private static HashMap<String, Long> dateCache = new HashMap<String, Long>();
/**
* Convert a timestamp (represented as a Date) to a String.
* @param d the date to convert
* @param includeMilliseconds controls whether milliseconds are included in the result
* @return the formatted timestamp
*/
public static String convert(Date d, boolean includeMilliseconds) {
return getFormatter(includeMilliseconds).format(d);
}
private static DateFormat getFormatter(boolean includeMillis) {
UtcTimestampConverter converter = utcTimestampConverter.get();
if (converter == null) {
converter = new UtcTimestampConverter();
utcTimestampConverter.set(converter);
}
return includeMillis ? converter.utcTimestampFormatMillis : converter.utcTimestampFormat;
}
//
// Performance optimization: the calendar for the start of the day is cached.
// The time is converted to millis and then added to the millis specified by
// the base calendar.
//
/**
* Convert a timestamp string into a Date.
* @param value the timestamp String
* @return the parsed timestamp
* @exception FieldConvertError raised if timestamp is an incorrect format.
*/
public static Date convert(String value) throws FieldConvertError {
verifyFormat(value);
long timeOffset = (parseLong(value.substring(9, 11)) * 3600000L)
+ (parseLong(value.substring(12, 14)) * 60000L)
+ (parseLong(value.substring(15, 17)) * 1000L);
if (value.length() == 21) {
timeOffset += parseLong(value.substring(18, 21));
}
return new Date(getMillisForDay(value) + timeOffset);
}
private static Long getMillisForDay(String value) {
String dateString = value.substring(0, 8);
Long millis = dateCache.get(dateString);
if (millis == null) {
Calendar c = new GregorianCalendar(1970, 0, 1, 0, 0, 0);
c.setTimeZone(SystemTime.UTC_TIMEZONE);
int year = Integer.parseInt(value.substring(0, 4));
int month = Integer.parseInt(value.substring(4, 6));
int day = Integer.parseInt(value.substring(6, 8));
c.set(year, month - 1, day);
millis = c.getTimeInMillis();
dateCache.put(dateString, c.getTimeInMillis());
}
return millis;
}
private static void verifyFormat(String value) throws FieldConvertError {
String type = "timestamp";
if (value.length() != 17 && value.length() != 21) {
throwFieldConvertError(value, type);
}
assertDigitSequence(value, 0, 8, type);
assertSeparator(value, 8, '-', type);
assertDigitSequence(value, 9, 11, type);
assertSeparator(value, 11, ':', type);
assertDigitSequence(value, 12, 14, type);
assertSeparator(value, 14, ':', type);
assertDigitSequence(value, 15, 17, type);
if (value.length() == 21) {
assertSeparator(value, 17, '.', type);
assertDigitSequence(value, 18, 21, type);
} else if (value.length() != 17) {
throwFieldConvertError(value, type);
}
}
}