| /* |
| * Copyright (C) 2016 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include "compile/PseudolocaleGenerator.h" |
| |
| #include "test/Test.h" |
| #include "util/Util.h" |
| |
| using ::android::ConfigDescription; |
| |
| namespace aapt { |
| |
| TEST(PseudolocaleGeneratorTest, PseudolocalizeStyledString) { |
| StringPool pool; |
| StyleString original_style; |
| original_style.str = "Hello world!"; |
| original_style.spans = {Span{"i", 1, 10}, Span{"b", 2, 3}, Span{"b", 6, 7}}; |
| |
| std::unique_ptr<StyledString> new_string = PseudolocalizeStyledString( |
| util::make_unique<StyledString>(pool.MakeRef(original_style)).get(), |
| Pseudolocalizer::Method::kNone, &pool); |
| |
| EXPECT_EQ(original_style.str, new_string->value->value); |
| ASSERT_EQ(original_style.spans.size(), new_string->value->spans.size()); |
| |
| EXPECT_EQ(std::string("i"), *new_string->value->spans[0].name); |
| EXPECT_EQ(std::u16string(u"H").size(), new_string->value->spans[0].first_char); |
| EXPECT_EQ(std::u16string(u"Hello worl").size(), new_string->value->spans[0].last_char); |
| |
| EXPECT_EQ(std::string("b"), *new_string->value->spans[1].name); |
| EXPECT_EQ(std::u16string(u"He").size(), new_string->value->spans[1].first_char); |
| EXPECT_EQ(std::u16string(u"Hel").size(), new_string->value->spans[1].last_char); |
| |
| EXPECT_EQ(std::string("b"), *new_string->value->spans[2].name); |
| EXPECT_EQ(std::u16string(u"Hello ").size(), new_string->value->spans[2].first_char); |
| EXPECT_EQ(std::u16string(u"Hello w").size(), new_string->value->spans[2].last_char); |
| |
| original_style.spans.insert(original_style.spans.begin(), Span{"em", 0, 11u}); |
| |
| new_string = PseudolocalizeStyledString( |
| util::make_unique<StyledString>(pool.MakeRef(original_style)).get(), |
| Pseudolocalizer::Method::kAccent, &pool); |
| |
| EXPECT_EQ(std::string("[Ĥéļļö ŵöŕļð¡ one two]"), new_string->value->value); |
| ASSERT_EQ(original_style.spans.size(), new_string->value->spans.size()); |
| |
| EXPECT_EQ(std::u16string(u"[").size(), new_string->value->spans[0].first_char); |
| EXPECT_EQ(std::u16string(u"[Ĥéļļö ŵöŕļð").size(), new_string->value->spans[0].last_char); |
| |
| EXPECT_EQ(std::u16string(u"[Ĥ").size(), new_string->value->spans[1].first_char); |
| EXPECT_EQ(std::u16string(u"[Ĥéļļö ŵöŕļ").size(), new_string->value->spans[1].last_char); |
| |
| EXPECT_EQ(std::u16string(u"[Ĥé").size(), new_string->value->spans[2].first_char); |
| EXPECT_EQ(std::u16string(u"[Ĥéļ").size(), new_string->value->spans[2].last_char); |
| |
| EXPECT_EQ(std::u16string(u"[Ĥéļļö ").size(), new_string->value->spans[3].first_char); |
| EXPECT_EQ(std::u16string(u"[Ĥéļļö ŵ").size(), new_string->value->spans[3].last_char); |
| } |
| |
| TEST(PseudolocaleGeneratorTest, PseudolocalizeAdjacentNestedTags) { |
| StringPool pool; |
| StyleString original_style; |
| original_style.str = "bold"; |
| original_style.spans = {Span{"b", 0, 3}, Span{"i", 0, 3}}; |
| |
| std::unique_ptr<StyledString> new_string = PseudolocalizeStyledString( |
| util::make_unique<StyledString>(pool.MakeRef(original_style)).get(), |
| Pseudolocalizer::Method::kAccent, &pool); |
| ASSERT_NE(nullptr, new_string); |
| ASSERT_EQ(2u, new_string->value->spans.size()); |
| EXPECT_EQ(std::string("[ɓöļð one]"), new_string->value->value); |
| |
| EXPECT_EQ(std::string("b"), *new_string->value->spans[0].name); |
| EXPECT_EQ(std::u16string(u"[").size(), new_string->value->spans[0].first_char); |
| EXPECT_EQ(std::u16string(u"[ɓöļ").size(), new_string->value->spans[0].last_char); |
| |
| EXPECT_EQ(std::string("i"), *new_string->value->spans[1].name); |
| EXPECT_EQ(std::u16string(u"[").size(), new_string->value->spans[1].first_char); |
| EXPECT_EQ(std::u16string(u"[ɓöļ").size(), new_string->value->spans[1].last_char); |
| } |
| |
| TEST(PseudolocaleGeneratorTest, PseudolocalizeAdjacentTagsUnsorted) { |
| StringPool pool; |
| StyleString original_style; |
| original_style.str = "bold"; |
| original_style.spans = {Span{"i", 2, 3}, Span{"b", 0, 1}}; |
| |
| std::unique_ptr<StyledString> new_string = PseudolocalizeStyledString( |
| util::make_unique<StyledString>(pool.MakeRef(original_style)).get(), |
| Pseudolocalizer::Method::kAccent, &pool); |
| ASSERT_NE(nullptr, new_string); |
| ASSERT_EQ(2u, new_string->value->spans.size()); |
| EXPECT_EQ(std::string("[ɓöļð one]"), new_string->value->value); |
| |
| EXPECT_EQ(std::string("b"), *new_string->value->spans[0].name); |
| EXPECT_EQ(std::u16string(u"[").size(), new_string->value->spans[0].first_char); |
| EXPECT_EQ(std::u16string(u"[ɓ").size(), new_string->value->spans[0].last_char); |
| |
| EXPECT_EQ(std::string("i"), *new_string->value->spans[1].name); |
| EXPECT_EQ(std::u16string(u"[ɓö").size(), new_string->value->spans[1].first_char); |
| EXPECT_EQ(std::u16string(u"[ɓöļ").size(), new_string->value->spans[1].last_char); |
| } |
| |
| TEST(PseudolocaleGeneratorTest, PseudolocalizeNestedAndAdjacentTags) { |
| StringPool pool; |
| StyleString original_style; |
| original_style.str = "This sentence is not what you think it is at all."; |
| original_style.spans = {Span{"b", 16u, 19u}, Span{"em", 29u, 47u}, Span{"i", 38u, 40u}, |
| Span{"b", 44u, 47u}}; |
| |
| std::unique_ptr<StyledString> new_string = PseudolocalizeStyledString( |
| util::make_unique<StyledString>(pool.MakeRef(original_style)).get(), |
| Pseudolocalizer::Method::kAccent, &pool); |
| ASSERT_NE(nullptr, new_string); |
| ASSERT_EQ(4u, new_string->value->spans.size()); |
| EXPECT_EQ(std::string( |
| "[Ţĥîš šéñţéñçé îš ñöţ ŵĥåţ ýöû ţĥîñķ îţ îš åţ åļļ. one two three four five six]"), |
| new_string->value->value); |
| |
| EXPECT_EQ(std::string("b"), *new_string->value->spans[0].name); |
| EXPECT_EQ(std::u16string(u"[Ţĥîš šéñţéñçé îš").size(), new_string->value->spans[0].first_char); |
| EXPECT_EQ(std::u16string(u"[Ţĥîš šéñţéñçé îš ñö").size(), new_string->value->spans[0].last_char); |
| |
| EXPECT_EQ(std::string("em"), *new_string->value->spans[1].name); |
| EXPECT_EQ(std::u16string(u"[Ţĥîš šéñţéñçé îš ñöţ ŵĥåţ ýöû").size(), |
| new_string->value->spans[1].first_char); |
| EXPECT_EQ(std::u16string(u"[Ţĥîš šéñţéñçé îš ñöţ ŵĥåţ ýöû ţĥîñķ îţ îš åţ åļ").size(), |
| new_string->value->spans[1].last_char); |
| |
| EXPECT_EQ(std::string("i"), *new_string->value->spans[2].name); |
| EXPECT_EQ(std::u16string(u"[Ţĥîš šéñţéñçé îš ñöţ ŵĥåţ ýöû ţĥîñķ îţ").size(), |
| new_string->value->spans[2].first_char); |
| EXPECT_EQ(std::u16string(u"[Ţĥîš šéñţéñçé îš ñöţ ŵĥåţ ýöû ţĥîñķ îţ î").size(), |
| new_string->value->spans[2].last_char); |
| |
| EXPECT_EQ(std::string("b"), *new_string->value->spans[3].name); |
| EXPECT_EQ(std::u16string(u"[Ţĥîš šéñţéñçé îš ñöţ ŵĥåţ ýöû ţĥîñķ îţ îš åţ").size(), |
| new_string->value->spans[3].first_char); |
| EXPECT_EQ(std::u16string(u"[Ţĥîš šéñţéñçé îš ñöţ ŵĥåţ ýöû ţĥîñķ îţ îš åţ åļ").size(), |
| new_string->value->spans[3].last_char); |
| } |
| |
| TEST(PseudolocaleGeneratorTest, PseudolocalizePartsOfString) { |
| StringPool pool; |
| StyleString original_style; |
| original_style.str = "This should NOT be pseudolocalized."; |
| original_style.spans = {Span{"em", 4u, 14u}, Span{"i", 18u, 33u}}; |
| std::unique_ptr<StyledString> original_string = |
| util::make_unique<StyledString>(pool.MakeRef(original_style)); |
| original_string->untranslatable_sections = {UntranslatableSection{11u, 15u}}; |
| |
| std::unique_ptr<StyledString> new_string = |
| PseudolocalizeStyledString(original_string.get(), Pseudolocalizer::Method::kAccent, &pool); |
| ASSERT_NE(nullptr, new_string); |
| ASSERT_EQ(2u, new_string->value->spans.size()); |
| EXPECT_EQ(std::string("[Ţĥîš šĥöûļð NOT ɓé þšéûðöļöçåļîžéð. one two three four]"), |
| new_string->value->value); |
| |
| EXPECT_EQ(std::string("em"), *new_string->value->spans[0].name); |
| EXPECT_EQ(std::u16string(u"[Ţĥîš").size(), new_string->value->spans[0].first_char); |
| EXPECT_EQ(std::u16string(u"[Ţĥîš šĥöûļð NO").size(), new_string->value->spans[0].last_char); |
| |
| EXPECT_EQ(std::string("i"), *new_string->value->spans[1].name); |
| EXPECT_EQ(std::u16string(u"[Ţĥîš šĥöûļð NOT ɓé").size(), new_string->value->spans[1].first_char); |
| EXPECT_EQ(std::u16string(u"[Ţĥîš šĥöûļð NOT ɓé þšéûðöļöçåļîžé").size(), |
| new_string->value->spans[1].last_char); |
| } |
| |
| TEST(PseudolocaleGeneratorTest, PseudolocalizeOnlyDefaultConfigs) { |
| std::unique_ptr<ResourceTable> table = |
| test::ResourceTableBuilder() |
| .AddString("android:string/one", "one") |
| .AddString("android:string/two", ResourceId{}, |
| test::ParseConfigOrDie("en"), "two") |
| .AddString("android:string/three", "three") |
| .AddString("android:string/three", ResourceId{}, |
| test::ParseConfigOrDie("en-rXA"), "three") |
| .AddString("android:string/four", "four") |
| .Build(); |
| |
| String* val = test::GetValue<String>(table.get(), "android:string/four"); |
| ASSERT_NE(nullptr, val); |
| val->SetTranslatable(false); |
| |
| std::unique_ptr<IAaptContext> context = test::ContextBuilder().Build(); |
| PseudolocaleGenerator generator; |
| ASSERT_TRUE(generator.Consume(context.get(), table.get())); |
| |
| // Normal pseudolocalization should take place. |
| ASSERT_NE(nullptr, |
| test::GetValueForConfig<String>(table.get(), "android:string/one", |
| test::ParseConfigOrDie("en-rXA"))); |
| ASSERT_NE(nullptr, |
| test::GetValueForConfig<String>(table.get(), "android:string/one", |
| test::ParseConfigOrDie("ar-rXB"))); |
| |
| // No default config for android:string/two, so no pseudlocales should exist. |
| ASSERT_EQ(nullptr, |
| test::GetValueForConfig<String>(table.get(), "android:string/two", |
| test::ParseConfigOrDie("en-rXA"))); |
| ASSERT_EQ(nullptr, |
| test::GetValueForConfig<String>(table.get(), "android:string/two", |
| test::ParseConfigOrDie("ar-rXB"))); |
| |
| // Check that we didn't override manual pseudolocalization. |
| val = test::GetValueForConfig<String>(table.get(), "android:string/three", |
| test::ParseConfigOrDie("en-rXA")); |
| ASSERT_NE(nullptr, val); |
| EXPECT_EQ(std::string("three"), *val->value); |
| |
| ASSERT_NE(nullptr, |
| test::GetValueForConfig<String>(table.get(), "android:string/three", |
| test::ParseConfigOrDie("ar-rXB"))); |
| |
| // Check that four's translateable marker was honored. |
| ASSERT_EQ(nullptr, |
| test::GetValueForConfig<String>(table.get(), "android:string/four", |
| test::ParseConfigOrDie("en-rXA"))); |
| ASSERT_EQ(nullptr, |
| test::GetValueForConfig<String>(table.get(), "android:string/four", |
| test::ParseConfigOrDie("ar-rXB"))); |
| } |
| |
| TEST(PseudolocaleGeneratorTest, PluralsArePseudolocalized) { |
| std::unique_ptr<IAaptContext> context = test::ContextBuilder().Build(); |
| std::unique_ptr<ResourceTable> table = |
| test::ResourceTableBuilder().SetPackageId("com.pkg", 0x7F).Build(); |
| std::unique_ptr<Plural> plural = util::make_unique<Plural>(); |
| plural->values = {util::make_unique<String>(table->string_pool.MakeRef("zero")), |
| util::make_unique<String>(table->string_pool.MakeRef("one"))}; |
| ASSERT_TRUE(table->AddResource(test::ParseNameOrDie("com.pkg:plurals/foo"), ConfigDescription{}, |
| {}, std::move(plural), context->GetDiagnostics())); |
| std::unique_ptr<Plural> expected = util::make_unique<Plural>(); |
| expected->values = {util::make_unique<String>(table->string_pool.MakeRef("[žéŕö one]")), |
| util::make_unique<String>(table->string_pool.MakeRef("[öñé one]"))}; |
| |
| PseudolocaleGenerator generator; |
| ASSERT_TRUE(generator.Consume(context.get(), table.get())); |
| |
| const auto* actual = test::GetValueForConfig<Plural>(table.get(), "com.pkg:plurals/foo", |
| test::ParseConfigOrDie("en-rXA")); |
| EXPECT_TRUE(actual->Equals(expected.get())); |
| } |
| |
| TEST(PseudolocaleGeneratorTest, RespectUntranslateableSections) { |
| std::unique_ptr<IAaptContext> context = |
| test::ContextBuilder().SetCompilationPackage("android").Build(); |
| std::unique_ptr<ResourceTable> table = util::make_unique<ResourceTable>(); |
| |
| { |
| StyleString original_style; |
| original_style.str = "Hello world!"; |
| original_style.spans = {Span{"i", 1, 10}, Span{"b", 2, 3}, Span{"b", 6, 7}}; |
| |
| auto styled_string = |
| util::make_unique<StyledString>(table->string_pool.MakeRef(original_style)); |
| styled_string->untranslatable_sections.push_back(UntranslatableSection{6u, 8u}); |
| styled_string->untranslatable_sections.push_back(UntranslatableSection{8u, 11u}); |
| |
| auto string = util::make_unique<String>(table->string_pool.MakeRef(original_style.str)); |
| string->untranslatable_sections.push_back(UntranslatableSection{6u, 11u}); |
| |
| ASSERT_TRUE(table->AddResource(test::ParseNameOrDie("android:string/foo"), ConfigDescription{}, |
| {} /* product */, std::move(styled_string), |
| context->GetDiagnostics())); |
| ASSERT_TRUE(table->AddResource(test::ParseNameOrDie("android:string/bar"), ConfigDescription{}, |
| {} /* product */, std::move(string), context->GetDiagnostics())); |
| } |
| |
| PseudolocaleGenerator generator; |
| ASSERT_TRUE(generator.Consume(context.get(), table.get())); |
| |
| StyledString* new_styled_string = test::GetValueForConfig<StyledString>( |
| table.get(), "android:string/foo", test::ParseConfigOrDie("en-rXA")); |
| ASSERT_NE(nullptr, new_styled_string); |
| |
| // "world" should be untranslated. |
| EXPECT_NE(std::string::npos, new_styled_string->value->value.find("world")); |
| |
| String* new_string = test::GetValueForConfig<String>(table.get(), "android:string/bar", |
| test::ParseConfigOrDie("en-rXA")); |
| ASSERT_NE(nullptr, new_string); |
| |
| // "world" should be untranslated. |
| EXPECT_NE(std::string::npos, new_string->value->find("world")); |
| } |
| |
| } // namespace aapt |