123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147 |
- from __future__ import annotations
- import argparse
- from concurrent.futures import Future, ThreadPoolExecutor
- from os.path import abspath
- from subprocess import CalledProcessError, run
- from typing import Sequence
- from tools.lib import gitroot
- def worker(args: tuple[str, ...]) -> None:
- # pip-compile doesn't let you customize the header, so we write
- # one ourselves. However, pip-compile needs -o DEST otherwise
- # it will bump >= pins even if they're satisfied. So, we need to
- # unfortunately rewrite the whole file.
- dest = args[-1]
- try:
- run(args, check=True, capture_output=True)
- except CalledProcessError as e:
- raise e
- with open(dest, "rb+") as f:
- content = f.read()
- f.seek(0, 0)
- f.write(
- b"""# DO NOT MODIFY. This file was generated with `make freeze-requirements`.
- """
- + content
- )
- def check_futures(futures: list[Future[None]]) -> int:
- rc = 0
- for future in futures:
- try:
- future.result()
- except CalledProcessError as e:
- rc = 1
- print(
- f"""`{e.cmd}` returned code {e.returncode}
- stdout:
- {e.stdout.decode()}
- stderr:
- {e.stderr.decode()}
- """
- )
- return rc
- def main(argv: Sequence[str] | None = None) -> int:
- parser = argparse.ArgumentParser()
- parser.add_argument("repo", type=str, help="Repository name.")
- args = parser.parse_args(argv)
- repo = args.repo
- base_path = abspath(gitroot())
- base_cmd = (
- "pip-compile",
- "--no-header",
- "--no-annotate",
- "--allow-unsafe",
- "-q",
- )
- executor = ThreadPoolExecutor(max_workers=3)
- futures = []
- if repo != "getsentry":
- futures.append(
- executor.submit(
- worker,
- (
- *base_cmd,
- f"{base_path}/requirements-base.txt",
- "-o",
- f"{base_path}/requirements-frozen.txt",
- ),
- )
- )
- futures.append(
- executor.submit(
- worker,
- (
- *base_cmd,
- f"{base_path}/requirements-base.txt",
- f"{base_path}/requirements-dev.txt",
- "-o",
- f"{base_path}/requirements-dev-frozen.txt",
- ),
- )
- )
- else:
- futures.append(
- executor.submit(
- worker,
- (
- *base_cmd,
- f"{base_path}/requirements-base.txt",
- # This is downloaded with bin/bump-sentry.
- f"{base_path}/sentry-requirements-frozen.txt",
- "-o",
- f"{base_path}/requirements-frozen.txt",
- ),
- )
- )
- # getsentry shares sentry's requirements-dev.
- futures.append(
- executor.submit(
- worker,
- (
- *base_cmd,
- f"{base_path}/requirements-base.txt",
- # This is downloaded with bin/bump-sentry.
- f"{base_path}/sentry-requirements-dev-frozen.txt",
- "-o",
- f"{base_path}/requirements-dev-frozen.txt",
- ),
- )
- )
- if repo == "sentry":
- # requirements-dev-only-frozen.txt is only used in sentry
- # (and reused in getsentry) as a fast path for some CI jobs.
- futures.append(
- executor.submit(
- worker,
- (
- *base_cmd,
- f"{base_path}/requirements-dev.txt",
- "-o",
- f"{base_path}/requirements-dev-only-frozen.txt",
- ),
- )
- )
- rc = check_futures(futures)
- executor.shutdown()
- return rc
- if __name__ == "__main__":
- raise SystemExit(main())
|