
This patch adds `lib.repoRevToName` function that generalizes away most of the code used for derivation name generation by `fetch*` functions (`fetchzip`, `fetchFromGitHub`, etc, except those which are delayed until latter commits for mass-rebuild reasons). It's first argument controls how the resulting name will look (see below). Since `lib` has no equivalent of Nixpkgs' `config`, this patch adds `config.fetchedSourceNameDefault` option to Nixpkgs and then re-exposes `lib.repoRevToName config.fetchedSourceNameDefault` expression as `pkgs.repoRevToNameMaybe` which is then used in `fetch*` derivations. The result is that different values of `config.fetchedSourceNameDefault` now control how the `src` derivations produced by `fetch*` functions are to be named, e.g.: - `fetchedSourceNameDefault = "source"` (the default): ``` $ nix-instantiate -A fuse.src /nix/store/<hash>-source.drv ``` - `fetchedSourceNameDefault = "versioned"`: ``` $ nix-instantiate -A fuse.src /nix/store/<hash>-libfuse-2.9.9-source.drv ``` - `fetchedSourceNameDefault = "full"`: ``` $ nix-instantiate -A fuse.src /nix/store/<hash>-libfuse-2.9.9-github-source.drv ``` See the documentation of `config.fetchedSourceNameDefault` for more info.
166 lines
4.6 KiB
Nix
166 lines
4.6 KiB
Nix
{
|
|
lib,
|
|
repoRevToNameMaybe,
|
|
fetchgit,
|
|
fetchzip,
|
|
}:
|
|
|
|
lib.makeOverridable (
|
|
{
|
|
owner,
|
|
repo,
|
|
tag ? null,
|
|
rev ? null,
|
|
name ? repoRevToNameMaybe repo (lib.revOrTag rev tag) "github",
|
|
fetchSubmodules ? false,
|
|
leaveDotGit ? null,
|
|
deepClone ? false,
|
|
private ? false,
|
|
forceFetchGit ? false,
|
|
fetchLFS ? false,
|
|
sparseCheckout ? [ ],
|
|
githubBase ? "github.com",
|
|
varPrefix ? null,
|
|
meta ? { },
|
|
... # For hash agility
|
|
}@args:
|
|
|
|
assert (
|
|
lib.assertMsg (lib.xor (tag == null) (
|
|
rev == null
|
|
)) "fetchFromGitHub requires one of either `rev` or `tag` to be provided (not both)."
|
|
);
|
|
|
|
let
|
|
|
|
position = (
|
|
if args.meta.description or null != null then
|
|
builtins.unsafeGetAttrPos "description" args.meta
|
|
else if tag != null then
|
|
builtins.unsafeGetAttrPos "tag" args
|
|
else
|
|
builtins.unsafeGetAttrPos "rev" args
|
|
);
|
|
baseUrl = "https://${githubBase}/${owner}/${repo}";
|
|
newMeta =
|
|
meta
|
|
// {
|
|
homepage = meta.homepage or baseUrl;
|
|
}
|
|
// lib.optionalAttrs (position != null) {
|
|
# to indicate where derivation originates, similar to make-derivation.nix's mkDerivation
|
|
position = "${position.file}:${toString position.line}";
|
|
};
|
|
passthruAttrs = removeAttrs args [
|
|
"owner"
|
|
"repo"
|
|
"tag"
|
|
"rev"
|
|
"fetchSubmodules"
|
|
"forceFetchGit"
|
|
"private"
|
|
"githubBase"
|
|
"varPrefix"
|
|
];
|
|
varBase = "NIX${lib.optionalString (varPrefix != null) "_${varPrefix}"}_GITHUB_PRIVATE_";
|
|
useFetchGit =
|
|
fetchSubmodules
|
|
|| (leaveDotGit == true)
|
|
|| deepClone
|
|
|| forceFetchGit
|
|
|| fetchLFS
|
|
|| (sparseCheckout != [ ]);
|
|
# We prefer fetchzip in cases we don't need submodules as the hash
|
|
# is more stable in that case.
|
|
fetcher =
|
|
if useFetchGit then
|
|
fetchgit
|
|
# fetchzip may not be overridable when using external tools, for example nix-prefetch
|
|
else if fetchzip ? override then
|
|
fetchzip.override { withUnzip = false; }
|
|
else
|
|
fetchzip;
|
|
privateAttrs = lib.optionalAttrs private {
|
|
netrcPhase =
|
|
# When using private repos:
|
|
# - Fetching with git works using https://github.com but not with the GitHub API endpoint
|
|
# - Fetching a tarball from a private repo requires to use the GitHub API endpoint
|
|
let
|
|
machineName = if githubBase == "github.com" && !useFetchGit then "api.github.com" else githubBase;
|
|
in
|
|
''
|
|
if [ -z "''$${varBase}USERNAME" -o -z "''$${varBase}PASSWORD" ]; then
|
|
echo "Error: Private fetchFromGitHub requires the nix building process (nix-daemon in multi user mode) to have the ${varBase}USERNAME and ${varBase}PASSWORD env vars set." >&2
|
|
exit 1
|
|
fi
|
|
cat > netrc <<EOF
|
|
machine ${machineName}
|
|
login ''$${varBase}USERNAME
|
|
password ''$${varBase}PASSWORD
|
|
EOF
|
|
'';
|
|
netrcImpureEnvVars = [
|
|
"${varBase}USERNAME"
|
|
"${varBase}PASSWORD"
|
|
];
|
|
};
|
|
|
|
gitRepoUrl = "${baseUrl}.git";
|
|
|
|
revWithTag = if tag != null then "refs/tags/${tag}" else rev;
|
|
|
|
fetcherArgs =
|
|
(
|
|
if useFetchGit then
|
|
{
|
|
inherit
|
|
tag
|
|
rev
|
|
deepClone
|
|
fetchSubmodules
|
|
sparseCheckout
|
|
fetchLFS
|
|
;
|
|
url = gitRepoUrl;
|
|
}
|
|
// lib.optionalAttrs (leaveDotGit != null) { inherit leaveDotGit; }
|
|
else
|
|
{
|
|
# Use the API endpoint for private repos, as the archive URI doesn't
|
|
# support access with GitHub's fine-grained access tokens.
|
|
#
|
|
# Use the archive URI for non-private repos, as the API endpoint has
|
|
# relatively restrictive rate limits for unauthenticated users.
|
|
url =
|
|
if private then
|
|
let
|
|
endpoint = "/repos/${owner}/${repo}/tarball/${revWithTag}";
|
|
in
|
|
if githubBase == "github.com" then
|
|
"https://api.github.com${endpoint}"
|
|
else
|
|
"https://${githubBase}/api/v3${endpoint}"
|
|
else
|
|
"${baseUrl}/archive/${revWithTag}.tar.gz";
|
|
extension = "tar.gz";
|
|
|
|
passthru = {
|
|
inherit gitRepoUrl;
|
|
};
|
|
}
|
|
)
|
|
// privateAttrs
|
|
// passthruAttrs
|
|
// {
|
|
inherit name;
|
|
};
|
|
in
|
|
|
|
fetcher fetcherArgs
|
|
// {
|
|
meta = newMeta;
|
|
inherit owner repo tag;
|
|
rev = revWithTag;
|
|
}
|
|
)
|