1 /*
2  * Copyright (C) 2019 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 package android.processor.compat.changeid;
18 
19 import static javax.lang.model.element.ElementKind.CLASS;
20 import static javax.lang.model.element.ElementKind.PARAMETER;
21 import static javax.tools.Diagnostic.Kind.ERROR;
22 import static javax.tools.StandardLocation.CLASS_OUTPUT;
23 
24 import android.processor.compat.SingleAnnotationProcessor;
25 import android.processor.compat.SourcePosition;
26 
27 import com.google.common.collect.ImmutableSet;
28 import com.google.common.collect.Table;
29 
30 import java.io.IOException;
31 import java.io.OutputStream;
32 import java.util.List;
33 import java.util.Objects;
34 import java.util.regex.Pattern;
35 
36 import javax.annotation.processing.SupportedAnnotationTypes;
37 import javax.annotation.processing.SupportedSourceVersion;
38 import javax.lang.model.SourceVersion;
39 import javax.lang.model.element.AnnotationMirror;
40 import javax.lang.model.element.AnnotationValue;
41 import javax.lang.model.element.Element;
42 import javax.lang.model.element.ElementKind;
43 import javax.lang.model.element.Modifier;
44 import javax.lang.model.element.PackageElement;
45 import javax.lang.model.element.TypeElement;
46 import javax.lang.model.element.VariableElement;
47 import javax.lang.model.type.TypeKind;
48 import javax.tools.FileObject;
49 
50 /**
51  * Annotation processor for ChangeId annotations.
52  *
53  * This processor outputs an XML file containing all the changeIds defined by this
54  * annotation. The file is bundled into the pratform image and used by the system server.
55  * Design doc: go/gating-and-logging.
56  */
57 @SupportedAnnotationTypes({"android.compat.annotation.ChangeId"})
58 @SupportedSourceVersion(SourceVersion.RELEASE_17)
59 public class ChangeIdProcessor extends SingleAnnotationProcessor {
60 
61     private static final String CONFIG_XML = "compat_config.xml";
62 
63     private static final String IGNORED_CLASS = "android.compat.Compatibility";
64     private static final ImmutableSet<String> IGNORED_METHOD_NAMES =
65             ImmutableSet.of("reportUnconditionalChange", "isChangeEnabled");
66 
67     private static final String CHANGE_ID_QUALIFIED_CLASS_NAME =
68             "android.compat.annotation.ChangeId";
69 
70     private static final String DISABLED_CLASS_NAME = "android.compat.annotation.Disabled";
71     private static final String ENABLED_AFTER_CLASS_NAME = "android.compat.annotation.EnabledAfter";
72     private static final String ENABLED_SINCE_CLASS_NAME = "android.compat.annotation.EnabledSince";
73     private static final String LOGGING_CLASS_NAME = "android.compat.annotation.LoggingOnly";
74     private static final String TARGET_SDK_VERSION = "targetSdkVersion";
75     private static final String OVERRIDABLE_CLASS_NAME = "android.compat.annotation.Overridable";
76 
77     private static final Pattern JAVADOC_SANITIZER = Pattern.compile("^\\s", Pattern.MULTILINE);
78     private static final Pattern HIDE_TAG_MATCHER = Pattern.compile("(\\s|^)@hide(\\s|$)");
79 
80     @Override
process(TypeElement annotation, Table<PackageElement, String, List<Element>> annotatedElements)81     protected void process(TypeElement annotation,
82             Table<PackageElement, String, List<Element>> annotatedElements) {
83         for (PackageElement packageElement : annotatedElements.rowKeySet()) {
84             for (String enclosingElementName : annotatedElements.row(packageElement).keySet()) {
85                 XmlWriter writer = new XmlWriter();
86                 for (Element element : annotatedElements.get(packageElement,
87                         enclosingElementName)) {
88                     Change change =
89                             createChange(packageElement.toString(), enclosingElementName, element);
90                     writer.addChange(change);
91                 }
92 
93                 try {
94                     FileObject resource = processingEnv.getFiler().createResource(
95                             CLASS_OUTPUT, packageElement.toString(),
96                             enclosingElementName + "_" + CONFIG_XML);
97                     try (OutputStream outputStream = resource.openOutputStream()) {
98                         writer.write(outputStream);
99                     }
100                 } catch (IOException e) {
101                     messager.printMessage(ERROR, "Failed to write output: " + e);
102                 }
103             }
104         }
105     }
106 
107     @Override
ignoreAnnotatedElement(Element element, AnnotationMirror mirror)108     protected boolean ignoreAnnotatedElement(Element element, AnnotationMirror mirror) {
109         // Ignore the annotations on method parameters in known methods in package android.compat
110         // (libcore/luni/src/main/java/android/compat/Compatibility.java)
111         // without generating an error.
112         if (element.getKind() == PARAMETER) {
113             Element enclosingMethod = element.getEnclosingElement();
114             Element enclosingElement = enclosingMethod.getEnclosingElement();
115             if (enclosingElement.getKind() == CLASS) {
116                 if (enclosingElement.toString().equals(IGNORED_CLASS) &&
117                         IGNORED_METHOD_NAMES.contains(enclosingMethod.getSimpleName().toString())) {
118                     return true;
119                 }
120             }
121         }
122         return !isValidChangeId(element);
123     }
124 
125     /**
126      * Checks if the provided java element is a valid change id (i.e. a long parameter with a
127      * constant value).
128      *
129      * @param element java element to check.
130      * @return true if the provided element is a legal change id that should be added to the
131      * produced XML file. If true is returned it's guaranteed that the following operations are
132      * safe.
133      */
isValidChangeId(Element element)134     private boolean isValidChangeId(Element element) {
135         if (element.getKind() != ElementKind.FIELD) {
136             messager.printMessage(
137                     ERROR,
138                     "Non FIELD element annotated with @ChangeId.",
139                     element);
140             return false;
141         }
142         if (!(element instanceof VariableElement)) {
143             messager.printMessage(
144                     ERROR,
145                     "Non variable annotated with @ChangeId.",
146                     element);
147             return false;
148         }
149         if (((VariableElement) element).getConstantValue() == null) {
150             messager.printMessage(
151                     ERROR,
152                     "Non constant/final variable annotated with @ChangeId.",
153                     element);
154             return false;
155         }
156         if (element.asType().getKind() != TypeKind.LONG) {
157             messager.printMessage(
158                     ERROR,
159                     "Variables annotated with @ChangeId must be of type long.",
160                     element);
161             return false;
162         }
163         if (!element.getModifiers().contains(Modifier.STATIC)) {
164             messager.printMessage(
165                     ERROR,
166                     "Non static variable annotated with @ChangeId.",
167                     element);
168             return false;
169         }
170        return true;
171     }
172 
createChange(String packageName, String enclosingElementName, Element element)173     private Change createChange(String packageName, String enclosingElementName, Element element) {
174         Change.Builder builder = new Change.Builder()
175                 .id((Long) ((VariableElement) element).getConstantValue())
176                 .name(element.getSimpleName().toString());
177 
178         AnnotationMirror changeId = null;
179         for (AnnotationMirror mirror : element.getAnnotationMirrors()) {
180             final String type =
181                     ((TypeElement) mirror.getAnnotationType().asElement()).getQualifiedName()
182                             .toString();
183             final AnnotationValue sdkValue =
184                     getAnnotationValue(element, mirror, TARGET_SDK_VERSION);
185             switch (type) {
186                 case DISABLED_CLASS_NAME:
187                     builder.disabled();
188                     break;
189                 case LOGGING_CLASS_NAME:
190                     builder.loggingOnly();
191                     break;
192                 case ENABLED_AFTER_CLASS_NAME:
193                     builder.enabledAfter((Integer)(Objects.requireNonNull(sdkValue).getValue()));
194                     break;
195                 case ENABLED_SINCE_CLASS_NAME:
196                     builder.enabledSince((Integer)(Objects.requireNonNull(sdkValue).getValue()));
197                     break;
198                 case OVERRIDABLE_CLASS_NAME:
199                     builder.overridable();
200                     break;
201                 case CHANGE_ID_QUALIFIED_CLASS_NAME:
202                     changeId = mirror;
203                     break;
204             }
205         }
206 
207         String comment =
208                 elements.getDocComment(element);
209         if (comment != null) {
210             comment = HIDE_TAG_MATCHER.matcher(comment).replaceAll("");
211             comment = JAVADOC_SANITIZER.matcher(comment).replaceAll("");
212             comment = comment.replaceAll("\\n", " ");
213             builder.description(comment.trim());
214         }
215 
216         return verifyChange(element,
217                 builder.javaClass(enclosingElementName)
218                         .javaPackage(packageName)
219                         .qualifiedClass(packageName + "." + enclosingElementName)
220                         .sourcePosition(getLineNumber(element, changeId))
221                         .build());
222     }
223 
getLineNumber(Element element, AnnotationMirror mirror)224     private String getLineNumber(Element element, AnnotationMirror mirror) {
225         SourcePosition position = Objects.requireNonNull(getSourcePosition(element, mirror));
226         return String.format("%s:%d", position.getFilename(), position.getStartLineNumber());
227     }
228 
verifyChange(Element element, Change change)229     private Change verifyChange(Element element, Change change) {
230         if (change.disabled && (change.enabledAfter != null || change.enabledSince != null)) {
231             messager.printMessage(
232                     ERROR,
233                     "ChangeId cannot be annotated with both @Disabled and "
234                             + "(@EnabledAfter | @EnabledSince).",
235                     element);
236         }
237         if (change.loggingOnly && (change.disabled || change.enabledAfter != null
238                                     || change.enabledSince != null)) {
239             messager.printMessage(
240                     ERROR,
241                     "ChangeId cannot be annotated with both @LoggingOnly and "
242                             + "(@EnabledAfter | @EnabledSince | @Disabled).",
243                     element);
244         }
245         if (change.enabledAfter != null && change.enabledSince != null) {
246             messager.printMessage(
247                     ERROR,
248                     "ChangeId cannot be annotated with both @EnabledAfter and "
249                             + "@EnabledSince. Prefer using the latter.",
250                     element);
251         }
252         return change;
253     }
254 }
255