OLD | NEW |
---|---|
1 # Copyright 2013 The Chromium Authors. All rights reserved. | 1 # Copyright 2013 The Chromium Authors. All rights reserved. |
2 # Use of this source code is governed by a BSD-style license that can be | 2 # Use of this source code is governed by a BSD-style license that can be |
3 # found in the LICENSE file. | 3 # found in the LICENSE file. |
4 | 4 |
5 import ast | 5 import ast |
6 import contextlib | 6 import contextlib |
7 import fnmatch | 7 import fnmatch |
8 import json | 8 import json |
9 import os | 9 import os |
10 import pipes | 10 import pipes |
11 import re | 11 import re |
12 import shlex | 12 import shlex |
13 import shutil | 13 import shutil |
14 import subprocess | 14 import subprocess |
15 import sys | 15 import sys |
16 import tempfile | 16 import tempfile |
17 import zipfile | 17 import zipfile |
18 | 18 |
19 | 19 |
20 CHROMIUM_SRC = os.path.normpath( | 20 CHROMIUM_SRC = os.path.normpath( |
21 os.path.join(os.path.dirname(__file__), | 21 os.path.join(os.path.dirname(__file__), |
22 os.pardir, os.pardir, os.pardir, os.pardir)) | 22 os.pardir, os.pardir, os.pardir, os.pardir)) |
23 COLORAMA_ROOT = os.path.join(CHROMIUM_SRC, | 23 COLORAMA_ROOT = os.path.join(CHROMIUM_SRC, |
24 'third_party', 'colorama', 'src') | 24 'third_party', 'colorama', 'src') |
25 # aapt should ignore OWNERS files in addition the default ignore pattern. | 25 # aapt should ignore OWNERS files in addition the default ignore pattern. |
26 AAPT_IGNORE_PATTERN = ('!OWNERS:!.svn:!.git:!.ds_store:!*.scc:.*:<dir>_*:' + | 26 AAPT_IGNORE_PATTERN = ('!OWNERS:!.svn:!.git:!.ds_store:!*.scc:.*:<dir>_*:' + |
27 '!CVS:!thumbs.db:!picasa.ini:!*~:!*.d.stamp') | 27 '!CVS:!thumbs.db:!picasa.ini:!*~:!*.d.stamp') |
28 HERMETIC_TIMESTAMP = (2001, 1, 1, 0, 0, 0) | |
28 | 29 |
29 | 30 |
30 @contextlib.contextmanager | 31 @contextlib.contextmanager |
31 def TempDir(): | 32 def TempDir(): |
32 dirname = tempfile.mkdtemp() | 33 dirname = tempfile.mkdtemp() |
33 try: | 34 try: |
34 yield dirname | 35 yield dirname |
35 finally: | 36 finally: |
36 shutil.rmtree(dirname) | 37 shutil.rmtree(dirname) |
37 | 38 |
(...skipping 173 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
211 if no_clobber: | 212 if no_clobber: |
212 output_path = os.path.join(path, name) | 213 output_path = os.path.join(path, name) |
213 if os.path.exists(output_path): | 214 if os.path.exists(output_path): |
214 raise Exception( | 215 raise Exception( |
215 'Path already exists from zip: %s %s %s' | 216 'Path already exists from zip: %s %s %s' |
216 % (zip_path, name, output_path)) | 217 % (zip_path, name, output_path)) |
217 | 218 |
218 z.extractall(path=path) | 219 z.extractall(path=path) |
219 | 220 |
220 | 221 |
221 def DoZip(inputs, output, base_dir): | 222 def DoZip(inputs, output, base_dir=None): |
223 """Creates a zip file from a list of files. | |
224 | |
225 Args: | |
226 inputs: A list of paths to zip, or a list of (zip_path, fs_path) tuples. | |
227 output: Destination .zip file. | |
228 base_dir: Prefix to strip from inputs. | |
229 """ | |
230 input_tuples = [] | |
231 for tup in inputs: | |
232 if isinstance(tup, basestring): | |
233 tup = (os.path.relpath(tup, base_dir), tup) | |
234 input_tuples.append(tup) | |
235 | |
236 # Sort by zip path to ensure stable zip ordering. | |
237 input_tuples.sort(key=lambda tup: tup[0]) | |
222 with zipfile.ZipFile(output, 'w') as outfile: | 238 with zipfile.ZipFile(output, 'w') as outfile: |
223 for f in inputs: | 239 for zip_path, fs_path in input_tuples: |
224 CheckZipPath(os.path.relpath(f, base_dir)) | 240 CheckZipPath(zip_path) |
225 outfile.write(f, os.path.relpath(f, base_dir)) | 241 zipinfo = zipfile.ZipInfo(filename=zip_path, date_time=HERMETIC_TIMESTAMP) |
jbudorick
2015/09/01 18:11:33
Maybe I'm not following ... if the inputs are the
agrieve
2015/09/01 19:06:16
The most relevant example is in our process_resour
| |
242 with file(fs_path) as f: | |
243 contents = f.read() | |
244 outfile.writestr(zipinfo, contents) | |
226 | 245 |
227 | 246 |
228 def ZipDir(output, base_dir): | 247 def ZipDir(output, base_dir): |
229 with zipfile.ZipFile(output, 'w') as outfile: | 248 """Creates a zip file from a directory.""" |
230 for root, _, files in os.walk(base_dir): | 249 inputs = [] |
231 for f in files: | 250 for root, _, files in os.walk(base_dir): |
232 path = os.path.join(root, f) | 251 for f in files: |
233 archive_path = os.path.relpath(path, base_dir) | 252 inputs.append(os.path.join(root, f)) |
234 CheckZipPath(archive_path) | 253 DoZip(inputs, output, base_dir) |
235 outfile.write(path, archive_path) | |
236 | 254 |
237 | 255 |
238 def MergeZips(output, inputs, exclude_patterns=None): | 256 def MergeZips(output, inputs, exclude_patterns=None, path_transform=None): |
257 path_transform = path_transform or (lambda p, z: p) | |
239 added_names = set() | 258 added_names = set() |
240 def Allow(name): | 259 def Allow(name): |
241 if exclude_patterns is not None: | 260 if exclude_patterns is not None: |
242 for p in exclude_patterns: | 261 for p in exclude_patterns: |
243 if fnmatch.fnmatch(name, p): | 262 if fnmatch.fnmatch(name, p): |
244 return False | 263 return False |
245 return True | 264 return True |
246 | 265 |
247 with zipfile.ZipFile(output, 'w') as out_zip: | 266 with zipfile.ZipFile(output, 'w') as out_zip: |
248 for in_file in inputs: | 267 for in_file in inputs: |
249 with zipfile.ZipFile(in_file, 'r') as in_zip: | 268 with zipfile.ZipFile(in_file, 'r') as in_zip: |
250 for name in in_zip.namelist(): | 269 for name in in_zip.namelist(): |
251 if name not in added_names and Allow(name): | 270 dst_name = path_transform(name, in_file) |
252 out_zip.writestr(name, in_zip.read(name)) | 271 if dst_name not in added_names and Allow(name): |
253 added_names.add(name) | 272 zipinfo = zipfile.ZipInfo(filename=dst_name, |
273 date_time=HERMETIC_TIMESTAMP) | |
274 out_zip.writestr(zipinfo, in_zip.read(name)) | |
275 added_names.add(dst_name) | |
254 | 276 |
255 | 277 |
256 def PrintWarning(message): | 278 def PrintWarning(message): |
257 print 'WARNING: ' + message | 279 print 'WARNING: ' + message |
258 | 280 |
259 | 281 |
260 def PrintBigWarning(message): | 282 def PrintBigWarning(message): |
261 print '***** ' * 8 | 283 print '***** ' * 8 |
262 PrintWarning(message) | 284 PrintWarning(message) |
263 print '***** ' * 8 | 285 print '***** ' * 8 |
(...skipping 103 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
367 file_jsons[file_path] = ReadJson(file_path) | 389 file_jsons[file_path] = ReadJson(file_path) |
368 | 390 |
369 expansion = file_jsons[file_path] | 391 expansion = file_jsons[file_path] |
370 for k in lookup_path[1:]: | 392 for k in lookup_path[1:]: |
371 expansion = expansion[k] | 393 expansion = expansion[k] |
372 | 394 |
373 new_args[i] = arg[:match.start()] + str(expansion) | 395 new_args[i] = arg[:match.start()] + str(expansion) |
374 | 396 |
375 return new_args | 397 return new_args |
376 | 398 |
OLD | NEW |