#!/usr/bin/env python3 # Copyright 2017 The Chromium Authors # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. """Merges dependency Android manifests into a root manifest.""" import argparse import collections import contextlib import os import sys import tempfile import xml.etree.ElementTree as ElementTree from util import build_utils from util import manifest_utils import action_helpers # build_utils adds //build to sys.path. _MANIFEST_MERGER_MAIN_CLASS = 'com.android.manifmerger.Merger' @contextlib.contextmanager def _ProcessMainManifest(manifest_path, min_sdk_version, target_sdk_version, max_sdk_version, manifest_package): """Patches the main Android manifest""" doc, manifest, _ = manifest_utils.ParseManifest(manifest_path) manifest_utils.SetUsesSdk(manifest, target_sdk_version, min_sdk_version, max_sdk_version) assert manifest_utils.GetPackage(manifest) or manifest_package, \ 'Must set manifest package in GN or in AndroidManifest.xml' if manifest_package: manifest.set('package', manifest_package) tmp_prefix = manifest_path.replace(os.path.sep, '-') with tempfile.NamedTemporaryFile(prefix=tmp_prefix) as patched_manifest: manifest_utils.SaveManifest(doc, patched_manifest.name) yield patched_manifest.name, manifest_utils.GetPackage(manifest) @contextlib.contextmanager def _ProcessOtherManifest(manifest_path, target_sdk_version, seen_package_names): """Patches non-main AndroidManifest.xml if necessary.""" # 1. Ensure targetSdkVersion is set to the expected value to avoid # spurious permissions being added (b/222331337). # 2. Ensure all manifests have a unique package name so that the merger # does not fail when this happens. doc, manifest, _ = manifest_utils.ParseManifest(manifest_path) changed_api = manifest_utils.SetTargetApiIfUnset(manifest, target_sdk_version) package_name = manifest_utils.GetPackage(manifest) package_count = seen_package_names[package_name] seen_package_names[package_name] += 1 if package_count > 0: manifest.set('package', f'{package_name}_{package_count}') if package_count > 0 or changed_api: tmp_prefix = manifest_path.replace(os.path.sep, '-') with tempfile.NamedTemporaryFile(prefix=tmp_prefix) as patched_manifest: manifest_utils.SaveManifest(doc, patched_manifest.name) yield patched_manifest.name else: yield manifest_path def main(argv): argv = build_utils.ExpandFileArgs(argv) parser = argparse.ArgumentParser(description=__doc__) action_helpers.add_depfile_arg(parser) parser.add_argument('--manifest-merger-jar', help='Path to SDK\'s manifest merger jar.', required=True) parser.add_argument('--root-manifest', help='Root manifest which to merge into', required=True) parser.add_argument('--output', help='Output manifest path', required=True) parser.add_argument('--extras', help='GN list of additional manifest to merge') parser.add_argument( '--min-sdk-version', required=True, help='android:minSdkVersion for merging.') parser.add_argument( '--target-sdk-version', required=True, help='android:targetSdkVersion for merging.') parser.add_argument( '--max-sdk-version', help='android:maxSdkVersion for merging.') parser.add_argument( '--manifest-package', help='Package name of the merged AndroidManifest.xml.') parser.add_argument('--warnings-as-errors', action='store_true', help='Treat all warnings as errors.') args = parser.parse_args(argv) with action_helpers.atomic_output(args.output) as output: cmd = build_utils.JavaCmd() + [ '-cp', args.manifest_merger_jar, _MANIFEST_MERGER_MAIN_CLASS, '--out', output.name, '--property', 'MIN_SDK_VERSION=' + args.min_sdk_version, '--property', 'TARGET_SDK_VERSION=' + args.target_sdk_version, ] if args.max_sdk_version: cmd += [ '--property', 'MAX_SDK_VERSION=' + args.max_sdk_version, ] extras = action_helpers.parse_gn_list(args.extras) with contextlib.ExitStack() as stack: root_manifest, package = stack.enter_context( _ProcessMainManifest(args.root_manifest, args.min_sdk_version, args.target_sdk_version, args.max_sdk_version, args.manifest_package)) if extras: seen_package_names = collections.Counter() extras_processed = [ stack.enter_context( _ProcessOtherManifest(e, args.target_sdk_version, seen_package_names)) for e in extras ] cmd += ['--libs', ':'.join(extras_processed)] cmd += [ '--main', root_manifest, '--property', 'PACKAGE=' + package, '--remove-tools-declarations', ] build_utils.CheckOutput( cmd, # https://issuetracker.google.com/issues/63514300: # The merger doesn't set a nonzero exit code for failures. fail_func=lambda returncode, stderr: returncode != 0 or build_utils. IsTimeStale(output.name, [root_manifest] + extras), fail_on_output=args.warnings_as_errors) if args.depfile: action_helpers.write_depfile(args.depfile, args.output, inputs=extras) if __name__ == '__main__': main(sys.argv[1:])