All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.hsqldb.HsqlDateTime Maven / Gradle / Ivy

There is a newer version: 4.7.5
Show newest version
/* Copyright (c) 2001-2011, The HSQL Development Group
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * Redistributions of source code must retain the above copyright notice, this
 * list of conditions and the following disclaimer.
 *
 * Redistributions in binary form must reproduce the above copyright notice,
 * this list of conditions and the following disclaimer in the documentation
 * and/or other materials provided with the distribution.
 *
 * Neither the name of the HSQL Development Group nor the names of its
 * contributors may be used to endorse or promote products derived from this
 * software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL HSQL DEVELOPMENT GROUP, HSQLDB.ORG,
 * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */


package org.hsqldb;

import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.Locale;
import java.util.TimeZone;

import org.hsqldb.error.Error;
import org.hsqldb.error.ErrorCode;
import org.hsqldb.lib.ArrayUtil;
import org.hsqldb.lib.StringUtil;
import org.hsqldb.types.DTIType;
import org.hsqldb.types.TimestampData;
import org.hsqldb.types.Types;

/**
 * collection of static methods to convert Date and Timestamp strings
 * into corresponding Java objects and perform other Calendar related
 * operation.

* * Was reviewed for 1.7.2 resulting in centralising all DATETIME related * operstions.

* * From version 2.0.0, HSQLDB supports TIME ZONE with datetime types. The * values are stored internally as UTC seconds from 1970, regardless of the * time zone of the JVM, and converted as and when required, to the local * timezone. * * @author Fred Toussi (fredt@users dot sourceforge.net) * @version 2.3.0 * @since 1.7.0 */ public class HsqlDateTime { /** * A reusable static value for today's date. Should only be accessed * by getToday() */ private static Locale defaultLocale = Locale.UK; private static long currentDateMillis; public static final Calendar tempCalDefault = new GregorianCalendar(); public static final Calendar tempCalGMT = new GregorianCalendar(TimeZone.getTimeZone("GMT"), defaultLocale); private static final Date tempDate = new Date(0); private static final String sdfdPattern = "yyyy-MM-dd"; static SimpleDateFormat sdfd = new SimpleDateFormat(sdfdPattern); private static final String sdftPattern = "HH:mm:ss"; static SimpleDateFormat sdft = new SimpleDateFormat(sdftPattern); private static final String sdftsPattern = "yyyy-MM-dd HH:mm:ss"; static SimpleDateFormat sdfts = new SimpleDateFormat(sdftsPattern); private static final String sdftsSysPattern = "yyyy-MM-dd HH:mm:ss.SSS"; static SimpleDateFormat sdftsSys = new SimpleDateFormat(sdftsSysPattern); static { tempCalGMT.setLenient(false); sdfd.setCalendar(new GregorianCalendar(TimeZone.getTimeZone("GMT"), defaultLocale)); sdfd.setLenient(false); sdft.setCalendar(new GregorianCalendar(TimeZone.getTimeZone("GMT"), defaultLocale)); sdft.setLenient(false); sdfts.setCalendar(new GregorianCalendar(TimeZone.getTimeZone("GMT"), defaultLocale)); sdfts.setLenient(false); } static { currentDateMillis = getNormalisedDate(System.currentTimeMillis()); } public static long getDateSeconds(String s) { try { synchronized (sdfd) { java.util.Date d = sdfd.parse(s); return d.getTime() / 1000; } } catch (Exception e) { throw Error.error(ErrorCode.X_22007); } } public static String getDateString(long seconds) { synchronized (sdfd) { sysDate.setTime(seconds * 1000); return sdfd.format(sysDate); } } public static long getTimestampSeconds(String s) { try { synchronized (sdfts) { java.util.Date d = sdfts.parse(s); return d.getTime() / 1000; } } catch (Exception e) { throw Error.error(ErrorCode.X_22007); } } public static void getTimestampString(StringBuffer sb, long seconds, int nanos, int scale) { synchronized (sdfts) { tempDate.setTime(seconds * 1000); sb.append(sdfts.format(tempDate)); if (scale > 0) { sb.append('.'); sb.append(StringUtil.toZeroPaddedString(nanos, 9, scale)); } } } public static String getTimestampString(long millis) { synchronized (sdfts) { sysDate.setTime(millis); return sdfts.format(sysDate); } } public static synchronized long getCurrentDateMillis(long millis) { if (millis - currentDateMillis >= 24 * 3600 * 1000) { currentDateMillis = getNormalisedDate(millis); } return currentDateMillis; } private static java.util.Date sysDate = new java.util.Date(); public static String getSystemTimeString() { synchronized (sdftsSys) { sysDate.setTime(System.currentTimeMillis()); return sdftsSys.format(sysDate); } } private static void resetToDate(Calendar cal) { cal.set(Calendar.HOUR_OF_DAY, 0); cal.set(Calendar.MINUTE, 0); cal.set(Calendar.SECOND, 0); cal.set(Calendar.MILLISECOND, 0); } private static void resetToTime(Calendar cal) { cal.set(Calendar.YEAR, 1970); cal.set(Calendar.MONTH, 0); cal.set(Calendar.DATE, 1); cal.set(Calendar.MILLISECOND, 0); } public static long convertMillisToCalendar(Calendar calendar, long millis) { synchronized (tempCalGMT) { synchronized (calendar) { calendar.clear(); tempCalGMT.setTimeInMillis(millis); calendar.set(tempCalGMT.get(Calendar.YEAR), tempCalGMT.get(Calendar.MONTH), tempCalGMT.get(Calendar.DAY_OF_MONTH), tempCalGMT.get(Calendar.HOUR_OF_DAY), tempCalGMT.get(Calendar.MINUTE), tempCalGMT.get(Calendar.SECOND)); return calendar.getTimeInMillis(); } } } public static long convertMillisFromCalendar(Calendar calendar, long millis) { synchronized (tempCalGMT) { synchronized (calendar) { tempCalGMT.clear(); calendar.setTimeInMillis(millis); tempCalGMT.set(calendar.get(Calendar.YEAR), calendar.get(Calendar.MONTH), calendar.get(Calendar.DAY_OF_MONTH), calendar.get(Calendar.HOUR_OF_DAY), calendar.get(Calendar.MINUTE), calendar.get(Calendar.SECOND)); return tempCalGMT.getTimeInMillis(); } } } /** * Sets the time in the given Calendar using the given milliseconds value; wrapper method to * allow use of more efficient JDK1.4 method on JDK1.4 (was protected in earlier versions). * * @param cal the Calendar * @param millis the time value in milliseconds */ public static void setTimeInMillis(Calendar cal, long millis) { //#ifdef JAVA4 // Use method directly cal.setTimeInMillis(millis); //#else /* // Have to go indirect synchronized (tempDate) { tempDate.setTime(millis); cal.setTime(tempDate); } */ //#endif JAVA4 } /** * Gets the time from the given Calendar as a milliseconds value; wrapper method to * allow use of more efficient JDK1.4 method on JDK1.4 (was protected in earlier versions). * * @param cal the Calendar * @return the time value in milliseconds */ public static long getTimeInMillis(Calendar cal) { //#ifdef JAVA4 // Use method directly return cal.getTimeInMillis(); //#else /* // Have to go indirect return cal.getTime().getTime(); */ //#endif JAVA4 } public static long convertToNormalisedTime(long t) { return convertToNormalisedTime(t, tempCalGMT); } public static long convertToNormalisedTime(long t, Calendar cal) { synchronized (cal) { setTimeInMillis(cal, t); resetToDate(cal); long t1 = getTimeInMillis(cal); return t - t1; } } public static long convertToNormalisedDate(long t, Calendar cal) { synchronized (cal) { setTimeInMillis(cal, t); resetToDate(cal); return getTimeInMillis(cal); } } public static long getNormalisedTime(long t) { Calendar cal = tempCalGMT; synchronized (cal) { setTimeInMillis(cal, t); resetToTime(cal); return getTimeInMillis(cal); } } public static long getNormalisedTime(Calendar cal, long t) { synchronized (cal) { setTimeInMillis(cal, t); resetToTime(cal); return getTimeInMillis(cal); } } public static long getNormalisedDate(long d) { synchronized (tempCalGMT) { setTimeInMillis(tempCalGMT, d); resetToDate(tempCalGMT); return getTimeInMillis(tempCalGMT); } } public static long getNormalisedDate(Calendar cal, long d) { synchronized (cal) { setTimeInMillis(cal, d); resetToDate(cal); return getTimeInMillis(cal); } } public static int getZoneSeconds(Calendar cal) { return (cal.get(Calendar.ZONE_OFFSET) + cal.get(Calendar.DST_OFFSET)) / 1000; } public static int getZoneMillis(Calendar cal, long millis) { //#ifdef JAVA4 // get zone for the specific date return cal.getTimeZone().getOffset(millis); //#else /* // get zone for the specific date setTimeInMillis(cal, millis); return (cal.get(Calendar.ZONE_OFFSET) + cal.get(Calendar.DST_OFFSET) ); */ //#endif JAVA4 } /** * Returns the indicated part of the given millisecond date object. * @param m the millisecond time value from which to extract the indicated part * @param part an integer code corresponding to the desired date part * @return the indicated part of the given java.util.Date object */ public static int getDateTimePart(long m, int part) { synchronized (tempCalGMT) { tempCalGMT.setTimeInMillis(m); return tempCalGMT.get(part); } } /** * truncates millisecond date object */ public static long getTruncatedPart(long m, int part) { synchronized (tempCalGMT) { tempCalGMT.setTimeInMillis(m); switch (part) { case DTIType.WEEK_OF_YEAR : { int dayWeek = tempCalGMT.get(Calendar.DAY_OF_WEEK); tempCalGMT.add(Calendar.DAY_OF_YEAR, 1 - dayWeek); resetToDate(tempCalGMT); break; } default : { zeroFromPart(tempCalGMT, part); break; } } return tempCalGMT.getTimeInMillis(); } } /** * rounded millisecond date object */ public static long getRoundedPart(long m, int part) { synchronized (tempCalGMT) { tempCalGMT.setTimeInMillis(m); switch (part) { case Types.SQL_INTERVAL_YEAR : if (tempCalGMT.get(Calendar.MONTH) > 6) { tempCalGMT.add(Calendar.YEAR, 1); } break; case Types.SQL_INTERVAL_MONTH : if (tempCalGMT.get(Calendar.DAY_OF_MONTH) > 15) { tempCalGMT.add(Calendar.MONTH, 1); } break; case Types.SQL_INTERVAL_DAY : if (tempCalGMT.get(Calendar.HOUR_OF_DAY) > 11) { tempCalGMT.add(Calendar.DAY_OF_MONTH, 1); } break; case Types.SQL_INTERVAL_HOUR : if (tempCalGMT.get(Calendar.MINUTE) > 29) { tempCalGMT.add(Calendar.HOUR_OF_DAY, 1); } break; case Types.SQL_INTERVAL_MINUTE : if (tempCalGMT.get(Calendar.SECOND) > 29) { tempCalGMT.add(Calendar.MINUTE, 1); } break; case Types.SQL_INTERVAL_SECOND : if (tempCalGMT.get(Calendar.MILLISECOND) > 499) { tempCalGMT.add(Calendar.SECOND, 1); } break; case DTIType.WEEK_OF_YEAR : { int dayYear = tempCalGMT.get(Calendar.DAY_OF_YEAR); int year = tempCalGMT.get(Calendar.YEAR); int week = tempCalGMT.get(Calendar.WEEK_OF_YEAR); int day = tempCalGMT.get(Calendar.DAY_OF_WEEK); tempCalGMT.clear(); tempCalGMT.set(Calendar.YEAR, year); if (day > 3) { week++; } if (week == 1 && (dayYear > 356 || dayYear < 7)) { tempCalGMT.set(Calendar.DAY_OF_YEAR, dayYear); while (true) { if (tempCalGMT.get(Calendar.DAY_OF_WEEK) == 1) { return tempCalGMT.getTimeInMillis(); } tempCalGMT.add(Calendar.DAY_OF_YEAR, -1); } } tempCalGMT.set(Calendar.WEEK_OF_YEAR, week); return tempCalGMT.getTimeInMillis(); } } zeroFromPart(tempCalGMT, part); return tempCalGMT.getTimeInMillis(); } } static void zeroFromPart(Calendar cal, int part) { switch (part) { case Types.SQL_INTERVAL_YEAR : cal.set(Calendar.MONTH, 0); case Types.SQL_INTERVAL_MONTH : cal.set(Calendar.DAY_OF_MONTH, 1); case Types.SQL_INTERVAL_DAY : cal.set(Calendar.HOUR_OF_DAY, 0); case Types.SQL_INTERVAL_HOUR : cal.set(Calendar.MINUTE, 0); case Types.SQL_INTERVAL_MINUTE : cal.set(Calendar.SECOND, 0); case Types.SQL_INTERVAL_SECOND : cal.set(Calendar.MILLISECOND, 0); } } //J- private static final char[][] dateTokens = { { 'R', 'R', 'R', 'R' }, { 'I', 'Y', 'Y', 'Y' }, { 'Y', 'Y', 'Y', 'Y' }, { 'I', 'Y' }, { 'Y', 'Y' }, { 'B', 'C' }, { 'B', '.', 'C', '.' }, { 'A', 'D' }, { 'A', '.', 'D', '.' }, { 'M', 'O', 'N' }, { 'M', 'O', 'N', 'T', 'H' }, { 'M', 'M' }, { 'D', 'A', 'Y' }, { 'D', 'Y' }, { 'W', 'W' }, { 'I', 'W' }, { 'D', 'D' }, { 'D', 'D', 'D' }, { 'W' }, { 'H', 'H', '2', '4' }, { 'H', 'H', '1', '2' }, { 'H', 'H' }, { 'M', 'I' }, { 'S', 'S' }, { 'A', 'M' }, { 'P', 'M' }, { 'A', '.', 'M', '.' }, { 'P', '.', 'M', '.' }, { 'F', 'F' } }; private static final String[] javaDateTokens = { "yyyy", "'*IYYY'", "yyyy", "'*IY'", "yy", "G", "G", "G", "G", "MMM", "MMMMM", "MM", "EEEE", "EE", "'*WW'", "w", "dd", "D", "'*W'", "HH", "KK", "KK", "mm", "ss", "aaa", "aaa", "aaa", "aaa", "S" }; private static final int[] sqlIntervalCodes = { -1, -1, Types.SQL_INTERVAL_YEAR, -1, Types.SQL_INTERVAL_YEAR, -1, -1, -1, -1, Types.SQL_INTERVAL_MONTH, Types.SQL_INTERVAL_MONTH, Types.SQL_INTERVAL_MONTH, -1, -1, DTIType.WEEK_OF_YEAR, -1, Types.SQL_INTERVAL_DAY, Types.SQL_INTERVAL_DAY, -1, Types.SQL_INTERVAL_HOUR, -1, Types.SQL_INTERVAL_HOUR, Types.SQL_INTERVAL_MINUTE, Types.SQL_INTERVAL_SECOND, -1,-1,-1,-1, -1 }; //J+ /** Indicates end-of-input */ private static final char e = 0xffff; public static TimestampData toDate(String string, String pattern, SimpleDateFormat format) { Date date; String javaPattern = HsqlDateTime.toJavaDatePattern(pattern); int matchIndex = javaPattern.indexOf("*IY"); if (matchIndex >= 0) { throw Error.error(ErrorCode.X_22511); } matchIndex = javaPattern.indexOf("*WW"); if (matchIndex >= 0) { throw Error.error(ErrorCode.X_22511); } matchIndex = javaPattern.indexOf("*W"); if (matchIndex >= 0) { throw Error.error(ErrorCode.X_22511); } try { format.applyPattern(javaPattern); date = format.parse(string); } catch (Exception e) { throw Error.error(ErrorCode.X_22007, e.toString()); } int nanos = ((int) (date.getTime() % 1000)) * 1000000; return new TimestampData(date.getTime() / 1000, nanos, 0); } public static String toFormattedDate(Date date, String pattern, SimpleDateFormat format) { String javaPattern = HsqlDateTime.toJavaDatePattern(pattern); try { format.applyPattern(javaPattern); } catch (Exception e) { throw Error.error(ErrorCode.X_22511); } String result = format.format(date); int matchIndex = result.indexOf("*IY"); if (matchIndex >= 0) { Calendar cal = format.getCalendar(); int matchLength = 3; int temp = result.indexOf("*IYYY"); if (temp >= 0) { matchLength = 5; matchIndex = temp; } int year = cal.get(Calendar.YEAR); int weekOfYear = cal.get(Calendar.WEEK_OF_YEAR); if (weekOfYear == 1 && cal.get(Calendar.DAY_OF_YEAR) > 360) { year++; } String yearString = String.valueOf(year); if (matchLength == 3) { yearString = yearString.substring(yearString.length() - 2); } StringBuilder sb = new StringBuilder(result); sb.replace(matchIndex, matchIndex + matchLength, yearString); result = sb.toString(); } matchIndex = result.indexOf("*WW"); if (matchIndex >= 0) { Calendar cal = format.getCalendar(); int matchLength = 3; int dayOfYear = cal.get(Calendar.DAY_OF_YEAR); int weekOfYear = ((dayOfYear - 1) / 7) + 1; StringBuilder sb = new StringBuilder(result); sb.replace(matchIndex, matchIndex + matchLength, String.valueOf(weekOfYear)); result = sb.toString(); } matchIndex = result.indexOf("*W"); if (matchIndex >= 0) { Calendar cal = format.getCalendar(); int matchLength = 2; int dayOfMonth = cal.get(Calendar.DAY_OF_MONTH); int weekOfMonth = ((dayOfMonth - 1) / 7) + 1; StringBuilder sb = new StringBuilder(result); sb.replace(matchIndex, matchIndex + matchLength, String.valueOf(weekOfMonth)); result = sb.toString(); } return result; } /** * Converts the given format into a pattern accepted by java.text.SimpleDataFormat * * @param format */ public static String toJavaDatePattern(String format) { int len = format.length(); char ch; StringBuffer sb = new StringBuffer(len); Tokenizer tokenizer = new Tokenizer(); for (int i = 0; i <= len; i++) { ch = (i == len) ? e : format.charAt(i); if (tokenizer.isInQuotes()) { if (tokenizer.isQuoteChar(ch)) { ch = '\''; } else if (ch == '\'') { // double the single quote sb.append(ch); } sb.append(ch); continue; } if (!tokenizer.next(ch, i)) { if (tokenizer.consumed) { int index = tokenizer.getLastMatch(); sb.append(javaDateTokens[index]); i = tokenizer.matchOffset; } else { if (tokenizer.isQuoteChar(ch)) { ch = '\''; sb.append(ch); } else if (tokenizer.isLiteral(ch)) { sb.append(ch); } else if (ch == e) { // } else { throw Error.error(ErrorCode.X_22007, format.substring(i)); } } tokenizer.reset(); } } if (tokenizer.isInQuotes()) { throw Error.error(ErrorCode.X_22007); } String javaPattern = sb.toString(); return javaPattern; } public static int toStandardIntervalPart(String format) { int len = format.length(); char ch; Tokenizer tokenizer = new Tokenizer(); for (int i = 0; i <= len; i++) { ch = (i == len) ? e : format.charAt(i); if (!tokenizer.next(ch, i)) { int index = tokenizer.getLastMatch(); if (index >= 0) { return sqlIntervalCodes[index]; } return -1; } } return -1; } /** * This class can match 64 tokens at maximum. */ static class Tokenizer { private int lastMatched; private int matchOffset; private int offset; private long state; private boolean consumed; private boolean isInQuotes; private boolean matched; // private final char quoteChar; private final char[] literalChars; private static char[] defaultLiterals = new char[] { ' ', ',', '-', '.', '/', ':', ';' }; char[][] tokens; public Tokenizer() { this.quoteChar = '\"'; this.literalChars = defaultLiterals; tokens = dateTokens; reset(); } /** * Resets for next reuse. * */ public void reset() { lastMatched = -1; offset = -1; state = 0; consumed = false; matched = false; } /** * Returns the length of a token to match. */ public int length() { return offset; } /** * Returns an index of the last matched token. */ public int getLastMatch() { return lastMatched; } /** * Indicates whether the last character has been consumed by the matcher. */ public boolean isConsumed() { return consumed; } /** * Indicates whether the last character has been consumed by the matcher. */ public boolean wasMatched() { return matched; } /** * Indicates if tokenizing a quoted string */ public boolean isInQuotes() { return isInQuotes; } /** * returns true if character is the quote char and sets state */ public boolean isQuoteChar(char ch) { if (quoteChar == ch) { isInQuotes = !isInQuotes; return true; } return false; } /** * Returns true if ch is in the list of literals */ public boolean isLiteral(char ch) { return ArrayUtil.isInSortedArray(ch, literalChars); } /** * Checks whether the specified bit is not set. * * @param bit */ private boolean isZeroBit(int bit) { return (state & (1L << bit)) == 0; } /** * Sets the specified bit. * @param bit */ private void setBit(int bit) { state |= (1L << bit); } /** * Matches the specified character against tokens. * * @param ch * @param tokens */ public boolean next(char ch, int position) { int index = ++offset; int len = offset + 1; int left = 0; matched = false; for (int i = tokens.length; --i >= 0; ) { if (isZeroBit(i)) { if (tokens[i][index] == ch) { if (tokens[i].length == len) { setBit(i); lastMatched = i; consumed = true; matched = true; matchOffset = position; } else { ++left; } } else { setBit(i); } } } return left > 0; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy