Skip to content

Commit cd16d34

Browse files
committed
[GR-32236] Allow to use time-zone-related data from ZoneRulesProvider.
PullRequest: js/2476
2 parents f968db9 + b83c59f commit cd16d34

File tree

10 files changed

+317
-9
lines changed

10 files changed

+317
-9
lines changed

CHANGELOG.md

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ See [version roadmap](https://www.graalvm.org/release-notes/version-roadmap/) fo
77

88
## Version 22.2.0
99
* Removed experimental option `commonjs-global-properties`. The same functionality can be achieved in user code with a direct call to `require()` after context creation.
10+
* Added an experimental option `--js.zone-rules-based-time-zones` that allows to use timezone-related data from `ZoneRulesProvider` (instead of ICU4J data files).
1011

1112
## Version 22.1.0
1213
* Updated Node.js to version 16.14.2.

graal-js/ci.jsonnet

Lines changed: 5 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -151,7 +151,11 @@ local ci = import '../ci.jsonnet';
151151
// interop benchmarks
152152
graalJs + common.jdk17 + common.bench + common.x52 + interopJmhBenchmarks + {name: 'js-bench-interop-jmh-jdk17-linux-amd64'},
153153

154-
// POST-MERGE - PGO profiles
154+
// POST-MERGE
155+
// PGO profiles
155156
graalJs + common.jdk17 + common.postMerge + common.linux + downstreamSubstratevmEE + {environment+: {TAGS: 'pgo_collect_js'}} + {name: 'js-postmerge-pgo-profiles-jdk17-linux-amd64'},
157+
158+
// js.zone-rules-based-time-zones
159+
graalJs + common.jdk17 + common.postMerge + common.linux + gateTags('zonerulesbasedtimezones') + {name: 'js-postmerge-zonerulesbasedtimezones-jdk17-linux-amd64'},
156160
],
157161
}

graal-js/mx.graal-js/mx_graal_js.py

Lines changed: 4 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -74,7 +74,9 @@ def _graal_js_gate_runner(args, tasks):
7474
with Task('UnitTests', tasks, tags=[GraalJsDefaultTags.default, GraalJsDefaultTags.all]) as t:
7575
if t:
7676
noWebAssemblyTestSuite = '^(?!' + webassemblyTestSuite + ')'
77-
unittest(['--regex', noWebAssemblyTestSuite, '--enable-timing', '--very-verbose', '--suite', _suite.name])
77+
commonOptions = ['--enable-timing', '--very-verbose', '--suite', _suite.name]
78+
unittest(['--regex', noWebAssemblyTestSuite] + commonOptions)
79+
unittest(['--regex', 'ZoneRulesProviderTest', '-Djava.time.zone.DefaultZoneRulesProvider=com.oracle.truffle.js.test.runtime.SimpleZoneRulesProvider'] + commonOptions)
7880

