mirror of
https://github.com/NixOS/nixpkgs.git
synced 2025-07-14 14:10:33 +03:00

Be able to build arbitrary Julia environments in Nixpkgs, in the same style as python.withPackages.
99 lines
2.9 KiB
Python
Executable file
99 lines
2.9 KiB
Python
Executable file
|
|
import json
|
|
from pathlib import Path
|
|
import multiprocessing
|
|
import subprocess
|
|
import sys
|
|
import toml
|
|
import yaml
|
|
|
|
import dag
|
|
|
|
dependencies_path = Path(sys.argv[1])
|
|
closure_yaml_path = Path(sys.argv[2])
|
|
julia_path = Path(sys.argv[3])
|
|
extract_artifacts_script = Path(sys.argv[4])
|
|
extra_libs = json.loads(sys.argv[5])
|
|
out_path = Path(sys.argv[6])
|
|
|
|
with open(dependencies_path, "r") as f:
|
|
dependencies = yaml.safe_load(f)
|
|
dependency_uuids = dependencies.keys()
|
|
|
|
with open(closure_yaml_path, "r") as f:
|
|
# Build up a map of UUID -> closure information
|
|
closure_yaml_list = yaml.safe_load(f) or []
|
|
closure_yaml = {}
|
|
for item in closure_yaml_list:
|
|
closure_yaml[item["uuid"]] = item
|
|
|
|
# Build up a dependency graph of UUIDs
|
|
closure_dependencies_dag = dag.DAG()
|
|
for uuid, contents in closure_yaml.items():
|
|
if contents.get("depends_on"):
|
|
closure_dependencies_dag.add_node(uuid, dependencies=contents["depends_on"].values())
|
|
|
|
with open(out_path, "w") as f:
|
|
f.write("{ lib, fetchurl, glibc, pkgs, stdenv }:\n\n")
|
|
f.write("rec {\n")
|
|
|
|
def process_item(item):
|
|
uuid, src = item
|
|
lines = []
|
|
artifacts = toml.loads(subprocess.check_output([julia_path, extract_artifacts_script, uuid, src]).decode())
|
|
if not artifacts: return f' uuid-{uuid} = {{}};\n'
|
|
|
|
lines.append(f' uuid-{uuid} = {{')
|
|
|
|
for artifact_name, details in artifacts.items():
|
|
if len(details["download"]) == 0: continue
|
|
download = details["download"][0]
|
|
url = download["url"]
|
|
sha256 = download["sha256"]
|
|
|
|
git_tree_sha1 = details["git-tree-sha1"]
|
|
|
|
depends_on = set()
|
|
if closure_dependencies_dag.has_node(uuid):
|
|
depends_on = set(closure_dependencies_dag.get_dependencies(uuid)).intersection(dependency_uuids)
|
|
|
|
other_libs = extra_libs.get(uuid, [])
|
|
|
|
fixup = f"""fixupPhase = let
|
|
libs = lib.concatMap (lib.mapAttrsToList (k: v: v.path))
|
|
[{" ".join(["uuid-" + x for x in depends_on])}];
|
|
in ''
|
|
find $out -type f -executable -exec \
|
|
patchelf --set-rpath \$ORIGIN:\$ORIGIN/../lib:${{lib.makeLibraryPath (["$out" glibc] ++ libs ++ (with pkgs; [{" ".join(other_libs)}]))}} {{}} \;
|
|
find $out -type f -executable -exec \
|
|
patchelf --set-interpreter ${{glibc}}/lib/ld-linux-x86-64.so.2 {{}} \;
|
|
''"""
|
|
|
|
derivation = f"""{{
|
|
name = "{artifact_name}";
|
|
src = fetchurl {{
|
|
url = "{url}";
|
|
sha256 = "{sha256}";
|
|
}};
|
|
sourceRoot = ".";
|
|
dontConfigure = true;
|
|
dontBuild = true;
|
|
installPhase = "cp -r . $out";
|
|
{fixup};
|
|
}}"""
|
|
|
|
lines.append(f""" "{artifact_name}" = {{
|
|
sha1 = "{git_tree_sha1}";
|
|
path = stdenv.mkDerivation {derivation};
|
|
}};\n""")
|
|
|
|
lines.append(' };\n')
|
|
|
|
return "\n".join(lines)
|
|
|
|
with multiprocessing.Pool(10) as pool:
|
|
for s in pool.map(process_item, dependencies.items()):
|
|
f.write(s)
|
|
|
|
f.write(f"""
|
|
}}\n""")
|