xref: /aosp_15_r20/frameworks/base/tools/aapt2/test/Common.h (revision d57664e9bc4670b3ecf6748a746a57c557b6bc9e)
1*d57664e9SAndroid Build Coastguard Worker /*
2*d57664e9SAndroid Build Coastguard Worker  * Copyright (C) 2015 The Android Open Source Project
3*d57664e9SAndroid Build Coastguard Worker  *
4*d57664e9SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*d57664e9SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*d57664e9SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*d57664e9SAndroid Build Coastguard Worker  *
8*d57664e9SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*d57664e9SAndroid Build Coastguard Worker  *
10*d57664e9SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*d57664e9SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*d57664e9SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*d57664e9SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*d57664e9SAndroid Build Coastguard Worker  * limitations under the License.
15*d57664e9SAndroid Build Coastguard Worker  */
16*d57664e9SAndroid Build Coastguard Worker 
17*d57664e9SAndroid Build Coastguard Worker #ifndef AAPT_TEST_COMMON_H
18*d57664e9SAndroid Build Coastguard Worker #define AAPT_TEST_COMMON_H
19*d57664e9SAndroid Build Coastguard Worker 
20*d57664e9SAndroid Build Coastguard Worker #include <iostream>
21*d57664e9SAndroid Build Coastguard Worker 
22*d57664e9SAndroid Build Coastguard Worker #include "android-base/logging.h"
23*d57664e9SAndroid Build Coastguard Worker #include "android-base/macros.h"
24*d57664e9SAndroid Build Coastguard Worker #include "androidfw/ConfigDescription.h"
25*d57664e9SAndroid Build Coastguard Worker #include "androidfw/StringPiece.h"
26*d57664e9SAndroid Build Coastguard Worker #include "gmock/gmock.h"
27*d57664e9SAndroid Build Coastguard Worker #include "gtest/gtest.h"
28*d57664e9SAndroid Build Coastguard Worker 
29*d57664e9SAndroid Build Coastguard Worker #include "Debug.h"
30*d57664e9SAndroid Build Coastguard Worker #include "ResourceTable.h"
31*d57664e9SAndroid Build Coastguard Worker #include "ResourceUtils.h"
32*d57664e9SAndroid Build Coastguard Worker #include "ResourceValues.h"
33*d57664e9SAndroid Build Coastguard Worker #include "ValueVisitor.h"
34*d57664e9SAndroid Build Coastguard Worker #include "io/File.h"
35*d57664e9SAndroid Build Coastguard Worker #include "process/IResourceTableConsumer.h"
36*d57664e9SAndroid Build Coastguard Worker 
37*d57664e9SAndroid Build Coastguard Worker namespace aapt {
38*d57664e9SAndroid Build Coastguard Worker namespace test {
39*d57664e9SAndroid Build Coastguard Worker 
40*d57664e9SAndroid Build Coastguard Worker struct TestDiagnosticsImpl : public android::IDiagnostics {
LogTestDiagnosticsImpl41*d57664e9SAndroid Build Coastguard Worker   void Log(Level level, android::DiagMessageActual& actual_msg) override {
42*d57664e9SAndroid Build Coastguard Worker     switch (level) {
43*d57664e9SAndroid Build Coastguard Worker       case Level::Note:
44*d57664e9SAndroid Build Coastguard Worker         return;
45*d57664e9SAndroid Build Coastguard Worker 
46*d57664e9SAndroid Build Coastguard Worker       case Level::Warn:
47*d57664e9SAndroid Build Coastguard Worker         std::cerr << actual_msg.source << ": warn: " << actual_msg.message << "." << std::endl;
48*d57664e9SAndroid Build Coastguard Worker         log << actual_msg.source << ": warn: " << actual_msg.message << "." << std::endl;
49*d57664e9SAndroid Build Coastguard Worker         break;
50*d57664e9SAndroid Build Coastguard Worker 
51*d57664e9SAndroid Build Coastguard Worker       case Level::Error:
52*d57664e9SAndroid Build Coastguard Worker         std::cerr << actual_msg.source << ": error: " << actual_msg.message << "." << std::endl;
53*d57664e9SAndroid Build Coastguard Worker         log << actual_msg.source << ": error: " << actual_msg.message << "." << std::endl;
54*d57664e9SAndroid Build Coastguard Worker         break;
55*d57664e9SAndroid Build Coastguard Worker     }
56*d57664e9SAndroid Build Coastguard Worker   }
57*d57664e9SAndroid Build Coastguard Worker 
GetLogTestDiagnosticsImpl58*d57664e9SAndroid Build Coastguard Worker   std::string GetLog() {
59*d57664e9SAndroid Build Coastguard Worker     return log.str();
60*d57664e9SAndroid Build Coastguard Worker   }
61*d57664e9SAndroid Build Coastguard Worker 
62*d57664e9SAndroid Build Coastguard Worker  private:
63*d57664e9SAndroid Build Coastguard Worker   std::ostringstream log;
64*d57664e9SAndroid Build Coastguard Worker };
65*d57664e9SAndroid Build Coastguard Worker 
66*d57664e9SAndroid Build Coastguard Worker android::IDiagnostics* GetDiagnostics();
67*d57664e9SAndroid Build Coastguard Worker 
ParseNameOrDie(android::StringPiece str)68*d57664e9SAndroid Build Coastguard Worker inline ResourceName ParseNameOrDie(android::StringPiece str) {
69*d57664e9SAndroid Build Coastguard Worker   ResourceNameRef ref;
70*d57664e9SAndroid Build Coastguard Worker   CHECK(ResourceUtils::ParseResourceName(str, &ref)) << "invalid resource name: " << str;
71*d57664e9SAndroid Build Coastguard Worker   return ref.ToResourceName();
72*d57664e9SAndroid Build Coastguard Worker }
73*d57664e9SAndroid Build Coastguard Worker 
ParseConfigOrDie(android::StringPiece str)74*d57664e9SAndroid Build Coastguard Worker inline android::ConfigDescription ParseConfigOrDie(android::StringPiece str) {
75*d57664e9SAndroid Build Coastguard Worker   android::ConfigDescription config;
76*d57664e9SAndroid Build Coastguard Worker   CHECK(android::ConfigDescription::Parse(str, &config)) << "invalid configuration: " << str;
77*d57664e9SAndroid Build Coastguard Worker   return config;
78*d57664e9SAndroid Build Coastguard Worker }
79*d57664e9SAndroid Build Coastguard Worker 
80*d57664e9SAndroid Build Coastguard Worker template <typename T = Value>
GetValueForConfigAndProduct(ResourceTable * table,android::StringPiece res_name,const android::ConfigDescription & config,android::StringPiece product)81*d57664e9SAndroid Build Coastguard Worker T* GetValueForConfigAndProduct(ResourceTable* table, android::StringPiece res_name,
82*d57664e9SAndroid Build Coastguard Worker                                const android::ConfigDescription& config,
83*d57664e9SAndroid Build Coastguard Worker                                android::StringPiece product) {
84*d57664e9SAndroid Build Coastguard Worker   std::optional<ResourceTable::SearchResult> result = table->FindResource(ParseNameOrDie(res_name));
85*d57664e9SAndroid Build Coastguard Worker   if (result) {
86*d57664e9SAndroid Build Coastguard Worker     ResourceConfigValue* config_value = result.value().entry->FindValue(config, product);
87*d57664e9SAndroid Build Coastguard Worker     if (config_value) {
88*d57664e9SAndroid Build Coastguard Worker       return ValueCast<T>(config_value->value.get());
89*d57664e9SAndroid Build Coastguard Worker     }
90*d57664e9SAndroid Build Coastguard Worker   }
91*d57664e9SAndroid Build Coastguard Worker   return nullptr;
92*d57664e9SAndroid Build Coastguard Worker }
93*d57664e9SAndroid Build Coastguard Worker 
94*d57664e9SAndroid Build Coastguard Worker template <>
95*d57664e9SAndroid Build Coastguard Worker Value* GetValueForConfigAndProduct<Value>(ResourceTable* table, android::StringPiece res_name,
96*d57664e9SAndroid Build Coastguard Worker                                           const android::ConfigDescription& config,
97*d57664e9SAndroid Build Coastguard Worker                                           android::StringPiece product);
98*d57664e9SAndroid Build Coastguard Worker 
99*d57664e9SAndroid Build Coastguard Worker template <typename T = Value>
GetValueForConfig(ResourceTable * table,android::StringPiece res_name,const android::ConfigDescription & config)100*d57664e9SAndroid Build Coastguard Worker T* GetValueForConfig(ResourceTable* table, android::StringPiece res_name,
101*d57664e9SAndroid Build Coastguard Worker                      const android::ConfigDescription& config) {
102*d57664e9SAndroid Build Coastguard Worker   return GetValueForConfigAndProduct<T>(table, res_name, config, {});
103*d57664e9SAndroid Build Coastguard Worker }
104*d57664e9SAndroid Build Coastguard Worker 
105*d57664e9SAndroid Build Coastguard Worker template <typename T = Value>
GetValue(ResourceTable * table,android::StringPiece res_name)106*d57664e9SAndroid Build Coastguard Worker T* GetValue(ResourceTable* table, android::StringPiece res_name) {
107*d57664e9SAndroid Build Coastguard Worker   return GetValueForConfig<T>(table, res_name, {});
108*d57664e9SAndroid Build Coastguard Worker }
109*d57664e9SAndroid Build Coastguard Worker 
110*d57664e9SAndroid Build Coastguard Worker class TestFile : public io::IFile {
111*d57664e9SAndroid Build Coastguard Worker  public:
TestFile(android::StringPiece path)112*d57664e9SAndroid Build Coastguard Worker   explicit TestFile(android::StringPiece path) : source_(path) {
113*d57664e9SAndroid Build Coastguard Worker   }
114*d57664e9SAndroid Build Coastguard Worker 
OpenAsData()115*d57664e9SAndroid Build Coastguard Worker   std::unique_ptr<io::IData> OpenAsData() override {
116*d57664e9SAndroid Build Coastguard Worker     return {};
117*d57664e9SAndroid Build Coastguard Worker   }
118*d57664e9SAndroid Build Coastguard Worker 
OpenInputStream()119*d57664e9SAndroid Build Coastguard Worker   std::unique_ptr<android::InputStream> OpenInputStream() override {
120*d57664e9SAndroid Build Coastguard Worker     return OpenAsData();
121*d57664e9SAndroid Build Coastguard Worker   }
122*d57664e9SAndroid Build Coastguard Worker 
GetSource()123*d57664e9SAndroid Build Coastguard Worker   const android::Source& GetSource() const override {
124*d57664e9SAndroid Build Coastguard Worker     return source_;
125*d57664e9SAndroid Build Coastguard Worker   }
126*d57664e9SAndroid Build Coastguard Worker 
GetModificationTime(struct tm * buf)127*d57664e9SAndroid Build Coastguard Worker   bool GetModificationTime(struct tm* buf) const override {
128*d57664e9SAndroid Build Coastguard Worker     return false;
129*d57664e9SAndroid Build Coastguard Worker   };
130*d57664e9SAndroid Build Coastguard Worker 
131*d57664e9SAndroid Build Coastguard Worker  private:
132*d57664e9SAndroid Build Coastguard Worker   DISALLOW_COPY_AND_ASSIGN(TestFile);
133*d57664e9SAndroid Build Coastguard Worker 
134*d57664e9SAndroid Build Coastguard Worker   android::Source source_;
135*d57664e9SAndroid Build Coastguard Worker };
136*d57664e9SAndroid Build Coastguard Worker 
137*d57664e9SAndroid Build Coastguard Worker }  // namespace test
138*d57664e9SAndroid Build Coastguard Worker 
139*d57664e9SAndroid Build Coastguard Worker // Workaround gtest bug (https://github.com/google/googletest/issues/443)
140*d57664e9SAndroid Build Coastguard Worker // that does not select base class operator<< for derived class T.
141*d57664e9SAndroid Build Coastguard Worker template <typename T>
142*d57664e9SAndroid Build Coastguard Worker typename std::enable_if<std::is_base_of<Value, T>::value, std::ostream&>::type operator<<(
143*d57664e9SAndroid Build Coastguard Worker     std::ostream& out, const T& value) {
144*d57664e9SAndroid Build Coastguard Worker   value.Print(&out);
145*d57664e9SAndroid Build Coastguard Worker   return out;
146*d57664e9SAndroid Build Coastguard Worker }
147*d57664e9SAndroid Build Coastguard Worker 
148*d57664e9SAndroid Build Coastguard Worker template std::ostream& operator<<<Item>(std::ostream&, const Item&);
149*d57664e9SAndroid Build Coastguard Worker template std::ostream& operator<<<Reference>(std::ostream&, const Reference&);
150*d57664e9SAndroid Build Coastguard Worker template std::ostream& operator<<<Id>(std::ostream&, const Id&);
151*d57664e9SAndroid Build Coastguard Worker template std::ostream& operator<<<RawString>(std::ostream&, const RawString&);
152*d57664e9SAndroid Build Coastguard Worker template std::ostream& operator<<<String>(std::ostream&, const String&);
153*d57664e9SAndroid Build Coastguard Worker template std::ostream& operator<<<StyledString>(std::ostream&, const StyledString&);
154*d57664e9SAndroid Build Coastguard Worker template std::ostream& operator<<<FileReference>(std::ostream&, const FileReference&);
155*d57664e9SAndroid Build Coastguard Worker template std::ostream& operator<<<BinaryPrimitive>(std::ostream&, const BinaryPrimitive&);
156*d57664e9SAndroid Build Coastguard Worker template std::ostream& operator<<<Attribute>(std::ostream&, const Attribute&);
157*d57664e9SAndroid Build Coastguard Worker template std::ostream& operator<<<Style>(std::ostream&, const Style&);
158*d57664e9SAndroid Build Coastguard Worker template std::ostream& operator<<<Array>(std::ostream&, const Array&);
159*d57664e9SAndroid Build Coastguard Worker template std::ostream& operator<<<Plural>(std::ostream&, const Plural&);
160*d57664e9SAndroid Build Coastguard Worker 
161*d57664e9SAndroid Build Coastguard Worker // Add a print method to Maybe.
162*d57664e9SAndroid Build Coastguard Worker template <typename T>
PrintTo(const std::optional<T> & value,std::ostream * out)163*d57664e9SAndroid Build Coastguard Worker void PrintTo(const std::optional<T>& value, std::ostream* out) {
164*d57664e9SAndroid Build Coastguard Worker   if (value) {
165*d57664e9SAndroid Build Coastguard Worker     *out << ::testing::PrintToString(value.value());
166*d57664e9SAndroid Build Coastguard Worker   } else {
167*d57664e9SAndroid Build Coastguard Worker     *out << "Nothing";
168*d57664e9SAndroid Build Coastguard Worker   }
169*d57664e9SAndroid Build Coastguard Worker }
170*d57664e9SAndroid Build Coastguard Worker 
171*d57664e9SAndroid Build Coastguard Worker namespace test {
172*d57664e9SAndroid Build Coastguard Worker 
173*d57664e9SAndroid Build Coastguard Worker MATCHER_P(StrEq, a,
174*d57664e9SAndroid Build Coastguard Worker           std::string(negation ? "isn't" : "is") + " equal to " +
175*d57664e9SAndroid Build Coastguard Worker               ::testing::PrintToString(android::StringPiece16(a))) {
176*d57664e9SAndroid Build Coastguard Worker   return android::StringPiece16(arg) == a;
177*d57664e9SAndroid Build Coastguard Worker }
178*d57664e9SAndroid Build Coastguard Worker 
179*d57664e9SAndroid Build Coastguard Worker template <typename T>
180*d57664e9SAndroid Build Coastguard Worker class ValueEqImpl : public ::testing::MatcherInterface<T> {
181*d57664e9SAndroid Build Coastguard Worker  public:
ValueEqImpl(const Value * expected)182*d57664e9SAndroid Build Coastguard Worker   explicit ValueEqImpl(const Value* expected) : expected_(expected) {
183*d57664e9SAndroid Build Coastguard Worker   }
184*d57664e9SAndroid Build Coastguard Worker 
MatchAndExplain(T x,::testing::MatchResultListener * listener)185*d57664e9SAndroid Build Coastguard Worker   bool MatchAndExplain(T x, ::testing::MatchResultListener* listener) const override {
186*d57664e9SAndroid Build Coastguard Worker     return expected_->Equals(&x);
187*d57664e9SAndroid Build Coastguard Worker   }
188*d57664e9SAndroid Build Coastguard Worker 
DescribeTo(::std::ostream * os)189*d57664e9SAndroid Build Coastguard Worker   void DescribeTo(::std::ostream* os) const override {
190*d57664e9SAndroid Build Coastguard Worker     *os << "is equal to " << *expected_;
191*d57664e9SAndroid Build Coastguard Worker   }
192*d57664e9SAndroid Build Coastguard Worker 
DescribeNegationTo(::std::ostream * os)193*d57664e9SAndroid Build Coastguard Worker   void DescribeNegationTo(::std::ostream* os) const override {
194*d57664e9SAndroid Build Coastguard Worker     *os << "is not equal to " << *expected_;
195*d57664e9SAndroid Build Coastguard Worker   }
196*d57664e9SAndroid Build Coastguard Worker 
197*d57664e9SAndroid Build Coastguard Worker  private:
198*d57664e9SAndroid Build Coastguard Worker   DISALLOW_COPY_AND_ASSIGN(ValueEqImpl);
199*d57664e9SAndroid Build Coastguard Worker 
200*d57664e9SAndroid Build Coastguard Worker   const Value* expected_;
201*d57664e9SAndroid Build Coastguard Worker };
202*d57664e9SAndroid Build Coastguard Worker 
203*d57664e9SAndroid Build Coastguard Worker template <typename TValue>
204*d57664e9SAndroid Build Coastguard Worker class ValueEqMatcher {
205*d57664e9SAndroid Build Coastguard Worker  public:
206*d57664e9SAndroid Build Coastguard Worker   // NOLINTNEXTLINE(google-explicit-constructor)
ValueEqMatcher(TValue expected)207*d57664e9SAndroid Build Coastguard Worker   ValueEqMatcher(TValue expected) : expected_(std::move(expected)) {
208*d57664e9SAndroid Build Coastguard Worker   }
209*d57664e9SAndroid Build Coastguard Worker 
210*d57664e9SAndroid Build Coastguard Worker   template <typename T>
211*d57664e9SAndroid Build Coastguard Worker   // NOLINTNEXTLINE(google-explicit-constructor)
212*d57664e9SAndroid Build Coastguard Worker   operator ::testing::Matcher<T>() const {
213*d57664e9SAndroid Build Coastguard Worker     return ::testing::Matcher<T>(new ValueEqImpl<T>(&expected_));
214*d57664e9SAndroid Build Coastguard Worker   }
215*d57664e9SAndroid Build Coastguard Worker 
216*d57664e9SAndroid Build Coastguard Worker  private:
217*d57664e9SAndroid Build Coastguard Worker   TValue expected_;
218*d57664e9SAndroid Build Coastguard Worker };
219*d57664e9SAndroid Build Coastguard Worker 
220*d57664e9SAndroid Build Coastguard Worker template <typename TValue>
221*d57664e9SAndroid Build Coastguard Worker class ValueEqPointerMatcher {
222*d57664e9SAndroid Build Coastguard Worker  public:
223*d57664e9SAndroid Build Coastguard Worker   // NOLINTNEXTLINE(google-explicit-constructor)
ValueEqPointerMatcher(const TValue * expected)224*d57664e9SAndroid Build Coastguard Worker   ValueEqPointerMatcher(const TValue* expected) : expected_(expected) {
225*d57664e9SAndroid Build Coastguard Worker   }
226*d57664e9SAndroid Build Coastguard Worker 
227*d57664e9SAndroid Build Coastguard Worker   template <typename T>
228*d57664e9SAndroid Build Coastguard Worker   // NOLINTNEXTLINE(google-explicit-constructor)
229*d57664e9SAndroid Build Coastguard Worker   operator ::testing::Matcher<T>() const {
230*d57664e9SAndroid Build Coastguard Worker     return ::testing::Matcher<T>(new ValueEqImpl<T>(expected_));
231*d57664e9SAndroid Build Coastguard Worker   }
232*d57664e9SAndroid Build Coastguard Worker 
233*d57664e9SAndroid Build Coastguard Worker  private:
234*d57664e9SAndroid Build Coastguard Worker   const TValue* expected_;
235*d57664e9SAndroid Build Coastguard Worker };
236*d57664e9SAndroid Build Coastguard Worker 
237*d57664e9SAndroid Build Coastguard Worker template <typename TValue,
238*d57664e9SAndroid Build Coastguard Worker           typename = typename std::enable_if<!std::is_pointer<TValue>::value, void>::type>
ValueEq(TValue value)239*d57664e9SAndroid Build Coastguard Worker inline ValueEqMatcher<TValue> ValueEq(TValue value) {
240*d57664e9SAndroid Build Coastguard Worker   return ValueEqMatcher<TValue>(std::move(value));
241*d57664e9SAndroid Build Coastguard Worker }
242*d57664e9SAndroid Build Coastguard Worker 
243*d57664e9SAndroid Build Coastguard Worker template <typename TValue>
ValueEq(const TValue * value)244*d57664e9SAndroid Build Coastguard Worker inline ValueEqPointerMatcher<TValue> ValueEq(const TValue* value) {
245*d57664e9SAndroid Build Coastguard Worker   return ValueEqPointerMatcher<TValue>(value);
246*d57664e9SAndroid Build Coastguard Worker }
247*d57664e9SAndroid Build Coastguard Worker 
248*d57664e9SAndroid Build Coastguard Worker MATCHER_P(StrValueEq, a,
249*d57664e9SAndroid Build Coastguard Worker           std::string(negation ? "isn't" : "is") + " equal to " + ::testing::PrintToString(a)) {
250*d57664e9SAndroid Build Coastguard Worker   return *(arg.value) == a;
251*d57664e9SAndroid Build Coastguard Worker }
252*d57664e9SAndroid Build Coastguard Worker 
253*d57664e9SAndroid Build Coastguard Worker MATCHER_P(HasValue, name,
254*d57664e9SAndroid Build Coastguard Worker           std::string(negation ? "does not have" : "has") + " value " +
255*d57664e9SAndroid Build Coastguard Worker               ::testing::PrintToString(name)) {
256*d57664e9SAndroid Build Coastguard Worker   return GetValueForConfig<Value>(&(*arg), name, {}) != nullptr;
257*d57664e9SAndroid Build Coastguard Worker }
258*d57664e9SAndroid Build Coastguard Worker 
259*d57664e9SAndroid Build Coastguard Worker MATCHER_P2(HasValue, name, config,
260*d57664e9SAndroid Build Coastguard Worker            std::string(negation ? "does not have" : "has") + " value " +
261*d57664e9SAndroid Build Coastguard Worker                ::testing::PrintToString(name) + " for config " + ::testing::PrintToString(config)) {
262*d57664e9SAndroid Build Coastguard Worker   return GetValueForConfig<Value>(&(*arg), name, config) != nullptr;
263*d57664e9SAndroid Build Coastguard Worker }
264*d57664e9SAndroid Build Coastguard Worker 
265*d57664e9SAndroid Build Coastguard Worker }  // namespace test
266*d57664e9SAndroid Build Coastguard Worker }  // namespace aapt
267*d57664e9SAndroid Build Coastguard Worker 
268*d57664e9SAndroid Build Coastguard Worker #endif /* AAPT_TEST_COMMON_H */
269