7981
with Task('WebAssemblyTests', tasks, tags=['webassembly', GraalJsDefaultTags.all]) as t:
8082
if t:
@@ -86,6 +88,7 @@ def _graal_js_gate_runner(args, tasks):
8688
'directbytebuffer': ['-Dpolyglot.js.direct-byte-buffer=true', 'gate'],
8789
'cloneuninitialized': ['-Dpolyglot.js.test-clone-uninitialized=true', 'gate'],
8890
'lazytranslation': ['-Dpolyglot.js.lazy-translation=true', 'gate'],
91+
'zonerulesbasedtimezones': ['-Dpolyglot.js.zone-rules-based-time-zones=true', 'gate'],
8992
'shareengine': ['gate', 'shareengine'],
9093
'latestversion': ['gate', 'minesversion=2022'],
9194
'instrument': ['gate', 'instrument', 'timeoutoverall=1800']
Lines changed: 76 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
/*
2+
* Copyright (c) 2022, 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* The Universal Permissive License (UPL), Version 1.0
6+
*
7+
* Subject to the condition set forth below, permission is hereby granted to any
8+
* person obtaining a copy of this software, associated documentation and/or
9+
* data (collectively the "Software"), free of charge and under any and all
10+
* copyright rights in the Software, and any and all patent rights owned or
11+
* freely licensable by each licensor hereunder covering either (i) the
12+
* unmodified Software as contributed to or provided by such licensor, or (ii)
13+
* the Larger Works (as defined below), to deal in both
14+
*
15+
* (a) the Software, and
16+
*
17+
* (b) any piece of software and/or hardware listed in the lrgrwrks.txt file if
18+
* one is included with the Software each a "Larger Work" to which the Software
19+
* is contributed by such licensors),
20+
*
21+
* without restriction, including without limitation the rights to copy, create
22+
* derivative works of, display, perform, and distribute the Software and make,
23+
* use, sell, offer for sale, import, export, have made, and have sold the
24+
* Software and the Larger Work(s), and to sublicense the foregoing rights on
25+
* either these or other terms.
26+
*
27+
* This license is subject to the following condition:
28+
*
29+
* The above copyright notice and either this complete permission notice or at a
30+
* minimum a reference to the UPL must be included in all copies or substantial
31+
* portions of the Software.
32+
*
33+
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
34+
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
35+
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
36+
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
37+
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
38+
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
39+
* SOFTWARE.
40+
*/
41+
package com.oracle.truffle.js.test.runtime;
42+
43+
import java.time.ZoneOffset;
44+
import java.time.zone.ZoneRules;
45+
import java.time.zone.ZoneRulesProvider;
46+
import java.util.Arrays;
47+
import java.util.HashSet;
48+
import java.util.NavigableMap;
49+
import java.util.Set;
50+
import java.util.TimeZone;
51+
import java.util.TreeMap;
52+
53+
/**
54+
* Simple implementation of {@code ZoneRulesProvider} for testing purposes. The data provided by
55+
* this implementation have nothing to do with the real tzdb data.
56+
*/
57+
public class SimpleZoneRulesProvider extends ZoneRulesProvider {
58+
59+
@Override
60+
protected Set<String> provideZoneIds() {
61+
return new HashSet<>(Arrays.asList(TimeZone.getAvailableIDs()));
62+
}
63+
64+
@Override
65+
protected ZoneRules provideRules(String zoneId, boolean forCaching) {
66+
return ZoneRules.of(ZoneOffset.ofHours(7));
67+
}
68+
69+
@Override
70+
protected NavigableMap<String, ZoneRules> provideVersions(String zoneId) {
71+
NavigableMap<String, ZoneRules> map = new TreeMap<>();
72+
map.put("theOnlyVersion", provideRules(zoneId, false));
73+
return map;
74+
}
75+
76+
}
Lines changed: 100 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,100 @@
1+
/*
2+
* Copyright (c) 2022, 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* The Universal Permissive License (UPL), Version 1.0
6+
*
7+
* Subject to the condition set forth below, permission is hereby granted to any
8+
* person obtaining a copy of this software, associated documentation and/or
9+
* data (collectively the "Software"), free of charge and under any and all
10+
* copyright rights in the Software, and any and all patent rights owned or
11+
* freely licensable by each licensor hereunder covering either (i) the
12+
* unmodified Software as contributed to or provided by such licensor, or (ii)
13+
* the Larger Works (as defined below), to deal in both
14+
*
15+
* (a) the Software, and
16+
*
17+
* (b) any piece of software and/or hardware listed in the lrgrwrks.txt file if
18+
* one is included with the Software each a "Larger Work" to which the Software
19+
* is contributed by such licensors),
20+
*
21+
* without restriction, including without limitation the rights to copy, create
22+
* derivative works of, display, perform, and distribute the Software and make,
23+
* use, sell, offer for sale, import, export, have made, and have sold the
24+
* Software and the Larger Work(s), and to sublicense the foregoing rights on
25+
* either these or other terms.
26+
*
27+
* This license is subject to the following condition:
28+
*
29+
* The above copyright notice and either this complete permission notice or at a
30+
* minimum a reference to the UPL must be included in all copies or substantial
31+
* portions of the Software.
32+
*
33+
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
34+
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
35+
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
36+
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
37+
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
38+
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
39+
* SOFTWARE.
40+
*/
41+
package com.oracle.truffle.js.test.runtime;
42+
43+
import static org.junit.Assert.assertEquals;
44+
import static org.junit.Assume.assumeTrue;
45+
46+
import com.oracle.truffle.js.lang.JavaScriptLanguage;
47+
import com.oracle.truffle.js.runtime.JSContextOptions;
48+
import com.oracle.truffle.js.test.JSTest;
49+
import org.graalvm.polyglot.Context;
50+
import org.junit.Before;
51+
import org.junit.Test;
52+
53+
/**
54+
* Tests of {@code js.zone-rules-based-time-zones} option.
55+
*/
56+
public class ZoneRulesProviderTest {
57+
58+
@Before
59+
public void setUp() {
60+
String provider = System.getProperty("java.time.zone.DefaultZoneRulesProvider");
61+
assumeTrue(SimpleZoneRulesProvider.class.getName().equals(provider));
62+
}
63+
64+
@Test
65+
public void testNoIntl() {
66+
try (Context context = newContext(false)) {
67+
context.eval(JavaScriptLanguage.ID, "date = new Date(Date.UTC(2020, 3, 25, 12, 13, 14))");
68+
// SimpleZoneRulesProvider uses +7 offset for all time-zones
69+
assertResult(context, "date.toString()", "Sat Apr 25 2020 19:13:14 GMT+0700 (CET)");
70+
assertResult(context, "date.toLocaleString()", "Sat Apr 25 2020 19:13:14 GMT+0700 (CET)");
71+
assertResult(context, "date.toTimeString()", "19:13:14 GMT+0700 (CET)");
72+
assertResult(context, "date.toLocaleTimeString()", "19:13:14");
73+
}
74+
}
75+
76+
@Test
77+
public void testIntl() {
78+
try (Context context = newContext(true)) {
79+
context.eval(JavaScriptLanguage.ID, "date = new Date(Date.UTC(2020, 3, 25, 12, 13, 14))");
80+
// SimpleZoneRulesProvider uses +7 offset for all time-zones
81+
assertResult(context, "date.toLocaleString()", "4/25/2020, 7:13:14 PM");
82+
assertResult(context, "date.toLocaleTimeString()", "7:13:14 PM");
83+
assertResult(context, "new Intl.DateTimeFormat('en', { dateStyle: 'long', timeStyle: 'long' }).format(date)", "April 25, 2020 at 7:13:14 PM GMT+7");
84+
}
85+
}
86+
87+
private static Context newContext(boolean intl) {
88+
return JSTest.newContextBuilder().//
89+
option(JSContextOptions.TIME_ZONE_NAME, "Europe/Prague").//
90+
option(JSContextOptions.INTL_402_NAME, Boolean.toString(intl)).//
91+
option(JSContextOptions.ZONE_RULES_BASED_TIME_ZONES_NAME, "true").//
92+
build();
93+
}
94+
95+
private static void assertResult(Context context, String code, String expectedResult) {
96+
String actualResult = context.eval(JavaScriptLanguage.ID, code).toString();
97+
assertEquals(expectedResult, actualResult);
98+
}
99+
100+
}

