| Adam Lesinski | ca5638f | 2015-10-21 14:42:43 -0700 | [diff] [blame] | 1 | /* | 
|  | 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 |  | 
|  | 17 | #include "java/AnnotationProcessor.h" | 
| Adam Lesinski | ca5638f | 2015-10-21 14:42:43 -0700 | [diff] [blame] | 18 |  | 
|  | 19 | #include <algorithm> | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 20 | #include <array> | 
| Narayan Kamath | 1c1544f | 2020-02-13 14:33:47 +0000 | [diff] [blame] | 21 | #include <regex> | 
| Adam Lesinski | ca5638f | 2015-10-21 14:42:43 -0700 | [diff] [blame] | 22 |  | 
| Adam Lesinski | e967d3f | 2017-07-24 18:19:36 -0700 | [diff] [blame] | 23 | #include "text/Unicode.h" | 
|  | 24 | #include "text/Utf8Iterator.h" | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 25 | #include "util/Util.h" | 
|  | 26 |  | 
| Adam Lesinski | a693c4a | 2017-11-09 11:29:39 -0800 | [diff] [blame] | 27 | using ::aapt::text::Printer; | 
| Adam Lesinski | e967d3f | 2017-07-24 18:19:36 -0700 | [diff] [blame] | 28 | using ::aapt::text::Utf8Iterator; | 
|  | 29 | using ::android::StringPiece; | 
| Adam Lesinski | d5083f6 | 2017-01-16 15:07:21 -0800 | [diff] [blame] | 30 |  | 
| Adam Lesinski | ca5638f | 2015-10-21 14:42:43 -0700 | [diff] [blame] | 31 | namespace aapt { | 
|  | 32 |  | 
| Yurii Zubrytskyi | a577514 | 2022-11-02 17:49:49 -0700 | [diff] [blame] | 33 | StringPiece AnnotationProcessor::ExtractFirstSentence(StringPiece comment) { | 
| Adam Lesinski | e967d3f | 2017-07-24 18:19:36 -0700 | [diff] [blame] | 34 | Utf8Iterator iter(comment); | 
|  | 35 | while (iter.HasNext()) { | 
|  | 36 | const char32_t codepoint = iter.Next(); | 
|  | 37 | if (codepoint == U'.') { | 
|  | 38 | const size_t current_position = iter.Position(); | 
|  | 39 | if (!iter.HasNext() || text::IsWhitespace(iter.Next())) { | 
|  | 40 | return comment.substr(0, current_position); | 
|  | 41 | } | 
|  | 42 | } | 
|  | 43 | } | 
|  | 44 | return comment; | 
|  | 45 | } | 
|  | 46 |  | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 47 | struct AnnotationRule { | 
|  | 48 | enum : uint32_t { | 
|  | 49 | kDeprecated = 0x01, | 
|  | 50 | kSystemApi = 0x02, | 
|  | 51 | kTestApi = 0x04, | 
| Mark Punzalan | 636b628 | 2023-10-11 16:54:38 +0000 | [diff] [blame] | 52 | kFlaggedApi = 0x08, | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 53 | }; | 
|  | 54 |  | 
|  | 55 | StringPiece doc_str; | 
|  | 56 | uint32_t bit_mask; | 
|  | 57 | StringPiece annotation; | 
| Mark Punzalan | 636b628 | 2023-10-11 16:54:38 +0000 | [diff] [blame] | 58 | bool preserve_params; | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 59 | }; | 
|  | 60 |  | 
| Mark Punzalan | 636b628 | 2023-10-11 16:54:38 +0000 | [diff] [blame] | 61 | static std::array<AnnotationRule, 3> sAnnotationRules = {{ | 
|  | 62 | {"@SystemApi", AnnotationRule::kSystemApi, "@android.annotation.SystemApi", true}, | 
|  | 63 | {"@TestApi", AnnotationRule::kTestApi, "@android.annotation.TestApi", false}, | 
|  | 64 | {"@FlaggedApi", AnnotationRule::kFlaggedApi, "@android.annotation.FlaggedApi", true}, | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 65 | }}; | 
|  | 66 |  | 
|  | 67 | void AnnotationProcessor::AppendCommentLine(std::string comment) { | 
| Yurii Zubrytskyi | a577514 | 2022-11-02 17:49:49 -0700 | [diff] [blame] | 68 | static constexpr std::string_view sDeprecated = "@deprecated"; | 
| Adam Lesinski | ca5638f | 2015-10-21 14:42:43 -0700 | [diff] [blame] | 69 |  | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 70 | // Treat deprecated specially, since we don't remove it from the source comment. | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 71 | if (comment.find(sDeprecated) != std::string::npos) { | 
| Narayan Kamath | 1c1544f | 2020-02-13 14:33:47 +0000 | [diff] [blame] | 72 | annotation_parameter_map_[AnnotationRule::kDeprecated] = ""; | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 73 | } | 
| Adam Lesinski | ca5638f | 2015-10-21 14:42:43 -0700 | [diff] [blame] | 74 |  | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 75 | for (const AnnotationRule& rule : sAnnotationRules) { | 
|  | 76 | std::string::size_type idx = comment.find(rule.doc_str.data()); | 
|  | 77 | if (idx != std::string::npos) { | 
| Narayan Kamath | 1c1544f | 2020-02-13 14:33:47 +0000 | [diff] [blame] | 78 | // Captures all parameters associated with the specified annotation rule | 
| Mark Punzalan | 636b628 | 2023-10-11 16:54:38 +0000 | [diff] [blame] | 79 | // by matching the first pair of parentheses after the rule. | 
|  | 80 | std::regex re(std::string(rule.doc_str).append(R"(\s*\((.+)\))")); | 
| Narayan Kamath | 1c1544f | 2020-02-13 14:33:47 +0000 | [diff] [blame] | 81 | std::smatch match_result; | 
|  | 82 | const bool is_match = std::regex_search(comment, match_result, re); | 
| Mark Punzalan | 636b628 | 2023-10-11 16:54:38 +0000 | [diff] [blame] | 83 | if (is_match && rule.preserve_params) { | 
| Narayan Kamath | 1c1544f | 2020-02-13 14:33:47 +0000 | [diff] [blame] | 84 | annotation_parameter_map_[rule.bit_mask] = match_result[1].str(); | 
|  | 85 | comment.erase(comment.begin() + match_result.position(), | 
|  | 86 | comment.begin() + match_result.position() + match_result.length()); | 
|  | 87 | } else { | 
|  | 88 | annotation_parameter_map_[rule.bit_mask] = ""; | 
|  | 89 | comment.erase(comment.begin() + idx, comment.begin() + idx + rule.doc_str.size()); | 
|  | 90 | } | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 91 | } | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 92 | } | 
| Adam Lesinski | 626b3db | 2016-04-07 13:24:59 -0700 | [diff] [blame] | 93 |  | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 94 | // Check if after removal of annotations the line is empty. | 
|  | 95 | const StringPiece trimmed = util::TrimWhitespace(comment); | 
|  | 96 | if (trimmed.empty()) { | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 97 | return; | 
|  | 98 | } | 
| Adam Lesinski | ca5638f | 2015-10-21 14:42:43 -0700 | [diff] [blame] | 99 |  | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 100 | // If there was trimming to do, copy the string. | 
|  | 101 | if (trimmed.size() != comment.size()) { | 
| Yurii Zubrytskyi | a577514 | 2022-11-02 17:49:49 -0700 | [diff] [blame] | 102 | comment = std::string(trimmed); | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 103 | } | 
|  | 104 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 105 | if (!has_comments_) { | 
|  | 106 | has_comments_ = true; | 
|  | 107 | comment_ << "/**"; | 
|  | 108 | } | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 109 | comment_ << "\n * " << std::move(comment); | 
| Adam Lesinski | ca5638f | 2015-10-21 14:42:43 -0700 | [diff] [blame] | 110 | } | 
|  | 111 |  | 
| Yurii Zubrytskyi | a577514 | 2022-11-02 17:49:49 -0700 | [diff] [blame] | 112 | void AnnotationProcessor::AppendComment(StringPiece comment) { | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 113 | // We need to process line by line to clean-up whitespace and append prefixes. | 
|  | 114 | for (StringPiece line : util::Tokenize(comment, '\n')) { | 
|  | 115 | line = util::TrimWhitespace(line); | 
|  | 116 | if (!line.empty()) { | 
| Yurii Zubrytskyi | a577514 | 2022-11-02 17:49:49 -0700 | [diff] [blame] | 117 | AppendCommentLine(std::string(line)); | 
| Adam Lesinski | ca5638f | 2015-10-21 14:42:43 -0700 | [diff] [blame] | 118 | } | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 119 | } | 
| Adam Lesinski | ca5638f | 2015-10-21 14:42:43 -0700 | [diff] [blame] | 120 | } | 
|  | 121 |  | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 122 | void AnnotationProcessor::AppendNewLine() { | 
|  | 123 | if (has_comments_) { | 
|  | 124 | comment_ << "\n *"; | 
|  | 125 | } | 
|  | 126 | } | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 127 |  | 
| Makoto Onuki | de6e6f2 | 2020-06-22 10:17:02 -0700 | [diff] [blame] | 128 | void AnnotationProcessor::Print(Printer* printer, bool strip_api_annotations) const { | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 129 | if (has_comments_) { | 
|  | 130 | std::string result = comment_.str(); | 
| Yurii Zubrytskyi | a577514 | 2022-11-02 17:49:49 -0700 | [diff] [blame] | 131 | for (StringPiece line : util::Tokenize(result, '\n')) { | 
| Adam Lesinski | a693c4a | 2017-11-09 11:29:39 -0800 | [diff] [blame] | 132 | printer->Println(line); | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 133 | } | 
| Adam Lesinski | a693c4a | 2017-11-09 11:29:39 -0800 | [diff] [blame] | 134 | printer->Println(" */"); | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 135 | } | 
|  | 136 |  | 
| Narayan Kamath | 1c1544f | 2020-02-13 14:33:47 +0000 | [diff] [blame] | 137 | if (annotation_parameter_map_.find(AnnotationRule::kDeprecated) != | 
|  | 138 | annotation_parameter_map_.end()) { | 
| Adam Lesinski | a693c4a | 2017-11-09 11:29:39 -0800 | [diff] [blame] | 139 | printer->Println("@Deprecated"); | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 140 | } | 
|  | 141 |  | 
| Makoto Onuki | de6e6f2 | 2020-06-22 10:17:02 -0700 | [diff] [blame] | 142 | if (strip_api_annotations) { | 
|  | 143 | return; | 
|  | 144 | } | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 145 | for (const AnnotationRule& rule : sAnnotationRules) { | 
| Narayan Kamath | 1c1544f | 2020-02-13 14:33:47 +0000 | [diff] [blame] | 146 | const auto& it = annotation_parameter_map_.find(rule.bit_mask); | 
|  | 147 | if (it != annotation_parameter_map_.end()) { | 
|  | 148 | printer->Print(rule.annotation); | 
|  | 149 | if (!it->second.empty()) { | 
|  | 150 | printer->Print("(").Print(it->second).Print(")"); | 
|  | 151 | } | 
|  | 152 | printer->Print("\n"); | 
| Adam Lesinski | 09f4d70 | 2017-08-08 10:39:55 -0700 | [diff] [blame] | 153 | } | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 154 | } | 
| Adam Lesinski | 7656554 | 2016-03-10 21:55:04 -0800 | [diff] [blame] | 155 | } | 
|  | 156 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 157 | }  // namespace aapt |