bitcoin/.github/ci-windows.py
MarcoFalke fae31b1e2f
ci: [refactor] Move github_import_vs_env to python script
Also, change the yaml anchor name and the step name.
Also, small refactors while touching the files.
2026-02-17 22:06:39 +01:00

222 lines
6.2 KiB
Python
Executable File

#!/usr/bin/env python3
# Copyright (c) The Bitcoin Core developers
# Distributed under the MIT software license, see the accompanying
# file COPYING or https://opensource.org/license/mit/.
import argparse
import os
import shlex
import subprocess
import sys
from pathlib import Path
def run(cmd, **kwargs):
print("+ " + shlex.join(cmd), flush=True)
kwargs.setdefault("check", True)
try:
return subprocess.run(cmd, **kwargs)
except Exception as e:
sys.exit(str(e))
GENERATE_OPTIONS = {
"standard": [
"-DBUILD_BENCH=ON",
"-DBUILD_KERNEL_LIB=ON",
"-DBUILD_UTIL_CHAINSTATE=ON",
"-DCMAKE_COMPILE_WARNING_AS_ERROR=ON",
],
"fuzz": [
"-DVCPKG_MANIFEST_NO_DEFAULT_FEATURES=ON",
"-DVCPKG_MANIFEST_FEATURES=wallet",
"-DBUILD_GUI=OFF",
"-DWITH_ZMQ=OFF",
"-DBUILD_FOR_FUZZING=ON",
"-DCMAKE_COMPILE_WARNING_AS_ERROR=ON",
],
}
def github_import_vs_env(_ci_type):
vswhere_path = Path(os.environ["ProgramFiles(x86)"]) / "Microsoft Visual Studio" / "Installer" / "vswhere.exe"
installation_path = run(
[str(vswhere_path), "-latest", "-property", "installationPath"],
capture_output=True,
text=True,
).stdout.strip()
vsdevcmd = Path(installation_path) / "Common7" / "Tools" / "vsdevcmd.bat"
comspec = os.environ["COMSPEC"]
output = run(
f'"{comspec}" /s /c ""{vsdevcmd}" -arch=x64 -no_logo && set"',
capture_output=True,
text=True,
).stdout
github_env = os.environ["GITHUB_ENV"]
with open(github_env, "a") as env_file:
for line in output.splitlines():
if "=" not in line:
continue
name, value = line.split("=", 1)
env_file.write(f"{name}={value}\n")
def generate(ci_type):
command = [
"cmake",
"-B",
"build",
"-Werror=dev",
"--preset",
"vs2026",
] + GENERATE_OPTIONS[ci_type]
run(command)
def build(_ci_type):
command = [
"cmake",
"--build",
"build",
"--config",
"Release",
]
if run(command + ["-j", str(os.process_cpu_count())], check=False).returncode != 0:
print("Build failure. Verbose build follows.")
run(command + ["-j1", "--verbose"])
def check_manifests(ci_type):
if ci_type != "standard":
print(f"Skipping manifest validation for '{ci_type}' ci type.")
return
release_dir = Path.cwd() / "build" / "bin" / "Release"
manifest_path = release_dir / "bitcoind.manifest"
cmd_bitcoind_manifest = [
"mt.exe",
"-nologo",
f"-inputresource:{release_dir / 'bitcoind.exe'}",
f"-out:{manifest_path}",
]
run(cmd_bitcoind_manifest)
print(manifest_path.read_text())
skips = { # Skip as they currently do not have manifests
"fuzz.exe",
"bench_bitcoin.exe",
"test_bitcoin-qt.exe",
"test_kernel.exe",
"bitcoin-chainstate.exe",
}
for entry in release_dir.iterdir():
if entry.suffix.lower() != ".exe":
continue
if entry.name in skips:
print(f"Skipping {entry.name} (no manifest present)")
continue
print(f"Checking {entry.name}")
cmd_check_manifest = [
"mt.exe",
"-nologo",
f"-inputresource:{entry}",
"-validate_manifest",
]
run(cmd_check_manifest)
def prepare_tests(ci_type):
if ci_type == "standard":
run([sys.executable, "-m", "pip", "install", "pyzmq"])
elif ci_type == "fuzz":
repo_dir = str(Path.cwd() / "qa-assets")
clone_cmd = [
"git",
"clone",
"--depth=1",
"https://github.com/bitcoin-core/qa-assets",
repo_dir,
]
run(clone_cmd)
print("Using qa-assets repo from commit ...")
run(["git", "-C", repo_dir, "log", "-1"])
def run_tests(ci_type):
build_dir = Path.cwd() / "build"
num_procs = str(os.process_cpu_count())
release_bin = build_dir / "bin" / "Release"
if ci_type == "standard":
test_envs = {
"BITCOIN_BIN": "bitcoin.exe",
"BITCOIND": "bitcoind.exe",
"BITCOINCLI": "bitcoin-cli.exe",
"BITCOIN_BENCH": "bench_bitcoin.exe",
"BITCOINTX": "bitcoin-tx.exe",
"BITCOINUTIL": "bitcoin-util.exe",
"BITCOINWALLET": "bitcoin-wallet.exe",
"BITCOINCHAINSTATE": "bitcoin-chainstate.exe",
}
for var, exe in test_envs.items():
os.environ[var] = str(release_bin / exe)
ctest_cmd = [
"ctest",
"--test-dir",
str(build_dir),
"--output-on-failure",
"--stop-on-failure",
"-j",
num_procs,
"--build-config",
"Release",
]
run(ctest_cmd)
test_cmd = [
sys.executable,
str(build_dir / "test" / "functional" / "test_runner.py"),
"--jobs",
num_procs,
"--quiet",
f"--tmpdirprefix={Path.cwd()}",
"--combinedlogslen=99999999",
*shlex.split(os.environ.get("TEST_RUNNER_EXTRA", "").strip()),
]
run(test_cmd)
elif ci_type == "fuzz":
os.environ["BITCOINFUZZ"] = str(release_bin / "fuzz.exe")
fuzz_cmd = [
sys.executable,
str(build_dir / "test" / "fuzz" / "test_runner.py"),
"--par",
num_procs,
"--loglevel",
"DEBUG",
str(Path.cwd() / "qa-assets" / "fuzz_corpora"),
]
run(fuzz_cmd)
def main():
parser = argparse.ArgumentParser(description="Utility to run Windows CI steps.")
parser.add_argument("ci_type", choices=GENERATE_OPTIONS, help="CI type to run.")
steps = list(map(lambda f: f.__name__, [
github_import_vs_env,
generate,
build,
check_manifests,
prepare_tests,
run_tests,
]))
parser.add_argument("step", choices=steps, help="CI step to perform.")
args = parser.parse_args()
exec(f'{args.step}("{args.ci_type}")')
if __name__ == "__main__":
main()