diff options
Diffstat (limited to 'scripts/build-apex-bundle.py')
-rw-r--r-- | scripts/build-apex-bundle.py | 114 |
1 files changed, 114 insertions, 0 deletions
diff --git a/scripts/build-apex-bundle.py b/scripts/build-apex-bundle.py new file mode 100644 index 000000000..dcdd9ef7d --- /dev/null +++ b/scripts/build-apex-bundle.py @@ -0,0 +1,114 @@ +#!/usr/bin/env python +# +# Copyright (C) 2022 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. +# +"""A tool to create an APEX bundle out of Soong-built base.zip""" + +from __future__ import print_function + +import argparse +import sys +import tempfile +import zipfile +import os +import json +import subprocess + + +def parse_args(): + """Parse commandline arguments.""" + parser = argparse.ArgumentParser() + parser.add_argument( + '--overwrite', + action='store_true', + help='If set, any previous existing output will be overwritten') + parser.add_argument('--output', help='specify the output .aab file') + parser.add_argument( + 'input', help='specify the input <apex name>-base.zip file') + return parser.parse_args() + + +def build_bundle(input, output, overwrite): + base_zip = zipfile.ZipFile(input) + + tmpdir = tempfile.mkdtemp() + tmp_base_zip = os.path.join(tmpdir, 'base.zip') + tmp_bundle_config = os.path.join(tmpdir, 'bundle_config.json') + + bundle_config = None + abi = [] + + # This block performs three tasks + # - extract/load bundle_config.json from input => bundle_config + # - get ABI from input => abi + # - discard bundle_config.json from input => tmp/base.zip + with zipfile.ZipFile(tmp_base_zip, 'a') as out: + for info in base_zip.infolist(): + + # discard bundle_config.json + if info.filename == 'bundle_config.json': + bundle_config = json.load(base_zip.open(info.filename)) + continue + + # get ABI from apex/{abi}.img + dir, basename = os.path.split(info.filename) + name, ext = os.path.splitext(basename) + if dir == 'apex' and ext == '.img': + abi.append(name) + + # copy entries to tmp/base.zip + out.writestr(info, base_zip.open(info.filename).read()) + + base_zip.close() + + if not bundle_config: + raise ValueError(f'bundle_config.json not found in {input}') + if len(abi) != 1: + raise ValueError(f'{input} should have only a single apex/*.img file') + + # add ABI to tmp/bundle_config.json + apex_config = bundle_config['apex_config'] + if 'supported_abi_set' not in apex_config: + apex_config['supported_abi_set'] = [] + supported_abi_set = apex_config['supported_abi_set'] + supported_abi_set.append({'abi': abi}) + + with open(tmp_bundle_config, 'w') as out: + json.dump(bundle_config, out) + + # invoke bundletool + cmd = [ + 'bundletool', 'build-bundle', '--config', tmp_bundle_config, '--modules', + tmp_base_zip, '--output', output + ] + if overwrite: + cmd.append('--overwrite') + subprocess.check_call(cmd) + + +def main(): + """Program entry point.""" + try: + args = parse_args() + build_bundle(args.input, args.output, args.overwrite) + + # pylint: disable=broad-except + except Exception as err: + print('error: ' + str(err), file=sys.stderr) + sys.exit(-1) + + +if __name__ == '__main__': + main() |