1 /*
2  * Copyright (C) 2010 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 com.android.cts.apicommon;
18 
19 import org.xml.sax.Attributes;
20 import org.xml.sax.SAXException;
21 import org.xml.sax.helpers.DefaultHandler;
22 
23 import java.util.ArrayList;
24 import java.util.List;
25 
26 /**
27  * {@link DefaultHandler} that builds an empty {@link ApiCoverage} object from scanning current.xml.
28  */
29 public class ApiXmlHandler extends DefaultHandler {
30 
31     private String mCurrentPackageName;
32 
33     private String mCurrentClassName;
34 
35     private String mCurrentInterfaceName;
36 
37     private boolean mIgnoreCurrentClass;
38 
39     private String mCurrentMethodName;
40 
41     private String mCurrentMethodReturnType;
42 
43     private boolean mCurrentMethodIsAbstract;
44 
45     private String mCurrentMethodVisibility;
46 
47     private boolean mCurrentMethodStaticMethod;
48 
49     private boolean mCurrentMethodFinalMethod;
50 
51     private boolean mDeprecated;
52 
53 
54     private List<String> mCurrentParameterTypes = new ArrayList<String>();
55 
56     private ApiCoverage mApiCoverage = new ApiCoverage();
57 
getApi()58     public ApiCoverage getApi() {
59         return mApiCoverage;
60     }
61 
62     @Override
startElement(String uri, String localName, String name, Attributes attributes)63     public void startElement(String uri, String localName, String name, Attributes attributes)
64             throws SAXException {
65         super.startElement(uri, localName, name, attributes);
66         if ("package".equalsIgnoreCase(localName)) {
67             mCurrentPackageName = getValue(attributes, "name");
68 
69             ApiPackage apiPackage = new ApiPackage(mCurrentPackageName);
70             mApiCoverage.addPackage(apiPackage);
71 
72         } else if ("class".equalsIgnoreCase(localName)
73                 || "interface".equalsIgnoreCase(localName)) {
74             if (isEnum(attributes)) {
75                 mIgnoreCurrentClass = true;
76                 return;
77             }
78             mIgnoreCurrentClass = false;
79             mCurrentClassName = getValue(attributes, "name");
80             mDeprecated = isDeprecated(attributes);
81             String superClass = attributes.getValue("extends");
82             ApiClass apiClass = new ApiClass(
83                     mCurrentClassName, mDeprecated, is(attributes, "abstract"), superClass);
84             ApiPackage apiPackage = mApiCoverage.getPackage(mCurrentPackageName);
85             apiPackage.addClass(apiClass);
86         } else if ("implements".equalsIgnoreCase(localName)) {
87             mCurrentInterfaceName = attributes.getValue("name");
88         } else if ("constructor".equalsIgnoreCase(localName)) {
89             mDeprecated = isDeprecated(attributes);
90             mCurrentParameterTypes.clear();
91         }  else if ("method".equalsIgnoreCase(localName)) {
92             mDeprecated = isDeprecated(attributes);
93             mCurrentMethodName = getValue(attributes, "name");
94             mCurrentMethodReturnType = getValue(attributes, "return");
95             mCurrentMethodIsAbstract = is(attributes, "abstract");
96             mCurrentMethodVisibility = getValue(attributes, "visibility");
97             mCurrentMethodStaticMethod = is(attributes, "static");
98             mCurrentMethodFinalMethod = is(attributes, "final");
99             mCurrentParameterTypes.clear();
100         } else if ("parameter".equalsIgnoreCase(localName)) {
101             mCurrentParameterTypes.add(getValue(attributes, "type"));
102         }
103     }
104 
105     @Override
endElement(String uri, String localName, String name)106     public void endElement(String uri, String localName, String name) throws SAXException {
107         super.endElement(uri, localName, name);
108         if (mIgnoreCurrentClass) {
109             // do not add anything for enum
110             return;
111         }
112         if ("constructor".equalsIgnoreCase(localName)) {
113             ApiConstructor apiConstructor = new ApiConstructor(mCurrentClassName,
114                     mCurrentParameterTypes, mDeprecated);
115             ApiPackage apiPackage = mApiCoverage.getPackage(mCurrentPackageName);
116             ApiClass apiClass = apiPackage.getClass(mCurrentClassName);
117             apiClass.addConstructor(apiConstructor);
118         } else if ("implements".equalsIgnoreCase(localName)) {
119             ApiPackage apiPackage = mApiCoverage.getPackage(mCurrentPackageName);
120             ApiClass apiClass = apiPackage.getClass(mCurrentClassName);
121             apiClass.addInterface(mCurrentInterfaceName);
122         } else if ("method".equalsIgnoreCase(localName)) {
123             ApiMethod apiMethod = new ApiMethod(
124                     mCurrentMethodName,
125                     mCurrentParameterTypes,
126                     mCurrentMethodReturnType,
127                     mDeprecated,
128                     mCurrentMethodVisibility,
129                     mCurrentMethodStaticMethod,
130                     mCurrentMethodFinalMethod,
131                     mCurrentMethodIsAbstract);
132             ApiPackage apiPackage = mApiCoverage.getPackage(mCurrentPackageName);
133             ApiClass apiClass = apiPackage.getClass(mCurrentClassName);
134             apiClass.addMethod(apiMethod);
135         }
136     }
137 
getValue(Attributes attributes, String key)138     public static String getValue(Attributes attributes, String key) {
139         // Strip away generics <...> and make inner classes always use a "." rather than "$".
140         return attributes.getValue(key)
141                 .replaceAll("<.+>", "")
142                 .replace("$", ".");
143     }
144 
isDeprecated(Attributes attributes)145     private boolean isDeprecated(Attributes attributes) {
146         return "deprecated".equals(attributes.getValue("deprecated"));
147     }
148 
is(Attributes attributes, String valueName)149     private static boolean is(Attributes attributes, String valueName) {
150         return "true".equals(attributes.getValue(valueName));
151     }
152 
isEnum(Attributes attributes)153     private boolean isEnum(Attributes attributes) {
154         return "java.lang.Enum".equals(attributes.getValue("extends"));
155     }
156 }
157