809995b526
Reviewed-by: ihse, joehw
503 lines
22 KiB
Java
503 lines
22 KiB
Java
/*
|
|
* Copyright (c) 2012, 2024, 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.text.*;
|
|
import java.text.spi.*;
|
|
import java.time.LocalDate;
|
|
import java.time.ZoneId;
|
|
import java.time.ZonedDateTime;
|
|
import java.time.format.DateTimeFormatter;
|
|
import java.time.format.FormatStyle;
|
|
import java.time.temporal.WeekFields;
|
|
import java.util.*;
|
|
import java.util.concurrent.CopyOnWriteArrayList;
|
|
import java.util.logging.Level;
|
|
import java.util.logging.LogRecord;
|
|
import java.util.logging.StreamHandler;
|
|
import java.util.spi.*;
|
|
import java.util.stream.IntStream;
|
|
import java.util.stream.Stream;
|
|
|
|
import jdk.test.lib.Utils;
|
|
import jdk.test.lib.process.ProcessTools;
|
|
import sun.util.locale.provider.LocaleProviderAdapter;
|
|
|
|
import static java.util.logging.LogManager.*;
|
|
|
|
public class LocaleProviders {
|
|
|
|
private static final boolean IS_WINDOWS = System.getProperty("os.name").startsWith("Windows");
|
|
private static final boolean IS_MAC = System.getProperty("os.name").startsWith("Mac");
|
|
|
|
public static void main(String[] args) {
|
|
String methodName = args[0];
|
|
|
|
switch (methodName) {
|
|
case "getPlatformLocale":
|
|
if (args[1].equals("format")) {
|
|
getPlatformLocale(Locale.Category.FORMAT);
|
|
} else {
|
|
getPlatformLocale(Locale.Category.DISPLAY);
|
|
}
|
|
break;
|
|
|
|
case "adapterTest":
|
|
adapterTest(args[1], args[2], (args.length >= 4 ? args[3] : ""));
|
|
break;
|
|
|
|
case "bug7198834Test":
|
|
bug7198834Test();
|
|
break;
|
|
|
|
case "tzNameTest":
|
|
tzNameTest(args[1]);
|
|
break;
|
|
|
|
case "bug8001440Test":
|
|
bug8001440Test();
|
|
break;
|
|
|
|
case "bug8010666Test":
|
|
bug8010666Test();
|
|
break;
|
|
|
|
case "bug8013086Test":
|
|
bug8013086Test(args[1], args[2]);
|
|
break;
|
|
|
|
case "bug8013903Test":
|
|
bug8013903Test();
|
|
break;
|
|
|
|
case "bug8027289Test":
|
|
bug8027289Test(args[1]);
|
|
break;
|
|
|
|
case "bug8220227Test":
|
|
bug8220227Test();
|
|
break;
|
|
|
|
case "bug8228465Test":
|
|
bug8228465Test();
|
|
break;
|
|
|
|
case "bug8232871Test":
|
|
bug8232871Test();
|
|
break;
|
|
|
|
case "bug8232860Test":
|
|
bug8232860Test();
|
|
break;
|
|
|
|
case "bug8245241Test":
|
|
bug8245241Test(args[1]);
|
|
break;
|
|
|
|
case "bug8248695Test":
|
|
bug8248695Test();
|
|
break;
|
|
|
|
case "bug8257964Test":
|
|
bug8257964Test();
|
|
break;
|
|
|
|
default:
|
|
throw new RuntimeException("Test method '"+methodName+"' not found.");
|
|
}
|
|
}
|
|
|
|
static void getPlatformLocale(Locale.Category cat) {
|
|
Locale defloc = Locale.getDefault(cat);
|
|
System.out.printf("%s,%s\n", defloc.getLanguage(), defloc.getCountry());
|
|
}
|
|
|
|
static void adapterTest(String expected, String lang, String ctry) {
|
|
Locale testLocale = Locale.of(lang, ctry);
|
|
LocaleProviderAdapter ldaExpected =
|
|
LocaleProviderAdapter.forType(LocaleProviderAdapter.Type.valueOf(expected));
|
|
if (!ldaExpected.getDateFormatProvider().isSupportedLocale(testLocale)) {
|
|
System.out.println("test locale: "+testLocale+" is not supported by the expected provider: "+ldaExpected+". Ignoring the test.");
|
|
return;
|
|
}
|
|
String preference = System.getProperty("java.locale.providers", "");
|
|
LocaleProviderAdapter lda = LocaleProviderAdapter.getAdapter(DateFormatProvider.class, testLocale);
|
|
LocaleProviderAdapter.Type type = lda.getAdapterType();
|
|
System.out.printf("testLocale: %s, got: %s, expected: %s\n", testLocale, type, expected);
|
|
if (!type.toString().equals(expected)) {
|
|
throw new RuntimeException("Returned locale data adapter is not correct.");
|
|
}
|
|
}
|
|
|
|
static void bug7198834Test() {
|
|
LocaleProviderAdapter lda = LocaleProviderAdapter.getAdapter(DateFormatProvider.class, Locale.US);
|
|
LocaleProviderAdapter.Type type = lda.getAdapterType();
|
|
if (type == LocaleProviderAdapter.Type.HOST && IS_WINDOWS) {
|
|
DateFormat df = DateFormat.getDateInstance(DateFormat.FULL, Locale.US);
|
|
String date = df.format(new Date());
|
|
if (date.charAt(date.length()-1) == ' ') {
|
|
throw new RuntimeException("Windows Host Locale Provider returns a trailing space.");
|
|
}
|
|
} else {
|
|
System.out.println("Windows HOST locale adapter not found. Ignoring this test.");
|
|
}
|
|
}
|
|
|
|
static void tzNameTest(String id) {
|
|
TimeZone tz = TimeZone.getTimeZone(id);
|
|
String tzName = tz.getDisplayName(false, TimeZone.SHORT, Locale.US);
|
|
if (tzName.startsWith("GMT")) {
|
|
throw new RuntimeException("JRE's localized time zone name for "+id+" could not be retrieved. Returned name was: "+tzName);
|
|
}
|
|
}
|
|
|
|
static void bug8001440Test() {
|
|
Locale locale = Locale.forLanguageTag("th-TH-u-nu-hoge");
|
|
NumberFormat nf = NumberFormat.getInstance(locale);
|
|
String nu = nf.format(1234560);
|
|
}
|
|
|
|
// This test assumes Windows localized language/country display names.
|
|
static void bug8010666Test() {
|
|
if (IS_WINDOWS) {
|
|
NumberFormat nf = NumberFormat.getInstance(Locale.US);
|
|
try {
|
|
double ver = nf.parse(System.getProperty("os.version"))
|
|
.doubleValue();
|
|
System.out.printf("Windows version: %.1f\n", ver);
|
|
if (ver >= 6.0) {
|
|
LocaleProviderAdapter lda =
|
|
LocaleProviderAdapter.getAdapter(
|
|
LocaleNameProvider.class, Locale.ENGLISH);
|
|
LocaleProviderAdapter.Type type = lda.getAdapterType();
|
|
if (type == LocaleProviderAdapter.Type.HOST) {
|
|
LocaleNameProvider lnp = lda.getLocaleNameProvider();
|
|
Locale mkmk = Locale.forLanguageTag("mk-MK");
|
|
String result = mkmk.getDisplayLanguage(Locale.ENGLISH);
|
|
String hostResult =
|
|
lnp.getDisplayLanguage(mkmk.getLanguage(),
|
|
Locale.ENGLISH);
|
|
System.out.printf(" Display language name for" +
|
|
" (mk_MK): result(HOST): \"%s\", returned: \"%s\"\n",
|
|
hostResult, result);
|
|
if (result == null ||
|
|
hostResult != null &&
|
|
!result.equals(hostResult)) {
|
|
throw new RuntimeException("Display language name" +
|
|
" mismatch for \"mk\". Returned name was" +
|
|
" \"" + result + "\", result(HOST): \"" +
|
|
hostResult + "\"");
|
|
}
|
|
result = Locale.US.getDisplayLanguage(Locale.ENGLISH);
|
|
hostResult =
|
|
lnp.getDisplayLanguage(Locale.US.getLanguage(),
|
|
Locale.ENGLISH);
|
|
System.out.printf(" Display language name for" +
|
|
" (en_US): result(HOST): \"%s\", returned: \"%s\"\n",
|
|
hostResult, result);
|
|
if (result == null ||
|
|
hostResult != null &&
|
|
!result.equals(hostResult)) {
|
|
throw new RuntimeException("Display language name" +
|
|
" mismatch for \"en\". Returned name was" +
|
|
" \"" + result + "\", result(HOST): \"" +
|
|
hostResult + "\"");
|
|
}
|
|
if (ver >= 6.1) {
|
|
result = Locale.US.getDisplayCountry(Locale.ENGLISH);
|
|
hostResult = lnp.getDisplayCountry(
|
|
Locale.US.getCountry(), Locale.ENGLISH);
|
|
System.out.printf(" Display country name for" +
|
|
" (en_US): result(HOST): \"%s\", returned: \"%s\"\n",
|
|
hostResult, result);
|
|
if (result == null ||
|
|
hostResult != null &&
|
|
!result.equals(hostResult)) {
|
|
throw new RuntimeException("Display country name" +
|
|
" mismatch for \"US\". Returned name was" +
|
|
" \"" + result + "\", result(HOST): \"" +
|
|
hostResult + "\"");
|
|
}
|
|
}
|
|
} else {
|
|
throw new RuntimeException("Windows Host" +
|
|
" LocaleProviderAdapter was not selected for" +
|
|
" English locale.");
|
|
}
|
|
}
|
|
} catch (ParseException pe) {
|
|
throw new RuntimeException("Parsing Windows version failed: "+pe.toString());
|
|
}
|
|
}
|
|
}
|
|
|
|
static void bug8013086Test(String lang, String ctry) {
|
|
try {
|
|
// Throws a NullPointerException if the test fails.
|
|
System.out.println(new SimpleDateFormat("z", Locale.of(lang, ctry)).parse("UTC"));
|
|
} catch (ParseException pe) {
|
|
// ParseException is fine in this test, as it's not "UTC"
|
|
}
|
|
}
|
|
|
|
static void bug8013903Test() {
|
|
if (IS_WINDOWS) {
|
|
Date sampleDate = new Date(0x10000000000L);
|
|
String expected = "\u5e73\u6210 16.11.03 (\u6c34) \u5348\u524d 11:53:47";
|
|
Locale l = Locale.of("ja", "JP", "JP");
|
|
SimpleDateFormat sdf = new SimpleDateFormat("GGGG yyyy.MMM.dd '('E')' a hh:mm:ss", l);
|
|
sdf.setTimeZone(TimeZone.getTimeZone("America/Los_Angeles"));
|
|
String result = sdf.format(sampleDate);
|
|
System.out.println(result);
|
|
// Windows display names. Subject to change if Windows changes its format.
|
|
if (!expected.equals(result)) {
|
|
throw new RuntimeException("Format failed. result: \"" +
|
|
result + "\", expected: \"" + expected);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void bug8027289Test(String expectedCodePoint) {
|
|
if (IS_WINDOWS) {
|
|
char[] expectedSymbol = Character.toChars(Integer.valueOf(expectedCodePoint, 16));
|
|
NumberFormat nf = NumberFormat.getCurrencyInstance(Locale.CHINA);
|
|
char formatted = nf.format(7000).charAt(0);
|
|
System.out.println("returned: " + formatted + ", expected: " + expectedSymbol[0]);
|
|
if (formatted != expectedSymbol[0]) {
|
|
throw new RuntimeException(
|
|
"Unexpected Chinese currency symbol. returned: "
|
|
+ formatted + ", expected: " + expectedSymbol[0]);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void bug8220227Test() {
|
|
if (IS_WINDOWS) {
|
|
Locale l = Locale.of("xx","XX");
|
|
String country = l.getDisplayCountry();
|
|
if (country.endsWith("(XX)")) {
|
|
throw new RuntimeException(
|
|
"Unexpected Region name: " + country);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void bug8228465Test() {
|
|
LocaleProviderAdapter lda = LocaleProviderAdapter.getAdapter(CalendarNameProvider.class, Locale.US);
|
|
LocaleProviderAdapter.Type type = lda.getAdapterType();
|
|
if (type == LocaleProviderAdapter.Type.HOST && IS_WINDOWS) {
|
|
var names = new GregorianCalendar()
|
|
.getDisplayNames(Calendar.ERA, Calendar.SHORT_FORMAT, Locale.US);
|
|
if (!names.keySet().contains("AD") ||
|
|
names.get("AD").intValue() != 1) {
|
|
throw new RuntimeException(
|
|
"Short Era name for 'AD' is missing or incorrect");
|
|
} else {
|
|
System.out.println("bug8228465Test succeeded.");
|
|
}
|
|
}
|
|
}
|
|
|
|
static void bug8232871Test() {
|
|
LocaleProviderAdapter lda = LocaleProviderAdapter.getAdapter(CalendarNameProvider.class, Locale.US);
|
|
LocaleProviderAdapter.Type type = lda.getAdapterType();
|
|
var lang = Locale.getDefault().getLanguage();
|
|
var cal = Calendar.getInstance();
|
|
var calType = cal.getCalendarType();
|
|
var expected = "\u4ee4\u548c1\u5e745\u67081\u65e5 \u6c34\u66dc\u65e5 \u5348\u524d0:00:00 \u30a2\u30e1\u30ea\u30ab\u592a\u5e73\u6d0b\u590f\u6642\u9593";
|
|
|
|
if (type == LocaleProviderAdapter.Type.HOST &&
|
|
IS_MAC &&
|
|
lang.equals("ja") &&
|
|
calType.equals("japanese")) {
|
|
cal.set(1, 4, 1, 0, 0, 0);
|
|
cal.setTimeZone(TimeZone.getTimeZone("America/Los_Angeles"));
|
|
DateFormat df = DateFormat.getDateTimeInstance(DateFormat.FULL, DateFormat.FULL,
|
|
Locale.JAPAN);
|
|
df.setCalendar(cal);
|
|
var result = df.format(cal.getTime());
|
|
if (result.equals(expected)) {
|
|
System.out.println("bug8232871Test succeeded.");
|
|
} else {
|
|
throw new RuntimeException(
|
|
"Japanese calendar names mismatch. result: " +
|
|
result +
|
|
", expected: " +
|
|
expected);
|
|
}
|
|
} else {
|
|
System.out.println("Test ignored. Either :-\n" +
|
|
"OS is not macOS, or\n" +
|
|
"provider is not HOST: " + type + ", or\n" +
|
|
"Language is not Japanese: " + lang + ", or\n" +
|
|
"native calendar is not JapaneseCalendar: " + calType);
|
|
}
|
|
}
|
|
|
|
static void bug8232860Test() {
|
|
var inputList = List.of(123, 123.4);
|
|
var nfExpectedList = List.of("123", "123.4");
|
|
var ifExpectedList = List.of("123", "123");
|
|
|
|
var defLoc = Locale.getDefault(Locale.Category.FORMAT);
|
|
var type = LocaleProviderAdapter.getAdapter(CalendarNameProvider.class, Locale.US)
|
|
.getAdapterType();
|
|
if (defLoc.equals(Locale.US) &&
|
|
type == LocaleProviderAdapter.Type.HOST &&
|
|
(IS_WINDOWS || IS_MAC)) {
|
|
final var numf = NumberFormat.getNumberInstance(Locale.US);
|
|
final var intf = NumberFormat.getIntegerInstance(Locale.US);
|
|
|
|
IntStream.range(0, inputList.size())
|
|
.forEach(i -> {
|
|
var input = inputList.get(i);
|
|
var nfExpected = nfExpectedList.get(i);
|
|
var result = numf.format(input);
|
|
if (!result.equals(nfExpected)) {
|
|
throw new RuntimeException("Incorrect number format. " +
|
|
"input: " + input + ", expected: " +
|
|
nfExpected + ", result: " + result);
|
|
}
|
|
|
|
var ifExpected = ifExpectedList.get(i);
|
|
result = intf.format(input);
|
|
if (!result.equals(ifExpected)) {
|
|
throw new RuntimeException("Incorrect integer format. " +
|
|
"input: " + input + ", expected: " +
|
|
ifExpected + ", result: " + result);
|
|
}
|
|
});
|
|
System.out.println("bug8232860Test succeeded.");
|
|
} else {
|
|
System.out.println("Test ignored. Either :-\n" +
|
|
"Default format locale is not Locale.US: " + defLoc + ", or\n" +
|
|
"OS is neither macOS/Windows, or\n" +
|
|
"provider is not HOST: " + type);
|
|
}
|
|
}
|
|
|
|
static void bug8245241Test(String expected) {
|
|
// this will ensure LocaleProviderAdapter initialization
|
|
DateFormat.getDateInstance();
|
|
LogConfig.handler.flush();
|
|
|
|
if (LogConfig.logRecordList.stream()
|
|
.noneMatch(r -> r.getLevel() == Level.INFO &&
|
|
r.getMessage().equals(expected))) {
|
|
throw new RuntimeException("Expected log was not emitted.");
|
|
}
|
|
}
|
|
|
|
// Set the root logger on loading the logging class
|
|
public static class LogConfig {
|
|
final static CopyOnWriteArrayList<LogRecord> logRecordList = new CopyOnWriteArrayList<>();
|
|
final static StreamHandler handler = new StreamHandler() {
|
|
@Override
|
|
public void publish(LogRecord record) {
|
|
logRecordList.add(record);
|
|
System.out.println("LogRecord: " + record.getMessage());
|
|
}
|
|
};
|
|
static {
|
|
getLogManager().getLogger("").addHandler(handler);
|
|
}
|
|
}
|
|
|
|
static void bug8248695Test() {
|
|
Locale l = Locale.getDefault(Locale.Category.FORMAT);
|
|
LocaleProviderAdapter lda = LocaleProviderAdapter.getAdapter(DateFormatProvider.class, l);
|
|
LocaleProviderAdapter.Type type = lda.getAdapterType();
|
|
if (type == LocaleProviderAdapter.Type.HOST) {
|
|
System.out.println("Locale: " + l);
|
|
var ld = LocalDate.now();
|
|
var zdt = ZonedDateTime.now(ZoneId.of("America/Los_Angeles"));
|
|
var df = DateTimeFormatter.ofLocalizedDate(FormatStyle.FULL).withLocale(l);
|
|
var tf = DateTimeFormatter.ofLocalizedTime(FormatStyle.FULL).withLocale(l);
|
|
var dtf = DateTimeFormatter.ofLocalizedDateTime(FormatStyle.FULL).withLocale(l);
|
|
|
|
// Checks there's no "unsupported temporal field" exception thrown, such as HourOfDay
|
|
System.out.println(df.format(ld));
|
|
System.out.println(tf.format(zdt));
|
|
|
|
// Checks there's no "Too many pattern letters: aa" exception thrown, if
|
|
// underlying OS provides the "am/pm" pattern.
|
|
System.out.println(dtf.format(zdt));
|
|
}
|
|
}
|
|
|
|
// Run only if the underlying platform locale is en-GB
|
|
// (Setting the java locale via command line properties does not substitute this)
|
|
static void bug8257964Test() {
|
|
var defLoc = Locale.getDefault(Locale.Category.FORMAT);
|
|
var type = LocaleProviderAdapter.getAdapter(CalendarNameProvider.class, Locale.UK)
|
|
.getAdapterType();
|
|
if (defLoc.equals(Locale.UK) &&
|
|
type == LocaleProviderAdapter.Type.HOST &&
|
|
(IS_WINDOWS || IS_MAC)) {
|
|
Calendar instance = Calendar.getInstance(Locale.UK);
|
|
int result = instance.getMinimalDaysInFirstWeek();
|
|
if (result != 4) {
|
|
throw new RuntimeException("MinimalDaysInFirstWeek for Locale.UK is incorrect. " +
|
|
"returned: " + result);
|
|
}
|
|
|
|
LocalDate date = LocalDate.of(2020,12,31);
|
|
result = date.get(WeekFields.of(Locale.UK).weekOfWeekBasedYear());
|
|
if (result != 53) {
|
|
throw new RuntimeException("weekNumber is incorrect. " +
|
|
"returned: " + result);
|
|
}
|
|
System.out.println("bug8257964Test succeeded.");
|
|
} else {
|
|
System.out.println("Test ignored. Either :-\n" +
|
|
"Default format locale is not Locale.UK: " + defLoc + ", or\n" +
|
|
"OS is neither macOS/Windows, or\n" +
|
|
"provider is not HOST: " + type);
|
|
}
|
|
}
|
|
|
|
/* Method is used by the LocaleProviders* related tests to launch a
|
|
* LocaleProviders test method with the appropriate LocaleProvider (e.g. CLDR,
|
|
* COMPAT, ETC.)
|
|
*/
|
|
static void test(String prefList, String methodName, String... params) throws Throwable {
|
|
|
|
List<String> command = List.of(
|
|
"-ea", "-esa",
|
|
"-cp", Utils.TEST_CLASS_PATH,
|
|
// Required for LocaleProvidersLogger
|
|
"-Djava.util.logging.config.class=LocaleProviders$LogConfig",
|
|
"-Djava.locale.providers=" + prefList,
|
|
"--add-exports=java.base/sun.util.locale.provider=ALL-UNNAMED",
|
|
"LocaleProviders", methodName);
|
|
|
|
// Build process with arguments, if required by the method
|
|
ProcessBuilder pb = ProcessTools.createTestJavaProcessBuilder(
|
|
Stream.concat(command.stream(), Stream.of(params)).toList());
|
|
|
|
// Evaluate process status
|
|
int exitCode = ProcessTools.executeCommand(pb).getExitValue();
|
|
if (exitCode != 0) {
|
|
throw new RuntimeException("Unexpected exit code: " + exitCode);
|
|
}
|
|
}
|
|
}
|