| OLD | NEW |
| 1 #!/usr/bin/env python | 1 #!/usr/bin/env python |
| 2 # Copyright (c) 2011 The Chromium Authors. All rights reserved. | 2 # Copyright (c) 2011 The Chromium Authors. All rights reserved. |
| 3 # Use of this source code is governed by a BSD-style license that can be | 3 # Use of this source code is governed by a BSD-style license that can be |
| 4 # found in the LICENSE file. | 4 # found in the LICENSE file. |
| 5 | 5 |
| 6 """Extracts a single file from a CAB archive.""" | 6 """Extracts a single file from a CAB archive.""" |
| 7 | 7 |
| 8 import os | 8 import os |
| 9 import shutil |
| 9 import subprocess | 10 import subprocess |
| 10 import sys | 11 import sys |
| 11 import tempfile | 12 import tempfile |
| 12 | 13 |
| 13 lock_file = os.path.join(tempfile.gettempdir(), 'expand.lock') | |
| 14 | |
| 15 | |
| 16 def acquire_lock(): | |
| 17 while True: | |
| 18 try: | |
| 19 fd = os.open(lock_file, os.O_CREAT | os.O_EXCL | os.O_RDWR) | |
| 20 return fd | |
| 21 except OSError as e: | |
| 22 if e.errno != errno.EEXIST: | |
| 23 raise | |
| 24 print 'Cab extraction could not get exclusive lock. Retrying in 100ms...' | |
| 25 time.sleep(0.1) | |
| 26 | |
| 27 | |
| 28 def release_lock(fd): | |
| 29 os.close(fd) | |
| 30 os.unlink(lock_file) | |
| 31 | |
| 32 | 14 |
| 33 def main(): | 15 def main(): |
| 34 if len(sys.argv) != 4: | 16 if len(sys.argv) != 4: |
| 35 print 'Usage: extract_from_cab.py cab_path archived_file output_dir' | 17 print 'Usage: extract_from_cab.py cab_path archived_file output_dir' |
| 36 return 1 | 18 return 1 |
| 37 | 19 |
| 38 [cab_path, archived_file, output_dir] = sys.argv[1:] | 20 [cab_path, archived_file, output_dir] = sys.argv[1:] |
| 39 | 21 |
| 40 lock_fd = acquire_lock() | 22 # Expand.exe does its work in a fixed-named temporary directory created within |
| 23 # the given output directory. This is a problem for concurrent extractions, so |
| 24 # create a unique temp dir within the desired output directory to work around |
| 25 # this limitation. |
| 26 temp_dir = tempfile.mkdtemp(dir=output_dir) |
| 27 |
| 41 try: | 28 try: |
| 42 # Invoke the Windows expand utility to extract the file. | 29 # Invoke the Windows expand utility to extract the file. |
| 43 level = subprocess.call( | 30 level = subprocess.call( |
| 44 ['expand', cab_path, '-F:' + archived_file, output_dir]) | 31 ['expand', cab_path, '-F:' + archived_file, temp_dir]) |
| 45 if level != 0: | 32 if level == 0: |
| 46 return level | 33 # Move the output file into place, preserving expand.exe's behavior of |
| 34 # paving over any preexisting file. |
| 35 output_file = os.path.join(output_dir, archived_file) |
| 36 try: |
| 37 os.remove(output_file) |
| 38 except OSError: |
| 39 pass |
| 40 os.rename(os.path.join(temp_dir, archived_file), output_file) |
| 47 finally: | 41 finally: |
| 48 release_lock(lock_fd) | 42 shutil.rmtree(temp_dir, True) |
| 43 |
| 44 if level != 0: |
| 45 return level |
| 49 | 46 |
| 50 # The expand utility preserves the modification date and time of the archived | 47 # The expand utility preserves the modification date and time of the archived |
| 51 # file. Touch the extracted file. This helps build systems that compare the | 48 # file. Touch the extracted file. This helps build systems that compare the |
| 52 # modification times of input and output files to determine whether to do an | 49 # modification times of input and output files to determine whether to do an |
| 53 # action. | 50 # action. |
| 54 os.utime(os.path.join(output_dir, archived_file), None) | 51 os.utime(os.path.join(output_dir, archived_file), None) |
| 55 return 0 | 52 return 0 |
| 56 | 53 |
| 57 | 54 |
| 58 if __name__ == '__main__': | 55 if __name__ == '__main__': |
| 59 sys.exit(main()) | 56 sys.exit(main()) |
| OLD | NEW |