diff --git a/src/java.base/share/classes/java/util/TimeZone.java b/src/java.base/share/classes/java/util/TimeZone.java
index 5e76b8490b7..6177c66de13 100644
--- a/src/java.base/share/classes/java/util/TimeZone.java
+++ b/src/java.base/share/classes/java/util/TimeZone.java
@@ -1,5 +1,5 @@
/*
- * Copyright (c) 1996, 2021, Oracle and/or its affiliates. All rights reserved.
+ * Copyright (c) 1996, 2022, 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
@@ -40,6 +40,7 @@ package java.util;
import java.io.Serializable;
import java.time.ZoneId;
+import java.time.ZoneOffset;
import jdk.internal.util.StaticProperty;
import sun.security.action.GetPropertyAction;
@@ -74,6 +75,7 @@ import sun.util.locale.provider.TimeZoneNameUtility;
*
*
* CustomID:
+ * {@code GMT} Sign Hours {@code :} Minutes {@code :} Seconds
* {@code GMT} Sign Hours {@code :} Minutes
* {@code GMT} Sign Hours Minutes
* {@code GMT} Sign Hours
@@ -84,11 +86,13 @@ import sun.util.locale.provider.TimeZoneNameUtility;
* Digit Digit
* Minutes:
* Digit Digit
+ * Seconds:
+ * Digit Digit
* Digit: one of
* {@code 0 1 2 3 4 5 6 7 8 9}
*
*
- * Hours must be between 0 to 23 and Minutes must be
+ * Hours must be between 0 to 23 and Minutes/Seconds must be
* between 00 to 59. For example, "GMT+10" and "GMT+0010" mean ten
* hours and ten minutes ahead of GMT, respectively.
*
@@ -102,17 +106,20 @@ import sun.util.locale.provider.TimeZoneNameUtility;
* zone ID is normalized in the following syntax:
*
* NormalizedCustomID:
- * {@code GMT} Sign TwoDigitHours {@code :} Minutes
+ * {@code GMT} Sign TwoDigitHours {@code :} Minutes [ColonSeconds]
* Sign: one of
* {@code + -}
* TwoDigitHours:
* Digit Digit
* Minutes:
* Digit Digit
+ * ColonSeconds:
+ * {@code :} Digit Digit
* Digit: one of
* {@code 0 1 2 3 4 5 6 7 8 9}
*
* For example, TimeZone.getTimeZone("GMT-8").getID() returns "GMT-08:00".
+ * ColonSeconds part only appears if the seconds value is non-zero.
*
* Three-letter time zone IDs
*
@@ -529,11 +536,11 @@ public abstract class TimeZone implements Serializable, Cloneable {
*/
public static TimeZone getTimeZone(ZoneId zoneId) {
String tzid = zoneId.getId(); // throws an NPE if null
- char c = tzid.charAt(0);
- if (c == '+' || c == '-') {
- tzid = "GMT" + tzid;
- } else if (c == 'Z' && tzid.length() == 1) {
- tzid = "UTC";
+ if (zoneId instanceof ZoneOffset zo) {
+ var totalMillis = zo.getTotalSeconds() * 1_000;
+ return new ZoneInfo(totalMillis == 0 ? "UTC" : GMT_ID + tzid, totalMillis);
+ } else if (tzid.startsWith("UT") && !tzid.equals("UTC")) {
+ tzid = tzid.replaceFirst("(UTC|UT)(.*)", "GMT$2");
}
return getTimeZone(tzid, true);
}
@@ -823,19 +830,24 @@ public abstract class TimeZone implements Serializable, Cloneable {
}
int hours = 0;
+ int minutes = 0;
int num = 0;
int countDelim = 0;
int len = 0;
while (index < length) {
c = id.charAt(index++);
if (c == ':') {
- if (countDelim > 0) {
+ if (countDelim > 1) {
return null;
}
- if (len > 2) {
+ if (len == 0 || len > 2) {
return null;
}
- hours = num;
+ if (countDelim == 0) {
+ hours = num;
+ } else if (countDelim == 1){
+ minutes = num;
+ }
countDelim++;
num = 0;
len = 0;
@@ -853,20 +865,31 @@ public abstract class TimeZone implements Serializable, Cloneable {
if (countDelim == 0) {
if (len <= 2) {
hours = num;
+ minutes = 0;
+ num = 0;
+ } else if (len <= 4) {
+ hours = num / 100;
+ minutes = num % 100;
num = 0;
} else {
- hours = num / 100;
- num %= 100;
+ return null;
+ }
+ } else if (countDelim == 1){
+ if (len == 2) {
+ minutes = num;
+ num = 0;
+ } else {
+ return null;
}
} else {
if (len != 2) {
return null;
}
}
- if (hours > 23 || num > 59) {
+ if (hours > 23 || minutes > 59 || num > 59) {
return null;
}
- int gmtOffset = (hours * 60 + num) * 60 * 1000;
+ int gmtOffset = (hours * 3_600 + minutes * 60 + num) * 1_000;
if (gmtOffset == 0) {
zi = ZoneInfoFile.getZoneInfo(GMT_ID);
diff --git a/src/java.base/share/classes/sun/util/calendar/ZoneInfoFile.java b/src/java.base/share/classes/sun/util/calendar/ZoneInfoFile.java
index 6d50b367da5..47d71399cfe 100644
--- a/src/java.base/share/classes/sun/util/calendar/ZoneInfoFile.java
+++ b/src/java.base/share/classes/sun/util/calendar/ZoneInfoFile.java
@@ -178,15 +178,16 @@ public final class ZoneInfoFile {
public static String toCustomID(int gmtOffset) {
char sign;
- int offset = gmtOffset / 60000;
+ int offset = gmtOffset / 1_000;
if (offset >= 0) {
sign = '+';
} else {
sign = '-';
offset = -offset;
}
- int hh = offset / 60;
- int mm = offset % 60;
+ int hh = offset / 3_600;
+ int mm = (offset % 3_600) / 60;
+ int ss = offset % 60;
char[] buf = new char[] { 'G', 'M', 'T', sign, '0', '0', ':', '0', '0' };
if (hh >= 10) {
@@ -197,7 +198,13 @@ public final class ZoneInfoFile {
buf[7] += (char)(mm / 10);
buf[8] += (char)(mm % 10);
}
- return new String(buf);
+ var id = new String(buf);
+ if (ss != 0) {
+ buf[7] = (char)('0' + ss / 10);
+ buf[8] = (char)('0' + ss % 10);
+ id += new String(buf, 6, 3);
+ }
+ return id;
}
///////////////////////////////////////////////////////////
diff --git a/test/jdk/java/util/TimeZone/TimeZoneTest.java b/test/jdk/java/util/TimeZone/TimeZoneTest.java
index d31d1722b7b..898a6dd35da 100644
--- a/test/jdk/java/util/TimeZone/TimeZoneTest.java
+++ b/test/jdk/java/util/TimeZone/TimeZoneTest.java
@@ -1,5 +1,5 @@
/*
- * Copyright (c) 1997, 2021, Oracle and/or its affiliates. All rights reserved.
+ * Copyright (c) 1997, 2022, 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
@@ -25,10 +25,11 @@
* @test
* @bug 4028006 4044013 4096694 4107276 4107570 4112869 4130885 7039469 7126465 7158483
* 8008577 8077685 8098547 8133321 8138716 8148446 8151876 8159684 8166875 8181157
- * 8228469 8274407
+ * 8228469 8274407 8285844
* @modules java.base/sun.util.resources
* @library /java/text/testlib
* @summary test TimeZone
+ * @run main TimeZoneTest -verbose
*/
import java.io.*;
@@ -220,12 +221,14 @@ public class TimeZoneTest extends IntlTest
}
}
- static final String formatMinutes(int min) {
+ static final String formatSeconds(int sec) {
char sign = '+';
- if (min < 0) { sign = '-'; min = -min; }
- int h = min/60;
- min = min%60;
- return "" + sign + h + ":" + ((min<10) ? "0" : "") + min;
+ if (sec < 0) { sign = '-'; sec = -sec; }
+ int h = sec / 3_600;
+ int m = sec % 3_600 / 60;
+ sec = sec % 60;
+ return "" + sign + h + ":" + ((m<10) ? "0" : "") + m +
+ (sec > 0 ? ":" + ((sec < 10) ? "0" : "") + sec : "");
}
/**
* As part of the VM fix (see CCC approved RFE 4028006, bug
@@ -240,21 +243,28 @@ public class TimeZoneTest extends IntlTest
*/
public void TestCustomParse() throws Exception {
Object[] DATA = {
- // ID Expected offset in minutes
- "GMT", null,
- "GMT+0", new Integer(0),
- "GMT+1", new Integer(60),
- "GMT-0030", new Integer(-30),
- "GMT+15:99", null,
- "GMT+", null,
- "GMT-", null,
- "GMT+0:", null,
- "GMT-:", null,
- "GMT+0010", new Integer(10), // Interpret this as 00:10
- "GMT-10", new Integer(-10*60),
- "GMT+30", null,
- "GMT-3:30", new Integer(-(3*60+30)),
- "GMT-230", new Integer(-(2*60+30)),
+ // ID Expected offset in seconds
+ "GMT", null,
+ "GMT+0", 0,
+ "GMT+1", 60 * 60,
+ "GMT-0030", -30 * 60,
+ "GMT+15:99", null,
+ "GMT+", null,
+ "GMT-", null,
+ "GMT+0:", null,
+ "GMT-:", null,
+ "GMT+0010", 10 * 60, // Interpret this as 00:10
+ "GMT-10", -10 * 60 * 60,
+ "GMT+30", null,
+ "GMT-3:30", -(3 * 60 + 30) * 60,
+ "GMT-230", -(2 * 60 + 30) * 60,
+ "GMT+00:00:01", 1,
+ "GMT-00:00:01", -1,
+ "GMT+00000", null,
+ "GMT+00:00:01:", null,
+ "GMT+00:00:012", null,
+ "GMT+00:00:0", null,
+ "GMT+00:00:", null,
};
for (int i=0; i " + zone.getID() + " GMT" + offset);
if (exp == null) {
throw new Exception("Expected parse failure for " + id +
@@ -280,7 +290,7 @@ public class TimeZoneTest extends IntlTest
", id " + zone.getID());
}
else if (ioffset != exp.intValue()) {
- throw new Exception("Expected offset of " + formatMinutes(exp.intValue()) +
+ throw new Exception("Expected offset of " + formatSeconds(exp.intValue()) +
", id Custom, for " + id +
", got offset of " + offset +
", id " + zone.getID());
diff --git a/test/jdk/java/util/TimeZone/ZoneIdRoundTripTest.java b/test/jdk/java/util/TimeZone/ZoneIdRoundTripTest.java
new file mode 100644
index 00000000000..0f1eeb88328
--- /dev/null
+++ b/test/jdk/java/util/TimeZone/ZoneIdRoundTripTest.java
@@ -0,0 +1,70 @@
+/*
+ * Copyright (c) 2022, 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.
+ */
+
+import java.time.ZoneId;
+import java.time.ZoneOffset;
+import java.util.TimeZone;
+
+import org.testng.annotations.Test;
+import org.testng.annotations.DataProvider;
+import static org.testng.Assert.assertEquals;
+
+/**
+ * @test
+ * @bug 8285844
+ * @summary Checks round-trips between TimeZone and ZoneId are consistent
+ * @run testng ZoneIdRoundTripTest
+ */
+@Test
+public class ZoneIdRoundTripTest {
+
+ @DataProvider
+ private Object[][] testZoneIds() {
+ return new Object[][] {
+ {ZoneId.of("Z"), 0},
+ {ZoneId.of("UT"), 0},
+ {ZoneId.of("UTC"), 0},
+ {ZoneId.of("GMT"), 0},
+ {ZoneId.of("+00:01"), 60_000},
+ {ZoneId.of("-00:01"), -60_000},
+ {ZoneId.of("+00:00:01"), 1_000},
+ {ZoneId.of("-00:00:01"), -1_000},
+ {ZoneId.of("UT+00:00:01"), 1_000},
+ {ZoneId.of("UT-00:00:01"), -1_000},
+ {ZoneId.of("UTC+00:00:01"), 1_000},
+ {ZoneId.of("UTC-00:00:01"), -1_000},
+ {ZoneId.of("GMT+00:00:01"), 1_000},
+ {ZoneId.of("GMT-00:00:01"), -1_000},
+ {ZoneOffset.of("+00:00:01"), 1_000},
+ {ZoneOffset.of("-00:00:01"), -1_000},
+ };
+ }
+
+ @Test(dataProvider="testZoneIds")
+ public void test_ZoneIdRoundTrip(ZoneId zid, int offset) {
+ var tz = TimeZone.getTimeZone(zid);
+ assertEquals(tz.getRawOffset(), offset);
+ assertEquals(tz.toZoneId().normalized(), zid.normalized());
+ }
+}
+