| /* |
| * 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. |
| */ |
| |
| #ifndef AAPT_XML_PULL_PARSER_H |
| #define AAPT_XML_PULL_PARSER_H |
| |
| #include <expat.h> |
| |
| #include <algorithm> |
| #include <istream> |
| #include <ostream> |
| #include <queue> |
| #include <stack> |
| #include <string> |
| #include <vector> |
| |
| #include "android-base/macros.h" |
| #include "androidfw/StringPiece.h" |
| |
| #include "Resource.h" |
| #include "process/IResourceTableConsumer.h" |
| #include "util/Maybe.h" |
| #include "xml/XmlUtil.h" |
| |
| namespace aapt { |
| namespace xml { |
| |
| class XmlPullParser : public IPackageDeclStack { |
| public: |
| enum class Event { |
| kBadDocument, |
| kStartDocument, |
| kEndDocument, |
| |
| kStartNamespace, |
| kEndNamespace, |
| kStartElement, |
| kEndElement, |
| kText, |
| kComment, |
| }; |
| |
| /** |
| * Skips to the next direct descendant node of the given start_depth, |
| * skipping namespace nodes. |
| * |
| * When NextChildNode() returns true, you can expect Comments, Text, and |
| * StartElement events. |
| */ |
| static bool NextChildNode(XmlPullParser* parser, size_t start_depth); |
| static bool SkipCurrentElement(XmlPullParser* parser); |
| static bool IsGoodEvent(Event event); |
| |
| explicit XmlPullParser(std::istream& in); |
| ~XmlPullParser(); |
| |
| /** |
| * Returns the current event that is being processed. |
| */ |
| Event event() const; |
| |
| const std::string& error() const; |
| |
| /** |
| * Note, unlike XmlPullParser, the first call to next() will return |
| * StartElement of the first element. |
| */ |
| Event Next(); |
| |
| // |
| // These are available for all nodes. |
| // |
| |
| const std::string& comment() const; |
| size_t line_number() const; |
| size_t depth() const; |
| |
| /** |
| * Returns the character data for a Text event. |
| */ |
| const std::string& text() const; |
| |
| // |
| // Namespace prefix and URI are available for StartNamespace and EndNamespace. |
| // |
| |
| const std::string& namespace_prefix() const; |
| const std::string& namespace_uri() const; |
| |
| // |
| // These are available for StartElement and EndElement. |
| // |
| |
| const std::string& element_namespace() const; |
| const std::string& element_name() const; |
| |
| /* |
| * Uses the current stack of namespaces to resolve the package. Eg: |
| * xmlns:app = "http://schemas.android.com/apk/res/com.android.app" |
| * ... |
| * android:text="@app:string/message" |
| * |
| * In this case, 'app' will be converted to 'com.android.app'. |
| * |
| * If xmlns:app="http://schemas.android.com/apk/res-auto", then |
| * 'package' will be set to 'defaultPackage'. |
| */ |
| Maybe<ExtractedPackage> TransformPackageAlias( |
| const android::StringPiece& alias, const android::StringPiece& local_package) const override; |
| |
| // |
| // Remaining methods are for retrieving information about attributes |
| // associated with a StartElement. |
| // |
| // Attributes must be in sorted order (according to the less than operator |
| // of struct Attribute). |
| // |
| |
| struct Attribute { |
| std::string namespace_uri; |
| std::string name; |
| std::string value; |
| |
| int compare(const Attribute& rhs) const; |
| bool operator<(const Attribute& rhs) const; |
| bool operator==(const Attribute& rhs) const; |
| bool operator!=(const Attribute& rhs) const; |
| }; |
| |
| using const_iterator = std::vector<Attribute>::const_iterator; |
| |
| const_iterator begin_attributes() const; |
| const_iterator end_attributes() const; |
| size_t attribute_count() const; |
| const_iterator FindAttribute(android::StringPiece namespace_uri, android::StringPiece name) const; |
| |
| private: |
| DISALLOW_COPY_AND_ASSIGN(XmlPullParser); |
| |
| static void XMLCALL StartNamespaceHandler(void* user_data, const char* prefix, |
| const char* uri); |
| static void XMLCALL StartElementHandler(void* user_data, const char* name, |
| const char** attrs); |
| static void XMLCALL CharacterDataHandler(void* user_data, const char* s, |
| int len); |
| static void XMLCALL EndElementHandler(void* user_data, const char* name); |
| static void XMLCALL EndNamespaceHandler(void* user_data, const char* prefix); |
| static void XMLCALL CommentDataHandler(void* user_data, const char* comment); |
| |
| struct EventData { |
| Event event; |
| size_t line_number; |
| size_t depth; |
| std::string data1; |
| std::string data2; |
| std::vector<Attribute> attributes; |
| }; |
| |
| std::istream& in_; |
| XML_Parser parser_; |
| char buffer_[16384]; |
| std::queue<EventData> event_queue_; |
| std::string error_; |
| const std::string empty_; |
| size_t depth_; |
| std::stack<std::string> namespace_uris_; |
| |
| struct PackageDecl { |
| std::string prefix; |
| ExtractedPackage package; |
| }; |
| std::vector<PackageDecl> package_aliases_; |
| }; |
| |
| /** |
| * Finds the attribute in the current element within the global namespace. |
| */ |
| Maybe<android::StringPiece> FindAttribute(const XmlPullParser* parser, |
| const android::StringPiece& name); |
| |
| /** |
| * Finds the attribute in the current element within the global namespace. The |
| * attribute's value |
| * must not be the empty string. |
| */ |
| Maybe<android::StringPiece> FindNonEmptyAttribute(const XmlPullParser* parser, |
| const android::StringPiece& name); |
| |
| // |
| // Implementation |
| // |
| |
| inline ::std::ostream& operator<<(::std::ostream& out, |
| XmlPullParser::Event event) { |
| switch (event) { |
| case XmlPullParser::Event::kBadDocument: |
| return out << "BadDocument"; |
| case XmlPullParser::Event::kStartDocument: |
| return out << "StartDocument"; |
| case XmlPullParser::Event::kEndDocument: |
| return out << "EndDocument"; |
| case XmlPullParser::Event::kStartNamespace: |
| return out << "StartNamespace"; |
| case XmlPullParser::Event::kEndNamespace: |
| return out << "EndNamespace"; |
| case XmlPullParser::Event::kStartElement: |
| return out << "StartElement"; |
| case XmlPullParser::Event::kEndElement: |
| return out << "EndElement"; |
| case XmlPullParser::Event::kText: |
| return out << "Text"; |
| case XmlPullParser::Event::kComment: |
| return out << "Comment"; |
| } |
| return out; |
| } |
| |
| inline bool XmlPullParser::NextChildNode(XmlPullParser* parser, |
| size_t start_depth) { |
| Event event; |
| |
| // First get back to the start depth. |
| while (IsGoodEvent(event = parser->Next()) && |
| parser->depth() > start_depth + 1) { |
| } |
| |
| // Now look for the first good node. |
| while ((event != Event::kEndElement || parser->depth() > start_depth) && |
| IsGoodEvent(event)) { |
| switch (event) { |
| case Event::kText: |
| case Event::kComment: |
| case Event::kStartElement: |
| return true; |
| default: |
| break; |
| } |
| event = parser->Next(); |
| } |
| return false; |
| } |
| |
| inline bool XmlPullParser::SkipCurrentElement(XmlPullParser* parser) { |
| int depth = 1; |
| while (depth > 0) { |
| switch (parser->Next()) { |
| case Event::kEndDocument: |
| return true; |
| case Event::kBadDocument: |
| return false; |
| case Event::kStartElement: |
| depth++; |
| break; |
| case Event::kEndElement: |
| depth--; |
| break; |
| default: |
| break; |
| } |
| } |
| return true; |
| } |
| |
| inline bool XmlPullParser::IsGoodEvent(XmlPullParser::Event event) { |
| return event != Event::kBadDocument && event != Event::kEndDocument; |
| } |
| |
| inline int XmlPullParser::Attribute::compare(const Attribute& rhs) const { |
| int cmp = namespace_uri.compare(rhs.namespace_uri); |
| if (cmp != 0) return cmp; |
| return name.compare(rhs.name); |
| } |
| |
| inline bool XmlPullParser::Attribute::operator<(const Attribute& rhs) const { |
| return compare(rhs) < 0; |
| } |
| |
| inline bool XmlPullParser::Attribute::operator==(const Attribute& rhs) const { |
| return compare(rhs) == 0; |
| } |
| |
| inline bool XmlPullParser::Attribute::operator!=(const Attribute& rhs) const { |
| return compare(rhs) != 0; |
| } |
| |
| inline XmlPullParser::const_iterator XmlPullParser::FindAttribute( |
| android::StringPiece namespace_uri, android::StringPiece name) const { |
| const auto end_iter = end_attributes(); |
| const auto iter = std::lower_bound( |
| begin_attributes(), end_iter, |
| std::pair<android::StringPiece, android::StringPiece>(namespace_uri, name), |
| [](const Attribute& attr, |
| const std::pair<android::StringPiece, android::StringPiece>& rhs) -> bool { |
| int cmp = attr.namespace_uri.compare( |
| 0, attr.namespace_uri.size(), rhs.first.data(), rhs.first.size()); |
| if (cmp < 0) return true; |
| if (cmp > 0) return false; |
| cmp = attr.name.compare(0, attr.name.size(), rhs.second.data(), |
| rhs.second.size()); |
| if (cmp < 0) return true; |
| return false; |
| }); |
| |
| if (iter != end_iter && namespace_uri == iter->namespace_uri && |
| name == iter->name) { |
| return iter; |
| } |
| return end_iter; |
| } |
| |
| } // namespace xml |
| } // namespace aapt |
| |
| #endif // AAPT_XML_PULL_PARSER_H |