-
-
Notifications
You must be signed in to change notification settings - Fork 14.7k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
WIP: rustPlatform.mkCargoVendorDeps: init
- Loading branch information
Showing
3 changed files
with
313 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,221 @@ | ||
import glob | ||
import hashlib | ||
import json | ||
import multiprocessing as mp | ||
import os | ||
import re | ||
import shutil | ||
import subprocess | ||
import sys | ||
|
||
import requests | ||
import tomli | ||
|
||
|
||
def eprint(text: str): | ||
print(text, file=sys.stderr) | ||
|
||
|
||
def load_file(path: str) -> dict: | ||
with open(path, "rb") as f: | ||
return tomli.load(f) | ||
|
||
|
||
def parse_git_source(source: str): | ||
matches = re.match('git\\+([^?]+)(\\?(rev|tag|branch)=(.*))?#(.*)', source) | ||
if matches is None: | ||
raise Exception(f"Unable to process git source: {source}") | ||
url, _, type_, value, git_sha_rev = matches.groups() | ||
return {"url": url, "type": type_, "value": value, "git_sha_rev": git_sha_rev} | ||
|
||
|
||
def download_file_with_checksum(url: str, destination_path: str, expected_checksum: str): | ||
"""Downloads a file and computes its SHA-256 checksum while writing it to disk.""" | ||
sha256_hash = hashlib.sha256() | ||
with requests.get(url, stream=True) as response: | ||
if response.status_code == 200: | ||
with open(destination_path, 'wb') as file: | ||
for chunk in response.iter_content(1024): # Download in chunks | ||
if chunk: # Filter out keep-alive chunks | ||
file.write(chunk) | ||
sha256_hash.update(chunk) # Update checksum while writing to file | ||
else: | ||
raise Exception(f"Failed to download file from {url}. Status code: {response.status_code}") | ||
|
||
# Compute the final checksum | ||
calculated_checksum = sha256_hash.hexdigest() | ||
|
||
if calculated_checksum != expected_checksum: | ||
raise Exception(f"Hash mismatch! File fetched from {url} had checksum {calculated_checksum}, expected {expected_checksum}.") | ||
|
||
|
||
# TODO: support other repositories | ||
# or maybe allow getting download_url through an API request | ||
def download_tarball(args: tuple[dict, str]): | ||
pkg, out_dir = args | ||
download_url = "https://crates.io/api/v1/crates" | ||
url = f"{download_url}/{pkg["name"]}/{pkg["version"]}/download" | ||
filename = f"{pkg["name"]}-{pkg["version"]}.tar.gz" | ||
eprint(f"Fetching {pkg["name"]} v{pkg["version"]} from {url} to {filename}") | ||
|
||
# TODO: allow legacy checksum specification, see importCargoLock for example | ||
# also, don't forget about the other usage of the checksum | ||
download_file_with_checksum(url, os.path.join(out_dir, "registry", filename), pkg["checksum"]) | ||
|
||
|
||
def create_vendor_staging(lockfile_path: str, out_dir: str): | ||
cargo_toml = load_file(lockfile_path) | ||
|
||
git_packages: list[dict] = [] | ||
registry_packages: list[dict] = [] | ||
|
||
for pkg in cargo_toml["package"]: | ||
# ignore local dependenices | ||
if "source" not in pkg.keys(): | ||
eprint(f"skipping local dependency: {pkg["name"]}") | ||
continue | ||
source = pkg["source"] | ||
|
||
if source.startswith("git+"): | ||
git_packages.append(pkg) | ||
elif source.startswith("registry+"): | ||
registry_packages.append(pkg) | ||
else: | ||
raise Exception(f"Can't process source: {source}.") | ||
|
||
git_sha_rev_to_url = {} | ||
for pkg in git_packages: | ||
source_info = parse_git_source(pkg["source"]) | ||
git_sha_rev_to_url[source_info["git_sha_rev"]] = source_info["url"] | ||
|
||
os.makedirs(out_dir, exist_ok=True) | ||
shutil.copy(lockfile_path, os.path.join(out_dir, "Cargo.lock")) | ||
|
||
if len(git_packages) != 0: | ||
os.mkdir(os.path.join(out_dir, "git")) | ||
|
||
for git_sha_rev, url in git_sha_rev_to_url.items(): | ||
eprint(f"fetching {url}#{git_sha_rev}") | ||
res_dir = os.path.join(out_dir, "git", git_sha_rev) | ||
subprocess.check_output(["nix-prefetch-git", "--builder", "--quiet", "--url", url, "--rev", git_sha_rev, "--out", res_dir]) | ||
|
||
if len(registry_packages) != 0: | ||
os.mkdir(os.path.join(out_dir, "registry")) | ||
|
||
# run download jobs in parallel | ||
pool = mp.Pool(min(10, mp.cpu_count())) | ||
pool.map(download_tarball, ((pkg, out_dir) for pkg in registry_packages)) | ||
pool.close() | ||
|
||
|
||
def get_manifest_metadata(manifest_path: str): | ||
output = subprocess.check_output(["cargo", "metadata", "--format-version", "1", "--no-deps", "--manifest-path", manifest_path]) | ||
return json.loads(output) | ||
|
||
|
||
def try_get_crate_manifest_path_from_mainfest_path(manifest_path: str, crate_name: str) -> str | None: | ||
metadata = get_manifest_metadata(manifest_path) | ||
crate_manifest_path = next((pkg["manifest_path"] for pkg in metadata["packages"] if pkg["name"] == crate_name), None) | ||
return crate_manifest_path | ||
|
||
|
||
def find_crate_manifest_in_tree(tree: str, crate_name: str): | ||
# in some cases Cargo.toml is not located at the top level, so we also look at subdirectories | ||
manifest_paths = (os.path.join(tree, path) for path in glob.iglob("**/Cargo.toml", root_dir=tree, recursive=True)) | ||
|
||
for manifest_path in manifest_paths: | ||
res = try_get_crate_manifest_path_from_mainfest_path(manifest_path, crate_name) | ||
if res is not None: | ||
return res | ||
|
||
raise Exception(f"Couldn't find manifest for crate {crate_name} inside {tree}") | ||
|
||
|
||
def create_vendor(vendor_staging_dir: str, out_dir: str): | ||
lockfile_path = os.path.join(vendor_staging_dir, "Cargo.lock") | ||
cargo_toml = load_file(lockfile_path) | ||
os.makedirs(out_dir, exist_ok=True) | ||
shutil.copy(lockfile_path, os.path.join(out_dir, "Cargo.lock")) | ||
|
||
configLines = [ | ||
'[source.crates-io]', | ||
'replace-with = "vendored-sources"', | ||
'[source.vendored-sources]', | ||
'directory = "@vendor@"', | ||
] | ||
|
||
seen_keys = set() | ||
for pkg in cargo_toml["package"]: | ||
# ignore local dependenices | ||
if "source" not in pkg.keys(): | ||
continue | ||
source = pkg["source"] | ||
|
||
if source.startswith("git+"): | ||
|
||
source_info = parse_git_source(pkg["source"]) | ||
git_sha_rev = source_info["git_sha_rev"] | ||
|
||
git_tree = os.path.join(vendor_staging_dir, "git", git_sha_rev) | ||
crate_manifest_path = find_crate_manifest_in_tree(git_tree, pkg["name"]) | ||
crate_tree = os.path.dirname(crate_manifest_path) | ||
|
||
dir_name = f"{pkg["name"]}-{pkg["version"]}" | ||
crate_out_dir = os.path.join(out_dir, dir_name) | ||
shutil.copytree(crate_tree, crate_out_dir) | ||
os.chmod(crate_out_dir, 0o755) | ||
|
||
with open(os.path.join(crate_out_dir, ".cargo-checksum.json"), "w") as f: | ||
f.write('{"files":{},"package":null}') | ||
|
||
with open(crate_manifest_path, "r") as f: | ||
manifest_data = f.read() | ||
if "workspace" in manifest_data: | ||
workspace_root = get_manifest_metadata(crate_manifest_path)["workspace_root"] | ||
os.chmod(os.path.join(crate_out_dir, "Cargo.toml"), 0o644) | ||
subprocess.check_output(["replace-workspace-values", os.path.join(crate_out_dir, "Cargo.toml"), os.path.join(workspace_root, "Cargo.toml")]) | ||
eprint(crate_out_dir) | ||
|
||
key = source_info["url"] + ("" if (source_info["type"] is None) else ("?" + source_info["type"] + "=" + source_info["value"])) | ||
|
||
if key not in seen_keys: | ||
configLines.append(f'[source."{key}"]') | ||
configLines.append(f'git = "{source_info["url"]}"') | ||
if source_info["type"] is not None: | ||
configLines.append(f'{source_info["type"]} = "{source_info["value"]}"') | ||
configLines.append('replace-with = "vendored-sources"') | ||
|
||
seen_keys.add(key) | ||
|
||
elif source.startswith("registry+"): | ||
filename = f"{pkg["name"]}-{pkg["version"]}.tar.gz" | ||
dir_name = f"{pkg["name"]}-{pkg["version"]}" | ||
crate_out_dir = os.path.join(out_dir, dir_name) | ||
tarball_path = os.path.join(vendor_staging_dir, "registry", filename) | ||
os.mkdir(crate_out_dir) | ||
subprocess.check_output(["tar", "xf", tarball_path, "-C", crate_out_dir, "--strip-components=1"]) | ||
with open(os.path.join(crate_out_dir, ".cargo-checksum.json"), "w") as f: | ||
f.write(f'{{"files":{{}},"package":"{pkg["checksum"]}"}}') | ||
eprint(crate_out_dir) | ||
|
||
else: | ||
raise Exception(f"Can't process source: {source}.") | ||
|
||
os.mkdir(os.path.join(out_dir, ".cargo")) | ||
with open(os.path.join(out_dir, ".cargo", "config"), "w") as config_file: | ||
config_file.writelines(line + '\n' for line in configLines) | ||
|
||
|
||
def main() -> None: | ||
method = sys.argv[1] | ||
|
||
if method == "make-vendor-staging": | ||
create_vendor_staging(lockfile_path=sys.argv[2], out_dir=sys.argv[3]) | ||
elif method == "make-vendor": | ||
create_vendor(vendor_staging_dir=sys.argv[2], out_dir=sys.argv[3]) | ||
else: | ||
raise Exception(f"Unknown method: {method}") | ||
|
||
|
||
if __name__ == "__main__": | ||
main() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,90 @@ | ||
{ | ||
lib, | ||
stdenvNoCC, | ||
runCommand, | ||
writers, | ||
python3Packages, | ||
cargo, | ||
nix-prefetch-git, | ||
cacert, | ||
}: | ||
|
||
let | ||
replaceWorkspaceValues = writers.writePython3Bin "replace-workspace-values" { | ||
libraries = with python3Packages; [ | ||
tomli | ||
tomli-w | ||
]; | ||
flakeIgnore = [ | ||
"E501" | ||
"W503" | ||
]; | ||
} (builtins.readFile ./replace-workspace-values.py); | ||
|
||
mkCargoVendorDepsScript = writers.writePython3 "mk-cargo-vendor-deps-script" { | ||
libraries = with python3Packages; [ | ||
tomli | ||
requests | ||
]; | ||
flakeIgnore = [ | ||
"E501" | ||
"W503" | ||
]; | ||
} (builtins.readFile ./mk-cargo-vendor-deps-script.py); | ||
in | ||
|
||
{ | ||
name ? if args ? pname && args ? version then "${args.pname}-${args.version}" else "cargo-deps", | ||
hash ? lib.fakeHash, | ||
nativeBuildInputs ? [ ], | ||
... | ||
}@args: | ||
|
||
# TODO: add asserts | ||
|
||
let | ||
removedArgs = [ | ||
"name" | ||
"pname" | ||
"version" | ||
"nativeBuildInputs" | ||
"hash" | ||
]; | ||
|
||
vendorStaging = stdenvNoCC.mkDerivation ( | ||
{ | ||
name = "${name}-vendor-staging"; | ||
|
||
nativeBuildInputs = [ | ||
nix-prefetch-git | ||
cacert | ||
] ++ nativeBuildInputs; | ||
|
||
buildPhase = '' | ||
runHook preBuild | ||
${mkCargoVendorDepsScript} make-vendor-staging Cargo.lock "$out" | ||
runHook postBuild | ||
''; | ||
|
||
dontInstall = true; | ||
dontFixup = true; | ||
|
||
outputHash = hash; | ||
outputHashAlgo = "sha256"; | ||
outputHashMode = "recursive"; | ||
} | ||
// builtins.removeAttrs args removedArgs | ||
); | ||
in | ||
|
||
runCommand "${name}-vendor" | ||
{ | ||
inherit vendorStaging; | ||
nativeBuildInputs = [ | ||
cargo | ||
replaceWorkspaceValues | ||
]; | ||
} | ||
'' | ||
${mkCargoVendorDepsScript} make-vendor "$vendorStaging" "$out" | ||
'' |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters