blob: 169896109d43e7af83904b8e63c01b6ec244ff49 [file] [log] [blame]
/*
* Copyright (C) 2012 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.
*/
package com.android.tools.lint.checks;
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;
import java.util.HashMap;
import java.util.Map;
/**
* Parser for the simplified XML API format version 1.
*/
public class ApiParser extends DefaultHandler {
private static final String NODE_API = "api";
private static final String NODE_CLASS = "class";
private static final String NODE_FIELD = "field";
private static final String NODE_METHOD = "method";
private static final String NODE_EXTENDS = "extends";
private static final String NODE_IMPLEMENTS = "implements";
private static final String ATTR_NAME = "name";
private static final String ATTR_SINCE = "since";
private static final String ATTR_DEPRECATED = "deprecated";
private final Map<String, ApiClass> mClasses = new HashMap<String, ApiClass>(1000);
private final Map<String, ApiPackage> mPackages = new HashMap<String, ApiPackage>();
private ApiClass mCurrentClass;
ApiParser() {
}
Map<String, ApiClass> getClasses() {
return mClasses;
}
Map<String, ApiPackage> getPackages() { return mPackages; }
@Override
public void startElement(String uri, String localName, String qName, Attributes attributes)
throws SAXException {
if (localName == null || localName.isEmpty()) {
localName = qName;
}
try {
//noinspection StatementWithEmptyBody
if (NODE_API.equals(localName)) {
// do nothing.
} else if (NODE_CLASS.equals(localName)) {
String name = attributes.getValue(ATTR_NAME);
int since = Integer.parseInt(attributes.getValue(ATTR_SINCE));
String deprecatedAttr = attributes.getValue(ATTR_DEPRECATED);
int deprecatedIn;
if (deprecatedAttr != null) {
deprecatedIn = Integer.parseInt(deprecatedAttr);
} else {
deprecatedIn = 0;
}
mCurrentClass = addClass(name, since, deprecatedIn);
} else if (NODE_EXTENDS.equals(localName)) {
String name = attributes.getValue(ATTR_NAME);
int since = getSince(attributes);
mCurrentClass.addSuperClass(name, since);
} else if (NODE_IMPLEMENTS.equals(localName)) {
String name = attributes.getValue(ATTR_NAME);
int since = getSince(attributes);
mCurrentClass.addInterface(name, since);
} else if (NODE_METHOD.equals(localName)) {
String name = attributes.getValue(ATTR_NAME);
int since = getSince(attributes);
int deprecatedIn = getDeprecatedIn(attributes);
mCurrentClass.addMethod(name, since, deprecatedIn);
} else if (NODE_FIELD.equals(localName)) {
String name = attributes.getValue(ATTR_NAME);
int since = getSince(attributes);
int deprecatedIn = getDeprecatedIn(attributes);
mCurrentClass.addField(name, since, deprecatedIn);
}
} finally {
super.startElement(uri, localName, qName, attributes);
}
}
private ApiClass addClass(String name, int apiLevel, int deprecatedIn) {
// There should not be any duplicates
ApiClass theClass = mClasses.get(name);
assert theClass == null;
theClass = new ApiClass(name, apiLevel, deprecatedIn);
mClasses.put(name, theClass);
String pkg = theClass.getPackage();
if (pkg != null) {
ApiPackage apiPackage = mPackages.get(pkg);
if (apiPackage == null) {
apiPackage = new ApiPackage(pkg);
mPackages.put(pkg, apiPackage);
}
apiPackage.addClass(theClass);
}
return theClass;
}
private int getSince(Attributes attributes) {
int since = mCurrentClass.getSince();
String sinceAttr = attributes.getValue(ATTR_SINCE);
if (sinceAttr != null) {
since = Integer.parseInt(sinceAttr);
}
return since;
}
private int getDeprecatedIn(Attributes attributes) {
int deprecatedIn = mCurrentClass.getDeprecatedIn();
String deprecatedAttr = attributes.getValue(ATTR_DEPRECATED);
if (deprecatedAttr != null) {
deprecatedIn = Integer.parseInt(deprecatedAttr);
}
return deprecatedIn;
}
}