Commit b7e108d6 authored by Frank Tang's avatar Frank Tang Committed by Commit Bot

[Intl] Use correct fallback values for options in Locale constructor

Fixes intl402/Locale/constructor-options-{casefirst,hourcycle,numeric}-invalid

Bug: v8:7684
Cq-Include-Trybots: luci.v8.try:v8_linux_noi18n_rel_ng
Change-Id: I43317f4bb1bb8422940faab1e5afa4162ed9ea11
Reviewed-on: https://chromium-review.googlesource.com/1137476Reviewed-by: 's avatarSathya Gunasekaran <gsathya@chromium.org>
Commit-Queue: Frank Tang <ftang@chromium.org>
Cr-Commit-Position: refs/heads/master@{#54504}
parent b102970c
...@@ -11,6 +11,7 @@ ...@@ -11,6 +11,7 @@
#include <map> #include <map>
#include <memory> #include <memory>
#include <string> #include <string>
#include <vector>
#include "src/api.h" #include "src/api.h"
#include "src/global-handles.h" #include "src/global-handles.h"
...@@ -32,6 +33,14 @@ namespace v8 { ...@@ -32,6 +33,14 @@ namespace v8 {
namespace internal { namespace internal {
namespace { namespace {
struct OptionData {
const char* name;
const char* key;
const std::vector<const char*>* possible_values;
bool is_bool_value;
};
// Inserts tags from options into locale string. // Inserts tags from options into locale string.
Maybe<bool> InsertOptionsIntoLocale(Isolate* isolate, Maybe<bool> InsertOptionsIntoLocale(Isolate* isolate,
Handle<JSReceiver> options, Handle<JSReceiver> options,
...@@ -39,32 +48,47 @@ Maybe<bool> InsertOptionsIntoLocale(Isolate* isolate, ...@@ -39,32 +48,47 @@ Maybe<bool> InsertOptionsIntoLocale(Isolate* isolate,
CHECK(isolate); CHECK(isolate);
CHECK(icu_locale); CHECK(icu_locale);
static const std::array<std::pair<const char*, const char*>, 6> static std::vector<const char*> hour_cycle_values = {"h11", "h12", "h23",
kOptionToUnicodeTagMap = {{{"calendar", "ca"}, "h24"};
{"collation", "co"}, static std::vector<const char*> case_first_values = {"upper", "lower",
{"hourCycle", "hc"}, "false"};
{"caseFirst", "kf"}, static std::vector<const char*> empty_values = {};
{"numeric", "kn"}, static const std::array<OptionData, 6> kOptionToUnicodeTagMap = {
{"numberingSystem", "nu"}}}; {{"calendar", "ca", &empty_values, false},
{"collation", "co", &empty_values, false},
{"hourCycle", "hc", &hour_cycle_values, false},
{"caseFirst", "kf", &case_first_values, false},
{"numeric", "kn", &empty_values, true},
{"numberingSystem", "nu", &empty_values, false}}};
// TODO(cira): Pass in values as per the spec to make this to be // TODO(cira): Pass in values as per the spec to make this to be
// spec compliant. // spec compliant.
std::vector<const char*> values = {};
for (const auto& option_to_bcp47 : kOptionToUnicodeTagMap) { for (const auto& option_to_bcp47 : kOptionToUnicodeTagMap) {
std::unique_ptr<char[]> value_str = nullptr; std::unique_ptr<char[]> value_str = nullptr;
Maybe<bool> maybe_found = Intl::GetStringOption( bool value_bool = false;
isolate, options, option_to_bcp47.first, values, "locale", &value_str); Maybe<bool> maybe_found =
option_to_bcp47.is_bool_value
? Intl::GetBoolOption(isolate, options, option_to_bcp47.name,
"locale", &value_bool)
: Intl::GetStringOption(isolate, options, option_to_bcp47.name,
*(option_to_bcp47.possible_values),
"locale", &value_str);
if (maybe_found.IsNothing()) return maybe_found; if (maybe_found.IsNothing()) return maybe_found;
// TODO(cira): Use fallback value if value is not found to make // TODO(cira): Use fallback value if value is not found to make
// this spec compliant. // this spec compliant.
if (!maybe_found.FromJust()) continue; if (!maybe_found.FromJust()) continue;
if (option_to_bcp47.is_bool_value) {
value_str = value_bool ? isolate->factory()->true_string()->ToCString()
: isolate->factory()->false_string()->ToCString();
}
DCHECK_NOT_NULL(value_str.get()); DCHECK_NOT_NULL(value_str.get());
// Convert bcp47 key and value into legacy ICU format so we can use // Convert bcp47 key and value into legacy ICU format so we can use
// uloc_setKeywordValue. // uloc_setKeywordValue.
const char* key = uloc_toLegacyKey(option_to_bcp47.second); const char* key = uloc_toLegacyKey(option_to_bcp47.key);
DCHECK_NOT_NULL(key); DCHECK_NOT_NULL(key);
// Overwrite existing, or insert new key-value to the locale string. // Overwrite existing, or insert new key-value to the locale string.
...@@ -113,17 +137,11 @@ bool PopulateLocaleWithUnicodeTags(Isolate* isolate, const char* icu_locale, ...@@ -113,17 +137,11 @@ bool PopulateLocaleWithUnicodeTags(Isolate* isolate, const char* icu_locale,
if (bcp47_key) { if (bcp47_key) {
const char* bcp47_value = uloc_toUnicodeLocaleType(bcp47_key, value); const char* bcp47_value = uloc_toUnicodeLocaleType(bcp47_key, value);
if (bcp47_value) { if (bcp47_value) {
// It's either Boolean value.
if (strncmp(bcp47_key, "kn", 2) == 0) {
bool numeric = strcmp(bcp47_value, "true") == 0 ? true : false;
Handle<Object> numeric_handle = factory->ToBoolean(numeric);
locale_holder->set_numeric(*numeric_handle);
continue;
}
// Or a string.
Handle<String> bcp47_handle = Handle<String> bcp47_handle =
factory->NewStringFromAsciiChecked(bcp47_value); factory->NewStringFromAsciiChecked(bcp47_value);
if (strncmp(bcp47_key, "ca", 2) == 0) { if (strncmp(bcp47_key, "kn", 2) == 0) {
locale_holder->set_numeric(*bcp47_handle);
} else if (strncmp(bcp47_key, "ca", 2) == 0) {
locale_holder->set_calendar(*bcp47_handle); locale_holder->set_calendar(*bcp47_handle);
} else if (strncmp(bcp47_key, "kf", 2) == 0) { } else if (strncmp(bcp47_key, "kf", 2) == 0) {
locale_holder->set_case_first(*bcp47_handle); locale_holder->set_case_first(*bcp47_handle);
......
...@@ -24,7 +24,7 @@ assertEquals('buddhist', locale.calendar); ...@@ -24,7 +24,7 @@ assertEquals('buddhist', locale.calendar);
assertEquals('phonebk', locale.collation); assertEquals('phonebk', locale.collation);
assertEquals('h23', locale.hourCycle); assertEquals('h23', locale.hourCycle);
assertEquals('upper', locale.caseFirst); assertEquals('upper', locale.caseFirst);
assertEquals(true, locale.numeric); assertEquals('true', locale.numeric);
assertEquals('roman', locale.numberingSystem); assertEquals('roman', locale.numberingSystem);
// Not defined, expected to undefined. // Not defined, expected to undefined.
assertEquals(undefined, locale.currency); assertEquals(undefined, locale.currency);
......
...@@ -448,12 +448,9 @@ ...@@ -448,12 +448,9 @@
'intl402/Locale/constructor-getter-order': [FAIL], 'intl402/Locale/constructor-getter-order': [FAIL],
'intl402/Locale/constructor-locale-object': [FAIL], 'intl402/Locale/constructor-locale-object': [FAIL],
'intl402/Locale/constructor-non-iana-canon': [FAIL], 'intl402/Locale/constructor-non-iana-canon': [FAIL],
'intl402/Locale/constructor-options-casefirst-invalid': [FAIL],
'intl402/Locale/constructor-options-hourcycle-invalid': [FAIL],
'intl402/Locale/constructor-options-language-grandfathered': [FAIL], 'intl402/Locale/constructor-options-language-grandfathered': [FAIL],
'intl402/Locale/constructor-options-language-invalid': [FAIL], 'intl402/Locale/constructor-options-language-invalid': [FAIL],
'intl402/Locale/constructor-options-language-valid': [FAIL], 'intl402/Locale/constructor-options-language-valid': [FAIL],
'intl402/Locale/constructor-options-numeric-valid': [FAIL],
'intl402/Locale/constructor-options-region-invalid': [FAIL], 'intl402/Locale/constructor-options-region-invalid': [FAIL],
'intl402/Locale/constructor-options-region-valid': [FAIL], 'intl402/Locale/constructor-options-region-valid': [FAIL],
'intl402/Locale/constructor-options-script-invalid': [FAIL], 'intl402/Locale/constructor-options-script-invalid': [FAIL],
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment