|  | #!/usr/bin/env python | 
|  | # Copyright (c) 2014 The WebRTC project authors. All Rights Reserved. | 
|  | # | 
|  | # Use of this source code is governed by a BSD-style license | 
|  | # that can be found in the LICENSE file in the root of the source | 
|  | # tree. An additional intellectual property rights grant can be found | 
|  | # in the file PATENTS.  All contributing project authors may | 
|  | # be found in the AUTHORS file in the root of the source tree. | 
|  |  | 
|  | """Script to download a Chromium checkout into the workspace. | 
|  |  | 
|  | The script downloads a full Chromium Git clone and its DEPS. | 
|  |  | 
|  | The following environment variable can be used to alter the behavior: | 
|  | * CHROMIUM_NO_HISTORY - If set to 1, a Git checkout with no history will be | 
|  | downloaded. This is consumes less bandwidth and disk space but is known to be | 
|  | slower in general if you have a high-speed connection. | 
|  |  | 
|  | After a successful sync has completed, a .last_sync_chromium file is written to | 
|  | the chromium directory. While it exists, no more gclient sync operations will be | 
|  | performed until the --target-revision changes or the SCRIPT_VERSION constant is | 
|  | incremented. The file can be removed manually to force a new sync. | 
|  | """ | 
|  |  | 
|  | import argparse | 
|  | import os | 
|  | import shutil | 
|  | import subprocess | 
|  | import sys | 
|  | import textwrap | 
|  |  | 
|  | # Bump this whenever the algorithm changes and you need bots/devs to re-sync, | 
|  | # ignoring the .last_sync_chromium file | 
|  | SCRIPT_VERSION = 8 | 
|  |  | 
|  | ROOT_DIR = os.path.dirname(os.path.abspath(__file__)) | 
|  | CHROMIUM_NO_HISTORY = 'CHROMIUM_NO_HISTORY' | 
|  |  | 
|  | # Duplicated from depot_tools/gclient.py since we cannot depend on that: | 
|  | DEPS_OS_CHOICES = { | 
|  | "win32": "win", | 
|  | "win": "win", | 
|  | "cygwin": "win", | 
|  | "darwin": "mac", | 
|  | "mac": "mac", | 
|  | "unix": "unix", | 
|  | "linux": "unix", | 
|  | "linux2": "unix", | 
|  | "linux3": "unix", | 
|  | "android": "android", | 
|  | } | 
|  |  | 
|  | def _parse_gclient_dict(): | 
|  | gclient_dict = {} | 
|  | try: | 
|  | main_gclient = os.path.join(os.path.dirname(ROOT_DIR), '.gclient') | 
|  | with open(main_gclient, 'rb') as deps_content: | 
|  | exec(deps_content, gclient_dict) | 
|  | except Exception as e: | 
|  | print >> sys.stderr, 'error while parsing .gclient:', e | 
|  | return gclient_dict | 
|  |  | 
|  |  | 
|  | def get_cache_dir(): | 
|  | return _parse_gclient_dict().get('cache_dir') | 
|  |  | 
|  |  | 
|  | def get_target_os_list(): | 
|  | # Always add the currently running OS since the --deps option will override | 
|  | # that if specified: | 
|  | target_os_list = [DEPS_OS_CHOICES.get(sys.platform, 'unix')] | 
|  | # Add any target_os entries from .gclient. | 
|  | target_os_list += _parse_gclient_dict().get('target_os', []) | 
|  | return ','.join(target_os_list) | 
|  |  | 
|  |  | 
|  | def main(): | 
|  | CR_DIR = os.path.join(ROOT_DIR, 'chromium') | 
|  |  | 
|  | p = argparse.ArgumentParser() | 
|  | p.add_argument('--target-revision', required=True, | 
|  | help='The target chromium git revision [REQUIRED]') | 
|  | p.add_argument('--chromium-dir', default=CR_DIR, | 
|  | help=('The path to the chromium directory to sync ' | 
|  | '(default: %(default)r)')) | 
|  | opts = p.parse_args() | 
|  | opts.chromium_dir = os.path.abspath(opts.chromium_dir) | 
|  |  | 
|  | target_os_list = get_target_os_list() | 
|  |  | 
|  | # Do a quick check to see if we were successful last time to make runhooks | 
|  | # sooper fast. | 
|  | flag_file = os.path.join(opts.chromium_dir, '.last_sync_chromium') | 
|  | flag_file_content = '\n'.join([ | 
|  | str(SCRIPT_VERSION), | 
|  | opts.target_revision, | 
|  | repr(target_os_list), | 
|  | ]) | 
|  | if (os.path.exists(os.path.join(opts.chromium_dir, 'src')) and | 
|  | os.path.exists(flag_file)): | 
|  | with open(flag_file, 'r') as f: | 
|  | if f.read() == flag_file_content: | 
|  | print 'Chromium already up to date: ', opts.target_revision | 
|  | return 0 | 
|  | os.unlink(flag_file) | 
|  |  | 
|  | # Workaround to avoid sync failure due move in | 
|  | # https://codereview.chromium.org/1155743013 | 
|  | # TODO(kjellander): Remove this after the summer of 2015. | 
|  | freetype_src = os.path.join(CR_DIR, 'src', 'third_party', 'freetype-android', | 
|  | 'src') | 
|  | if os.path.isdir(freetype_src): | 
|  | shutil.rmtree(freetype_src) | 
|  |  | 
|  | # Avoid downloading NaCl toolchain as part of the Chromium hooks. | 
|  | gclient_cmd = 'gclient.bat' if sys.platform.startswith('win') else 'gclient' | 
|  | args = [ | 
|  | gclient_cmd, 'sync', '--force', '--nohooks', '--revision', | 
|  | 'src@' + opts.target_revision | 
|  | ] | 
|  |  | 
|  | if os.environ.get('CHROME_HEADLESS') == '1': | 
|  | # Running on a buildbot. | 
|  | args.append('-vvv') | 
|  |  | 
|  | if sys.platform.startswith('win'): | 
|  | cache_path = os.path.join(os.path.splitdrive(ROOT_DIR)[0] + os.path.sep, | 
|  | 'b', 'git-cache') | 
|  | else: | 
|  | cache_path = '/b/git-cache' | 
|  | else: | 
|  | # Verbose, but not as verbose as on the buildbots. | 
|  | args.append('-v') | 
|  |  | 
|  | # Support developers setting the cache_dir in .gclient. | 
|  | cache_path = get_cache_dir() | 
|  |  | 
|  | # Allow for users with poor internet connections to download a Git clone | 
|  | # without history (saves several gigs but is generally slower and doesn't work | 
|  | # with the Git cache). | 
|  | if os.environ.get(CHROMIUM_NO_HISTORY) == '1': | 
|  | if cache_path: | 
|  | print >> sys.stderr, ( | 
|  | 'You cannot use "no-history" mode for syncing Chrome (i.e. set the ' | 
|  | '%s environment variable to 1) when you have cache_dir configured in ' | 
|  | 'your .gclient.' % CHROMIUM_NO_HISTORY) | 
|  | return 1 | 
|  | args.append('--no-history') | 
|  | gclient_entries_file = os.path.join(opts.chromium_dir, '.gclient_entries') | 
|  | else: | 
|  | # Write a temporary .gclient file that has the cache_dir variable added. | 
|  | gclientfile = os.path.join(opts.chromium_dir, '.gclient') | 
|  | with open(gclientfile, 'rb') as spec: | 
|  | spec = spec.read().splitlines() | 
|  | spec[-1] = 'cache_dir = %r' % (cache_path,) | 
|  | with open(gclientfile + '.tmp', 'wb') as f: | 
|  | f.write('\n'.join(spec)) | 
|  |  | 
|  | args += [ | 
|  | '--gclientfile', '.gclient.tmp', | 
|  | '--delete_unversioned_trees', '--reset', '--upstream' | 
|  | ] | 
|  | gclient_entries_file = os.path.join(opts.chromium_dir, | 
|  | '.gclient.tmp_entries') | 
|  |  | 
|  | # To avoid gclient sync problems when DEPS entries have been removed we must | 
|  | # wipe the gclient's entries file that contains cached URLs for all DEPS. | 
|  | if os.path.exists(gclient_entries_file): | 
|  | os.unlink(gclient_entries_file) | 
|  |  | 
|  | if target_os_list: | 
|  | args += ['--deps=' + target_os_list] | 
|  |  | 
|  | print textwrap.dedent("""\ | 
|  | +---------------------------------------------------------------------+ | 
|  | | NOTICE: This sync of Chromium will take a long time as several      | | 
|  | |         gigabytes of data are downloaded. If this is your initial   | | 
|  | |         sync and it's interrupted, try running 'gclient sync' again.| | 
|  | |         If that fails, wipe everything clean and start over again.  | | 
|  | +---------------------------------------------------------------------+""") | 
|  | print 'Running "%s" in %s' % (' '.join(args), opts.chromium_dir) | 
|  | ret = subprocess.call(args, cwd=opts.chromium_dir) | 
|  | if ret == 0: | 
|  | with open(flag_file, 'wb') as f: | 
|  | f.write(flag_file_content) | 
|  |  | 
|  | return ret | 
|  |  | 
|  |  | 
|  | if __name__ == '__main__': | 
|  | sys.exit(main()) |