326e35ffaf
AAPT2 Macros are compile-time resources definitions that are expanded when referenced during the link phase. A macro must be defined in the res/values.xml directory. A macro definition for a macro named "foo" looks like the following: <macro name="foo">contents</macro> When "@macro/foo" is used in the res/values directory or in a compiled XML file, the contents of the macro replace the macro reference and then the substituted contents are compiled and linked. If the macro contents reference xml namespaces from its original definition, the namespaces of the original macro definition will be used to determine which package is being referenced. Macros can be used anywhere resources can be referenced using the @package:type/entry syntax. Macros are not included in the final resource table or the R.java since they are not actual resources. Bug: 175616308 Test: aapt2_tests Change-Id: I48b29ab6564357b32b4b4e32bff7ef06036382bc
1420 lines
53 KiB
C++
1420 lines
53 KiB
C++
/*
|
||
* Copyright (C) 2015 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 "ResourceParser.h"
|
||
|
||
#include <sstream>
|
||
#include <string>
|
||
|
||
#include "ResourceTable.h"
|
||
#include "ResourceUtils.h"
|
||
#include "ResourceValues.h"
|
||
#include "io/StringStream.h"
|
||
#include "test/Test.h"
|
||
#include "xml/XmlPullParser.h"
|
||
|
||
using ::aapt::io::StringInputStream;
|
||
using ::aapt::test::StrValueEq;
|
||
using ::aapt::test::ValueEq;
|
||
using ::android::ConfigDescription;
|
||
using ::android::Res_value;
|
||
using ::android::ResTable_map;
|
||
using ::android::StringPiece;
|
||
using ::testing::Eq;
|
||
using ::testing::IsEmpty;
|
||
using ::testing::IsNull;
|
||
using ::testing::NotNull;
|
||
using ::testing::Pointee;
|
||
using ::testing::SizeIs;
|
||
using ::testing::StrEq;
|
||
|
||
using PolicyFlags = android::ResTable_overlayable_policy_header::PolicyFlags;
|
||
|
||
namespace aapt {
|
||
|
||
constexpr const char* kXmlPreamble = "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n";
|
||
|
||
TEST(ResourceParserSingleTest, FailToParseWithNoRootResourcesElement) {
|
||
std::unique_ptr<IAaptContext> context = test::ContextBuilder().Build();
|
||
ResourceTable table;
|
||
ResourceParser parser(context->GetDiagnostics(), &table, Source{"test"}, {});
|
||
|
||
std::string input = kXmlPreamble;
|
||
input += R"(<attr name="foo"/>)";
|
||
StringInputStream in(input);
|
||
xml::XmlPullParser xml_parser(&in);
|
||
ASSERT_FALSE(parser.Parse(&xml_parser));
|
||
}
|
||
|
||
class ResourceParserTest : public ::testing::Test {
|
||
public:
|
||
void SetUp() override {
|
||
context_ = test::ContextBuilder().Build();
|
||
}
|
||
|
||
::testing::AssertionResult TestParse(const StringPiece& str) {
|
||
return TestParse(str, ConfigDescription{});
|
||
}
|
||
|
||
::testing::AssertionResult TestParse(const StringPiece& str, const ConfigDescription& config) {
|
||
ResourceParserOptions parserOptions;
|
||
ResourceParser parser(context_->GetDiagnostics(), &table_, Source{"test"}, config,
|
||
parserOptions);
|
||
|
||
std::string input = kXmlPreamble;
|
||
input += "<resources>\n";
|
||
input.append(str.data(), str.size());
|
||
input += "\n</resources>";
|
||
StringInputStream in(input);
|
||
xml::XmlPullParser xmlParser(&in);
|
||
if (parser.Parse(&xmlParser)) {
|
||
return ::testing::AssertionSuccess();
|
||
}
|
||
return ::testing::AssertionFailure();
|
||
}
|
||
|
||
protected:
|
||
ResourceTable table_;
|
||
std::unique_ptr<IAaptContext> context_;
|
||
};
|
||
|
||
TEST_F(ResourceParserTest, ParseQuotedString) {
|
||
ASSERT_TRUE(TestParse(R"(<string name="foo"> " hey there " </string>)"));
|
||
|
||
String* str = test::GetValue<String>(&table_, "string/foo");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str, StrValueEq(" hey there "));
|
||
EXPECT_THAT(str->untranslatable_sections, IsEmpty());
|
||
|
||
ASSERT_TRUE(TestParse(R"(<string name="bar">Isn\'t it cool?</string>)"));
|
||
str = test::GetValue<String>(&table_, "string/bar");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str, StrValueEq("Isn't it cool?"));
|
||
|
||
ASSERT_TRUE(TestParse(R"(<string name="baz">"Isn't it cool?"</string>)"));
|
||
str = test::GetValue<String>(&table_, "string/baz");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str, StrValueEq("Isn't it cool?"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseEscapedString) {
|
||
ASSERT_TRUE(TestParse(R"(<string name="foo">\?123</string>)"));
|
||
|
||
String* str = test::GetValue<String>(&table_, "string/foo");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str, StrValueEq("?123"));
|
||
EXPECT_THAT(str->untranslatable_sections, IsEmpty());
|
||
|
||
ASSERT_TRUE(TestParse(R"(<string name="bar">This isn\’t a bad string</string>)"));
|
||
str = test::GetValue<String>(&table_, "string/bar");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str, StrValueEq("This isn’t a bad string"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseFormattedString) {
|
||
ASSERT_FALSE(TestParse(R"(<string name="foo">%d %s</string>)"));
|
||
ASSERT_TRUE(TestParse(R"(<string name="foo">%1$d %2$s</string>)"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseStyledString) {
|
||
// Use a surrogate pair unicode point so that we can verify that the span
|
||
// indices use UTF-16 length and not UTF-8 length.
|
||
std::string input =
|
||
"<string name=\"foo\">This is my aunt\u2019s <b>fickle <small>string</small></b></string>";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
StyledString* str = test::GetValue<StyledString>(&table_, "string/foo");
|
||
ASSERT_THAT(str, NotNull());
|
||
|
||
EXPECT_THAT(str->value->value, StrEq("This is my aunt\u2019s fickle string"));
|
||
EXPECT_THAT(str->value->spans, SizeIs(2));
|
||
EXPECT_THAT(str->untranslatable_sections, IsEmpty());
|
||
|
||
EXPECT_THAT(*str->value->spans[0].name, StrEq("b"));
|
||
EXPECT_THAT(str->value->spans[0].first_char, Eq(18u));
|
||
EXPECT_THAT(str->value->spans[0].last_char, Eq(30u));
|
||
|
||
EXPECT_THAT(*str->value->spans[1].name, StrEq("small"));
|
||
EXPECT_THAT(str->value->spans[1].first_char, Eq(25u));
|
||
EXPECT_THAT(str->value->spans[1].last_char, Eq(30u));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseStringWithWhitespace) {
|
||
ASSERT_TRUE(TestParse(R"(<string name="foo"> This is what I think </string>)"));
|
||
|
||
String* str = test::GetValue<String>(&table_, "string/foo");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str->value, StrEq("This is what I think"));
|
||
EXPECT_THAT(str->untranslatable_sections, IsEmpty());
|
||
|
||
ASSERT_TRUE(TestParse(R"(<string name="foo2">" This is what I think "</string>)"));
|
||
|
||
str = test::GetValue<String>(&table_, "string/foo2");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str, StrValueEq(" This is what I think "));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseStringTruncateASCII) {
|
||
// Tuncate leading and trailing whitespace
|
||
EXPECT_TRUE(TestParse(R"(<string name="foo"> Hello </string>)"));
|
||
|
||
String* str = test::GetValue<String>(&table_, "string/foo");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str->value, StrEq("Hello"));
|
||
EXPECT_THAT(str->untranslatable_sections, IsEmpty());
|
||
|
||
// AAPT does not truncate unicode whitespace
|
||
EXPECT_TRUE(TestParse(R"(<string name="foo2">\u0020\Hello\u0020</string>)"));
|
||
|
||
str = test::GetValue<String>(&table_, "string/foo2");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str->value, StrEq(" Hello "));
|
||
EXPECT_THAT(str->untranslatable_sections, IsEmpty());
|
||
|
||
// Preserve non-ASCII whitespace including extended ASCII characters
|
||
EXPECT_TRUE(TestParse(R"(<string name="foo3"> Hello World </string>)"));
|
||
|
||
str = test::GetValue<String>(&table_, "string/foo3");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str->value, StrEq("\xC2\xA0Hello\xE2\x80\xAFWorld\xC2\xA0"));
|
||
EXPECT_THAT(str->untranslatable_sections, IsEmpty());
|
||
|
||
EXPECT_TRUE(TestParse(R"(<string name="foo4">2005年6月1日</string>)"));
|
||
|
||
str = test::GetValue<String>(&table_, "string/foo4");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str->value, StrEq("2005年6月1日"));
|
||
EXPECT_THAT(str->untranslatable_sections, IsEmpty());
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseStyledStringWithWhitespace) {
|
||
std::string input = R"(<string name="foo"> <b> My <i> favorite</i> string </b> </string>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
StyledString* str = test::GetValue<StyledString>(&table_, "string/foo");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(str->value->value, StrEq(" My favorite string "));
|
||
EXPECT_THAT(str->untranslatable_sections, IsEmpty());
|
||
|
||
ASSERT_THAT(str->value->spans, SizeIs(2u));
|
||
EXPECT_THAT(*str->value->spans[0].name, StrEq("b"));
|
||
EXPECT_THAT(str->value->spans[0].first_char, Eq(1u));
|
||
EXPECT_THAT(str->value->spans[0].last_char, Eq(21u));
|
||
|
||
EXPECT_THAT(*str->value->spans[1].name, StrEq("i"));
|
||
EXPECT_THAT(str->value->spans[1].first_char, Eq(5u));
|
||
EXPECT_THAT(str->value->spans[1].last_char, Eq(13u));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseStringTranslatableAttribute) {
|
||
// If there is no translate attribute the default is 'true'
|
||
EXPECT_TRUE(TestParse(R"(<string name="foo1">Translate</string>)"));
|
||
String* str = test::GetValue<String>(&table_, "string/foo1");
|
||
ASSERT_THAT(str, NotNull());
|
||
ASSERT_TRUE(str->IsTranslatable());
|
||
|
||
// Explicit 'true' translate attribute
|
||
EXPECT_TRUE(TestParse(R"(<string name="foo2" translatable="true">Translate</string>)"));
|
||
str = test::GetValue<String>(&table_, "string/foo2");
|
||
ASSERT_THAT(str, NotNull());
|
||
ASSERT_TRUE(str->IsTranslatable());
|
||
|
||
// Explicit 'false' translate attribute
|
||
EXPECT_TRUE(TestParse(R"(<string name="foo3" translatable="false">Do not translate</string>)"));
|
||
str = test::GetValue<String>(&table_, "string/foo3");
|
||
ASSERT_THAT(str, NotNull());
|
||
ASSERT_FALSE(str->IsTranslatable());
|
||
|
||
// Invalid value for the translate attribute, should be boolean ('true' or 'false')
|
||
EXPECT_FALSE(TestParse(R"(<string name="foo4" translatable="yes">Translate</string>)"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, IgnoreXliffTagsOtherThanG) {
|
||
std::string input = R"(
|
||
<string name="foo" xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
||
There are <xliff:source>no</xliff:source> apples</string>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
String* str = test::GetValue<String>(&table_, "string/foo");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str, StrValueEq("There are no apples"));
|
||
EXPECT_THAT(str->untranslatable_sections, IsEmpty());
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, NestedXliffGTagsAreIllegal) {
|
||
std::string input = R"(
|
||
<string name="foo" xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
||
Do not <xliff:g>translate <xliff:g>this</xliff:g></xliff:g></string>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, RecordUntranslateableXliffSectionsInString) {
|
||
std::string input = R"(
|
||
<string name="foo" xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
||
There are <xliff:g id="count">%1$d</xliff:g> apples</string>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
String* str = test::GetValue<String>(&table_, "string/foo");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str, StrValueEq("There are %1$d apples"));
|
||
|
||
ASSERT_THAT(str->untranslatable_sections, SizeIs(1));
|
||
EXPECT_THAT(str->untranslatable_sections[0].start, Eq(10u));
|
||
EXPECT_THAT(str->untranslatable_sections[0].end, Eq(14u));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, RecordUntranslateableXliffSectionsInStyledString) {
|
||
std::string input = R"(
|
||
<string name="foo" xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
||
There are <b><xliff:g id="count">%1$d</xliff:g></b> apples</string>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
StyledString* str = test::GetValue<StyledString>(&table_, "string/foo");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(str->value->value, Eq(" There are %1$d apples"));
|
||
|
||
ASSERT_THAT(str->untranslatable_sections, SizeIs(1));
|
||
EXPECT_THAT(str->untranslatable_sections[0].start, Eq(11u));
|
||
EXPECT_THAT(str->untranslatable_sections[0].end, Eq(15u));
|
||
|
||
ASSERT_THAT(str->value->spans, SizeIs(1u));
|
||
EXPECT_THAT(*str->value->spans[0].name, StrEq("b"));
|
||
EXPECT_THAT(str->value->spans[0].first_char, Eq(11u));
|
||
EXPECT_THAT(str->value->spans[0].last_char, Eq(14u));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseNull) {
|
||
std::string input = R"(<integer name="foo">@null</integer>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
// The Android runtime treats a value of android::Res_value::TYPE_NULL as
|
||
// a non-existing value, and this causes problems in styles when trying to
|
||
// resolve an attribute. Null values must be encoded as android::Res_value::TYPE_REFERENCE
|
||
// with a data value of 0.
|
||
Reference* null_ref = test::GetValue<Reference>(&table_, "integer/foo");
|
||
ASSERT_THAT(null_ref, NotNull());
|
||
EXPECT_FALSE(null_ref->name);
|
||
EXPECT_FALSE(null_ref->id);
|
||
EXPECT_THAT(null_ref->reference_type, Eq(Reference::Type::kResource));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseEmpty) {
|
||
std::string input = R"(<integer name="foo">@empty</integer>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
BinaryPrimitive* integer = test::GetValue<BinaryPrimitive>(&table_, "integer/foo");
|
||
ASSERT_THAT(integer, NotNull());
|
||
EXPECT_THAT(integer->value.dataType, Eq(Res_value::TYPE_NULL));
|
||
EXPECT_THAT(integer->value.data, Eq(Res_value::DATA_NULL_EMPTY));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseAttr) {
|
||
std::string input = R"(
|
||
<attr name="foo" format="string"/>
|
||
<attr name="bar"/>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Attribute* attr = test::GetValue<Attribute>(&table_, "attr/foo");
|
||
ASSERT_THAT(attr, NotNull());
|
||
EXPECT_THAT(attr->type_mask, Eq(ResTable_map::TYPE_STRING));
|
||
|
||
attr = test::GetValue<Attribute>(&table_, "attr/bar");
|
||
ASSERT_THAT(attr, NotNull());
|
||
EXPECT_THAT(attr->type_mask, Eq(ResTable_map::TYPE_ANY));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseMacro) {
|
||
std::string input = R"(<macro name="foo">12345</macro>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Macro* macro = test::GetValue<Macro>(&table_, "macro/foo");
|
||
ASSERT_THAT(macro, NotNull());
|
||
EXPECT_THAT(macro->raw_value, Eq("12345"));
|
||
EXPECT_THAT(macro->style_string.str, Eq("12345"));
|
||
EXPECT_THAT(macro->style_string.spans, IsEmpty());
|
||
EXPECT_THAT(macro->untranslatable_sections, IsEmpty());
|
||
EXPECT_THAT(macro->alias_namespaces, IsEmpty());
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseMacroUntranslatableSection) {
|
||
std::string input = R"(<macro name="foo" xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
|
||
This being <b><xliff:g>human</xliff:g></b> is a guest house.</macro>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Macro* macro = test::GetValue<Macro>(&table_, "macro/foo");
|
||
ASSERT_THAT(macro, NotNull());
|
||
EXPECT_THAT(macro->raw_value, Eq("\nThis being human is a guest house."));
|
||
EXPECT_THAT(macro->style_string.str, Eq(" This being human is a guest house."));
|
||
EXPECT_THAT(macro->style_string.spans.size(), Eq(1));
|
||
EXPECT_THAT(macro->style_string.spans[0].name, Eq("b"));
|
||
EXPECT_THAT(macro->style_string.spans[0].first_char, Eq(12));
|
||
EXPECT_THAT(macro->style_string.spans[0].last_char, Eq(16));
|
||
ASSERT_THAT(macro->untranslatable_sections.size(), Eq(1));
|
||
EXPECT_THAT(macro->untranslatable_sections[0].start, Eq(12));
|
||
EXPECT_THAT(macro->untranslatable_sections[0].end, Eq(17));
|
||
EXPECT_THAT(macro->alias_namespaces, IsEmpty());
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseMacroNamespaces) {
|
||
std::string input = R"(<macro name="foo" xmlns:app="http://schemas.android.com/apk/res/android">
|
||
@app:string/foo</macro>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Macro* macro = test::GetValue<Macro>(&table_, "macro/foo");
|
||
ASSERT_THAT(macro, NotNull());
|
||
EXPECT_THAT(macro->raw_value, Eq("\n@app:string/foo"));
|
||
EXPECT_THAT(macro->style_string.str, Eq("@app:string/foo"));
|
||
EXPECT_THAT(macro->style_string.spans, IsEmpty());
|
||
EXPECT_THAT(macro->untranslatable_sections, IsEmpty());
|
||
EXPECT_THAT(macro->alias_namespaces.size(), Eq(1));
|
||
EXPECT_THAT(macro->alias_namespaces[0].alias, Eq("app"));
|
||
EXPECT_THAT(macro->alias_namespaces[0].package_name, Eq("android"));
|
||
EXPECT_THAT(macro->alias_namespaces[0].is_private, Eq(false));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseMacroReference) {
|
||
std::string input = R"(<string name="res_string">@macro/foo</string>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Reference* macro = test::GetValue<Reference>(&table_, "string/res_string");
|
||
ASSERT_THAT(macro, NotNull());
|
||
EXPECT_THAT(macro->type_flags, Eq(ResTable_map::TYPE_STRING));
|
||
EXPECT_THAT(macro->allow_raw, Eq(false));
|
||
|
||
input = R"(<style name="foo">
|
||
<item name="bar">@macro/foo</item>
|
||
</style>)";
|
||
|
||
ASSERT_TRUE(TestParse(input));
|
||
Style* style = test::GetValue<Style>(&table_, "style/foo");
|
||
ASSERT_THAT(style, NotNull());
|
||
EXPECT_THAT(style->entries.size(), Eq(1));
|
||
|
||
macro = ValueCast<Reference>(style->entries[0].value.get());
|
||
ASSERT_THAT(macro, NotNull());
|
||
EXPECT_THAT(macro->type_flags, Eq(0U));
|
||
EXPECT_THAT(macro->allow_raw, Eq(true));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseMacroNoNameFail) {
|
||
std::string input = R"(<macro>12345</macro>)";
|
||
ASSERT_FALSE(TestParse(input));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseMacroNonDefaultConfigurationFail) {
|
||
const ConfigDescription watch_config = test::ParseConfigOrDie("watch");
|
||
std::string input = R"(<macro name="foo">12345</macro>)";
|
||
ASSERT_FALSE(TestParse(input, watch_config));
|
||
}
|
||
|
||
// Old AAPT allowed attributes to be defined under different configurations, but ultimately
|
||
// stored them with the default configuration. Check that we have the same behavior.
|
||
TEST_F(ResourceParserTest, ParseAttrAndDeclareStyleableUnderConfigButRecordAsNoConfig) {
|
||
const ConfigDescription watch_config = test::ParseConfigOrDie("watch");
|
||
std::string input = R"(
|
||
<attr name="foo" />
|
||
<declare-styleable name="bar">
|
||
<attr name="baz" format="reference"/>
|
||
</declare-styleable>)";
|
||
ASSERT_TRUE(TestParse(input, watch_config));
|
||
|
||
EXPECT_THAT(test::GetValueForConfig<Attribute>(&table_, "attr/foo", watch_config), IsNull());
|
||
EXPECT_THAT(test::GetValueForConfig<Attribute>(&table_, "attr/baz", watch_config), IsNull());
|
||
EXPECT_THAT(test::GetValueForConfig<Styleable>(&table_, "styleable/bar", watch_config), IsNull());
|
||
|
||
EXPECT_THAT(test::GetValue<Attribute>(&table_, "attr/foo"), NotNull());
|
||
EXPECT_THAT(test::GetValue<Attribute>(&table_, "attr/baz"), NotNull());
|
||
EXPECT_THAT(test::GetValue<Styleable>(&table_, "styleable/bar"), NotNull());
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseAttrWithMinMax) {
|
||
std::string input = R"(<attr name="foo" min="10" max="23" format="integer"/>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Attribute* attr = test::GetValue<Attribute>(&table_, "attr/foo");
|
||
ASSERT_THAT(attr, NotNull());
|
||
EXPECT_THAT(attr->type_mask, Eq(ResTable_map::TYPE_INTEGER));
|
||
EXPECT_THAT(attr->min_int, Eq(10));
|
||
EXPECT_THAT(attr->max_int, Eq(23));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, FailParseAttrWithMinMaxButNotInteger) {
|
||
ASSERT_FALSE(TestParse(R"(<attr name="foo" min="10" max="23" format="string"/>)"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseUseAndDeclOfAttr) {
|
||
std::string input = R"(
|
||
<declare-styleable name="Styleable">
|
||
<attr name="foo" />
|
||
</declare-styleable>
|
||
<attr name="foo" format="string"/>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Attribute* attr = test::GetValue<Attribute>(&table_, "attr/foo");
|
||
ASSERT_THAT(attr, NotNull());
|
||
EXPECT_THAT(attr->type_mask, Eq(ResTable_map::TYPE_STRING));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseDoubleUseOfAttr) {
|
||
std::string input = R"(
|
||
<declare-styleable name="Theme">
|
||
<attr name="foo" />
|
||
</declare-styleable>
|
||
<declare-styleable name="Window">
|
||
<attr name="foo" format="boolean"/>
|
||
</declare-styleable>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Attribute* attr = test::GetValue<Attribute>(&table_, "attr/foo");
|
||
ASSERT_THAT(attr, NotNull());
|
||
EXPECT_THAT(attr->type_mask, Eq(ResTable_map::TYPE_BOOLEAN));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseEnumAttr) {
|
||
std::string input = R"(
|
||
<attr name="foo">
|
||
<enum name="bar" value="0"/>
|
||
<enum name="bat" value="0x1"/>
|
||
<enum name="baz" value="2"/>
|
||
</attr>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Attribute* enum_attr = test::GetValue<Attribute>(&table_, "attr/foo");
|
||
ASSERT_THAT(enum_attr, NotNull());
|
||
EXPECT_THAT(enum_attr->type_mask, Eq(ResTable_map::TYPE_ENUM));
|
||
ASSERT_THAT(enum_attr->symbols, SizeIs(3));
|
||
|
||
ASSERT_TRUE(enum_attr->symbols[0].symbol.name);
|
||
EXPECT_THAT(enum_attr->symbols[0].symbol.name.value().entry, Eq("bar"));
|
||
EXPECT_THAT(enum_attr->symbols[0].value, Eq(0u));
|
||
EXPECT_THAT(enum_attr->symbols[0].type, Eq(Res_value::TYPE_INT_DEC));
|
||
|
||
ASSERT_TRUE(enum_attr->symbols[1].symbol.name);
|
||
EXPECT_THAT(enum_attr->symbols[1].symbol.name.value().entry, Eq("bat"));
|
||
EXPECT_THAT(enum_attr->symbols[1].value, Eq(1u));
|
||
EXPECT_THAT(enum_attr->symbols[1].type, Eq(Res_value::TYPE_INT_HEX));
|
||
|
||
ASSERT_TRUE(enum_attr->symbols[2].symbol.name);
|
||
EXPECT_THAT(enum_attr->symbols[2].symbol.name.value().entry, Eq("baz"));
|
||
EXPECT_THAT(enum_attr->symbols[2].value, Eq(2u));
|
||
EXPECT_THAT(enum_attr->symbols[2].type, Eq(Res_value::TYPE_INT_DEC));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseFlagAttr) {
|
||
std::string input = R"(
|
||
<attr name="foo">
|
||
<flag name="bar" value="0"/>
|
||
<flag name="bat" value="1"/>
|
||
<flag name="baz" value="2"/>
|
||
</attr>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Attribute* flag_attr = test::GetValue<Attribute>(&table_, "attr/foo");
|
||
ASSERT_THAT(flag_attr, NotNull());
|
||
EXPECT_THAT(flag_attr->type_mask, Eq(ResTable_map::TYPE_FLAGS));
|
||
ASSERT_THAT(flag_attr->symbols, SizeIs(3));
|
||
|
||
ASSERT_TRUE(flag_attr->symbols[0].symbol.name);
|
||
EXPECT_THAT(flag_attr->symbols[0].symbol.name.value().entry, Eq("bar"));
|
||
EXPECT_THAT(flag_attr->symbols[0].value, Eq(0u));
|
||
|
||
ASSERT_TRUE(flag_attr->symbols[1].symbol.name);
|
||
EXPECT_THAT(flag_attr->symbols[1].symbol.name.value().entry, Eq("bat"));
|
||
EXPECT_THAT(flag_attr->symbols[1].value, Eq(1u));
|
||
|
||
ASSERT_TRUE(flag_attr->symbols[2].symbol.name);
|
||
EXPECT_THAT(flag_attr->symbols[2].symbol.name.value().entry, Eq("baz"));
|
||
EXPECT_THAT(flag_attr->symbols[2].value, Eq(2u));
|
||
|
||
std::unique_ptr<BinaryPrimitive> flag_value =
|
||
ResourceUtils::TryParseFlagSymbol(flag_attr, "baz|bat");
|
||
ASSERT_THAT(flag_value, NotNull());
|
||
EXPECT_THAT(flag_value->value.data, Eq(1u | 2u));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, FailToParseEnumAttrWithNonUniqueKeys) {
|
||
std::string input = R"(
|
||
<attr name="foo">
|
||
<enum name="bar" value="0"/>
|
||
<enum name="bat" value="1"/>
|
||
<enum name="bat" value="2"/>
|
||
</attr>)";
|
||
ASSERT_FALSE(TestParse(input));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseStyle) {
|
||
std::string input = R"(
|
||
<style name="foo" parent="@style/fu">
|
||
<item name="bar">#ffffffff</item>
|
||
<item name="bat">@string/hey</item>
|
||
<item name="baz"><b>hey</b></item>
|
||
</style>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Style* style = test::GetValue<Style>(&table_, "style/foo");
|
||
ASSERT_THAT(style, NotNull());
|
||
ASSERT_TRUE(style->parent);
|
||
EXPECT_THAT(style->parent.value().name, Eq(make_value(test::ParseNameOrDie("style/fu"))));
|
||
ASSERT_THAT(style->entries, SizeIs(3));
|
||
|
||
EXPECT_THAT(style->entries[0].key.name, Eq(make_value(test::ParseNameOrDie("attr/bar"))));
|
||
EXPECT_THAT(style->entries[1].key.name, Eq(make_value(test::ParseNameOrDie("attr/bat"))));
|
||
EXPECT_THAT(style->entries[2].key.name, Eq(make_value(test::ParseNameOrDie("attr/baz"))));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseStyleWithShorthandParent) {
|
||
ASSERT_TRUE(TestParse(R"(<style name="foo" parent="com.app:Theme"/>)"));
|
||
|
||
Style* style = test::GetValue<Style>(&table_, "style/foo");
|
||
ASSERT_THAT(style, NotNull());
|
||
ASSERT_TRUE(style->parent);
|
||
EXPECT_THAT(style->parent.value().name, Eq(make_value(test::ParseNameOrDie("com.app:style/Theme"))));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseStyleWithPackageAliasedParent) {
|
||
std::string input = R"(
|
||
<style xmlns:app="http://schemas.android.com/apk/res/android"
|
||
name="foo" parent="app:Theme"/>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Style* style = test::GetValue<Style>(&table_, "style/foo");
|
||
ASSERT_THAT(style, NotNull());
|
||
ASSERT_TRUE(style->parent);
|
||
ASSERT_TRUE(style->parent.value().name);
|
||
EXPECT_THAT(style->parent.value().name, Eq(make_value(test::ParseNameOrDie("android:style/Theme"))));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseStyleWithPackageAliasedItems) {
|
||
std::string input = R"(
|
||
<style xmlns:app="http://schemas.android.com/apk/res/android" name="foo">
|
||
<item name="app:bar">0</item>
|
||
</style>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Style* style = test::GetValue<Style>(&table_, "style/foo");
|
||
ASSERT_THAT(style, NotNull());
|
||
ASSERT_THAT(style->entries, SizeIs(1));
|
||
EXPECT_THAT(style->entries[0].key.name, Eq(make_value(test::ParseNameOrDie("android:attr/bar"))));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseStyleWithRawStringItem) {
|
||
std::string input = R"(
|
||
<style name="foo">
|
||
<item name="bar">
|
||
com.helloworld.AppClass
|
||
</item>
|
||
</style>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Style* style = test::GetValue<Style>(&table_, "style/foo");
|
||
ASSERT_THAT(style, NotNull());
|
||
EXPECT_THAT(style->entries[0].value, NotNull());
|
||
RawString* value = ValueCast<RawString>(style->entries[0].value.get());
|
||
EXPECT_THAT(value, NotNull());
|
||
EXPECT_THAT(*value->value, StrEq(R"(com.helloworld.AppClass)"));
|
||
}
|
||
|
||
|
||
TEST_F(ResourceParserTest, ParseStyleWithInferredParent) {
|
||
ASSERT_TRUE(TestParse(R"(<style name="foo.bar"/>)"));
|
||
|
||
Style* style = test::GetValue<Style>(&table_, "style/foo.bar");
|
||
ASSERT_THAT(style, NotNull());
|
||
ASSERT_TRUE(style->parent);
|
||
EXPECT_THAT(style->parent.value().name, Eq(make_value(test::ParseNameOrDie("style/foo"))));
|
||
EXPECT_TRUE(style->parent_inferred);
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseStyleWithInferredParentOverridenByEmptyParentAttribute) {
|
||
ASSERT_TRUE(TestParse(R"(<style name="foo.bar" parent=""/>)"));
|
||
|
||
Style* style = test::GetValue<Style>(&table_, "style/foo.bar");
|
||
ASSERT_THAT(style, NotNull());
|
||
EXPECT_FALSE(style->parent);
|
||
EXPECT_FALSE(style->parent_inferred);
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseStyleWithPrivateParentReference) {
|
||
ASSERT_TRUE(TestParse(R"(<style name="foo" parent="*android:style/bar" />)"));
|
||
|
||
Style* style = test::GetValue<Style>(&table_, "style/foo");
|
||
ASSERT_THAT(style, NotNull());
|
||
ASSERT_TRUE(style->parent);
|
||
EXPECT_TRUE(style->parent.value().private_reference);
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseAutoGeneratedIdReference) {
|
||
ASSERT_TRUE(TestParse(R"(<string name="foo">@+id/bar</string>)"));
|
||
ASSERT_THAT(test::GetValue<Id>(&table_, "id/bar"), NotNull());
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseAttributesDeclareStyleable) {
|
||
std::string input = R"(
|
||
<declare-styleable name="foo">
|
||
<attr name="bar" />
|
||
<attr name="bat" format="string|reference"/>
|
||
<attr name="baz">
|
||
<enum name="foo" value="1"/>
|
||
</attr>
|
||
</declare-styleable>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Maybe<ResourceTable::SearchResult> result =
|
||
table_.FindResource(test::ParseNameOrDie("styleable/foo"));
|
||
ASSERT_TRUE(result);
|
||
EXPECT_THAT(result.value().entry->visibility.level, Eq(Visibility::Level::kPublic));
|
||
|
||
Attribute* attr = test::GetValue<Attribute>(&table_, "attr/bar");
|
||
ASSERT_THAT(attr, IsNull());
|
||
|
||
attr = test::GetValue<Attribute>(&table_, "attr/bat");
|
||
ASSERT_THAT(attr, NotNull());
|
||
EXPECT_TRUE(attr->IsWeak());
|
||
|
||
attr = test::GetValue<Attribute>(&table_, "attr/baz");
|
||
ASSERT_THAT(attr, NotNull());
|
||
EXPECT_TRUE(attr->IsWeak());
|
||
EXPECT_THAT(attr->symbols, SizeIs(1));
|
||
|
||
EXPECT_THAT(test::GetValue<Id>(&table_, "id/foo"), NotNull());
|
||
|
||
Styleable* styleable = test::GetValue<Styleable>(&table_, "styleable/foo");
|
||
ASSERT_THAT(styleable, NotNull());
|
||
ASSERT_THAT(styleable->entries, SizeIs(3));
|
||
|
||
EXPECT_THAT(styleable->entries[0].name, Eq(make_value(test::ParseNameOrDie("attr/bar"))));
|
||
EXPECT_THAT(styleable->entries[1].name, Eq(make_value(test::ParseNameOrDie("attr/bat"))));
|
||
EXPECT_THAT(styleable->entries[2].name, Eq(make_value(test::ParseNameOrDie("attr/baz"))));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseDeclareStyleablePreservingVisibility) {
|
||
StringInputStream input(R"(
|
||
<resources>
|
||
<declare-styleable name="foo">
|
||
<attr name="myattr" />
|
||
</declare-styleable>
|
||
<declare-styleable name="bar">
|
||
<attr name="myattr" />
|
||
</declare-styleable>
|
||
<public type="styleable" name="bar" />
|
||
</resources>)");
|
||
ResourceParser parser(context_->GetDiagnostics(), &table_, Source{"test"},
|
||
ConfigDescription::DefaultConfig(),
|
||
ResourceParserOptions{.preserve_visibility_of_styleables = true});
|
||
|
||
xml::XmlPullParser xml_parser(&input);
|
||
ASSERT_TRUE(parser.Parse(&xml_parser));
|
||
|
||
EXPECT_EQ(
|
||
table_.FindResource(test::ParseNameOrDie("styleable/foo")).value().entry->visibility.level,
|
||
Visibility::Level::kUndefined);
|
||
EXPECT_EQ(
|
||
table_.FindResource(test::ParseNameOrDie("styleable/bar")).value().entry->visibility.level,
|
||
Visibility::Level::kPublic);
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParsePrivateAttributesDeclareStyleable) {
|
||
std::string input = R"(
|
||
<declare-styleable xmlns:privAndroid="http://schemas.android.com/apk/prv/res/android"
|
||
name="foo">
|
||
<attr name="*android:bar" />
|
||
<attr name="privAndroid:bat" />
|
||
</declare-styleable>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
Styleable* styleable = test::GetValue<Styleable>(&table_, "styleable/foo");
|
||
ASSERT_THAT(styleable, NotNull());
|
||
ASSERT_THAT(styleable->entries, SizeIs(2));
|
||
|
||
EXPECT_TRUE(styleable->entries[0].private_reference);
|
||
ASSERT_TRUE(styleable->entries[0].name);
|
||
EXPECT_THAT(styleable->entries[0].name.value().package, Eq("android"));
|
||
|
||
EXPECT_TRUE(styleable->entries[1].private_reference);
|
||
ASSERT_TRUE(styleable->entries[1].name);
|
||
EXPECT_THAT(styleable->entries[1].name.value().package, Eq("android"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseArray) {
|
||
std::string input = R"(
|
||
<array name="foo">
|
||
<item>@string/ref</item>
|
||
<item>hey</item>
|
||
<item>23</item>
|
||
</array>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Array* array = test::GetValue<Array>(&table_, "array/foo");
|
||
ASSERT_THAT(array, NotNull());
|
||
ASSERT_THAT(array->elements, SizeIs(3));
|
||
|
||
EXPECT_THAT(ValueCast<Reference>(array->elements[0].get()), NotNull());
|
||
EXPECT_THAT(ValueCast<String>(array->elements[1].get()), NotNull());
|
||
EXPECT_THAT(ValueCast<BinaryPrimitive>(array->elements[2].get()), NotNull());
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseStringArray) {
|
||
std::string input = R"(
|
||
<string-array name="foo">
|
||
<item>"Werk"</item>"
|
||
</string-array>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
EXPECT_THAT(test::GetValue<Array>(&table_, "array/foo"), NotNull());
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseArrayWithFormat) {
|
||
std::string input = R"(
|
||
<array name="foo" format="string">
|
||
<item>100</item>
|
||
</array>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Array* array = test::GetValue<Array>(&table_, "array/foo");
|
||
ASSERT_THAT(array, NotNull());
|
||
ASSERT_THAT(array->elements, SizeIs(1));
|
||
|
||
String* str = ValueCast<String>(array->elements[0].get());
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str, StrValueEq("100"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseArrayWithBadFormat) {
|
||
std::string input = R"(
|
||
<array name="foo" format="integer">
|
||
<item>Hi</item>
|
||
</array>)";
|
||
ASSERT_FALSE(TestParse(input));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParsePlural) {
|
||
std::string input = R"(
|
||
<plurals name="foo">
|
||
<item quantity="other">apples</item>
|
||
<item quantity="one">apple</item>
|
||
</plurals>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Plural* plural = test::GetValue<Plural>(&table_, "plurals/foo");
|
||
ASSERT_THAT(plural, NotNull());
|
||
EXPECT_THAT(plural->values[Plural::Zero], IsNull());
|
||
EXPECT_THAT(plural->values[Plural::Two], IsNull());
|
||
EXPECT_THAT(plural->values[Plural::Few], IsNull());
|
||
EXPECT_THAT(plural->values[Plural::Many], IsNull());
|
||
|
||
EXPECT_THAT(plural->values[Plural::One], NotNull());
|
||
EXPECT_THAT(plural->values[Plural::Other], NotNull());
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseCommentsWithResource) {
|
||
std::string input = R"(
|
||
<!--This is a comment-->
|
||
<string name="foo">Hi</string>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
String* value = test::GetValue<String>(&table_, "string/foo");
|
||
ASSERT_THAT(value, NotNull());
|
||
EXPECT_THAT(value->GetComment(), Eq("This is a comment"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, DoNotCombineMultipleComments) {
|
||
std::string input = R"(
|
||
<!--One-->
|
||
<!--Two-->
|
||
<string name="foo">Hi</string>)";
|
||
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
String* value = test::GetValue<String>(&table_, "string/foo");
|
||
ASSERT_THAT(value, NotNull());
|
||
EXPECT_THAT(value->GetComment(), Eq("Two"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, IgnoreCommentBeforeEndTag) {
|
||
std::string input = R"(
|
||
<!--One-->
|
||
<string name="foo">
|
||
Hi
|
||
<!--Two-->
|
||
</string>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
String* value = test::GetValue<String>(&table_, "string/foo");
|
||
ASSERT_THAT(value, NotNull());
|
||
EXPECT_THAT(value->GetComment(), Eq("One"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseNestedComments) {
|
||
// We only care about declare-styleable and enum/flag attributes because
|
||
// comments from those end up in R.java
|
||
std::string input = R"(
|
||
<declare-styleable name="foo">
|
||
<!-- The name of the bar -->
|
||
<attr name="barName" format="string|reference" />
|
||
</declare-styleable>
|
||
|
||
<attr name="foo">
|
||
<!-- The very first -->
|
||
<enum name="one" value="1" />
|
||
</attr>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Styleable* styleable = test::GetValue<Styleable>(&table_, "styleable/foo");
|
||
ASSERT_THAT(styleable, NotNull());
|
||
ASSERT_THAT(styleable->entries, SizeIs(1));
|
||
EXPECT_THAT(styleable->entries[0].GetComment(), Eq("The name of the bar"));
|
||
|
||
Attribute* attr = test::GetValue<Attribute>(&table_, "attr/foo");
|
||
ASSERT_THAT(attr, NotNull());
|
||
ASSERT_THAT(attr->symbols, SizeIs(1));
|
||
EXPECT_THAT(attr->symbols[0].symbol.GetComment(), Eq("The very first"));
|
||
}
|
||
|
||
// Declaring an ID as public should not require a separate definition (as an ID has no value).
|
||
TEST_F(ResourceParserTest, ParsePublicIdAsDefinition) {
|
||
ASSERT_TRUE(TestParse(R"(<public type="id" name="foo"/>)"));
|
||
ASSERT_THAT(test::GetValue<Id>(&table_, "id/foo"), NotNull());
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, KeepAllProducts) {
|
||
std::string input = R"(
|
||
<string name="foo" product="phone">hi</string>
|
||
<string name="foo" product="no-sdcard">ho</string>
|
||
<string name="bar" product="">wee</string>
|
||
<string name="baz">woo</string>
|
||
<string name="bit" product="phablet">hoot</string>
|
||
<string name="bot" product="default">yes</string>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
ASSERT_THAT(test::GetValueForConfigAndProduct<String>(&table_, "string/foo", ConfigDescription::DefaultConfig(), "phone"), NotNull());
|
||
ASSERT_THAT(test::GetValueForConfigAndProduct<String>(&table_, "string/foo",ConfigDescription::DefaultConfig(), "no-sdcard"), NotNull());
|
||
ASSERT_THAT(test::GetValueForConfigAndProduct<String>(&table_, "string/bar", ConfigDescription::DefaultConfig(), ""), NotNull());
|
||
ASSERT_THAT(test::GetValueForConfigAndProduct<String>(&table_, "string/baz", ConfigDescription::DefaultConfig(), ""), NotNull());
|
||
ASSERT_THAT(test::GetValueForConfigAndProduct<String>(&table_, "string/bit", ConfigDescription::DefaultConfig(), "phablet"), NotNull());
|
||
ASSERT_THAT(test::GetValueForConfigAndProduct<String>(&table_, "string/bot", ConfigDescription::DefaultConfig(), "default"), NotNull());
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, AutoIncrementIdsInPublicGroup) {
|
||
std::string input = R"(
|
||
<public-group type="attr" first-id="0x01010040">
|
||
<public name="foo" />
|
||
<public name="bar" />
|
||
</public-group>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Maybe<ResourceTable::SearchResult> result = table_.FindResource(test::ParseNameOrDie("attr/foo"));
|
||
ASSERT_TRUE(result);
|
||
ASSERT_TRUE(result.value().entry->id);
|
||
EXPECT_THAT(result.value().entry->id.value(), Eq(ResourceId(0x01010040)));
|
||
|
||
result = table_.FindResource(test::ParseNameOrDie("attr/bar"));
|
||
ASSERT_TRUE(result);
|
||
ASSERT_TRUE(result.value().entry->id);
|
||
EXPECT_THAT(result.value().entry->id.value(), Eq(ResourceId(0x01010041)));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, StagingPublicGroup) {
|
||
std::string input = R"(
|
||
<staging-public-group type="attr" first-id="0x01ff0049">
|
||
<public name="foo" />
|
||
<public name="bar" />
|
||
</staging-public-group>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Maybe<ResourceTable::SearchResult> result = table_.FindResource(test::ParseNameOrDie("attr/foo"));
|
||
ASSERT_TRUE(result);
|
||
|
||
ASSERT_TRUE(result.value().entry->id);
|
||
EXPECT_THAT(result.value().entry->id.value(), Eq(ResourceId(0x01ff0049)));
|
||
EXPECT_THAT(result.value().entry->visibility.level, Eq(Visibility::Level::kPublic));
|
||
EXPECT_TRUE(result.value().entry->visibility.staged_api);
|
||
|
||
result = table_.FindResource(test::ParseNameOrDie("attr/bar"));
|
||
ASSERT_TRUE(result);
|
||
|
||
ASSERT_TRUE(result.value().entry->id);
|
||
EXPECT_THAT(result.value().entry->id.value(), Eq(ResourceId(0x01ff004a)));
|
||
EXPECT_THAT(result.value().entry->visibility.level, Eq(Visibility::Level::kPublic));
|
||
EXPECT_TRUE(result.value().entry->visibility.staged_api);
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, StrongestSymbolVisibilityWins) {
|
||
std::string input = R"(
|
||
<!-- private -->
|
||
<java-symbol type="string" name="foo" />
|
||
<!-- public -->
|
||
<public type="string" name="foo" id="0x01020000" />
|
||
<!-- private2 -->
|
||
<java-symbol type="string" name="foo" />)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Maybe<ResourceTable::SearchResult> result =
|
||
table_.FindResource(test::ParseNameOrDie("string/foo"));
|
||
ASSERT_TRUE(result);
|
||
|
||
ResourceEntry* entry = result.value().entry;
|
||
ASSERT_THAT(entry, NotNull());
|
||
EXPECT_THAT(entry->visibility.level, Eq(Visibility::Level::kPublic));
|
||
EXPECT_THAT(entry->visibility.comment, StrEq("public"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ExternalTypesShouldOnlyBeReferences) {
|
||
ASSERT_TRUE(TestParse(R"(<item type="layout" name="foo">@layout/bar</item>)"));
|
||
ASSERT_FALSE(TestParse(R"(<item type="layout" name="bar">"this is a string"</item>)"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, AddResourcesElementShouldAddEntryWithUndefinedSymbol) {
|
||
ASSERT_TRUE(TestParse(R"(<add-resource name="bar" type="string" />)"));
|
||
|
||
Maybe<ResourceTable::SearchResult> result =
|
||
table_.FindResource(test::ParseNameOrDie("string/bar"));
|
||
ASSERT_TRUE(result);
|
||
const ResourceEntry* entry = result.value().entry;
|
||
ASSERT_THAT(entry, NotNull());
|
||
EXPECT_THAT(entry->visibility.level, Eq(Visibility::Level::kUndefined));
|
||
EXPECT_TRUE(entry->allow_new);
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseItemElementWithFormat) {
|
||
ASSERT_TRUE(TestParse(R"(<item name="foo" type="integer" format="float">0.3</item>)"));
|
||
|
||
BinaryPrimitive* val = test::GetValue<BinaryPrimitive>(&table_, "integer/foo");
|
||
ASSERT_THAT(val, NotNull());
|
||
EXPECT_THAT(val->value.dataType, Eq(Res_value::TYPE_FLOAT));
|
||
|
||
ASSERT_FALSE(TestParse(R"(<item name="bar" type="integer" format="fraction">100</item>)"));
|
||
}
|
||
|
||
// An <item> without a format specifier accepts all types of values.
|
||
TEST_F(ResourceParserTest, ParseItemElementWithoutFormat) {
|
||
ASSERT_TRUE(TestParse(R"(<item name="foo" type="integer">100%p</item>)"));
|
||
|
||
BinaryPrimitive* val = test::GetValue<BinaryPrimitive>(&table_, "integer/foo");
|
||
ASSERT_THAT(val, NotNull());
|
||
EXPECT_THAT(val->value.dataType, Eq(Res_value::TYPE_FRACTION));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseConfigVaryingItem) {
|
||
ASSERT_TRUE(TestParse(R"(<item name="foo" type="configVarying">Hey</item>)"));
|
||
ASSERT_THAT(test::GetValue<String>(&table_, "configVarying/foo"), NotNull());
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseBagElement) {
|
||
std::string input = R"(
|
||
<bag name="bag" type="configVarying">
|
||
<item name="test">Hello!</item>
|
||
</bag>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
Style* val = test::GetValue<Style>(&table_, "configVarying/bag");
|
||
ASSERT_THAT(val, NotNull());
|
||
ASSERT_THAT(val->entries, SizeIs(1));
|
||
|
||
EXPECT_THAT(val->entries[0].key, Eq(Reference(test::ParseNameOrDie("attr/test"))));
|
||
EXPECT_THAT(ValueCast<RawString>(val->entries[0].value.get()), NotNull());
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseElementWithNoValue) {
|
||
std::string input = R"(
|
||
<item type="drawable" format="reference" name="foo" />
|
||
<string name="foo" />)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
ASSERT_THAT(test::GetValue(&table_, "drawable/foo"), Pointee(ValueEq(Reference())));
|
||
|
||
String* str = test::GetValue<String>(&table_, "string/foo");
|
||
ASSERT_THAT(str, NotNull());
|
||
EXPECT_THAT(*str, StrValueEq(""));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParsePlatformIndependentNewline) {
|
||
ASSERT_TRUE(TestParse(R"(<string name="foo">%1$s %n %2$s</string>)"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseOverlayable) {
|
||
std::string input = R"(
|
||
<overlayable name="Name" actor="overlay://theme">
|
||
<policy type="signature">
|
||
<item type="string" name="foo" />
|
||
<item type="drawable" name="bar" />
|
||
</policy>
|
||
</overlayable>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
auto search_result = table_.FindResource(test::ParseNameOrDie("string/foo"));
|
||
ASSERT_TRUE(search_result);
|
||
ASSERT_THAT(search_result.value().entry, NotNull());
|
||
ASSERT_TRUE(search_result.value().entry->overlayable_item);
|
||
OverlayableItem& result_overlayable_item = search_result.value().entry->overlayable_item.value();
|
||
EXPECT_THAT(result_overlayable_item.overlayable->name, Eq("Name"));
|
||
EXPECT_THAT(result_overlayable_item.overlayable->actor, Eq("overlay://theme"));
|
||
EXPECT_THAT(result_overlayable_item.policies, Eq(PolicyFlags::SIGNATURE));
|
||
|
||
search_result = table_.FindResource(test::ParseNameOrDie("drawable/bar"));
|
||
ASSERT_TRUE(search_result);
|
||
ASSERT_THAT(search_result.value().entry, NotNull());
|
||
ASSERT_TRUE(search_result.value().entry->overlayable_item);
|
||
result_overlayable_item = search_result.value().entry->overlayable_item.value();
|
||
EXPECT_THAT(result_overlayable_item.overlayable->name, Eq("Name"));
|
||
EXPECT_THAT(result_overlayable_item.overlayable->actor, Eq("overlay://theme"));
|
||
EXPECT_THAT(result_overlayable_item.policies, Eq(PolicyFlags::SIGNATURE));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseOverlayableRequiresName) {
|
||
EXPECT_FALSE(TestParse(R"(<overlayable actor="overlay://theme" />)"));
|
||
EXPECT_TRUE(TestParse(R"(<overlayable name="Name" />)"));
|
||
EXPECT_TRUE(TestParse(R"(<overlayable name="Name" actor="overlay://theme" />)"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseOverlayableBadActorFail) {
|
||
EXPECT_FALSE(TestParse(R"(<overlayable name="Name" actor="overley://theme" />)"));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseOverlayablePolicy) {
|
||
std::string input = R"(
|
||
<overlayable name="Name">
|
||
<policy type="product">
|
||
<item type="string" name="bar" />
|
||
</policy>
|
||
<policy type="system">
|
||
<item type="string" name="fiz" />
|
||
</policy>
|
||
<policy type="vendor">
|
||
<item type="string" name="fuz" />
|
||
</policy>
|
||
<policy type="public">
|
||
<item type="string" name="faz" />
|
||
</policy>
|
||
<policy type="signature">
|
||
<item type="string" name="foz" />
|
||
</policy>
|
||
<policy type="odm">
|
||
<item type="string" name="biz" />
|
||
</policy>
|
||
<policy type="oem">
|
||
<item type="string" name="buz" />
|
||
</policy>
|
||
<policy type="actor">
|
||
<item type="string" name="actor" />
|
||
</policy>
|
||
</overlayable>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
auto search_result = table_.FindResource(test::ParseNameOrDie("string/bar"));
|
||
ASSERT_TRUE(search_result);
|
||
ASSERT_THAT(search_result.value().entry, NotNull());
|
||
ASSERT_TRUE(search_result.value().entry->overlayable_item);
|
||
OverlayableItem result_overlayable_item = search_result.value().entry->overlayable_item.value();
|
||
EXPECT_THAT(result_overlayable_item.overlayable->name, Eq("Name"));
|
||
EXPECT_THAT(result_overlayable_item.policies, Eq(PolicyFlags::PRODUCT_PARTITION));
|
||
|
||
search_result = table_.FindResource(test::ParseNameOrDie("string/fiz"));
|
||
ASSERT_TRUE(search_result);
|
||
ASSERT_THAT(search_result.value().entry, NotNull());
|
||
ASSERT_TRUE(search_result.value().entry->overlayable_item);
|
||
result_overlayable_item = search_result.value().entry->overlayable_item.value();
|
||
EXPECT_THAT(result_overlayable_item.overlayable->name, Eq("Name"));
|
||
EXPECT_THAT(result_overlayable_item.policies, Eq(PolicyFlags::SYSTEM_PARTITION));
|
||
|
||
search_result = table_.FindResource(test::ParseNameOrDie("string/fuz"));
|
||
ASSERT_TRUE(search_result);
|
||
ASSERT_THAT(search_result.value().entry, NotNull());
|
||
ASSERT_TRUE(search_result.value().entry->overlayable_item);
|
||
result_overlayable_item = search_result.value().entry->overlayable_item.value();
|
||
EXPECT_THAT(result_overlayable_item.overlayable->name, Eq("Name"));
|
||
EXPECT_THAT(result_overlayable_item.policies, Eq(PolicyFlags::VENDOR_PARTITION));
|
||
|
||
search_result = table_.FindResource(test::ParseNameOrDie("string/faz"));
|
||
ASSERT_TRUE(search_result);
|
||
ASSERT_THAT(search_result.value().entry, NotNull());
|
||
ASSERT_TRUE(search_result.value().entry->overlayable_item);
|
||
result_overlayable_item = search_result.value().entry->overlayable_item.value();
|
||
EXPECT_THAT(result_overlayable_item.overlayable->name, Eq("Name"));
|
||
EXPECT_THAT(result_overlayable_item.policies, Eq(PolicyFlags::PUBLIC));
|
||
|
||
search_result = table_.FindResource(test::ParseNameOrDie("string/foz"));
|
||
ASSERT_TRUE(search_result);
|
||
ASSERT_THAT(search_result.value().entry, NotNull());
|
||
ASSERT_TRUE(search_result.value().entry->overlayable_item);
|
||
result_overlayable_item = search_result.value().entry->overlayable_item.value();
|
||
EXPECT_THAT(result_overlayable_item.overlayable->name, Eq("Name"));
|
||
EXPECT_THAT(result_overlayable_item.policies, Eq(PolicyFlags::SIGNATURE));
|
||
|
||
search_result = table_.FindResource(test::ParseNameOrDie("string/biz"));
|
||
ASSERT_TRUE(search_result);
|
||
ASSERT_THAT(search_result.value().entry, NotNull());
|
||
ASSERT_TRUE(search_result.value().entry->overlayable_item);
|
||
result_overlayable_item = search_result.value().entry->overlayable_item.value();
|
||
EXPECT_THAT(result_overlayable_item.overlayable->name, Eq("Name"));
|
||
EXPECT_THAT(result_overlayable_item.policies, Eq(PolicyFlags::ODM_PARTITION));
|
||
|
||
search_result = table_.FindResource(test::ParseNameOrDie("string/buz"));
|
||
ASSERT_TRUE(search_result);
|
||
ASSERT_THAT(search_result.value().entry, NotNull());
|
||
ASSERT_TRUE(search_result.value().entry->overlayable_item);
|
||
result_overlayable_item = search_result.value().entry->overlayable_item.value();
|
||
EXPECT_THAT(result_overlayable_item.overlayable->name, Eq("Name"));
|
||
EXPECT_THAT(result_overlayable_item.policies, Eq(PolicyFlags::OEM_PARTITION));
|
||
|
||
search_result = table_.FindResource(test::ParseNameOrDie("string/actor"));
|
||
ASSERT_TRUE(search_result);
|
||
ASSERT_THAT(search_result.value().entry, NotNull());
|
||
ASSERT_TRUE(search_result.value().entry->overlayable_item);
|
||
result_overlayable_item = search_result.value().entry->overlayable_item.value();
|
||
EXPECT_THAT(result_overlayable_item.overlayable->name, Eq("Name"));
|
||
EXPECT_THAT(result_overlayable_item.policies, Eq(PolicyFlags::ACTOR_SIGNATURE));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseOverlayableNoPolicyError) {
|
||
std::string input = R"(
|
||
<overlayable name="Name">
|
||
<item type="string" name="foo" />
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
|
||
input = R"(
|
||
<overlayable name="Name">
|
||
<policy>
|
||
<item name="foo" />
|
||
</policy>
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseOverlayableBadPolicyError) {
|
||
std::string input = R"(
|
||
<overlayable name="Name">
|
||
<policy type="illegal_policy">
|
||
<item type="string" name="foo" />
|
||
</policy>
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
|
||
input = R"(
|
||
<overlayable name="Name">
|
||
<policy type="product">
|
||
<item name="foo" />
|
||
</policy>
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
|
||
input = R"(
|
||
<overlayable name="Name">
|
||
<policy type="vendor">
|
||
<item type="string" />
|
||
</policy>
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseOverlayableMultiplePolicy) {
|
||
std::string input = R"(
|
||
<overlayable name="Name">
|
||
<policy type="vendor|public">
|
||
<item type="string" name="foo" />
|
||
</policy>
|
||
<policy type="product|system">
|
||
<item type="string" name="bar" />
|
||
</policy>
|
||
</overlayable>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
auto search_result = table_.FindResource(test::ParseNameOrDie("string/foo"));
|
||
ASSERT_TRUE(search_result);
|
||
ASSERT_THAT(search_result.value().entry, NotNull());
|
||
ASSERT_TRUE(search_result.value().entry->overlayable_item);
|
||
OverlayableItem result_overlayable_item = search_result.value().entry->overlayable_item.value();
|
||
EXPECT_THAT(result_overlayable_item.overlayable->name, Eq("Name"));
|
||
EXPECT_THAT(result_overlayable_item.policies, Eq(PolicyFlags::VENDOR_PARTITION
|
||
| PolicyFlags::PUBLIC));
|
||
|
||
search_result = table_.FindResource(test::ParseNameOrDie("string/bar"));
|
||
ASSERT_TRUE(search_result);
|
||
ASSERT_THAT(search_result.value().entry, NotNull());
|
||
ASSERT_TRUE(search_result.value().entry->overlayable_item);
|
||
result_overlayable_item = search_result.value().entry->overlayable_item.value();
|
||
EXPECT_THAT(result_overlayable_item.overlayable->name, Eq("Name"));
|
||
EXPECT_THAT(result_overlayable_item.policies, Eq(PolicyFlags::PRODUCT_PARTITION
|
||
| PolicyFlags::SYSTEM_PARTITION));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, DuplicateOverlayableIsError) {
|
||
std::string input = R"(
|
||
<overlayable name="Name">
|
||
<item type="string" name="foo" />
|
||
<item type="string" name="foo" />
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
|
||
input = R"(
|
||
<overlayable name="Name">
|
||
<item type="string" name="foo" />
|
||
</overlayable>
|
||
<overlayable name="Name">
|
||
<item type="string" name="foo" />
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
|
||
input = R"(
|
||
<overlayable name="Name">
|
||
<item type="string" name="foo" />
|
||
</overlayable>
|
||
<overlayable name="Other">
|
||
<item type="string" name="foo" />
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
|
||
input = R"(
|
||
<overlayable name="Name" actor="overlay://my.actor.one">
|
||
<item type="string" name="foo" />
|
||
</overlayable>
|
||
<overlayable name="Other" actor="overlay://my.actor.two">
|
||
<item type="string" name="foo" />
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
|
||
input = R"(
|
||
<overlayable name="Name">
|
||
<policy type="product">
|
||
<item type="string" name="foo" />
|
||
<item type="string" name="foo" />
|
||
</policy>
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
|
||
input = R"(
|
||
<overlayable name="Name">
|
||
<policy type="product">
|
||
<item type="string" name="foo" />
|
||
</policy>
|
||
<item type="string" name="foo" />
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
|
||
input = R"(
|
||
<overlayable name="Name">
|
||
<policy type="product">
|
||
<item type="string" name="foo" />
|
||
</policy>
|
||
<policy type="vendor">
|
||
<item type="string" name="foo" />
|
||
</policy>
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
|
||
input = R"(
|
||
<overlayable name="Name">
|
||
<policy type="product">
|
||
<item type="string" name="foo" />
|
||
</policy>
|
||
</overlayable>
|
||
|
||
<overlayable name="Name">
|
||
<policy type="product">
|
||
<item type="string" name="foo" />
|
||
</policy>
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, NestPolicyInOverlayableError) {
|
||
std::string input = R"(
|
||
<overlayable name="Name">
|
||
<policy type="vendor|product">
|
||
<policy type="public">
|
||
<item type="string" name="foo" />
|
||
</policy>
|
||
</policy>
|
||
</overlayable>)";
|
||
EXPECT_FALSE(TestParse(input));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseIdItem) {
|
||
std::string input = R"(
|
||
<item name="foo" type="id">@id/bar</item>
|
||
<item name="bar" type="id"/>
|
||
<item name="baz" type="id"></item>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
ASSERT_THAT(test::GetValue<Reference>(&table_, "id/foo"), NotNull());
|
||
ASSERT_THAT(test::GetValue<Id>(&table_, "id/bar"), NotNull());
|
||
ASSERT_THAT(test::GetValue<Id>(&table_, "id/baz"), NotNull());
|
||
|
||
input = R"(
|
||
<id name="foo2">@id/bar</id>
|
||
<id name="bar2"/>
|
||
<id name="baz2"></id>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
|
||
ASSERT_THAT(test::GetValue<Reference>(&table_, "id/foo2"), NotNull());
|
||
ASSERT_THAT(test::GetValue<Id>(&table_, "id/bar2"), NotNull());
|
||
ASSERT_THAT(test::GetValue<Id>(&table_, "id/baz2"), NotNull());
|
||
|
||
// Reject attribute references
|
||
input = R"(<item name="foo3" type="id">?attr/bar"</item>)";
|
||
ASSERT_FALSE(TestParse(input));
|
||
|
||
// Reject non-references
|
||
input = R"(<item name="foo4" type="id">0x7f010001</item>)";
|
||
ASSERT_FALSE(TestParse(input));
|
||
input = R"(<item name="foo5" type="id">@drawable/my_image</item>)";
|
||
ASSERT_FALSE(TestParse(input));
|
||
input = R"(<item name="foo6" type="id"><string name="biz"></string></item>)";
|
||
ASSERT_FALSE(TestParse(input));
|
||
|
||
// Ids that reference other resource ids cannot be public
|
||
input = R"(<public name="foo7" type="id">@id/bar7</item>)";
|
||
ASSERT_FALSE(TestParse(input));
|
||
}
|
||
|
||
TEST_F(ResourceParserTest, ParseCData) {
|
||
// Double quotes should still change the state of whitespace processing
|
||
std::string input = R"(<string name="foo">Hello<![CDATA[ "</string>' ]]> World</string>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
auto output = test::GetValue<String>(&table_, "string/foo");
|
||
ASSERT_THAT(output, NotNull());
|
||
EXPECT_THAT(*output, StrValueEq(std::string("Hello </string>' World").data()));
|
||
|
||
input = R"(<string name="foo2"><![CDATA[Hello
|
||
World]]></string>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
output = test::GetValue<String>(&table_, "string/foo2");
|
||
ASSERT_THAT(output, NotNull());
|
||
EXPECT_THAT(*output, StrValueEq(std::string("Hello World").data()));
|
||
|
||
// Cdata blocks should have their whitespace trimmed
|
||
input = R"(<string name="foo3"> <![CDATA[ text ]]> </string>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
output = test::GetValue<String>(&table_, "string/foo3");
|
||
ASSERT_THAT(output, NotNull());
|
||
EXPECT_THAT(*output, StrValueEq(std::string("text").data()));
|
||
|
||
input = R"(<string name="foo4"> <![CDATA[]]> </string>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
output = test::GetValue<String>(&table_, "string/foo4");
|
||
ASSERT_THAT(output, NotNull());
|
||
EXPECT_THAT(*output, StrValueEq(std::string("").data()));
|
||
|
||
input = R"(<string name="foo5"> <![CDATA[ ]]> </string>)";
|
||
ASSERT_TRUE(TestParse(input));
|
||
output = test::GetValue<String>(&table_, "string/foo5");
|
||
ASSERT_THAT(output, NotNull());
|
||
EXPECT_THAT(*output, StrValueEq(std::string("").data()));
|
||
|
||
// Single quotes must still be escaped
|
||
input = R"(<string name="foo6"><![CDATA[some text and ' apostrophe]]></string>)";
|
||
ASSERT_FALSE(TestParse(input));
|
||
}
|
||
|
||
} // namespace aapt
|