Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(441)

Side by Side Diff: build/extract_from_cab.py

Issue 8890072: Yet another fix for concurrent calls to expand.exe. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: ToT sync Created 9 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « no previous file | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
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())
OLDNEW
« no previous file with comments | « no previous file | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698