com.android.manifmerger.ManifestModel Maven / Gradle / Ivy
/*
* Copyright (C) 2014 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.manifmerger;
import static com.android.SdkConstants.ANDROID_URI;
import static com.android.SdkConstants.ATTR_NAME;
import static com.android.manifmerger.AttributeModel.Hexadecimal32BitsWithMinimumValue;
import static com.android.manifmerger.AttributeModel.MultiValueValidator;
import com.android.SdkConstants;
import com.android.annotations.NonNull;
import com.android.annotations.Nullable;
import com.android.annotations.concurrency.Immutable;
import com.android.utils.SdkUtils;
import com.android.xml.AndroidManifest;
import com.google.common.base.Joiner;
import com.google.common.base.Preconditions;
import com.google.common.base.Strings;
import com.google.common.collect.ImmutableList;
import org.w3c.dom.Attr;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
/**
* Model for the manifest file merging activities.
*
*
* This model will describe each element that is eligible for merging and associated merging
* policies. It is not reusable as most of its interfaces are private but a future enhancement
* could easily make this more generic/reusable if we need to merge more than manifest files.
*
*/
@Immutable
class ManifestModel {
/**
* Interface responsible for providing a key extraction capability from a xml element.
* Some elements store their keys as an attribute, some as a sub-element attribute, some don't
* have any key.
*/
@Immutable
interface NodeKeyResolver {
/**
* Returns the key associated with this xml element.
* @param xmlElement the xml element to get the key from
* @return the key as a string to uniquely identify xmlElement from similarly typed elements
* in the xml document or null if there is no key.
*/
@Nullable String getKey(Element xmlElement);
/**
* Returns the attribute(s) used to store the xml element key.
* @return the key attribute(s) name(s) or null of this element does not have a key.
*/
@NonNull
ImmutableList getKeyAttributesNames();
}
/**
* Implementation of {@link com.android.manifmerger.ManifestModel.NodeKeyResolver} that do not
* provide any key (the element has to be unique in the xml document).
*/
private static class NoKeyNodeResolver implements NodeKeyResolver {
@Override
@Nullable
public String getKey(Element xmlElement) {
return null;
}
@NonNull
@Override
public ImmutableList getKeyAttributesNames() {
return ImmutableList.of();
}
}
/**
* Implementation of {@link com.android.manifmerger.ManifestModel.NodeKeyResolver} that uses an
* attribute to resolve the key value.
*/
private static class AttributeBasedNodeKeyResolver implements NodeKeyResolver {
@Nullable private final String mNamespaceUri;
private final String mAttributeName;
/**
* Build a new instance capable of resolving an xml element key from the passed attribute
* namespace and local name.
* @param namespaceUri optional namespace for the attribute name.
* @param attributeName attribute name
*/
private AttributeBasedNodeKeyResolver(@Nullable String namespaceUri,
@NonNull String attributeName) {
this.mNamespaceUri = namespaceUri;
this.mAttributeName = Preconditions.checkNotNull(attributeName);
}
@Override
@Nullable
public String getKey(@NonNull Element xmlElement) {
String key = mNamespaceUri == null
? xmlElement.getAttribute(mAttributeName)
: xmlElement.getAttributeNS(mNamespaceUri, mAttributeName);
if (Strings.isNullOrEmpty(key)) return null;
return key;
}
@NonNull
@Override
public ImmutableList getKeyAttributesNames() {
return ImmutableList.of(mAttributeName);
}
}
/**
* Subclass of {@link com.android.manifmerger.ManifestModel.AttributeBasedNodeKeyResolver} that
* uses "android:name" as the attribute.
*/
private static final NodeKeyResolver DEFAULT_NAME_ATTRIBUTE_RESOLVER =
new AttributeBasedNodeKeyResolver(ANDROID_URI, SdkConstants.ATTR_NAME);
private static final NoKeyNodeResolver DEFAULT_NO_KEY_NODE_RESOLVER = new NoKeyNodeResolver();
/**
* A {@link com.android.manifmerger.ManifestModel.NodeKeyResolver} capable of extracting the
* element key first in an "android:name" attribute and if not value found there, in the
* "android:glEsVersion" attribute.
*/
@Nullable
private static final NodeKeyResolver NAME_AND_GLESVERSION_KEY_RESOLVER = new NodeKeyResolver() {
private final NodeKeyResolver nameAttrResolver = DEFAULT_NAME_ATTRIBUTE_RESOLVER;
private final NodeKeyResolver glEsVersionResolver =
new AttributeBasedNodeKeyResolver(ANDROID_URI,
AndroidManifest.ATTRIBUTE_GLESVERSION);
@Nullable
@Override
public String getKey(Element xmlElement) {
@Nullable String key = nameAttrResolver.getKey(xmlElement);
return Strings.isNullOrEmpty(key)
? glEsVersionResolver.getKey(xmlElement)
: key;
}
@NonNull
@Override
public ImmutableList getKeyAttributesNames() {
return ImmutableList.of(SdkConstants.ATTR_NAME, AndroidManifest.ATTRIBUTE_GLESVERSION);
}
};
/**
* Specific {@link com.android.manifmerger.ManifestModel.NodeKeyResolver} for intent-filter
* elements.
* Intent filters do not have a proper key, therefore their identity is really carried by
* the presence of the action and category sub-elements.
* We concatenate such elements sub-keys (after sorting them to work around declaration order)
* and use that for the intent-filter unique key.
*/
@Nullable
private static final NodeKeyResolver INTENT_FILTER_KEY_RESOLVER = new NodeKeyResolver() {
@Nullable
@Override
public String getKey(@NonNull Element element) {
@NonNull OrphanXmlElement xmlElement = new OrphanXmlElement(element);
assert(xmlElement.getType() == NodeTypes.INTENT_FILTER);
// concatenate all actions and categories attribute names.
@NonNull List allSubElementKeys = new ArrayList();
NodeList childNodes = element.getChildNodes();
for (int i = 0; i < childNodes.getLength(); i++) {
Node child = childNodes.item(i);
if (child.getNodeType() != Node.ELEMENT_NODE) continue;
@NonNull OrphanXmlElement subElement = new OrphanXmlElement((Element) child);
if (subElement.getType() == NodeTypes.ACTION
|| subElement.getType() == NodeTypes.CATEGORY) {
Attr nameAttribute = subElement.getXml()
.getAttributeNodeNS(ANDROID_URI, ATTR_NAME);
if (nameAttribute != null) {
allSubElementKeys.add(nameAttribute.getValue());
}
}
}
Collections.sort(allSubElementKeys);
return Joiner.on('+').join(allSubElementKeys);
}
@NonNull
@Override
public ImmutableList getKeyAttributesNames() {
return ImmutableList.of("action#name", "category#name");
}
};
/**
* Implementation of {@link com.android.manifmerger.ManifestModel.NodeKeyResolver} that
* combined two attributes values to create the key value.
*/
private static final class TwoAttributesBasedKeyResolver implements NodeKeyResolver {
private final NodeKeyResolver firstAttributeKeyResolver;
private final NodeKeyResolver secondAttributeKeyResolver;
private TwoAttributesBasedKeyResolver(NodeKeyResolver firstAttributeKeyResolver,
NodeKeyResolver secondAttributeKeyResolver) {
this.firstAttributeKeyResolver = firstAttributeKeyResolver;
this.secondAttributeKeyResolver = secondAttributeKeyResolver;
}
@Nullable
@Override
public String getKey(Element xmlElement) {
@Nullable String firstKey = firstAttributeKeyResolver.getKey(xmlElement);
@Nullable String secondKey = secondAttributeKeyResolver.getKey(xmlElement);
return Strings.isNullOrEmpty(firstKey)
? secondKey
: Strings.isNullOrEmpty(secondKey)
? firstKey
: firstKey + "+" + secondKey;
}
@NonNull
@Override
public ImmutableList getKeyAttributesNames() {
return ImmutableList.of(firstAttributeKeyResolver.getKeyAttributesNames().get(0),
secondAttributeKeyResolver.getKeyAttributesNames().get(0));
}
}
private static final AttributeModel.BooleanValidator BOOLEAN_VALIDATOR =
new AttributeModel.BooleanValidator();
private static final boolean MULTIPLE_DECLARATION_FOR_SAME_KEY_ALLOWED = true;
/**
* Definitions of the support node types in the Android Manifest file.
* {@link }
* for more details about the xml format.
*
* There is no DTD or schema associated with the file type so this is best effort in providing
* some metadata on the elements of the Android's xml file.
*
* Each xml element is defined as an enum value and for each node, extra metadata is added
*
* - {@link com.android.manifmerger.MergeType} to identify how the merging engine
* should process this element.
* - {@link com.android.manifmerger.ManifestModel.NodeKeyResolver} to resolve the
* element's key. Elements can have an attribute like "android:name", others can use
* a sub-element, and finally some do not have a key and are meant to be unique.
* - List of attributes models with special behaviors :
*
* - Smart substitution of class names to fully qualified class names using the
* document's package declaration. The list's size can be 0..n
* - Implicit default value when no defined on the xml element.
* - {@link AttributeModel.Validator} to validate attribute value against.
*
*
*
* It is of the outermost importance to keep this model correct as it is used by the merging
* engine to make all its decisions. There should not be special casing in the engine, all
* decisions must be represented here.
*
* If you find yourself needing to extend the model to support future requirements, do it here
* and modify the engine to make proper decision based on the added metadata.
*/
enum NodeTypes {
/**
* Action (contained in intent-filter)
*
* See also :
* {@link
* Action Xml documentation}
*/
ACTION(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER),
/**
* Activity (contained in application)
*
* See also :
* {@link
* Activity Xml documentation}
*/
ACTIVITY(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER,
AttributeModel.newModel("parentActivityName").setIsPackageDependent(),
AttributeModel.newModel(SdkConstants.ATTR_NAME).setIsPackageDependent()),
/**
* Activity-alias (contained in application)
*
* See also :
* {@link
* Activity-alias Xml documentation}
*/
ACTIVITY_ALIAS(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER,
AttributeModel.newModel("targetActivity").setIsPackageDependent(),
AttributeModel.newModel(SdkConstants.ATTR_NAME).setIsPackageDependent()),
/**
* Application (contained in manifest)
*
* See also :
* {@link
* Application Xml documentation}
*/
APPLICATION(MergeType.MERGE, DEFAULT_NO_KEY_NODE_RESOLVER,
AttributeModel.newModel("backupAgent").setIsPackageDependent(),
AttributeModel.newModel(SdkConstants.ATTR_NAME).setIsPackageDependent()),
/**
* Category (contained in intent-filter)
*
* See also :
* {@link
* Category Xml documentation}
*/
CATEGORY(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER),
/**
* Compatible-screens (contained in manifest)
*
* See also :
* {@link
* Category Xml documentation}
*/
COMPATIBLE_SCREENS(MergeType.MERGE, DEFAULT_NO_KEY_NODE_RESOLVER),
/**
* Data (contained in intent-filter)
*
* See also :
* {@link
* Category Xml documentation}
*/
DATA(MergeType.MERGE, DEFAULT_NO_KEY_NODE_RESOLVER),
/**
* Grant-uri-permission (contained in intent-filter)
*
* See also :
* {@link
* Category Xml documentation}
*/
GRANT_URI_PERMISSION(MergeType.MERGE, DEFAULT_NO_KEY_NODE_RESOLVER),
/**
* Instrumentation (contained in intent-filter)
*
* See also :
* {@link
* Instrunentation Xml documentation}
*/
INSTRUMENTATION(
MergeType.MERGE, DEFAULT_NO_KEY_NODE_RESOLVER,
AttributeModel.newModel("name").setMergingPolicy(AttributeModel.NO_MERGING_POLICY),
AttributeModel.newModel("targetPackage")
.setMergingPolicy(AttributeModel.NO_MERGING_POLICY),
AttributeModel.newModel("functionalTest")
.setMergingPolicy(AttributeModel.NO_MERGING_POLICY),
AttributeModel.newModel("handleProfiling")
.setMergingPolicy(AttributeModel.NO_MERGING_POLICY),
AttributeModel.newModel("label").setMergingPolicy(AttributeModel.NO_MERGING_POLICY)
),
/**
* Intent-filter (contained in activity, activity-alias, service, receiver)
*
* See also :
* {@link
* Intent-filter Xml documentation}
*/
INTENT_FILTER(MergeType.ALWAYS, INTENT_FILTER_KEY_RESOLVER,
MULTIPLE_DECLARATION_FOR_SAME_KEY_ALLOWED),
/**
* Manifest (top level node)
*
* See also :
* {@link
* Manifest Xml documentation}
*/
MANIFEST(MergeType.MERGE_CHILDREN_ONLY, DEFAULT_NO_KEY_NODE_RESOLVER),
/**
* Meta-data (contained in activity, activity-alias, application, provider, receiver)
*
* See also :
* {@link
* Meta-data Xml documentation}
*/
META_DATA(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER),
/**
* Path-permission (contained in provider)
*
* See also :
* {@link
* Meta-data Xml documentation}
*/
PATH_PERMISSION(MergeType.MERGE, DEFAULT_NO_KEY_NODE_RESOLVER),
/**
* Permission-group (contained in manifest).
*
* See also :
* {@link
* Permission-group Xml documentation}
*
*/
PERMISSION_GROUP(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER,
AttributeModel.newModel(SdkConstants.ATTR_NAME)),
/**
* Permission (contained in manifest).
*
* See also :
* {@link
* Permission Xml documentation}
*
*/
PERMISSION(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER,
AttributeModel.newModel(SdkConstants.ATTR_NAME),
AttributeModel.newModel("protectionLevel")
.setDefaultValue("normal")
// TODO : this will need to be populated from
// sdk/platforms/android-19/data/res/values.attrs_manifest.xml
.setOnReadValidator(new MultiValueValidator(
"normal", "dangerous", "signature", "signatureOrSystem"))),
/**
* Permission-tree (contained in manifest).
*
* See also :
* {@link
* Permission-tree Xml documentation}
*
*/
PERMISSION_TREE(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER,
AttributeModel.newModel(SdkConstants.ATTR_NAME)),
/**
* Provider (contained in application)
*
* See also :
* {@link
* Provider Xml documentation}
*/
PROVIDER(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER,
AttributeModel.newModel(SdkConstants.ATTR_NAME)
.setIsPackageDependent()),
/**
* Receiver (contained in application)
*
* See also :
* {@link
* Receiver Xml documentation}
*/
RECEIVER(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER,
AttributeModel.newModel(SdkConstants.ATTR_NAME).setIsPackageDependent()),
/**
* Screen (contained in compatible-screens)
*
* See also :
* {@link
* Receiver Xml documentation}
*/
SCREEN(MergeType.MERGE, new TwoAttributesBasedKeyResolver(
new AttributeBasedNodeKeyResolver(ANDROID_URI, "screenSize"),
new AttributeBasedNodeKeyResolver(ANDROID_URI, "screenDensity"))),
/**
* Service (contained in application)
*
* See also :
* {@link
* Service Xml documentation}
*/
SERVICE(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER,
AttributeModel.newModel(SdkConstants.ATTR_NAME).setIsPackageDependent()),
/**
* Supports-gl-texture (contained in manifest)
*
* See also :
* {@link
* Support-screens Xml documentation}
*/
SUPPORTS_GL_TEXTURE(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER),
/**
* Support-screens (contained in manifest)
*
* See also :
* {@link
* Support-screens Xml documentation}
*/
SUPPORTS_SCREENS(MergeType.MERGE, DEFAULT_NO_KEY_NODE_RESOLVER),
/**
* Uses-configuration (contained in manifest)
*
* See also :
* {@link
* Support-screens Xml documentation}
*/
USES_CONFIGURATION(MergeType.MERGE, DEFAULT_NO_KEY_NODE_RESOLVER),
/**
* Uses-feature (contained in manifest)
*
* See also :
* {@link
* Uses-feature Xml documentation}
*/
USES_FEATURE(MergeType.MERGE, NAME_AND_GLESVERSION_KEY_RESOLVER,
AttributeModel.newModel(AndroidManifest.ATTRIBUTE_REQUIRED)
.setDefaultValue(SdkConstants.VALUE_TRUE)
.setOnReadValidator(BOOLEAN_VALIDATOR)
.setMergingPolicy(AttributeModel.OR_MERGING_POLICY),
AttributeModel.newModel(AndroidManifest.ATTRIBUTE_GLESVERSION)
.setDefaultValue("0x00010000")
.setOnReadValidator(new Hexadecimal32BitsWithMinimumValue(0x00010000))),
/**
* Use-library (contained in application)
*
* See also :
* {@link
* Use-library Xml documentation}
*/
USES_LIBRARY(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER,
AttributeModel.newModel(AndroidManifest.ATTRIBUTE_REQUIRED)
.setDefaultValue(SdkConstants.VALUE_TRUE)
.setOnReadValidator(BOOLEAN_VALIDATOR)
.setMergingPolicy(AttributeModel.OR_MERGING_POLICY)),
/**
* Uses-permission (contained in application)
*
* See also :
* {@link
* Uses-permission Xml documentation}
*/
USES_PERMISSION(MergeType.MERGE, DEFAULT_NAME_ATTRIBUTE_RESOLVER),
/**
* Uses-sdk (contained in manifest)
*
* See also :
* {@link
* Uses-sdk Xml documentation}
*/
USES_SDK(MergeType.MERGE, DEFAULT_NO_KEY_NODE_RESOLVER,
AttributeModel.newModel("minSdkVersion")
.setDefaultValue(SdkConstants.VALUE_1)
.setMergingPolicy(AttributeModel.NO_MERGING_POLICY),
AttributeModel.newModel("maxSdkVersion")
.setMergingPolicy(AttributeModel.NO_MERGING_POLICY),
// TODO : model target's default value is minSdkVersion value.
AttributeModel.newModel("targetSdkVersion")
.setMergingPolicy(AttributeModel.NO_MERGING_POLICY)
),
/**
* Custom tag for any application specific element
*/
CUSTOM(MergeType.MERGE, DEFAULT_NO_KEY_NODE_RESOLVER);
private final MergeType mMergeType;
private final NodeKeyResolver mNodeKeyResolver;
private final ImmutableList mAttributeModels;
private final boolean mMultipleDeclarationAllowed;
NodeTypes(
@NonNull MergeType mergeType,
@NonNull NodeKeyResolver nodeKeyResolver,
@Nullable AttributeModel.Builder... attributeModelBuilders) {
this(mergeType, nodeKeyResolver, false, attributeModelBuilders);
}
NodeTypes(
@NonNull MergeType mergeType,
@NonNull NodeKeyResolver nodeKeyResolver,
boolean mutipleDeclarationAllowed,
@Nullable AttributeModel.Builder... attributeModelBuilders) {
this.mMergeType = Preconditions.checkNotNull(mergeType);
this.mNodeKeyResolver = Preconditions.checkNotNull(nodeKeyResolver);
@NonNull ImmutableList.Builder attributeModels =
new ImmutableList.Builder();
if (attributeModelBuilders != null) {
for (AttributeModel.Builder attributeModelBuilder : attributeModelBuilders) {
attributeModels.add(attributeModelBuilder.build());
}
}
this.mAttributeModels = attributeModels.build();
this.mMultipleDeclarationAllowed = mutipleDeclarationAllowed;
}
@NonNull
NodeKeyResolver getNodeKeyResolver() {
return mNodeKeyResolver;
}
ImmutableList getAttributeModels() {
return mAttributeModels.asList();
}
@Nullable
AttributeModel getAttributeModel(XmlNode.NodeName attributeName) {
// mAttributeModels could be replaced with a Map if the number of models grows.
for (AttributeModel attributeModel : mAttributeModels) {
if (attributeModel.getName().equals(attributeName)) {
return attributeModel;
}
}
return null;
}
/**
* Returns the Xml name for this node type
*/
String toXmlName() {
return SdkUtils.constantNameToXmlName(this.name());
}
/**
* Returns the {@link NodeTypes} instance from an xml element name (without namespace
* decoration). For instance, an xml element
*
* {@code
*
* ...
* }
*
* has a xml simple name of "activity" which will resolve to {@link NodeTypes#ACTIVITY} value.
*
* Note : a runtime exception will be generated if no mapping from the simple name to a
* {@link com.android.manifmerger.ManifestModel.NodeTypes} exists.
*
* @param xmlSimpleName the xml (lower-hyphen separated words) simple name.
* @return the {@link NodeTypes} associated with that element name.
*/
static NodeTypes fromXmlSimpleName(String xmlSimpleName) {
String constantName = SdkUtils.xmlNameToConstantName(xmlSimpleName);
try {
return NodeTypes.valueOf(constantName);
} catch (IllegalArgumentException e) {
// if this element name is not a known tag, we categorize it as 'custom' which will
// be simply merged. It will prevent us from catching simple spelling mistakes but
// extensibility is a must have feature.
return NodeTypes.CUSTOM;
}
}
MergeType getMergeType() {
return mMergeType;
}
/**
* Returns true if multiple declaration for the same type and key are allowed or false if
* there must be only one declaration of this element for a particular key value.
*/
boolean areMultipleDeclarationAllowed() {
return mMultipleDeclarationAllowed;
}
}
}