blob: 78e668573797e6945590303019c7e3ae06800346 [file] [log] [blame]
László Csomor5f99fda2017-08-11 09:28:12 +02001# pylint: disable=g-direct-third-party-import
Philipp Wollermann4c558982017-07-27 18:01:12 +02002# pylint: disable=g-bad-file-header
3# Copyright 2017 The Bazel Authors. All rights reserved.
4#
5# Licensed under the Apache License, Version 2.0 (the "License");
6# you may not use this file except in compliance with the License.
7# You may obtain a copy of the License at
8#
9# http:#www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an "AS IS" BASIS,
13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14# See the License for the specific language governing permissions and
15# limitations under the License.
16"""Creates the embedded_tools.zip that is part of the Bazel binary."""
17
hlopko5a661c72017-08-09 12:03:03 +020018import contextlib
Philipp Wollermann4c558982017-07-27 18:01:12 +020019import fnmatch
20import os
21import os.path
22import re
Philipp Wollermann4c558982017-07-27 18:01:12 +020023import sys
Philipp Wollermann4c558982017-07-27 18:01:12 +020024import zipfile
25
László Csomor5f99fda2017-08-11 09:28:12 +020026from src.create_embedded_tools_lib import copy_tar_to_zip
27from src.create_embedded_tools_lib import copy_zip_to_zip
28from src.create_embedded_tools_lib import is_executable
29
Philipp Wollermann4c558982017-07-27 18:01:12 +020030output_paths = [
iirinad4651e82019-03-15 03:15:37 -070031 ('*tools/jdk/BUILD', lambda x: 'tools/jdk/BUILD'),
Klaus Aehlig6f52fca2019-03-18 03:43:40 -070032 ('*tools/build_defs/repo/BUILD.repo',
33 lambda x: 'tools/build_defs/repo/BUILD'),
philwoe67c9612019-05-06 04:28:48 -070034 ('*tools/platforms/BUILD.tools', lambda x: 'platforms/BUILD'),
cpeyser8613c902017-09-01 00:15:29 +020035 ('*tools/platforms/*', lambda x: 'platforms/' + os.path.basename(x)),
Laszlo Csomorea642d62018-09-03 05:37:04 -070036 ('*tools/cpp/runfiles/generated_*',
37 lambda x: 'tools/cpp/runfiles/' + os.path.basename(x)[len('generated_'):]),
iirina5bd55432019-01-16 06:36:56 -080038 ('*jarjar_command_deploy.jar',
39 lambda x: 'tools/jdk/jarjar_command_deploy.jar'),
iirinad4651e82019-03-15 03:15:37 -070040 ('*BUILD.java_langtools', lambda x: 'third_party/java/jdk/langtools/BUILD'),
Philipp Wollermann4c558982017-07-27 18:01:12 +020041 ('*launcher.exe', lambda x: 'tools/launcher/launcher.exe'),
Yun Peng394211b2017-09-15 15:59:14 +020042 ('*def_parser.exe', lambda x: 'tools/def_parser/def_parser.exe'),
Philipp Wollermann4c558982017-07-27 18:01:12 +020043 ('*zipper.exe', lambda x: 'tools/zip/zipper/zipper.exe'),
44 ('*zipper', lambda x: 'tools/zip/zipper/zipper'),
iirinaba3ebcb2019-03-21 08:02:56 -070045 ('*third_party/jarjar/BUILD.tools', lambda x: 'third_party/jarjar/BUILD'),
46 ('*third_party/jarjar/LICENSE', lambda x: 'third_party/jarjar/LICENSE'),
Philipp Wollermann4c558982017-07-27 18:01:12 +020047 ('*src/objc_tools/*',
48 lambda x: 'tools/objc/precomp_' + os.path.basename(x)),
49 ('*xcode*StdRedirect.dylib', lambda x: 'tools/objc/StdRedirect.dylib'),
50 ('*xcode*make_hashed_objlist.py',
51 lambda x: 'tools/objc/make_hashed_objlist.py'),
52 ('*xcode*realpath', lambda x: 'tools/objc/realpath'),
53 ('*xcode*xcode-locator', lambda x: 'tools/objc/xcode-locator'),
philwoe67c9612019-05-06 04:28:48 -070054 ('*src/tools/xcode/*', lambda x: 'tools/objc/' + os.path.basename(x)),
Philipp Wollermann4c558982017-07-27 18:01:12 +020055 ('*external/openjdk_*/file/*.tar.gz', lambda x: 'jdk.tar.gz'),
56 ('*external/openjdk_*/file/*.zip', lambda x: 'jdk.zip'),
twerth683922c2018-12-20 05:12:08 -080057 ('*src/minimal_jdk.tar.gz', lambda x: 'jdk.tar.gz'),
58 ('*src/minimal_jdk.zip', lambda x: 'jdk.zip'),
laszlocsomora6545052019-11-05 05:18:32 -080059 ('*.bzl.tools', lambda x: x[:-6]),
Philipp Wollermann4c558982017-07-27 18:01:12 +020060 ('*', lambda x: re.sub(r'^.*bazel-out/[^/]*/bin/', '', x, count=1)),
61]
62
63
64def get_output_path(path):
65 for pattern, transformer in output_paths:
66 if fnmatch.fnmatch(path.replace('\\', '/'), pattern):
67 # BUILD.tools are stored as BUILD files.
68 return transformer(path).replace('/BUILD.tools', '/BUILD')
69
70
Philipp Wollermann4c558982017-07-27 18:01:12 +020071def get_input_files(argsfile):
philwoe67c9612019-05-06 04:28:48 -070072 """Returns a dict of archive_file to input_file.
Philipp Wollermann4c558982017-07-27 18:01:12 +020073
74 This describes the files that should be put into the generated archive.
75
76 Args:
77 argsfile: The file containing the list of input files.
philwoe67c9612019-05-06 04:28:48 -070078
79 Raises:
80 ValueError: When two input files map to the same output file.
Philipp Wollermann4c558982017-07-27 18:01:12 +020081 """
82 with open(argsfile, 'r') as f:
philwoe67c9612019-05-06 04:28:48 -070083 input_files = sorted(set(x.strip() for x in f.readlines()))
Philipp Wollermann4c558982017-07-27 18:01:12 +020084
85 result = {}
86 for input_file in input_files:
87 # If we have both a BUILD and a BUILD.tools file, take the latter only.
88 if (os.path.basename(input_file) == 'BUILD' and
89 input_file + '.tools' in input_files):
90 continue
91
philwoe67c9612019-05-06 04:28:48 -070092 # It's an error to have two files map to the same output file, because the
93 # result is hard to predict and can easily be wrong.
94 output_path = get_output_path(input_file)
95 if output_path in result:
96 raise ValueError(
97 'Duplicate output file: Both {} and {} map to {}'.format(
98 result[output_path], input_file, output_path))
99 result[output_path] = input_file
Philipp Wollermann4c558982017-07-27 18:01:12 +0200100
philwoe67c9612019-05-06 04:28:48 -0700101 return result
Philipp Wollermann4c558982017-07-27 18:01:12 +0200102
103
104def copy_jdk_into_archive(output_zip, archive_file, input_file):
Laszlo Csomorea642d62018-09-03 05:37:04 -0700105 """Extract the JDK and adds it to the archive under jdk/*."""
László Csomor5f99fda2017-08-11 09:28:12 +0200106
107 def _replace_dirname(filename):
108 # Rename the first folder to 'jdk', because Bazel looks for a
109 # bundled JDK in the embedded tools using that folder name.
110 return 'jdk/' + '/'.join(filename.split('/')[1:])
111
Philipp Wollermann4c558982017-07-27 18:01:12 +0200112 # The JDK is special - it's extracted instead of copied.
113 if archive_file.endswith('.tar.gz'):
László Csomor5f99fda2017-08-11 09:28:12 +0200114 copy_tar_to_zip(output_zip, input_file, _replace_dirname)
Philipp Wollermann4c558982017-07-27 18:01:12 +0200115 elif archive_file.endswith('.zip'):
László Csomor5f99fda2017-08-11 09:28:12 +0200116 copy_zip_to_zip(output_zip, input_file, _replace_dirname)
Philipp Wollermann4c558982017-07-27 18:01:12 +0200117
118
119def main():
120 output_zip = os.path.join(os.getcwd(), sys.argv[1])
121 input_files = get_input_files(sys.argv[2])
122
123 # Copy all the input_files into output_zip.
hlopko5a661c72017-08-09 12:03:03 +0200124 # Adding contextlib.closing to be python 2.6 (for centos 6.7) compatible
125 with contextlib.closing(
126 zipfile.ZipFile(output_zip, 'w', zipfile.ZIP_DEFLATED)) as output_zip:
Philipp Wollermann4c558982017-07-27 18:01:12 +0200127 zipinfo = zipfile.ZipInfo('WORKSPACE', (1980, 1, 1, 0, 0, 0))
128 zipinfo.external_attr = 0o644 << 16
129 output_zip.writestr(zipinfo, 'workspace(name = "bazel_tools")\n')
130
philwoe67c9612019-05-06 04:28:48 -0700131 # By sorting the file list, the resulting ZIP file will be reproducible and
132 # deterministic.
133 for archive_file, input_file in sorted(input_files.items()):
Philipp Wollermann4c558982017-07-27 18:01:12 +0200134 if os.path.basename(archive_file) in ('jdk.tar.gz', 'jdk.zip'):
135 copy_jdk_into_archive(output_zip, archive_file, input_file)
136 else:
137 zipinfo = zipfile.ZipInfo(archive_file, (1980, 1, 1, 0, 0, 0))
138 zipinfo.external_attr = 0o755 << 16 if is_executable(
139 input_file) else 0o644 << 16
140 zipinfo.compress_type = zipfile.ZIP_DEFLATED
141 with open(input_file, 'rb') as f:
142 output_zip.writestr(zipinfo, f.read())
143
144
145if __name__ == '__main__':
146 main()