xref: /aosp_15_r20/external/crosvm/infra/recipes.py (revision bb4ee6a4ae7042d18b07a98463b9c8b875e44b39)
1*bb4ee6a4SAndroid Build Coastguard Worker#!/bin/sh
2*bb4ee6a4SAndroid Build Coastguard Worker# Copyright 2019 The LUCI Authors. All rights reserved.
3*bb4ee6a4SAndroid Build Coastguard Worker# Use of this source code is governed under the Apache License, Version 2.0
4*bb4ee6a4SAndroid Build Coastguard Worker# that can be found in the LICENSE file.
5*bb4ee6a4SAndroid Build Coastguard Worker
6*bb4ee6a4SAndroid Build Coastguard Worker# We want to run python in unbuffered mode; however shebangs on linux grab the
7*bb4ee6a4SAndroid Build Coastguard Worker# entire rest of the shebang line as a single argument, leading to errors like:
8*bb4ee6a4SAndroid Build Coastguard Worker#
9*bb4ee6a4SAndroid Build Coastguard Worker#   /usr/bin/env: 'python3 -u': No such file or directory
10*bb4ee6a4SAndroid Build Coastguard Worker#
11*bb4ee6a4SAndroid Build Coastguard Worker# This little shell hack is a triple-quoted noop in python, but in sh it
12*bb4ee6a4SAndroid Build Coastguard Worker# evaluates to re-exec'ing this script in unbuffered mode.
13*bb4ee6a4SAndroid Build Coastguard Worker# pylint: disable=pointless-string-statement
14*bb4ee6a4SAndroid Build Coastguard Worker''''exec python3 -u -- "$0" ${1+"$@"} # '''
15*bb4ee6a4SAndroid Build Coastguard Worker"""Bootstrap script to clone and forward to the recipe engine tool.
16*bb4ee6a4SAndroid Build Coastguard Worker
17*bb4ee6a4SAndroid Build Coastguard Worker*******************
18*bb4ee6a4SAndroid Build Coastguard Worker** DO NOT MODIFY **
19*bb4ee6a4SAndroid Build Coastguard Worker*******************
20*bb4ee6a4SAndroid Build Coastguard Worker
21*bb4ee6a4SAndroid Build Coastguard WorkerThis is a copy of https://chromium.googlesource.com/infra/luci/recipes-py/+/main/recipes.py.
22*bb4ee6a4SAndroid Build Coastguard WorkerTo fix bugs, fix in the googlesource repo then run the autoroller.
23*bb4ee6a4SAndroid Build Coastguard Worker"""
24*bb4ee6a4SAndroid Build Coastguard Worker
25*bb4ee6a4SAndroid Build Coastguard Worker# pylint: disable=wrong-import-position
26*bb4ee6a4SAndroid Build Coastguard Workerimport argparse
27*bb4ee6a4SAndroid Build Coastguard Workerimport errno
28*bb4ee6a4SAndroid Build Coastguard Workerimport json
29*bb4ee6a4SAndroid Build Coastguard Workerimport logging
30*bb4ee6a4SAndroid Build Coastguard Workerimport os
31*bb4ee6a4SAndroid Build Coastguard Workerimport shutil
32*bb4ee6a4SAndroid Build Coastguard Workerimport subprocess
33*bb4ee6a4SAndroid Build Coastguard Workerimport sys
34*bb4ee6a4SAndroid Build Coastguard Worker
35*bb4ee6a4SAndroid Build Coastguard Workerimport urllib.parse as urlparse
36*bb4ee6a4SAndroid Build Coastguard Worker
37*bb4ee6a4SAndroid Build Coastguard Workerfrom collections import namedtuple
38*bb4ee6a4SAndroid Build Coastguard Worker
39*bb4ee6a4SAndroid Build Coastguard Worker
40*bb4ee6a4SAndroid Build Coastguard Worker# The dependency entry for the recipe_engine in the client repo's recipes.cfg
41*bb4ee6a4SAndroid Build Coastguard Worker#
42*bb4ee6a4SAndroid Build Coastguard Worker# url (str) - the url to the engine repo we want to use.
43*bb4ee6a4SAndroid Build Coastguard Worker# revision (str) - the git revision for the engine to get.
44*bb4ee6a4SAndroid Build Coastguard Worker# branch (str) - the branch to fetch for the engine as an absolute ref (e.g.
45*bb4ee6a4SAndroid Build Coastguard Worker#   refs/heads/main)
46*bb4ee6a4SAndroid Build Coastguard WorkerEngineDep = namedtuple('EngineDep', 'url revision branch')
47*bb4ee6a4SAndroid Build Coastguard Worker
48*bb4ee6a4SAndroid Build Coastguard Worker
49*bb4ee6a4SAndroid Build Coastguard Workerclass MalformedRecipesCfg(Exception):
50*bb4ee6a4SAndroid Build Coastguard Worker
51*bb4ee6a4SAndroid Build Coastguard Worker  def __init__(self, msg, path):
52*bb4ee6a4SAndroid Build Coastguard Worker    full_message = f'malformed recipes.cfg: {msg}: {path!r}'
53*bb4ee6a4SAndroid Build Coastguard Worker    super().__init__(full_message)
54*bb4ee6a4SAndroid Build Coastguard Worker
55*bb4ee6a4SAndroid Build Coastguard Worker
56*bb4ee6a4SAndroid Build Coastguard Workerdef parse(repo_root, recipes_cfg_path):
57*bb4ee6a4SAndroid Build Coastguard Worker  """Parse is a lightweight a recipes.cfg file parser.
58*bb4ee6a4SAndroid Build Coastguard Worker
59*bb4ee6a4SAndroid Build Coastguard Worker  Args:
60*bb4ee6a4SAndroid Build Coastguard Worker    repo_root (str) - native path to the root of the repo we're trying to run
61*bb4ee6a4SAndroid Build Coastguard Worker      recipes for.
62*bb4ee6a4SAndroid Build Coastguard Worker    recipes_cfg_path (str) - native path to the recipes.cfg file to process.
63*bb4ee6a4SAndroid Build Coastguard Worker
64*bb4ee6a4SAndroid Build Coastguard Worker  Returns (as tuple):
65*bb4ee6a4SAndroid Build Coastguard Worker    engine_dep (EngineDep|None): The recipe_engine dependency, or None, if the
66*bb4ee6a4SAndroid Build Coastguard Worker      current repo IS the recipe_engine.
67*bb4ee6a4SAndroid Build Coastguard Worker    recipes_path (str) - native path to where the recipes live inside of the
68*bb4ee6a4SAndroid Build Coastguard Worker      current repo (i.e. the folder containing `recipes/` and/or
69*bb4ee6a4SAndroid Build Coastguard Worker      `recipe_modules`)
70*bb4ee6a4SAndroid Build Coastguard Worker  """
71*bb4ee6a4SAndroid Build Coastguard Worker  with open(recipes_cfg_path, 'r', encoding='utf-8') as file:
72*bb4ee6a4SAndroid Build Coastguard Worker    recipes_cfg = json.load(file)
73*bb4ee6a4SAndroid Build Coastguard Worker
74*bb4ee6a4SAndroid Build Coastguard Worker  try:
75*bb4ee6a4SAndroid Build Coastguard Worker    if (version := recipes_cfg['api_version']) != 2:
76*bb4ee6a4SAndroid Build Coastguard Worker      raise MalformedRecipesCfg(f'unknown version {version}', recipes_cfg_path)
77*bb4ee6a4SAndroid Build Coastguard Worker
78*bb4ee6a4SAndroid Build Coastguard Worker    # If we're running ./recipes.py from the recipe_engine repo itself, then
79*bb4ee6a4SAndroid Build Coastguard Worker    # return None to signal that there's no EngineDep.
80*bb4ee6a4SAndroid Build Coastguard Worker    repo_name = recipes_cfg.get('repo_name')
81*bb4ee6a4SAndroid Build Coastguard Worker    if not repo_name:
82*bb4ee6a4SAndroid Build Coastguard Worker      repo_name = recipes_cfg['project_id']
83*bb4ee6a4SAndroid Build Coastguard Worker    if repo_name == 'recipe_engine':
84*bb4ee6a4SAndroid Build Coastguard Worker      return None, recipes_cfg.get('recipes_path', '')
85*bb4ee6a4SAndroid Build Coastguard Worker
86*bb4ee6a4SAndroid Build Coastguard Worker    engine = recipes_cfg['deps']['recipe_engine']
87*bb4ee6a4SAndroid Build Coastguard Worker
88*bb4ee6a4SAndroid Build Coastguard Worker    if 'url' not in engine:
89*bb4ee6a4SAndroid Build Coastguard Worker      raise MalformedRecipesCfg(
90*bb4ee6a4SAndroid Build Coastguard Worker          'Required field "url" in dependency "recipe_engine" not found',
91*bb4ee6a4SAndroid Build Coastguard Worker          recipes_cfg_path)
92*bb4ee6a4SAndroid Build Coastguard Worker
93*bb4ee6a4SAndroid Build Coastguard Worker    engine.setdefault('revision', '')
94*bb4ee6a4SAndroid Build Coastguard Worker    engine.setdefault('branch', 'refs/heads/main')
95*bb4ee6a4SAndroid Build Coastguard Worker    recipes_path = recipes_cfg.get('recipes_path', '')
96*bb4ee6a4SAndroid Build Coastguard Worker
97*bb4ee6a4SAndroid Build Coastguard Worker    # TODO(iannucci): only support absolute refs
98*bb4ee6a4SAndroid Build Coastguard Worker    if not engine['branch'].startswith('refs/'):
99*bb4ee6a4SAndroid Build Coastguard Worker      engine['branch'] = 'refs/heads/' + engine['branch']
100*bb4ee6a4SAndroid Build Coastguard Worker
101*bb4ee6a4SAndroid Build Coastguard Worker    recipes_path = os.path.join(repo_root,
102*bb4ee6a4SAndroid Build Coastguard Worker                                recipes_path.replace('/', os.path.sep))
103*bb4ee6a4SAndroid Build Coastguard Worker    return EngineDep(**engine), recipes_path
104*bb4ee6a4SAndroid Build Coastguard Worker  except KeyError as ex:
105*bb4ee6a4SAndroid Build Coastguard Worker    raise MalformedRecipesCfg(str(ex), recipes_cfg_path) from ex
106*bb4ee6a4SAndroid Build Coastguard Worker
107*bb4ee6a4SAndroid Build Coastguard Worker
108*bb4ee6a4SAndroid Build Coastguard WorkerIS_WIN = sys.platform.startswith(('win', 'cygwin'))
109*bb4ee6a4SAndroid Build Coastguard Worker
110*bb4ee6a4SAndroid Build Coastguard Worker_BAT = '.bat' if IS_WIN else ''
111*bb4ee6a4SAndroid Build Coastguard WorkerGIT = 'git' + _BAT
112*bb4ee6a4SAndroid Build Coastguard WorkerCIPD = 'cipd' + _BAT
113*bb4ee6a4SAndroid Build Coastguard WorkerREQUIRED_BINARIES = {GIT, CIPD}
114*bb4ee6a4SAndroid Build Coastguard Worker
115*bb4ee6a4SAndroid Build Coastguard Worker
116*bb4ee6a4SAndroid Build Coastguard Workerdef _is_executable(path):
117*bb4ee6a4SAndroid Build Coastguard Worker  return os.path.isfile(path) and os.access(path, os.X_OK)
118*bb4ee6a4SAndroid Build Coastguard Worker
119*bb4ee6a4SAndroid Build Coastguard Worker
120*bb4ee6a4SAndroid Build Coastguard Workerdef _subprocess_call(argv, **kwargs):
121*bb4ee6a4SAndroid Build Coastguard Worker  logging.info('Running %r', argv)
122*bb4ee6a4SAndroid Build Coastguard Worker  return subprocess.call(argv, **kwargs)
123*bb4ee6a4SAndroid Build Coastguard Worker
124*bb4ee6a4SAndroid Build Coastguard Worker
125*bb4ee6a4SAndroid Build Coastguard Workerdef _git_check_call(argv, **kwargs):
126*bb4ee6a4SAndroid Build Coastguard Worker  argv = [GIT] + argv
127*bb4ee6a4SAndroid Build Coastguard Worker  logging.info('Running %r', argv)
128*bb4ee6a4SAndroid Build Coastguard Worker  subprocess.check_call(argv, **kwargs)
129*bb4ee6a4SAndroid Build Coastguard Worker
130*bb4ee6a4SAndroid Build Coastguard Worker
131*bb4ee6a4SAndroid Build Coastguard Workerdef _git_output(argv, **kwargs):
132*bb4ee6a4SAndroid Build Coastguard Worker  argv = [GIT] + argv
133*bb4ee6a4SAndroid Build Coastguard Worker  logging.info('Running %r', argv)
134*bb4ee6a4SAndroid Build Coastguard Worker  return subprocess.check_output(argv, **kwargs)
135*bb4ee6a4SAndroid Build Coastguard Worker
136*bb4ee6a4SAndroid Build Coastguard Worker
137*bb4ee6a4SAndroid Build Coastguard Workerdef parse_args(argv):
138*bb4ee6a4SAndroid Build Coastguard Worker  """This extracts a subset of the arguments that this bootstrap script cares
139*bb4ee6a4SAndroid Build Coastguard Worker  about. Currently this consists of:
140*bb4ee6a4SAndroid Build Coastguard Worker    * an override for the recipe engine in the form of `-O recipe_engine=/path`
141*bb4ee6a4SAndroid Build Coastguard Worker    * the --package option.
142*bb4ee6a4SAndroid Build Coastguard Worker  """
143*bb4ee6a4SAndroid Build Coastguard Worker  override_prefix = 'recipe_engine='
144*bb4ee6a4SAndroid Build Coastguard Worker
145*bb4ee6a4SAndroid Build Coastguard Worker  parser = argparse.ArgumentParser(add_help=False)
146*bb4ee6a4SAndroid Build Coastguard Worker  parser.add_argument('-O', '--project-override', action='append')
147*bb4ee6a4SAndroid Build Coastguard Worker  parser.add_argument('--package', type=os.path.abspath)
148*bb4ee6a4SAndroid Build Coastguard Worker  args, _ = parser.parse_known_args(argv)
149*bb4ee6a4SAndroid Build Coastguard Worker  for override in args.project_override or ():
150*bb4ee6a4SAndroid Build Coastguard Worker    if override.startswith(override_prefix):
151*bb4ee6a4SAndroid Build Coastguard Worker      return override[len(override_prefix):], args.package
152*bb4ee6a4SAndroid Build Coastguard Worker  return None, args.package
153*bb4ee6a4SAndroid Build Coastguard Worker
154*bb4ee6a4SAndroid Build Coastguard Worker
155*bb4ee6a4SAndroid Build Coastguard Workerdef checkout_engine(engine_path, repo_root, recipes_cfg_path):
156*bb4ee6a4SAndroid Build Coastguard Worker  """Checks out the recipe_engine repo pinned in recipes.cfg.
157*bb4ee6a4SAndroid Build Coastguard Worker
158*bb4ee6a4SAndroid Build Coastguard Worker  Returns the path to the recipe engine repo.
159*bb4ee6a4SAndroid Build Coastguard Worker  """
160*bb4ee6a4SAndroid Build Coastguard Worker  dep, recipes_path = parse(repo_root, recipes_cfg_path)
161*bb4ee6a4SAndroid Build Coastguard Worker  if dep is None:
162*bb4ee6a4SAndroid Build Coastguard Worker    # we're running from the engine repo already!
163*bb4ee6a4SAndroid Build Coastguard Worker    return os.path.join(repo_root, recipes_path)
164*bb4ee6a4SAndroid Build Coastguard Worker
165*bb4ee6a4SAndroid Build Coastguard Worker  url = dep.url
166*bb4ee6a4SAndroid Build Coastguard Worker
167*bb4ee6a4SAndroid Build Coastguard Worker  if not engine_path and url.startswith('file://'):
168*bb4ee6a4SAndroid Build Coastguard Worker    engine_path = urlparse.urlparse(url).path
169*bb4ee6a4SAndroid Build Coastguard Worker
170*bb4ee6a4SAndroid Build Coastguard Worker  if not engine_path:
171*bb4ee6a4SAndroid Build Coastguard Worker    revision = dep.revision
172*bb4ee6a4SAndroid Build Coastguard Worker    branch = dep.branch
173*bb4ee6a4SAndroid Build Coastguard Worker
174*bb4ee6a4SAndroid Build Coastguard Worker    # Ensure that we have the recipe engine cloned.
175*bb4ee6a4SAndroid Build Coastguard Worker    engine_path = os.path.join(recipes_path, '.recipe_deps', 'recipe_engine')
176*bb4ee6a4SAndroid Build Coastguard Worker
177*bb4ee6a4SAndroid Build Coastguard Worker    # Note: this logic mirrors the logic in recipe_engine/fetch.py
178*bb4ee6a4SAndroid Build Coastguard Worker    _git_check_call(['init', engine_path], stdout=subprocess.DEVNULL)
179*bb4ee6a4SAndroid Build Coastguard Worker
180*bb4ee6a4SAndroid Build Coastguard Worker    try:
181*bb4ee6a4SAndroid Build Coastguard Worker      _git_check_call(['rev-parse', '--verify', f'{revision}^{{commit}}'],
182*bb4ee6a4SAndroid Build Coastguard Worker                      cwd=engine_path,
183*bb4ee6a4SAndroid Build Coastguard Worker                      stdout=subprocess.DEVNULL,
184*bb4ee6a4SAndroid Build Coastguard Worker                      stderr=subprocess.DEVNULL)
185*bb4ee6a4SAndroid Build Coastguard Worker    except subprocess.CalledProcessError:
186*bb4ee6a4SAndroid Build Coastguard Worker      _git_check_call(['fetch', '--quiet', url, branch],
187*bb4ee6a4SAndroid Build Coastguard Worker                      cwd=engine_path,
188*bb4ee6a4SAndroid Build Coastguard Worker                      stdout=subprocess.DEVNULL)
189*bb4ee6a4SAndroid Build Coastguard Worker
190*bb4ee6a4SAndroid Build Coastguard Worker    try:
191*bb4ee6a4SAndroid Build Coastguard Worker      _git_check_call(['diff', '--quiet', revision], cwd=engine_path)
192*bb4ee6a4SAndroid Build Coastguard Worker    except subprocess.CalledProcessError:
193*bb4ee6a4SAndroid Build Coastguard Worker      index_lock = os.path.join(engine_path, '.git', 'index.lock')
194*bb4ee6a4SAndroid Build Coastguard Worker      try:
195*bb4ee6a4SAndroid Build Coastguard Worker        os.remove(index_lock)
196*bb4ee6a4SAndroid Build Coastguard Worker      except OSError as exc:
197*bb4ee6a4SAndroid Build Coastguard Worker        if exc.errno != errno.ENOENT:
198*bb4ee6a4SAndroid Build Coastguard Worker          logging.warning('failed to remove %r, reset will fail: %s',
199*bb4ee6a4SAndroid Build Coastguard Worker                          index_lock, exc)
200*bb4ee6a4SAndroid Build Coastguard Worker      _git_check_call(['reset', '-q', '--hard', revision], cwd=engine_path)
201*bb4ee6a4SAndroid Build Coastguard Worker
202*bb4ee6a4SAndroid Build Coastguard Worker    # If the engine has refactored/moved modules we need to clean all .pyc files
203*bb4ee6a4SAndroid Build Coastguard Worker    # or things will get squirrely.
204*bb4ee6a4SAndroid Build Coastguard Worker    _git_check_call(['clean', '-qxf'], cwd=engine_path)
205*bb4ee6a4SAndroid Build Coastguard Worker
206*bb4ee6a4SAndroid Build Coastguard Worker  return engine_path
207*bb4ee6a4SAndroid Build Coastguard Worker
208*bb4ee6a4SAndroid Build Coastguard Worker
209*bb4ee6a4SAndroid Build Coastguard Workerdef main():
210*bb4ee6a4SAndroid Build Coastguard Worker  for required_binary in REQUIRED_BINARIES:
211*bb4ee6a4SAndroid Build Coastguard Worker    if not shutil.which(required_binary):
212*bb4ee6a4SAndroid Build Coastguard Worker      return f'Required binary is not found on PATH: {required_binary}'
213*bb4ee6a4SAndroid Build Coastguard Worker
214*bb4ee6a4SAndroid Build Coastguard Worker  if '--verbose' in sys.argv:
215*bb4ee6a4SAndroid Build Coastguard Worker    logging.getLogger().setLevel(logging.INFO)
216*bb4ee6a4SAndroid Build Coastguard Worker
217*bb4ee6a4SAndroid Build Coastguard Worker  args = sys.argv[1:]
218*bb4ee6a4SAndroid Build Coastguard Worker  engine_override, recipes_cfg_path = parse_args(args)
219*bb4ee6a4SAndroid Build Coastguard Worker
220*bb4ee6a4SAndroid Build Coastguard Worker  if recipes_cfg_path:
221*bb4ee6a4SAndroid Build Coastguard Worker    # calculate repo_root from recipes_cfg_path
222*bb4ee6a4SAndroid Build Coastguard Worker    repo_root = os.path.dirname(
223*bb4ee6a4SAndroid Build Coastguard Worker        os.path.dirname(os.path.dirname(recipes_cfg_path)))
224*bb4ee6a4SAndroid Build Coastguard Worker  else:
225*bb4ee6a4SAndroid Build Coastguard Worker    # find repo_root with git and calculate recipes_cfg_path
226*bb4ee6a4SAndroid Build Coastguard Worker    repo_root = (
227*bb4ee6a4SAndroid Build Coastguard Worker        _git_output(['rev-parse', '--show-toplevel'],
228*bb4ee6a4SAndroid Build Coastguard Worker                    cwd=os.path.abspath(os.path.dirname(__file__))).strip())
229*bb4ee6a4SAndroid Build Coastguard Worker    repo_root = os.path.abspath(repo_root).decode()
230*bb4ee6a4SAndroid Build Coastguard Worker    recipes_cfg_path = os.path.join(repo_root, 'infra', 'config', 'recipes.cfg')
231*bb4ee6a4SAndroid Build Coastguard Worker    args = ['--package', recipes_cfg_path] + args
232*bb4ee6a4SAndroid Build Coastguard Worker  engine_path = checkout_engine(engine_override, repo_root, recipes_cfg_path)
233*bb4ee6a4SAndroid Build Coastguard Worker
234*bb4ee6a4SAndroid Build Coastguard Worker  vpython = 'vpython3' + _BAT
235*bb4ee6a4SAndroid Build Coastguard Worker  if not shutil.which(vpython):
236*bb4ee6a4SAndroid Build Coastguard Worker    return f'Required binary is not found on PATH: {vpython}'
237*bb4ee6a4SAndroid Build Coastguard Worker
238*bb4ee6a4SAndroid Build Coastguard Worker  # We overwrite PYTHONPATH here on purpose; We don't want any conflicting
239*bb4ee6a4SAndroid Build Coastguard Worker  # environmental path leaking through into the recipe_engine which manages its
240*bb4ee6a4SAndroid Build Coastguard Worker  # environment entirely via vpython.
241*bb4ee6a4SAndroid Build Coastguard Worker  os.environ['PYTHONPATH'] = engine_path
242*bb4ee6a4SAndroid Build Coastguard Worker
243*bb4ee6a4SAndroid Build Coastguard Worker  spec = '.vpython3'
244*bb4ee6a4SAndroid Build Coastguard Worker  debugger = os.environ.get('RECIPE_DEBUGGER', '')
245*bb4ee6a4SAndroid Build Coastguard Worker  if debugger.startswith('pycharm'):
246*bb4ee6a4SAndroid Build Coastguard Worker    spec = '.pycharm.vpython3'
247*bb4ee6a4SAndroid Build Coastguard Worker  elif debugger.startswith('vscode'):
248*bb4ee6a4SAndroid Build Coastguard Worker    spec = '.vscode.vpython3'
249*bb4ee6a4SAndroid Build Coastguard Worker
250*bb4ee6a4SAndroid Build Coastguard Worker  argv = ([
251*bb4ee6a4SAndroid Build Coastguard Worker      vpython,
252*bb4ee6a4SAndroid Build Coastguard Worker      '-vpython-spec',
253*bb4ee6a4SAndroid Build Coastguard Worker      os.path.join(engine_path, spec),
254*bb4ee6a4SAndroid Build Coastguard Worker      '-u',
255*bb4ee6a4SAndroid Build Coastguard Worker      os.path.join(engine_path, 'recipe_engine', 'main.py'),
256*bb4ee6a4SAndroid Build Coastguard Worker  ] + args)
257*bb4ee6a4SAndroid Build Coastguard Worker
258*bb4ee6a4SAndroid Build Coastguard Worker  if IS_WIN:
259*bb4ee6a4SAndroid Build Coastguard Worker    # No real 'exec' on windows; set these signals to ignore so that they
260*bb4ee6a4SAndroid Build Coastguard Worker    # propagate to our children but we still wait for the child process to quit.
261*bb4ee6a4SAndroid Build Coastguard Worker    import signal  # pylint: disable=import-outside-toplevel
262*bb4ee6a4SAndroid Build Coastguard Worker    signal.signal(signal.SIGBREAK, signal.SIG_IGN)  # pylint: disable=no-member
263*bb4ee6a4SAndroid Build Coastguard Worker    signal.signal(signal.SIGINT, signal.SIG_IGN)
264*bb4ee6a4SAndroid Build Coastguard Worker    signal.signal(signal.SIGTERM, signal.SIG_IGN)
265*bb4ee6a4SAndroid Build Coastguard Worker    return _subprocess_call(argv)
266*bb4ee6a4SAndroid Build Coastguard Worker
267*bb4ee6a4SAndroid Build Coastguard Worker  os.execvp(argv[0], argv)
268*bb4ee6a4SAndroid Build Coastguard Worker  return -1  # should never occur
269*bb4ee6a4SAndroid Build Coastguard Worker
270*bb4ee6a4SAndroid Build Coastguard Worker
271*bb4ee6a4SAndroid Build Coastguard Workerif __name__ == '__main__':
272*bb4ee6a4SAndroid Build Coastguard Worker  sys.exit(main())
273