| #!/bin/sh |
| # Copyright 2019 The LUCI Authors. All rights reserved. |
| # Use of this source code is governed under the Apache License, Version 2.0 |
| # that can be found in the LICENSE file. |
| |
| # We want to run python in unbuffered mode; however shebangs on linux grab the |
| # entire rest of the shebang line as a single argument, leading to errors like: |
| # |
| # /usr/bin/env: 'python3 -u': No such file or directory |
| # |
| # This little shell hack is a triple-quoted noop in python, but in sh it |
| # evaluates to re-exec'ing this script in unbuffered mode. |
| # pylint: disable=pointless-string-statement |
| ''''exec python3 -u -- "$0" ${1+"$@"} # ''' |
| """Bootstrap script to clone and forward to the recipe engine tool. |
| |
| ******************* |
| ** DO NOT MODIFY ** |
| ******************* |
| |
| This is a copy of https://chromium.googlesource.com/infra/luci/recipes-py/+/main/recipes.py. |
| To fix bugs, fix in the googlesource repo then run the autoroller. |
| """ |
| |
| # pylint: disable=wrong-import-position |
| import argparse |
| import errno |
| import json |
| import logging |
| import os |
| import shutil |
| import subprocess |
| import sys |
| |
| import urllib.parse as urlparse |
| |
| from collections import namedtuple |
| |
| |
| # The dependency entry for the recipe_engine in the client repo's recipes.cfg |
| # |
| # url (str) - the url to the engine repo we want to use. |
| # revision (str) - the git revision for the engine to get. |
| # branch (str) - the branch to fetch for the engine as an absolute ref (e.g. |
| # refs/heads/main) |
| EngineDep = namedtuple('EngineDep', 'url revision branch') |
| |
| |
| class MalformedRecipesCfg(Exception): |
| |
| def __init__(self, msg, path): |
| full_message = f'malformed recipes.cfg: {msg}: {path!r}' |
| super().__init__(full_message) |
| |
| |
| def parse(repo_root, recipes_cfg_path): |
| """Parse is a lightweight a recipes.cfg file parser. |
| |
| Args: |
| repo_root (str) - native path to the root of the repo we're trying to run |
| recipes for. |
| recipes_cfg_path (str) - native path to the recipes.cfg file to process. |
| |
| Returns (as tuple): |
| engine_dep (EngineDep|None): The recipe_engine dependency, or None, if the |
| current repo IS the recipe_engine. |
| recipes_path (str) - native path to where the recipes live inside of the |
| current repo (i.e. the folder containing `recipes/` and/or |
| `recipe_modules`) |
| """ |
| with open(recipes_cfg_path, 'r', encoding='utf-8') as file: |
| recipes_cfg = json.load(file) |
| |
| try: |
| if (version := recipes_cfg['api_version']) != 2: |
| raise MalformedRecipesCfg(f'unknown version {version}', recipes_cfg_path) |
| |
| # If we're running ./recipes.py from the recipe_engine repo itself, then |
| # return None to signal that there's no EngineDep. |
| repo_name = recipes_cfg.get('repo_name') |
| if not repo_name: |
| repo_name = recipes_cfg['project_id'] |
| if repo_name == 'recipe_engine': |
| return None, recipes_cfg.get('recipes_path', '') |
| |
| engine = recipes_cfg['deps']['recipe_engine'] |
| |
| if 'url' not in engine: |
| raise MalformedRecipesCfg( |
| 'Required field "url" in dependency "recipe_engine" not found', |
| recipes_cfg_path) |
| |
| engine.setdefault('revision', '') |
| engine.setdefault('branch', 'refs/heads/main') |
| recipes_path = recipes_cfg.get('recipes_path', '') |
| |
| # TODO(iannucci): only support absolute refs |
| if not engine['branch'].startswith('refs/'): |
| engine['branch'] = 'refs/heads/' + engine['branch'] |
| |
| recipes_path = os.path.join(repo_root, |
| recipes_path.replace('/', os.path.sep)) |
| return EngineDep(**engine), recipes_path |
| except KeyError as ex: |
| raise MalformedRecipesCfg(str(ex), recipes_cfg_path) from ex |
| |
| |
| IS_WIN = sys.platform.startswith(('win', 'cygwin')) |
| |
| _BAT = '.bat' if IS_WIN else '' |
| GIT = 'git' + _BAT |
| CIPD = 'cipd' + _BAT |
| REQUIRED_BINARIES = {GIT, CIPD} |
| |
| |
| def _is_executable(path): |
| return os.path.isfile(path) and os.access(path, os.X_OK) |
| |
| |
| def _subprocess_call(argv, **kwargs): |
| logging.info('Running %r', argv) |
| return subprocess.call(argv, **kwargs) |
| |
| |
| def _git_check_call(argv, **kwargs): |
| argv = [GIT] + argv |
| logging.info('Running %r', argv) |
| subprocess.check_call(argv, **kwargs) |
| |
| |
| def _git_output(argv, **kwargs): |
| argv = [GIT] + argv |
| logging.info('Running %r', argv) |
| return subprocess.check_output(argv, **kwargs) |
| |
| |
| def parse_args(argv): |
| """This extracts a subset of the arguments that this bootstrap script cares |
| about. Currently this consists of: |
| * an override for the recipe engine in the form of `-O recipe_engine=/path` |
| * the --package option. |
| """ |
| override_prefix = 'recipe_engine=' |
| |
| parser = argparse.ArgumentParser(add_help=False) |
| parser.add_argument('-O', '--project-override', action='append') |
| parser.add_argument('--package', type=os.path.abspath) |
| args, _ = parser.parse_known_args(argv) |
| for override in args.project_override or (): |
| if override.startswith(override_prefix): |
| return override[len(override_prefix):], args.package |
| return None, args.package |
| |
| |
| def checkout_engine(engine_path, repo_root, recipes_cfg_path): |
| """Checks out the recipe_engine repo pinned in recipes.cfg. |
| |
| Returns the path to the recipe engine repo. |
| """ |
| dep, recipes_path = parse(repo_root, recipes_cfg_path) |
| if dep is None: |
| # we're running from the engine repo already! |
| return os.path.join(repo_root, recipes_path) |
| |
| url = dep.url |
| |
| if not engine_path and url.startswith('file://'): |
| engine_path = urlparse.urlparse(url).path |
| |
| if not engine_path: |
| revision = dep.revision |
| branch = dep.branch |
| |
| # Ensure that we have the recipe engine cloned. |
| engine_path = os.path.join(recipes_path, '.recipe_deps', 'recipe_engine') |
| |
| # Note: this logic mirrors the logic in recipe_engine/fetch.py |
| _git_check_call(['init', engine_path], stdout=subprocess.DEVNULL) |
| |
| try: |
| _git_check_call(['rev-parse', '--verify', f'{revision}^{{commit}}'], |
| cwd=engine_path, |
| stdout=subprocess.DEVNULL, |
| stderr=subprocess.DEVNULL) |
| except subprocess.CalledProcessError: |
| _git_check_call(['fetch', '--quiet', url, branch], |
| cwd=engine_path, |
| stdout=subprocess.DEVNULL) |
| |
| try: |
| _git_check_call(['diff', '--quiet', revision], cwd=engine_path) |
| except subprocess.CalledProcessError: |
| index_lock = os.path.join(engine_path, '.git', 'index.lock') |
| try: |
| os.remove(index_lock) |
| except OSError as exc: |
| if exc.errno != errno.ENOENT: |
| logging.warning('failed to remove %r, reset will fail: %s', |
| index_lock, exc) |
| _git_check_call(['reset', '-q', '--hard', revision], cwd=engine_path) |
| |
| # If the engine has refactored/moved modules we need to clean all .pyc files |
| # or things will get squirrely. |
| _git_check_call(['clean', '-qxf'], cwd=engine_path) |
| |
| return engine_path |
| |
| |
| def main(): |
| for required_binary in REQUIRED_BINARIES: |
| if not shutil.which(required_binary): |
| return f'Required binary is not found on PATH: {required_binary}' |
| |
| if '--verbose' in sys.argv: |
| logging.getLogger().setLevel(logging.INFO) |
| |
| args = sys.argv[1:] |
| engine_override, recipes_cfg_path = parse_args(args) |
| |
| if recipes_cfg_path: |
| # calculate repo_root from recipes_cfg_path |
| repo_root = os.path.dirname( |
| os.path.dirname(os.path.dirname(recipes_cfg_path))) |
| else: |
| # find repo_root with git and calculate recipes_cfg_path |
| repo_root = ( |
| _git_output(['rev-parse', '--show-toplevel'], |
| cwd=os.path.abspath(os.path.dirname(__file__))).strip()) |
| repo_root = os.path.abspath(repo_root).decode() |
| recipes_cfg_path = os.path.join(repo_root, 'infra', 'config', 'recipes.cfg') |
| args = ['--package', recipes_cfg_path] + args |
| engine_path = checkout_engine(engine_override, repo_root, recipes_cfg_path) |
| |
| vpython = 'vpython3' + _BAT |
| if not shutil.which(vpython): |
| return f'Required binary is not found on PATH: {vpython}' |
| |
| # We unset PYTHONPATH here in case the user has conflicting environmental |
| # things we don't want them to leak through into the recipe_engine which |
| # manages its environment entirely via vpython. |
| # |
| # NOTE: os.unsetenv unhelpfully doesn't exist on all platforms until python3.9 |
| # so we have to use the cutesy `pop` formulation below until then... |
| os.environ.pop("PYTHONPATH", None) |
| |
| spec = '.vpython3' |
| debugger = os.environ.get('RECIPE_DEBUGGER', '') |
| if debugger.startswith('pycharm'): |
| spec = '.pycharm.vpython3' |
| elif debugger.startswith('vscode'): |
| spec = '.vscode.vpython3' |
| |
| argv = ([ |
| vpython, |
| '-vpython-spec', |
| os.path.join(engine_path, spec), |
| '-u', |
| os.path.join(engine_path, 'recipe_engine', 'main.py'), |
| ] + args) |
| |
| if IS_WIN: |
| # No real 'exec' on windows; set these signals to ignore so that they |
| # propagate to our children but we still wait for the child process to quit. |
| import signal # pylint: disable=import-outside-toplevel |
| signal.signal(signal.SIGBREAK, signal.SIG_IGN) # pylint: disable=no-member |
| signal.signal(signal.SIGINT, signal.SIG_IGN) |
| signal.signal(signal.SIGTERM, signal.SIG_IGN) |
| return _subprocess_call(argv) |
| |
| os.execvp(argv[0], argv) |
| return -1 # should never occur |
| |
| |
| if __name__ == '__main__': |
| sys.exit(main()) |