Index: build/extract_from_cab.py |
diff --git a/build/extract_from_cab.py b/build/extract_from_cab.py |
new file mode 100755 |
index 0000000000000000000000000000000000000000..080370ca9adee2dd953456d45d29e1fa7398e12e |
--- /dev/null |
+++ b/build/extract_from_cab.py |
@@ -0,0 +1,63 @@ |
+#!/usr/bin/env python |
+# Copyright (c) 2012 The Chromium Authors. All rights reserved. |
+# Use of this source code is governed by a BSD-style license that can be |
+# found in the LICENSE file. |
+ |
+"""Extracts a single file from a CAB archive.""" |
+ |
+import os |
+import shutil |
+import subprocess |
+import sys |
+import tempfile |
+ |
+def run_quiet(*args): |
+ """Run 'expand' suppressing noisy output. Returns returncode from process.""" |
+ popen = subprocess.Popen(args, stdout=subprocess.PIPE) |
+ out, _ = popen.communicate() |
+ if popen.returncode: |
+ # expand emits errors to stdout, so if we fail, then print that out. |
+ print out |
+ return popen.returncode |
+ |
+def main(): |
+ if len(sys.argv) != 4: |
+ print 'Usage: extract_from_cab.py cab_path archived_file output_dir' |
+ return 1 |
+ |
+ [cab_path, archived_file, output_dir] = sys.argv[1:] |
+ |
+ # Expand.exe does its work in a fixed-named temporary directory created within |
+ # the given output directory. This is a problem for concurrent extractions, so |
+ # create a unique temp dir within the desired output directory to work around |
+ # this limitation. |
+ temp_dir = tempfile.mkdtemp(dir=output_dir) |
+ |
+ try: |
+ # Invoke the Windows expand utility to extract the file. |
+ level = run_quiet('expand', cab_path, '-F:' + archived_file, temp_dir) |
+ if level == 0: |
+ # Move the output file into place, preserving expand.exe's behavior of |
+ # paving over any preexisting file. |
+ output_file = os.path.join(output_dir, archived_file) |
+ try: |
+ os.remove(output_file) |
+ except OSError: |
+ pass |
+ os.rename(os.path.join(temp_dir, archived_file), output_file) |
+ finally: |
+ shutil.rmtree(temp_dir, True) |
+ |
+ if level != 0: |
+ return level |
+ |
+ # The expand utility preserves the modification date and time of the archived |
+ # file. Touch the extracted file. This helps build systems that compare the |
+ # modification times of input and output files to determine whether to do an |
+ # action. |
+ os.utime(os.path.join(output_dir, archived_file), None) |
+ return 0 |
+ |
+ |
+if __name__ == '__main__': |
+ sys.exit(main()) |