| /* |
| * 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 "java/AnnotationProcessor.h" |
| |
| #include <algorithm> |
| #include <array> |
| #include <regex> |
| |
| #include "text/Unicode.h" |
| #include "text/Utf8Iterator.h" |
| #include "util/Util.h" |
| |
| using ::aapt::text::Printer; |
| using ::aapt::text::Utf8Iterator; |
| using ::android::StringPiece; |
| |
| namespace aapt { |
| |
| StringPiece AnnotationProcessor::ExtractFirstSentence(StringPiece comment) { |
| Utf8Iterator iter(comment); |
| while (iter.HasNext()) { |
| const char32_t codepoint = iter.Next(); |
| if (codepoint == U'.') { |
| const size_t current_position = iter.Position(); |
| if (!iter.HasNext() || text::IsWhitespace(iter.Next())) { |
| return comment.substr(0, current_position); |
| } |
| } |
| } |
| return comment; |
| } |
| |
| struct AnnotationRule { |
| enum : uint32_t { |
| kDeprecated = 0x01, |
| kSystemApi = 0x02, |
| kTestApi = 0x04, |
| kFlaggedApi = 0x08, |
| }; |
| |
| StringPiece doc_str; |
| uint32_t bit_mask; |
| StringPiece annotation; |
| bool preserve_params; |
| }; |
| |
| static std::array<AnnotationRule, 3> sAnnotationRules = {{ |
| {"@SystemApi", AnnotationRule::kSystemApi, "@android.annotation.SystemApi", true}, |
| {"@TestApi", AnnotationRule::kTestApi, "@android.annotation.TestApi", false}, |
| {"@FlaggedApi", AnnotationRule::kFlaggedApi, "@android.annotation.FlaggedApi", true}, |
| }}; |
| |
| void AnnotationProcessor::AppendCommentLine(std::string comment, bool add_api_annotations) { |
| static constexpr std::string_view sDeprecated = "@deprecated"; |
| |
| if (add_api_annotations) { |
| // Treat deprecated specially, since we don't remove it from the source comment. |
| if (comment.find(sDeprecated) != std::string::npos) { |
| annotation_parameter_map_[AnnotationRule::kDeprecated] = ""; |
| } |
| |
| for (const AnnotationRule& rule : sAnnotationRules) { |
| std::string::size_type idx = comment.find(rule.doc_str.data()); |
| if (idx != std::string::npos) { |
| // Captures all parameters associated with the specified annotation rule |
| // by matching the first pair of parentheses after the rule. |
| std::regex re(std::string(rule.doc_str).append(R"(\s*\((.+)\))")); |
| std::smatch match_result; |
| const bool is_match = std::regex_search(comment, match_result, re); |
| if (is_match && rule.preserve_params) { |
| annotation_parameter_map_[rule.bit_mask] = match_result[1].str(); |
| comment.erase(comment.begin() + match_result.position(), |
| comment.begin() + match_result.position() + match_result.length()); |
| } else { |
| annotation_parameter_map_[rule.bit_mask] = ""; |
| comment.erase(comment.begin() + idx, comment.begin() + idx + rule.doc_str.size()); |
| } |
| } |
| } |
| } |
| |
| // Check if after removal of annotations the line is empty. |
| const StringPiece trimmed = util::TrimWhitespace(comment); |
| if (trimmed.empty()) { |
| return; |
| } |
| |
| // If there was trimming to do, copy the string. |
| if (trimmed.size() != comment.size()) { |
| comment = std::string(trimmed); |
| } |
| |
| if (!has_comments_) { |
| has_comments_ = true; |
| comment_ << "/**"; |
| } |
| comment_ << "\n * " << std::move(comment); |
| } |
| |
| void AnnotationProcessor::AppendComment(StringPiece comment, bool add_api_annotations) { |
| // We need to process line by line to clean-up whitespace and append prefixes. |
| for (StringPiece line : util::Tokenize(comment, '\n')) { |
| line = util::TrimWhitespace(line); |
| if (!line.empty()) { |
| AppendCommentLine(std::string(line), add_api_annotations); |
| } |
| } |
| } |
| |
| void AnnotationProcessor::AppendNewLine() { |
| if (has_comments_) { |
| comment_ << "\n *"; |
| } |
| } |
| |
| void AnnotationProcessor::Print(Printer* printer, bool strip_api_annotations) const { |
| if (has_comments_) { |
| std::string result = comment_.str(); |
| for (StringPiece line : util::Tokenize(result, '\n')) { |
| printer->Println(line); |
| } |
| printer->Println(" */"); |
| } |
| |
| if (annotation_parameter_map_.find(AnnotationRule::kDeprecated) != |
| annotation_parameter_map_.end()) { |
| printer->Println("@Deprecated"); |
| } |
| |
| if (strip_api_annotations) { |
| return; |
| } |
| for (const AnnotationRule& rule : sAnnotationRules) { |
| const auto& it = annotation_parameter_map_.find(rule.bit_mask); |
| if (it != annotation_parameter_map_.end()) { |
| printer->Print(rule.annotation); |
| if (!it->second.empty()) { |
| printer->Print("(").Print(it->second).Print(")"); |
| } |
| printer->Print("\n"); |
| } |
| } |
| } |
| |
| } // namespace aapt |