graal-js/src/com.oracle.truffle.js/src/com/oracle/truffle/js/nodes/intl/InitializeDateTimeFormatNode.java

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -210,7 +210,7 @@ private TimeZone toTimeZone(Object timeZoneValue) {
210210
errorBranch.enter();
211211
throw Errors.createRangeErrorInvalidTimeZone(name);
212212
}
213-
return IntlUtil.getICUTimeZone(tzId);
213+
return IntlUtil.getICUTimeZone(tzId, context);
214214
} else {
215215
return getRealm().getLocalTimeZone();
216216
}

graal-js/src/com.oracle.truffle.js/src/com/oracle/truffle/js/runtime/JSContextOptions.java

Lines changed: 14 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -226,6 +226,11 @@ public String apply(String tz) {
226226
}
227227
}));
228228

229+
public static final String ZONE_RULES_BASED_TIME_ZONES_NAME = JS_OPTION_PREFIX + "zone-rules-based-time-zones";
230+
@Option(name = ZONE_RULES_BASED_TIME_ZONES_NAME, category = OptionCategory.EXPERT, help = "Use ZoneRulesProvider instead of time-zone data from ICU4J.") //
231+
public static final OptionKey<Boolean> ZONE_RULES_BASED_TIME_ZONES = new OptionKey<>(false);
232+
@CompilationFinal private boolean zoneRulesBasedTimeZones;
233+
229234
public static final String TIMER_RESOLUTION_NAME = JS_OPTION_PREFIX + "timer-resolution";
230235
@Option(name = TIMER_RESOLUTION_NAME, category = OptionCategory.USER, usageSyntax = "<nanoseconds>", help = "Resolution of timers (performance.now() and Date built-ins) in nanoseconds. Fuzzy time is used when set to 0.") //
231236
public static final OptionKey<Long> TIMER_RESOLUTION = new OptionKey<>(1000000L);
@@ -667,6 +672,7 @@ private void cacheOptions() {
667672
});
668673
this.parseOnly = readBooleanOption(PARSE_ONLY);
669674
this.debug = readBooleanOption(DEBUG_BUILTIN);
675+
this.zoneRulesBasedTimeZones = readBooleanOption(ZONE_RULES_BASED_TIME_ZONES);
670676
this.timerResolution = patchLongOption(TIMER_RESOLUTION, TIMER_RESOLUTION_NAME, timerResolution, msg -> {
671677
timerResolutionCyclicAssumption.invalidate(msg);
672678
timerResolutionCurrentAssumption = timerResolutionCyclicAssumption.getAssumption();
@@ -857,6 +863,10 @@ public boolean isV8LegacyConst() {
857863
return v8LegacyConst;
858864
}
859865

866+
public boolean hasZoneRulesBasedTimeZones() {
867+
return zoneRulesBasedTimeZones;
868+
}
869+
860870
public boolean canAgentBlock() {
861871
return agentCanBlock;
862872
}
@@ -1153,6 +1163,7 @@ public int hashCode() {
11531163
hash = 53 * hash + (this.debug ? 1 : 0);
11541164
hash = 53 * hash + (this.directByteBuffer ? 1 : 0);
11551165
hash = 53 * hash + (this.parseOnly ? 1 : 0);
1166+
hash = 53 * hash + (this.zoneRulesBasedTimeZones ? 1 : 0);
11561167
hash = 53 * hash + (int) this.timerResolution;
11571168
hash = 53 * hash + (this.agentCanBlock ? 1 : 0);
11581169
hash = 53 * hash + (this.awaitOptimization ? 1 : 0);
@@ -1255,6 +1266,9 @@ public boolean equals(Object obj) {
12551266
if (this.parseOnly != other.parseOnly) {
12561267
return false;
12571268
}
1269+
if (this.zoneRulesBasedTimeZones != other.zoneRulesBasedTimeZones) {
1270+
return false;
1271+
}
12581272
if (this.timerResolution != other.timerResolution) {
12591273
return false;
12601274
}

graal-js/src/com.oracle.truffle.js/src/com/oracle/truffle/js/runtime/JSRealm.java

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -2596,7 +2596,7 @@ public TimeZone getLocalTimeZone() {
25962596

25972597
@TruffleBoundary
25982598
private TimeZone getICUTimeZoneFromEnv() {
2599-
return IntlUtil.getICUTimeZone(getLocalTimeZoneId());
2599+
return IntlUtil.getICUTimeZone(getLocalTimeZoneId(), getContext());
26002600
}
26012601

26022602
public ZoneId getLocalTimeZoneId() {
@@ -2990,7 +2990,7 @@ public void setLocalTimeZone(String tzId) {
29902990
try {
29912991
if (tzId != null) {
29922992
newZoneId = ZoneId.of(tzId);
2993-
newTimeZone = IntlUtil.getICUTimeZone(tzId);
2993+
newTimeZone = IntlUtil.getICUTimeZone(tzId, getContext());
29942994
} else {
29952995
// Reset to default time zone (fields are reinitialized on next use).
29962996
newZoneId = null;

graal-js/src/com.oracle.truffle.js/src/com/oracle/truffle/js/runtime/util/IntlUtil.java

Lines changed: 13 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@
4141
package com.oracle.truffle.js.runtime.util;
4242

4343
import java.time.ZoneId;
44+
import java.time.zone.ZoneRulesProvider;
4445
import java.util.ArrayList;
4546
import java.util.Arrays;
4647
import java.util.HashSet;
@@ -873,14 +874,22 @@ public static void ensureIsWellFormedUnitIdentifier(String unitIdentifier) {
873874
}
874875

875876
@TruffleBoundary
876-
public static TimeZone getICUTimeZone(String tzId) {
877+
public static TimeZone getICUTimeZone(String tzId, JSContext context) {
877878
assert tzId != null;
878-
return TimeZone.getTimeZone(tzId);
879+
if (context.getContextOptions().hasZoneRulesBasedTimeZones()) {
880+
return new ZoneRulesBasedTimeZone(tzId, ZoneRulesProvider.getRules(tzId, false));
881+
} else {
882+
return TimeZone.getTimeZone(tzId);
883+
}
879884
}
880885

881886
@TruffleBoundary
882-
public static TimeZone getICUTimeZone(ZoneId zoneId) {
883-
return getICUTimeZone(toICUTimeZoneId(zoneId));
887+
public static TimeZone getICUTimeZone(ZoneId zoneId, JSContext context) {
888+
if (context.getContextOptions().hasZoneRulesBasedTimeZones()) {
889+
return new ZoneRulesBasedTimeZone(zoneId.getId(), zoneId.getRules());
890+
} else {
891+
return TimeZone.getTimeZone(toICUTimeZoneId(zoneId));
892+
}
884893
}
885894

886895
private static String toICUTimeZoneId(ZoneId zoneId) {

0 commit comments

Comments
 (0)