blob: dac32397a7f721dc66997eddfaf6a16c110027fd [file] [log] [blame]
/*
* Copyright (c) 2005, 2019, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
* @test
* @bug 6231602
* @summary Make sure that ZONE_OFFSET and/or DST_OFFSET setting is
* taken into account for time calculations.
*/
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.Locale;
import java.util.TimeZone;
import static java.util.GregorianCalendar.*;
public class ZoneOffsets {
// This TimeZone always returns the dstOffset value.
@SuppressWarnings("serial")
private static class TestTimeZone extends TimeZone {
private int gmtOffset;
private int dstOffset;
TestTimeZone(int gmtOffset, String id, int dstOffset) {
this.gmtOffset = gmtOffset;
setID(id);
this.dstOffset = dstOffset;
}
public int getOffset(int era, int year, int month, int day,
int dayOfWeek, int milliseconds) {
return gmtOffset + dstOffset;
}
public int getOffset(long date) {
return gmtOffset + dstOffset;
}
public void setRawOffset(int offsetMillis) {
gmtOffset = offsetMillis;
}
public int getRawOffset() {
return gmtOffset;
}
public int getDSTSavings() {
return dstOffset;
}
public boolean useDaylightTime() {
return dstOffset != 0;
}
public boolean inDaylightTime(Date date) {
return dstOffset != 0;
}
public String toString() {
return "TestTimeZone[" + getID() + ", " + gmtOffset + ", " + dstOffset + "]";
}
}
private static Locale[] locales = {
Locale.getDefault(),
new Locale("th", "TH"),
new Locale("ja", "JP", "JP")};
private static final int HOUR = 60 * 60 * 1000;
private static int[][] offsets = {
{0, 0},
{0, HOUR},
{0, 2 * HOUR},
{-8 * HOUR, 0},
{-8 * HOUR, HOUR},
{-8 * HOUR, 2 * HOUR},
{9 * HOUR, 0},
{9 * HOUR, HOUR},
{9 * HOUR, 2 * HOUR}};
public static void main(String[] args) {
for (int l = 0; l < locales.length; l++) {
Locale loc = locales[l];
for (int i = 0; i < offsets.length; i++) {
test(loc, offsets[i][0], offsets[i][1]);
}
}
// The test case in the bug report.
GregorianCalendar cal = new GregorianCalendar();
cal.setLenient(false);
cal.setGregorianChange(new Date(Long.MIN_VALUE));
cal.clear();
cal.set(ZONE_OFFSET, 0);
cal.set(DST_OFFSET, 0);
cal.set(ERA, AD);
cal.set(2004, FEBRUARY, 3, 0, 0, 0);
cal.set(MILLISECOND, 0);
// The following line should not throw an IllegalArgumentException.
cal.getTime();
}
private static void test(Locale loc, int gmtOffset, int dstOffset) {
TimeZone tz1 = new TestTimeZone(gmtOffset,
"GMT" + (gmtOffset / HOUR) + "." + (dstOffset / HOUR),
dstOffset);
int someDifferentOffset = gmtOffset + 2 * HOUR;
TimeZone tz2 = new TestTimeZone(someDifferentOffset,
"GMT" + (someDifferentOffset / HOUR) + "." + (dstOffset / HOUR),
dstOffset);
int someDifferentDSTOffset = dstOffset == 2 * HOUR ? HOUR : dstOffset + HOUR;
TimeZone tz3 = new TestTimeZone(gmtOffset,
"GMT" + (gmtOffset / HOUR) + "." + (someDifferentDSTOffset / HOUR),
someDifferentDSTOffset);
// cal1 is the base line.
Calendar cal1 = Calendar.getInstance(tz1, loc);
cal1.clear();
cal1.set(2005, MARCH, 11);
long t1 = cal1.getTime().getTime();
int gmt = cal1.get(ZONE_OFFSET);
int dst = cal1.get(DST_OFFSET);
// Test 8 cases with cal2.
Calendar cal2 = Calendar.getInstance(tz2, loc);
cal2.clear();
cal2.set(2005, MARCH, 11);
// test1: set only ZONE_OFFSET
cal2.set(ZONE_OFFSET, gmtOffset);
if (t1 != cal2.getTime().getTime() || dst != cal2.get(DST_OFFSET)) {
error("Test1", loc, cal2, gmtOffset, dstOffset, t1);
}
cal2.setTimeZone(tz3);
cal2.clear();
cal2.set(2005, MARCH, 11);
// test2: set only DST_OFFSET
cal2.set(DST_OFFSET, dstOffset);
if (t1 != cal2.getTime().getTime() || gmt != cal2.get(ZONE_OFFSET)) {
error("Test2", loc, cal2, gmtOffset, dstOffset, t1);
}
cal2.setTimeZone(tz2);
cal2.clear();
cal2.set(2005, MARCH, 11);
// test3: set both ZONE_OFFSET and DST_OFFSET
cal2.set(ZONE_OFFSET, gmtOffset);
cal2.set(DST_OFFSET, dstOffset);
if (t1 != cal2.getTime().getTime()) {
error("Test3", loc, cal2, gmtOffset, dstOffset, t1);
}
cal2.setTimeZone(tz3);
cal2.clear();
cal2.set(2005, MARCH, 11);
// test4: set both ZONE_OFFSET and DST_OFFSET
cal2.set(ZONE_OFFSET, gmtOffset);
cal2.set(DST_OFFSET, dstOffset);
if (t1 != cal2.getTime().getTime()) {
error("Test4", loc, cal2, gmtOffset, dstOffset, t1);
}
// Test the same thing in non-lenient
cal2.setLenient(false);
cal2.setTimeZone(tz2);
cal2.clear();
cal2.set(2005, MARCH, 11);
adjustJapaneseEra(cal2);
// test5: set only ZONE_OFFSET in non-lenient
cal2.set(ZONE_OFFSET, gmtOffset);
if (t1 != cal2.getTime().getTime() || dst != cal2.get(DST_OFFSET)) {
error("Test5", loc, cal2, gmtOffset, dstOffset, t1);
}
cal2.setTimeZone(tz3);
cal2.clear();
cal2.set(2005, MARCH, 11);
adjustJapaneseEra(cal2);
// test6: set only DST_OFFSET in non-lenient
cal2.set(DST_OFFSET, dstOffset);
if (t1 != cal2.getTime().getTime() || gmt != cal2.get(ZONE_OFFSET)) {
error("Test6", loc, cal2, gmtOffset, dstOffset, t1);
}
cal2.setTimeZone(tz2);
cal2.clear();
cal2.set(2005, MARCH, 11);
adjustJapaneseEra(cal2);
// test7: set both ZONE_OFFSET and DST_OFFSET in non-lenient
cal2.set(ZONE_OFFSET, gmtOffset);
cal2.set(DST_OFFSET, dstOffset);
if (t1 != cal2.getTime().getTime()) {
error("Test7", loc, cal2, gmtOffset, dstOffset, t1);
}
cal2.setTimeZone(tz3);
cal2.clear();
cal2.set(2005, MARCH, 11);
adjustJapaneseEra(cal2);
// test8: set both ZONE_OFFSET and DST_OFFSET in non-lenient
cal2.set(ZONE_OFFSET, gmtOffset);
cal2.set(DST_OFFSET, dstOffset);
if (t1 != cal2.getTime().getTime()) {
error("Test8", loc, cal2, gmtOffset, dstOffset, t1);
}
}
private static void error(String msg, Locale loc, Calendar cal2, int gmtOffset, int dstOffset, long t1) {
System.err.println(cal2);
throw new RuntimeException(msg + ": Locale=" + loc
+ ", gmtOffset=" + gmtOffset + ", dstOffset=" + dstOffset
+ ", cal1 time=" + t1 + ", cal2 time=" + cal2.getTime().getTime());
}
private static void adjustJapaneseEra(Calendar cal) {
// In case of Japanese calendar, explicitly set the last era; REIWA so that
// year 2005 won't throw exception
if (!cal.isLenient() &&
cal.getCalendarType().equals("japanese") &&
System.currentTimeMillis() < 1556668800000L) { // Current time not in REIWA
cal.set(Calendar.ERA, 5);
cal.add(Calendar.YEAR, -30); // -30: Subtract year-length of HEISEI era
}
return;
}
}