| # pylint: disable=g-direct-third-party-import |
| # Copyright 2017 The Bazel Authors. All rights reserved. |
| # |
| # 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 for extracting resource files from an AAR. |
| |
| An AAR may contain resources under the /res directory. This tool extracts all |
| of the resources into a directory. If no resources exist, it creates an |
| empty.xml file that defines no resources. |
| |
| In the future, this script may be extended to also extract assets. |
| """ |
| |
| import os |
| import sys |
| import zipfile |
| |
| from tools.android import junction |
| from third_party.py import gflags |
| |
| FLAGS = gflags.FLAGS |
| |
| gflags.DEFINE_string("input_aar", None, "Input AAR") |
| gflags.MarkFlagAsRequired("input_aar") |
| gflags.DEFINE_string("output_res_dir", None, "Output resources directory") |
| gflags.MarkFlagAsRequired("output_res_dir") |
| gflags.DEFINE_string("output_assets_dir", None, "Output assets directory") |
| |
| |
| def ExtractResources(aar, output_res_dir): |
| """Extract resource from an `aar` file to the `output_res_dir` directory.""" |
| aar_contains_no_resources = True |
| output_res_dir_abs = os.path.abspath(output_res_dir) |
| for name in aar.namelist(): |
| if name.startswith("res/") and not name.endswith("/"): |
| ExtractOneFile(aar, name, output_res_dir_abs) |
| aar_contains_no_resources = False |
| if aar_contains_no_resources: |
| empty_xml_filename = output_res_dir + "/res/values/empty.xml" |
| WriteFileWithJunctions(empty_xml_filename, b"<resources/>") |
| |
| |
| def ExtractAssets(aar, output_assets_dir): |
| """Extracts assets from an `aar` file to the `output_assets_dir` directory.""" |
| aar_contains_no_assets = True |
| output_assets_dir_abs = os.path.abspath(output_assets_dir) |
| for name in aar.namelist(): |
| if name.startswith("assets/") and not name.endswith("/"): |
| ExtractOneFile(aar, name, output_assets_dir_abs) |
| aar_contains_no_assets = False |
| if aar_contains_no_assets: |
| # aapt will ignore this file and not print an error message, because it |
| # thinks that it is a swap file. We need to create at least one file so that |
| # Bazel does not complain that the output tree artifact was not created. |
| empty_asset_filename = (output_assets_dir + |
| "/assets/empty_asset_generated_by_bazel~") |
| WriteFileWithJunctions(empty_asset_filename, b"") |
| |
| |
| def WriteFileWithJunctions(filename, content): |
| """Writes file including creating any junctions or directories necessary.""" |
| def _WriteFile(filename): |
| with open(filename, "wb") as openfile: |
| openfile.write(content) |
| |
| if os.name == "nt": |
| # Create a junction to the parent directory, because its path might be too |
| # long. Creating the junction also creates all parent directories. |
| with junction.TempJunction(os.path.dirname(filename)) as junc: |
| filename = os.path.join(junc, os.path.basename(filename)) |
| # Write the file within scope of the TempJunction, otherwise the path in |
| # `filename` would no longer be valid. |
| _WriteFile(filename) |
| else: |
| os.makedirs(os.path.dirname(filename)) |
| _WriteFile(filename) |
| |
| |
| def ExtractOneFile(aar, name, abs_output_dir): |
| """Extract one file from the aar to the output directory.""" |
| if os.name == "nt": |
| fullpath = os.path.normpath(os.path.join(abs_output_dir, name)) |
| if name[-1] == "/": |
| # The zip entry is a directory. Create a junction to it, which also |
| # takes care of creating the directory and all of its parents in a |
| # longpath-safe manner. |
| # We must pretend to have extracted this directory, even if it's |
| # empty, therefore we mustn't rely on creating it as a parent |
| # directory of a subsequently extracted zip entry (because there may |
| # be no such subsequent entry). |
| with junction.TempJunction(fullpath.rstrip("/")) as juncpath: |
| pass |
| else: |
| # The zip entry is a file. Create a junction to its parent directory, |
| # then open the compressed entry as a file object, so we can extract |
| # the data even if the extracted file's path would be too long. |
| # The tradeoff is that we lose the permission bits of the compressed |
| # file, but Unix permissions don't mean much on Windows anyway. |
| with junction.TempJunction(os.path.dirname(fullpath)) as juncpath: |
| extracted_path = os.path.join(juncpath, os.path.basename(fullpath)) |
| with aar.open(name) as src_fd: |
| with open(extracted_path, "wb") as dest_fd: |
| dest_fd.write(src_fd.read()) |
| else: |
| aar.extract(name, abs_output_dir) |
| |
| |
| def main(): |
| with zipfile.ZipFile(FLAGS.input_aar, "r") as aar: |
| ExtractResources(aar, FLAGS.output_res_dir) |
| if FLAGS.output_assets_dir is not None: |
| ExtractAssets(aar, FLAGS.output_assets_dir) |
| |
| if __name__ == "__main__": |
| FLAGS(sys.argv) |
| main() |