blob: cad4c6c7c94fd03ef801efa25494164df3c3b0f8 [file] [log] [blame]
Adam Lesinskica5638f2015-10-21 14:42:43 -07001/*
2 * Copyright (C) 2015 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Adam Lesinskica5638f2015-10-21 14:42:43 -070017#include "java/ManifestClassGenerator.h"
Adam Lesinskica5638f2015-10-21 14:42:43 -070018
19#include <algorithm>
20
Adam Lesinskice5e56e2016-10-21 17:56:45 -070021#include "Source.h"
22#include "java/AnnotationProcessor.h"
23#include "java/ClassDefinition.h"
24#include "util/Maybe.h"
25#include "xml/XmlDom.h"
26
Adam Lesinskid5083f62017-01-16 15:07:21 -080027using android::StringPiece;
28
Adam Lesinskica5638f2015-10-21 14:42:43 -070029namespace aapt {
30
Adam Lesinskice5e56e2016-10-21 17:56:45 -070031static Maybe<StringPiece> ExtractJavaIdentifier(IDiagnostics* diag,
32 const Source& source,
Adam Lesinskid0f116b2016-07-08 15:00:32 -070033 const StringPiece& value) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070034 const StringPiece sep = ".";
35 auto iter = std::find_end(value.begin(), value.end(), sep.begin(), sep.end());
Adam Lesinskica5638f2015-10-21 14:42:43 -070036
Adam Lesinskice5e56e2016-10-21 17:56:45 -070037 StringPiece result;
38 if (iter != value.end()) {
39 result.assign(iter + sep.size(), value.end() - (iter + sep.size()));
40 } else {
41 result = value;
42 }
Adam Lesinskica5638f2015-10-21 14:42:43 -070043
Adam Lesinskice5e56e2016-10-21 17:56:45 -070044 if (result.empty()) {
45 diag->Error(DiagMessage(source) << "empty symbol");
46 return {};
47 }
Adam Lesinskica5638f2015-10-21 14:42:43 -070048
Adam Lesinskice5e56e2016-10-21 17:56:45 -070049 iter = util::FindNonAlphaNumericAndNotInSet(result, "_");
50 if (iter != result.end()) {
51 diag->Error(DiagMessage(source) << "invalid character '"
52 << StringPiece(iter, 1) << "' in '"
53 << result << "'");
54 return {};
55 }
Adam Lesinskica5638f2015-10-21 14:42:43 -070056
Adam Lesinskice5e56e2016-10-21 17:56:45 -070057 if (*result.begin() >= '0' && *result.begin() <= '9') {
58 diag->Error(DiagMessage(source) << "symbol can not start with a digit");
59 return {};
60 }
Adam Lesinskica5638f2015-10-21 14:42:43 -070061
Adam Lesinskice5e56e2016-10-21 17:56:45 -070062 return result;
Adam Lesinskica5638f2015-10-21 14:42:43 -070063}
64
Adam Lesinskice5e56e2016-10-21 17:56:45 -070065static bool WriteSymbol(const Source& source, IDiagnostics* diag,
66 xml::Element* el, ClassDefinition* class_def) {
67 xml::Attribute* attr = el->FindAttribute(xml::kSchemaAndroid, "name");
68 if (!attr) {
69 diag->Error(DiagMessage(source) << "<" << el->name
70 << "> must define 'android:name'");
71 return false;
72 }
Adam Lesinskica5638f2015-10-21 14:42:43 -070073
Adam Lesinskice5e56e2016-10-21 17:56:45 -070074 Maybe<StringPiece> result = ExtractJavaIdentifier(
75 diag, source.WithLine(el->line_number), attr->value);
76 if (!result) {
77 return false;
78 }
Adam Lesinskica5638f2015-10-21 14:42:43 -070079
Adam Lesinskice5e56e2016-10-21 17:56:45 -070080 std::unique_ptr<StringMember> string_member =
81 util::make_unique<StringMember>(result.value(), attr->value);
82 string_member->GetCommentBuilder()->AppendComment(el->comment);
Adam Lesinski6cbfb1d2016-03-31 13:33:02 -070083
Adam Lesinski6e241e72017-08-18 19:49:58 -070084 if (class_def->AddMember(std::move(string_member)) == ClassDefinition::Result::kOverridden) {
85 diag->Warn(DiagMessage(source.WithLine(el->line_number))
86 << "duplicate definitions of '" << result.value() << "', overriding previous");
87 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -070088 return true;
Adam Lesinskica5638f2015-10-21 14:42:43 -070089}
90
Adam Lesinskice5e56e2016-10-21 17:56:45 -070091std::unique_ptr<ClassDefinition> GenerateManifestClass(IDiagnostics* diag,
92 xml::XmlResource* res) {
93 xml::Element* el = xml::FindRootElement(res->root.get());
94 if (!el) {
95 diag->Error(DiagMessage(res->file.source) << "no root tag defined");
96 return {};
97 }
98
99 if (el->name != "manifest" && !el->namespace_uri.empty()) {
100 diag->Error(DiagMessage(res->file.source)
101 << "no <manifest> root tag defined");
102 return {};
103 }
104
105 std::unique_ptr<ClassDefinition> permission_class =
Adam Lesinskiceb9b2f2017-02-16 12:05:42 -0800106 util::make_unique<ClassDefinition>("permission", ClassQualifier::kStatic, false);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700107 std::unique_ptr<ClassDefinition> permission_group_class =
Adam Lesinskiceb9b2f2017-02-16 12:05:42 -0800108 util::make_unique<ClassDefinition>("permission_group", ClassQualifier::kStatic, false);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700109
110 bool error = false;
111 std::vector<xml::Element*> children = el->GetChildElements();
112 for (xml::Element* child_el : children) {
113 if (child_el->namespace_uri.empty()) {
114 if (child_el->name == "permission") {
115 error |= !WriteSymbol(res->file.source, diag, child_el,
116 permission_class.get());
117 } else if (child_el->name == "permission-group") {
118 error |= !WriteSymbol(res->file.source, diag, child_el,
119 permission_group_class.get());
120 }
Adam Lesinskica5638f2015-10-21 14:42:43 -0700121 }
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700122 }
Adam Lesinskica5638f2015-10-21 14:42:43 -0700123
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700124 if (error) {
125 return {};
126 }
Adam Lesinskica5638f2015-10-21 14:42:43 -0700127
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700128 std::unique_ptr<ClassDefinition> manifest_class =
Adam Lesinskiceb9b2f2017-02-16 12:05:42 -0800129 util::make_unique<ClassDefinition>("Manifest", ClassQualifier::kNone, false);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700130 manifest_class->AddMember(std::move(permission_class));
131 manifest_class->AddMember(std::move(permission_group_class));
132 return manifest_class;
Adam Lesinskica5638f2015-10-21 14:42:43 -0700133}
134
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700135} // namespace aapt