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

Side by Side Diff: gsutil.py

Issue 1346213003: gsutil: Parallel-safe, specify target, add clean. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/tools/depot_tools
Patch Set: Created 5 years, 3 months 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 | tests/gsutil_test.py » ('j') | 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 2014 The Chromium Authors. All rights reserved. 2 # Copyright 2014 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 """Run a pinned gsutil.""" 6 """Run a pinned gsutil."""
7 7
8 8
9 import argparse 9 import argparse
10 import base64 10 import base64
11 import hashlib 11 import hashlib
12 import json 12 import json
13 import os 13 import os
14 import shutil 14 import shutil
15 import subprocess 15 import subprocess
16 import sys 16 import sys
17 import time
17 import urllib2 18 import urllib2
18 import zipfile 19 import zipfile
19 20
20 21
21 GSUTIL_URL = 'https://storage.googleapis.com/pub/' 22 GSUTIL_URL = 'https://storage.googleapis.com/pub/'
22 API_URL = 'https://www.googleapis.com/storage/v1/b/pub/o/' 23 API_URL = 'https://www.googleapis.com/storage/v1/b/pub/o/'
23 24
24 THIS_DIR = os.path.dirname(os.path.abspath(__file__)) 25 THIS_DIR = os.path.dirname(os.path.abspath(__file__))
25 DEFAULT_BIN_DIR = os.path.join(THIS_DIR, 'external_bin', 'gsutil') 26 DEFAULT_BIN_DIR = os.path.join(THIS_DIR, 'external_bin', 'gsutil')
26 DEFAULT_FALLBACK_GSUTIL = os.path.join( 27 DEFAULT_FALLBACK_GSUTIL = os.path.join(
27 THIS_DIR, 'third_party', 'gsutil', 'gsutil') 28 THIS_DIR, 'third_party', 'gsutil', 'gsutil')
28 29
29
30 class InvalidGsutilError(Exception): 30 class InvalidGsutilError(Exception):
31 pass 31 pass
32 32
33 33
34 def download_gsutil(version, target_dir): 34 def download_gsutil(version, target_dir):
35 """Downloads gsutil into the target_dir.""" 35 """Downloads gsutil into the target_dir."""
36 filename = 'gsutil_%s.zip' % version 36 filename = 'gsutil_%s.zip' % version
37 target_filename = os.path.join(target_dir, filename) 37 target_filename = os.path.join(target_dir, filename)
38 38
39 # Check if the target exists already. 39 # Check if the target exists already.
(...skipping 26 matching lines...) Expand all
66 f.write(buf) 66 f.write(buf)
67 return target_filename 67 return target_filename
68 68
69 69
70 def check_gsutil(gsutil_bin): 70 def check_gsutil(gsutil_bin):
71 """Run gsutil version and make sure it runs.""" 71 """Run gsutil version and make sure it runs."""
72 return subprocess.call( 72 return subprocess.call(
73 [sys.executable, gsutil_bin, 'version'], 73 [sys.executable, gsutil_bin, 'version'],
74 stdout=subprocess.PIPE, stderr=subprocess.STDOUT) == 0 74 stdout=subprocess.PIPE, stderr=subprocess.STDOUT) == 0
75 75
76 def ensure_gsutil(version, target): 76 def ensure_gsutil(version, target, clean):
77 bin_dir = os.path.join(target, 'gsutil_%s' % version) 77 bin_dir = os.path.join(target, 'gsutil_%s' % version)
78 gsutil_bin = os.path.join(bin_dir, 'gsutil', 'gsutil') 78 gsutil_bin = os.path.join(bin_dir, 'gsutil', 'gsutil')
79 if os.path.isfile(gsutil_bin) and check_gsutil(gsutil_bin): 79 if not clean and os.path.isfile(gsutil_bin) and check_gsutil(gsutil_bin):
80 # Everything is awesome! we're all done here. 80 # Everything is awesome! we're all done here.
81 return gsutil_bin 81 return gsutil_bin
82 82
83 if os.path.isdir(bin_dir): 83 # Clean up if we're redownloading a corrupted gsutil.
84 # Clean up if we're redownloading a corrupted gsutil. 84 suffix = '%d_%d' % (time.time(), os.getpid())
vapier 2015/09/17 19:31:00 this isn't parallel safe. you cannot rely on PID'
dnj 2015/09/17 19:51:47 Using "tempfile.mkdtemp" now.
85 shutil.rmtree(bin_dir) 85 try:
86 cache_dir = os.path.join(target, '.cache_dir') 86 cleanup_path = '%s.%s' % (bin_dir, suffix)
87 shutil.move(bin_dir, cleanup_path)
88 shutil.rmtree(cleanup_path)
89 except IOError:
90 # Directory was not present.
91 pass
92
93 cache_dir = os.path.join(target, '.cache_dir.%s' % (suffix,))
Ryan Tseng 2015/09/17 19:26:45 this would be more aptly called "tempdir" now? ma
dnj 2015/09/17 19:51:46 Acknowledged.
87 if not os.path.isdir(cache_dir): 94 if not os.path.isdir(cache_dir):
88 os.makedirs(cache_dir) 95 os.makedirs(cache_dir)
89 target_zip_filename = download_gsutil(version, cache_dir) 96 target_zip_filename = download_gsutil(version, cache_dir)
90 with zipfile.ZipFile(target_zip_filename, 'r') as target_zip: 97 with zipfile.ZipFile(target_zip_filename, 'r') as target_zip:
91 target_zip.extractall(bin_dir) 98 target_zip.extractall(cache_dir)
99
100 try:
101 shutil.move(cache_dir, bin_dir)
102 except IOError:
103 # Something else did this in parallel.
104 pass
92 105
93 # Final check that the gsutil bin is okay. This should never fail. 106 # Final check that the gsutil bin is okay. This should never fail.
94 if not check_gsutil(gsutil_bin): 107 if not check_gsutil(gsutil_bin):
95 raise InvalidGsutilError() 108 raise InvalidGsutilError()
96 109
97 return gsutil_bin 110 return gsutil_bin
98 111
99 112
100 def run_gsutil(force_version, fallback, target, args): 113 def run_gsutil(force_version, fallback, target, args, clean=False):
101 if force_version: 114 if force_version:
102 gsutil_bin = ensure_gsutil(force_version, target) 115 gsutil_bin = ensure_gsutil(force_version, target, clean)
103 else: 116 else:
104 gsutil_bin = fallback 117 gsutil_bin = fallback
105 cmd = [sys.executable, gsutil_bin] + args 118 cmd = [sys.executable, gsutil_bin] + args
106 return subprocess.call(cmd) 119 return subprocess.call(cmd)
107 120
108 121
109 def parse_args(): 122 def parse_args():
123 bin_dir = os.environ.get('DEPOT_TOOLS_GSUTIL_BIN_DIR')
Ryan Tseng 2015/09/17 19:26:45 get(key, default)
dnj 2015/09/17 19:51:47 Done.
124 if not bin_dir:
125 bin_dir = DEFAULT_BIN_DIR
126
110 parser = argparse.ArgumentParser() 127 parser = argparse.ArgumentParser()
111 parser.add_argument('--force-version', default='4.13') 128 parser.add_argument('--force-version', default='4.13')
129 parser.add_argument('--clean', action='store_true',
Ryan Tseng 2015/09/17 19:26:45 What do you see the use case for this being?
dnj 2015/09/17 19:51:47 Useful for testing massively in parallel.
130 help='Clear any existing gsutil package, forcing a new download.')
112 parser.add_argument('--fallback', default=DEFAULT_FALLBACK_GSUTIL) 131 parser.add_argument('--fallback', default=DEFAULT_FALLBACK_GSUTIL)
113 parser.add_argument('--target', default=DEFAULT_BIN_DIR) 132 parser.add_argument('--target', default=bin_dir,
133 help='The target directory to download/store a gsutil version in. '
134 '(default is %(default)s).')
114 parser.add_argument('args', nargs=argparse.REMAINDER) 135 parser.add_argument('args', nargs=argparse.REMAINDER)
115 136
116 args, extras = parser.parse_known_args() 137 args, extras = parser.parse_known_args()
117 if args.args and args.args[0] == '--': 138 if args.args and args.args[0] == '--':
118 args.args.pop(0) 139 args.args.pop(0)
119 if extras: 140 if extras:
120 args.args = extras + args.args 141 args.args = extras + args.args
121 return args.force_version, args.fallback, args.target, args.args 142 return args
122 143
123 144
124 def main(): 145 def main():
125 force_version, fallback, target, args = parse_args() 146 args = parse_args()
126 return run_gsutil(force_version, fallback, target, args) 147 return run_gsutil(args.force_version, args.fallback, args.target, args.args,
148 clean=args.clean)
127 149
128 if __name__ == '__main__': 150 if __name__ == '__main__':
129 sys.exit(main()) 151 sys.exit(main())
OLDNEW
« no previous file with comments | « no previous file | tests/gsutil_test.py » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698