-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathbuild_with_docker.py
120 lines (92 loc) · 4.95 KB
/
build_with_docker.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
import logging
import os
import subprocess
import sys
from argparse import ArgumentParser
from pathlib import Path
from typing import List
logger = logging.getLogger("build-with-docker")
def main(cli_args: List[str]):
logging.basicConfig(level=logging.DEBUG)
parser = ArgumentParser()
parser.add_argument("--image", type=str, required=True)
parser.add_argument("--no-docker-interactive", action="store_true", default=False)
parser.add_argument("--no-docker-tty", action="store_true", default=False)
parser.add_argument("--project", type=str, help="source code folder (project)")
parser.add_argument("--packaged-src", type=str, help="source code packaged in a JSON file")
parser.add_argument("--contract", type=str, help="contract to build (contract name, as found in Cargo.toml)")
parser.add_argument("--output", type=str, default=Path(os.getcwd()) / "output")
parser.add_argument("--package-whole-project-src", action="store_true", default=False, help="deprecated parameter, not used anymore")
parser.add_argument("--cargo-target-dir", help="deprecated parameter, not used anymore")
parser.add_argument("--no-wasm-opt", action="store_true", default=False, help="do not optimize wasm files after the build (default: %(default)s)")
parser.add_argument("--build-root", type=str, required=False, help="root path (within container) for the build (default: %(default)s)")
parser.add_argument("--cargo-verbose", action="store_true", default=False, help="set 'CARGO_TERM_VERBOSE' environment variable (default: %(default)s)")
# Handle CLI arguments
parsed_args = parser.parse_args(cli_args)
image = parsed_args.image
docker_interactive = not parsed_args.no_docker_interactive
docker_tty = not parsed_args.no_docker_tty
project_path = Path(parsed_args.project).expanduser().resolve() if parsed_args.project else None
packaged_src_path = Path(parsed_args.packaged_src).expanduser().resolve() if parsed_args.packaged_src else None
contract_path = parsed_args.contract
output_path = Path(parsed_args.output).expanduser().resolve()
no_wasm_opt = parsed_args.no_wasm_opt
build_root = Path(parsed_args.build_root) if parsed_args.build_root else None
cargo_verbose = parsed_args.cargo_verbose
# Prepare (and check) output folder
output_path.mkdir(parents=True, exist_ok=True)
ensure_output_folder_is_empty(output_path)
# Prepare general docker arguments
docker_general_args = ["docker", "run"]
if docker_interactive:
docker_general_args += ["--interactive"]
if docker_tty:
docker_general_args += ["--tty"]
docker_general_args += ["--user", f"{str(os.getuid())}:{str(os.getgid())}"]
docker_general_args += ["--rm"]
# Prepare docker arguments related to mounting volumes
docker_mount_args: List[str] = ["--volume", f"{output_path}:/output"]
if project_path:
docker_mount_args.extend(["--volume", f"{project_path}:/project"])
if packaged_src_path:
docker_mount_args.extend(["--volume", f"{packaged_src_path}:/packaged-src.json"])
mounted_temporary_root = Path("/tmp/multiversx_sdk_rust_contract_builder")
mounted_cargo_target_dir = mounted_temporary_root / "cargo-target-dir"
mounted_cargo_registry = mounted_temporary_root / "cargo-registry"
mounted_cargo_git = mounted_temporary_root / "cargo-git"
mounted_cargo_target_dir.mkdir(parents=True, exist_ok=True)
mounted_cargo_registry.mkdir(parents=True, exist_ok=True)
mounted_cargo_git.mkdir(parents=True, exist_ok=True)
docker_mount_args += ["--volume", f"{mounted_cargo_target_dir}:/rust/cargo-target-dir"]
docker_mount_args += ["--volume", f"{mounted_cargo_registry}:/rust/registry"]
docker_mount_args += ["--volume", f"{mounted_cargo_git}:/rust/git"]
docker_env_args = ["--env", f"CARGO_TERM_VERBOSE={str(cargo_verbose).lower()}"]
# Prepare entrypoint arguments
entrypoint_args: List[str] = []
if project_path:
entrypoint_args.extend(["--project", "project"])
if packaged_src_path:
entrypoint_args.extend(["--packaged-src", "packaged-src.json"])
if no_wasm_opt:
entrypoint_args.append("--no-wasm-opt")
if contract_path:
entrypoint_args.extend(["--contract", contract_path])
if build_root:
entrypoint_args.extend(["--build-root", str(build_root)])
# Run docker container
args = docker_general_args + docker_mount_args + docker_env_args + [image] + entrypoint_args
logger.info(f"Running docker: {args}")
result = subprocess.run(args)
return result.returncode
def ensure_output_folder_is_empty(parent_output_folder: Path):
is_empty = len(os.listdir(parent_output_folder)) == 0
if not is_empty:
raise Exception(f"Output folder must be empty: {parent_output_folder}")
if __name__ == "__main__":
try:
return_code = main(sys.argv[1:])
exit(return_code)
except Exception as err:
print("An error occurred.")
print(err)
exit(1)