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