mirror of
https://github.com/nix-community/naersk
synced 2024-11-26 13:30:17 +00:00
179 lines
5.9 KiB
Nix
179 lines
5.9 KiB
Nix
{ lib, writeText, runCommand, remarshal }:
|
|
with
|
|
{ builtinz =
|
|
builtins //
|
|
import ./builtins
|
|
{ inherit lib writeText remarshal runCommand ; };
|
|
};
|
|
rec
|
|
{
|
|
# creates an attrset from package name to package version + sha256
|
|
# (note: this includes the package's dependencies)
|
|
mkVersions = cargolock:
|
|
if builtins.hasAttr "metadata" cargolock then
|
|
|
|
# TODO: this should nub by <pkg-name>-<pkg-version>
|
|
(lib.concatMap (x:
|
|
with { mdk = mkMetadataKey x.name x.version; };
|
|
( lib.optional (builtins.hasAttr mdk cargolock.metadata)
|
|
{ inherit (x) version name;
|
|
sha256 = cargolock.metadata.${mkMetadataKey x.name x.version};
|
|
}
|
|
) ++ (lib.concatMap (parseDependency cargolock) (x.dependencies or []))
|
|
|
|
)
|
|
cargolock.package)
|
|
else [];
|
|
|
|
# Turns "lib-name lib-ver (registry+...)" to [ { name = "lib-name", etc } ]
|
|
# iff the package is present in the Cargo.lock (otherwise returns [])
|
|
parseDependency = cargolock: str:
|
|
with rec
|
|
{ components = lib.splitString " " str;
|
|
name = lib.elemAt components 0;
|
|
version = lib.elemAt components 1;
|
|
mdk = mkMetadataKey name version;
|
|
};
|
|
( lib.optional (builtins.hasAttr mdk cargolock.metadata)
|
|
(
|
|
with
|
|
{ sha256 = cargolock.metadata.${mkMetadataKey name version};
|
|
};
|
|
{ inherit name version sha256; }
|
|
));
|
|
|
|
|
|
# crafts the key used to look up the sha256 in the cargo lock; no
|
|
# robustness guarantee
|
|
mkMetadataKey = name: version:
|
|
"checksum ${name} ${version} (registry+https://github.com/rust-lang/crates.io-index)";
|
|
|
|
# A cargo lock that only includes the transitive dependencies of the
|
|
# package (and the package itself). This package is Nix-generated and thus
|
|
# only the transitive dependencies contribute to the package's derivation.
|
|
cargolockFor = cargolock: name: version:
|
|
with rec
|
|
{ tdeps = transitiveDeps cargolock name version;
|
|
tdepPrefix = dep: "checksum ${dep.name} ${dep.version}";
|
|
isTransitiveDep = p: lib.any
|
|
(d: d.package.name == p.name && d.package.version == p.version)
|
|
tdeps;
|
|
isTransitiveDepChecksumKey = k:
|
|
lib.any (tdep: lib.hasPrefix (tdepPrefix tdep.package) k) tdeps;
|
|
};
|
|
cargolock //
|
|
{ package = lib.filter (p:
|
|
(p.name == name && p.version == version) ||
|
|
(isTransitiveDep p)) cargolock.package;
|
|
|
|
metadata = lib.filterAttrs (k: _: isTransitiveDepChecksumKey k)
|
|
cargolock.metadata;
|
|
};
|
|
|
|
# A stripped down Cargo.toml, similar to cargolockFor
|
|
cargotomlFor = name: version:
|
|
{ package =
|
|
{ name = "dummy";
|
|
version = "0.1.0";
|
|
edition = "2018";
|
|
};
|
|
dependencies =
|
|
{ ${name} = version; };
|
|
};
|
|
|
|
# A very minimal 'src' which makes cargo happy nonetheless
|
|
dummySrc =
|
|
{ cargoconfig # string
|
|
, cargotomls # attrset
|
|
, cargolock # attrset
|
|
}:
|
|
let
|
|
config = writeText "config" cargoconfig;
|
|
cargolock' = builtinz.writeTOML "Cargo.toml" cargolock;
|
|
cargotomlss = writeText "foo"
|
|
(lib.concatStrings (lib.mapAttrsToList
|
|
(k: v: "${k}\n${builtinz.writeTOML "Cargo.toml-ds-aa" v}\n")
|
|
cargotomls
|
|
));
|
|
|
|
in
|
|
runCommand "dummy-src" {}
|
|
''
|
|
mkdir -p $out/.cargo
|
|
cp -r ${config} $out/.cargo/config
|
|
cp ${cargolock'} $out/Cargo.lock
|
|
|
|
cat ${cargotomlss} | \
|
|
while IFS= read -r member; do
|
|
read -r cargotoml
|
|
final_dir="$out/$member"
|
|
mkdir -p "$final_dir"
|
|
final_path="$final_dir/Cargo.toml"
|
|
cp $cargotoml "$final_path"
|
|
|
|
# make sure cargo is happy
|
|
pushd $out/$member > /dev/null
|
|
|
|
# TODO: get rid of all these hacks
|
|
|
|
sed -i '/.*=\s*{.*path\s*=.*/d' Cargo.toml
|
|
sed -i '/^build\s*=/d' Cargo.toml
|
|
|
|
mkdir -p src
|
|
touch src/main.rs
|
|
touch src/lib.rs
|
|
# Dirty hack to "touch" any bench, test, etc path.
|
|
cat Cargo.toml | \
|
|
grep -oP '^path\s*=\s*"\K\w+.rs' | \
|
|
while IFS= read -r path; do
|
|
mkdir -p $(dirname $path)
|
|
touch $path
|
|
done || true
|
|
cat Cargo.toml | \
|
|
grep -oP '^name\s*=\s*"\K\w+' | \
|
|
while IFS= read -r name; do
|
|
mkdir -p $name
|
|
mkdir -p benches
|
|
touch benches/$name.rs
|
|
done || true
|
|
popd > /dev/null
|
|
done
|
|
'';
|
|
|
|
mkPackages = cargolock:
|
|
lib.foldl' lib.recursiveUpdate {} (
|
|
map (p: { ${p.name} = { ${p.version} = p; } ; })
|
|
cargolock.package);
|
|
|
|
directDependencies = cargolock: name: version:
|
|
with rec
|
|
{ packages = mkPackages cargolock;
|
|
package = packages.${name}.${version};
|
|
} ;
|
|
|
|
lib.optionals (builtins.hasAttr "dependencies" package)
|
|
(map parseDependency' package.dependencies);
|
|
|
|
transitiveDeps = cargolock: name: version:
|
|
with
|
|
{ wrap = p:
|
|
{ key = "${p.name}-${p.version}";
|
|
package = p;
|
|
};
|
|
packages = mkPackages cargolock;
|
|
};
|
|
builtins.genericClosure
|
|
{ startSet = [ (wrap packages.${name}.${version}) ];
|
|
operator = p: map (dep: wrap (packages.${dep.name}.${dep.version})) (
|
|
(lib.optionals (builtins.hasAttr "dependencies" p.package)
|
|
(map parseDependency' p.package.dependencies)));
|
|
};
|
|
|
|
# turns "<package> <version> ..." into { name = <package>, version = <version>; }
|
|
parseDependency' = str:
|
|
with { components = lib.splitString " " str; };
|
|
{ name = lib.elemAt components 0; version = lib.elemAt components 1; };
|
|
|
|
allRemoteDependencies = cargolock:
|
|
[];
|
|
}
|