2023-11-20 02:02:11 +00:00
|
|
|
# Functions for copying sources to the Nix store.
|
2017-07-29 00:05:35 +00:00
|
|
|
{ lib }:
|
2009-02-09 16:51:03 +00:00
|
|
|
|
2021-02-05 23:15:33 +00:00
|
|
|
# Tested in lib/tests/sources.sh
|
2020-10-20 11:47:24 +00:00
|
|
|
let
|
|
|
|
inherit (builtins)
|
|
|
|
match
|
2020-10-30 22:15:07 +00:00
|
|
|
split
|
2020-10-20 11:47:24 +00:00
|
|
|
storeDir
|
|
|
|
;
|
|
|
|
inherit (lib)
|
2021-02-21 11:36:42 +00:00
|
|
|
boolToString
|
2020-10-20 11:47:24 +00:00
|
|
|
filter
|
|
|
|
isString
|
2020-10-22 11:57:23 +00:00
|
|
|
readFile
|
2020-10-20 11:47:24 +00:00
|
|
|
;
|
2023-04-05 14:10:09 +00:00
|
|
|
inherit (lib.filesystem)
|
|
|
|
pathIsRegularFile
|
|
|
|
;
|
2020-01-10 20:24:31 +00:00
|
|
|
|
2022-01-30 01:01:32 +00:00
|
|
|
/*
|
|
|
|
A basic filter for `cleanSourceWith` that removes
|
|
|
|
directories of version control system, backup files (*~)
|
|
|
|
and some generated files.
|
|
|
|
*/
|
2016-11-18 03:21:18 +00:00
|
|
|
cleanSourceFilter =
|
|
|
|
name: type:
|
|
|
|
let
|
|
|
|
baseName = baseNameOf (toString name);
|
|
|
|
in
|
|
|
|
!(
|
2019-04-02 16:01:07 +00:00
|
|
|
# Filter out version control software files/directories
|
|
|
|
(
|
|
|
|
baseName == ".git"
|
|
|
|
|| type == "directory" && (baseName == ".svn" || baseName == "CVS" || baseName == ".hg")
|
2024-12-10 19:26:33 +00:00
|
|
|
)
|
2019-04-02 16:01:07 +00:00
|
|
|
||
|
2017-09-12 11:50:12 +00:00
|
|
|
# Filter out editor backup / swap files.
|
2016-11-18 03:21:18 +00:00
|
|
|
lib.hasSuffix "~" baseName
|
2020-10-20 11:47:24 +00:00
|
|
|
|| match "^\\.sw[a-z]$" baseName != null
|
|
|
|
|| match "^\\..*\\.sw[a-z]$" baseName != null
|
2024-12-10 19:26:33 +00:00
|
|
|
||
|
|
|
|
|
2016-11-18 03:21:18 +00:00
|
|
|
# Filter out generates files.
|
|
|
|
lib.hasSuffix ".o" baseName
|
|
|
|
|| lib.hasSuffix ".so" baseName
|
2024-12-10 19:26:33 +00:00
|
|
|
||
|
2016-11-18 03:21:18 +00:00
|
|
|
# Filter out nix-build result symlinks
|
2021-09-17 09:59:37 +00:00
|
|
|
(type == "symlink" && lib.hasPrefix "result" baseName)
|
|
|
|
||
|
|
|
|
# Filter out sockets and other types of files we can't have in the store.
|
|
|
|
(type == "unknown")
|
2016-11-18 03:21:18 +00:00
|
|
|
);
|
|
|
|
|
2022-01-30 01:01:32 +00:00
|
|
|
/*
|
|
|
|
Filters a source tree removing version control files and directories using cleanSourceFilter.
|
|
|
|
|
|
|
|
Example:
|
|
|
|
cleanSource ./.
|
|
|
|
*/
|
2018-01-02 05:29:20 +00:00
|
|
|
cleanSource =
|
|
|
|
src:
|
|
|
|
cleanSourceWith {
|
|
|
|
filter = cleanSourceFilter;
|
|
|
|
inherit src;
|
|
|
|
};
|
|
|
|
|
2022-01-30 01:01:32 +00:00
|
|
|
/*
|
|
|
|
Like `builtins.filterSource`, except it will compose with itself,
|
|
|
|
allowing you to chain multiple calls together without any
|
|
|
|
intermediate copies being put in the nix store.
|
|
|
|
|
|
|
|
Example:
|
|
|
|
lib.cleanSourceWith {
|
|
|
|
filter = f;
|
|
|
|
src = lib.cleanSourceWith {
|
|
|
|
filter = g;
|
|
|
|
src = ./.;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
# Succeeds!
|
|
|
|
|
|
|
|
builtins.filterSource f (builtins.filterSource g ./.)
|
|
|
|
# Fails!
|
|
|
|
*/
|
|
|
|
cleanSourceWith =
|
|
|
|
{
|
|
|
|
# A path or cleanSourceWith result to filter and/or rename.
|
|
|
|
src,
|
|
|
|
# Optional with default value: constant true (include everything)
|
|
|
|
# The function will be combined with the && operator such
|
|
|
|
# that src.filter is called lazily.
|
|
|
|
# For implementing a filter, see
|
|
|
|
# https://nixos.org/nix/manual/#builtin-filterSource
|
|
|
|
# Type: A function (path -> type -> bool)
|
|
|
|
filter ? _path: _type: true,
|
|
|
|
# Optional name to use as part of the store path.
|
|
|
|
# This defaults to `src.name` or otherwise `"source"`.
|
|
|
|
name ? null,
|
|
|
|
}:
|
2018-01-02 05:29:20 +00:00
|
|
|
let
|
2021-02-05 22:26:46 +00:00
|
|
|
orig = toSourceAttributes src;
|
|
|
|
in
|
|
|
|
fromSourceAttributes {
|
|
|
|
inherit (orig) origSrc;
|
|
|
|
filter = path: type: filter path type && orig.filter path type;
|
|
|
|
name = if name != null then name else orig.name;
|
2018-01-02 05:29:20 +00:00
|
|
|
};
|
2009-02-09 16:51:03 +00:00
|
|
|
|
2021-02-21 11:36:42 +00:00
|
|
|
/*
|
|
|
|
Add logging to a source, for troubleshooting the filtering behavior.
|
|
|
|
Type:
|
|
|
|
sources.trace :: sourceLike -> Source
|
|
|
|
*/
|
|
|
|
trace =
|
|
|
|
# Source to debug. The returned source will behave like this source, but also log its filter invocations.
|
|
|
|
src:
|
|
|
|
let
|
|
|
|
attrs = toSourceAttributes src;
|
|
|
|
in
|
|
|
|
fromSourceAttributes (
|
|
|
|
attrs
|
|
|
|
// {
|
|
|
|
filter =
|
|
|
|
path: type:
|
|
|
|
let
|
|
|
|
r = attrs.filter path type;
|
|
|
|
in
|
|
|
|
builtins.trace "${attrs.name}.filter ${path} = ${boolToString r}" r;
|
|
|
|
}
|
|
|
|
)
|
|
|
|
// {
|
|
|
|
satisfiesSubpathInvariant = src ? satisfiesSubpathInvariant && src.satisfiesSubpathInvariant;
|
|
|
|
};
|
|
|
|
|
2022-01-30 01:01:32 +00:00
|
|
|
/*
|
|
|
|
Filter sources by a list of regular expressions.
|
|
|
|
|
|
|
|
Example: src = sourceByRegex ./my-subproject [".*\.py$" "^database.sql$"]
|
|
|
|
*/
|
2019-07-19 14:23:11 +00:00
|
|
|
sourceByRegex =
|
|
|
|
src: regexes:
|
|
|
|
let
|
|
|
|
isFiltered = src ? _isLibCleanSourceWith;
|
|
|
|
origSrc = if isFiltered then src.origSrc else src;
|
|
|
|
in
|
|
|
|
lib.cleanSourceWith {
|
|
|
|
filter = (
|
|
|
|
path: type:
|
|
|
|
let
|
|
|
|
relPath = lib.removePrefix (toString origSrc + "/") (toString path);
|
2020-10-20 11:47:24 +00:00
|
|
|
in
|
|
|
|
lib.any (re: match re relPath != null) regexes
|
|
|
|
);
|
2019-07-19 14:23:11 +00:00
|
|
|
inherit src;
|
|
|
|
};
|
2009-02-09 16:51:03 +00:00
|
|
|
|
2021-03-28 19:49:20 +00:00
|
|
|
/*
|
|
|
|
Get all files ending with the specified suffices from the given
|
|
|
|
source directory or its descendants, omitting files that do not match
|
|
|
|
any suffix. The result of the example below will include files like
|
|
|
|
`./dir/module.c` and `./dir/subdir/doc.xml` if present.
|
|
|
|
|
|
|
|
Type: sourceLike -> [String] -> Source
|
|
|
|
|
|
|
|
Example:
|
|
|
|
sourceFilesBySuffices ./. [ ".xml" ".c" ]
|
|
|
|
*/
|
|
|
|
sourceFilesBySuffices =
|
|
|
|
# Path or source containing the files to be returned
|
|
|
|
src:
|
|
|
|
# A list of file suffix strings
|
|
|
|
exts:
|
2009-02-09 16:51:03 +00:00
|
|
|
let
|
|
|
|
filter =
|
|
|
|
name: type:
|
|
|
|
let
|
|
|
|
base = baseNameOf (toString name);
|
2014-08-25 12:33:17 +00:00
|
|
|
in
|
|
|
|
type == "directory" || lib.any (ext: lib.hasSuffix ext base) exts;
|
2021-03-28 19:49:20 +00:00
|
|
|
in
|
|
|
|
cleanSourceWith { inherit filter src; };
|
2009-02-09 16:51:03 +00:00
|
|
|
|
2022-11-18 15:17:51 +00:00
|
|
|
pathIsGitRepo = path: (_commitIdFromGitRepoOrError path) ? value;
|
2016-06-02 15:03:35 +00:00
|
|
|
|
2022-01-30 01:01:32 +00:00
|
|
|
/*
|
|
|
|
Get the commit id of a git repo.
|
|
|
|
|
|
|
|
Example: commitIdFromGitRepo <nixpkgs/.git>
|
|
|
|
*/
|
2022-11-07 11:52:13 +00:00
|
|
|
commitIdFromGitRepo =
|
|
|
|
path:
|
2022-11-18 15:17:51 +00:00
|
|
|
let
|
|
|
|
commitIdOrError = _commitIdFromGitRepoOrError path;
|
2022-11-07 11:52:13 +00:00
|
|
|
in
|
|
|
|
commitIdOrError.value or (throw commitIdOrError.error);
|
|
|
|
|
2022-11-18 15:17:51 +00:00
|
|
|
# Get the commit id of a git repo.
|
2022-11-07 11:52:13 +00:00
|
|
|
|
2022-11-18 15:17:51 +00:00
|
|
|
# Returns `{ value = commitHash }` or `{ error = "... message ..." }`.
|
2022-11-07 11:52:13 +00:00
|
|
|
|
2022-11-18 15:17:51 +00:00
|
|
|
# Example: commitIdFromGitRepo <nixpkgs/.git>
|
|
|
|
# not exported, used for commitIdFromGitRepo
|
|
|
|
_commitIdFromGitRepoOrError =
|
2018-11-07 09:07:42 +00:00
|
|
|
let
|
|
|
|
readCommitFromFile =
|
|
|
|
file: path:
|
2022-11-06 04:32:22 +00:00
|
|
|
let
|
|
|
|
fileName = path + "/${file}";
|
|
|
|
packedRefsName = path + "/packed-refs";
|
2020-01-14 20:08:38 +00:00
|
|
|
absolutePath =
|
|
|
|
base: path: if lib.hasPrefix "/" path then path else toString (/. + "${base}/${path}");
|
2020-01-10 20:24:31 +00:00
|
|
|
in
|
|
|
|
if
|
|
|
|
pathIsRegularFile path
|
|
|
|
# Resolve git worktrees. See gitrepository-layout(5)
|
|
|
|
then
|
|
|
|
let
|
|
|
|
m = match "^gitdir: (.*)$" (lib.fileContents path);
|
|
|
|
in
|
|
|
|
if m == null then
|
2022-11-07 11:52:13 +00:00
|
|
|
{ error = "File contains no gitdir reference: " + path; }
|
2020-01-10 20:24:31 +00:00
|
|
|
else
|
2021-03-27 20:16:31 +00:00
|
|
|
let
|
|
|
|
gitDir = absolutePath (dirOf path) (lib.head m);
|
|
|
|
commonDir'' =
|
|
|
|
if pathIsRegularFile "${gitDir}/commondir" then lib.fileContents "${gitDir}/commondir" else gitDir;
|
|
|
|
commonDir' = lib.removeSuffix "/" commonDir'';
|
|
|
|
commonDir = absolutePath gitDir commonDir';
|
|
|
|
refFile = lib.removePrefix "${commonDir}/" "${gitDir}/${file}";
|
2020-01-10 20:24:31 +00:00
|
|
|
in
|
|
|
|
readCommitFromFile refFile commonDir
|
2024-12-10 19:26:33 +00:00
|
|
|
|
2020-01-10 20:24:31 +00:00
|
|
|
else if
|
|
|
|
pathIsRegularFile fileName
|
|
|
|
# Sometimes git stores the commitId directly in the file but
|
|
|
|
# sometimes it stores something like: «ref: refs/heads/branch-name»
|
2016-05-24 22:34:28 +00:00
|
|
|
then
|
2016-07-31 12:58:54 +00:00
|
|
|
let
|
|
|
|
fileContent = lib.fileContents fileName;
|
|
|
|
matchRef = match "^ref: (.*)$" fileContent;
|
2020-01-10 20:24:31 +00:00
|
|
|
in
|
2018-11-07 09:07:42 +00:00
|
|
|
if matchRef == null then { value = fileContent; } else readCommitFromFile (lib.head matchRef) path
|
2024-12-10 19:26:33 +00:00
|
|
|
|
2020-01-10 20:24:31 +00:00
|
|
|
else if
|
|
|
|
pathIsRegularFile packedRefsName
|
2016-05-24 22:34:28 +00:00
|
|
|
# Sometimes, the file isn't there at all and has been packed away in the
|
|
|
|
# packed-refs file, so we have to grep through it:
|
|
|
|
then
|
2016-07-27 14:44:26 +00:00
|
|
|
let
|
|
|
|
fileContent = readFile packedRefsName;
|
2020-10-20 11:47:24 +00:00
|
|
|
matchRef = match "([a-z0-9]+) ${file}";
|
|
|
|
isRef = s: isString s && (matchRef s) != null;
|
2020-07-17 09:38:38 +00:00
|
|
|
# there is a bug in libstdc++ leading to stackoverflow for long strings:
|
|
|
|
# https://github.com/NixOS/nix/issues/2147#issuecomment-659868795
|
2020-10-20 11:47:24 +00:00
|
|
|
refs = filter isRef (split "\n" fileContent);
|
2020-07-17 09:38:38 +00:00
|
|
|
in
|
|
|
|
if refs == [ ] then
|
2022-11-07 11:52:13 +00:00
|
|
|
{ error = "Could not find " + file + " in " + packedRefsName; }
|
|
|
|
else
|
|
|
|
{ value = lib.head (matchRef (lib.head refs)); }
|
2024-12-10 19:26:33 +00:00
|
|
|
|
2022-11-07 11:52:13 +00:00
|
|
|
else
|
|
|
|
{ error = "Not a .git directory: " + toString path; };
|
2018-11-07 09:07:42 +00:00
|
|
|
in
|
|
|
|
readCommitFromFile "HEAD";
|
2018-01-11 15:17:56 +00:00
|
|
|
|
2020-10-20 11:47:24 +00:00
|
|
|
pathHasContext = builtins.hasContext or (lib.hasPrefix storeDir);
|
2018-01-11 15:17:56 +00:00
|
|
|
|
|
|
|
canCleanSource = src: src ? _isLibCleanSourceWith || !(pathHasContext (toString src));
|
2021-02-05 22:26:46 +00:00
|
|
|
|
|
|
|
# -------------------------------------------------------------------------- #
|
|
|
|
# Internal functions
|
|
|
|
#
|
|
|
|
|
|
|
|
# toSourceAttributes : sourceLike -> SourceAttrs
|
|
|
|
#
|
|
|
|
# Convert any source-like object into a simple, singular representation.
|
|
|
|
# We don't expose this representation in order to avoid having a fifth path-
|
|
|
|
# like class of objects in the wild.
|
|
|
|
# (Existing ones being: paths, strings, sources and x//{outPath})
|
|
|
|
# So instead of exposing internals, we build a library of combinator functions.
|
|
|
|
toSourceAttributes =
|
|
|
|
src:
|
|
|
|
let
|
|
|
|
isFiltered = src ? _isLibCleanSourceWith;
|
|
|
|
in
|
|
|
|
{
|
|
|
|
# The original path
|
|
|
|
origSrc = if isFiltered then src.origSrc else src;
|
|
|
|
filter = if isFiltered then src.filter else _: _: true;
|
|
|
|
name = if isFiltered then src.name else "source";
|
|
|
|
};
|
|
|
|
|
|
|
|
# fromSourceAttributes : SourceAttrs -> Source
|
|
|
|
#
|
|
|
|
# Inverse of toSourceAttributes for Source objects.
|
|
|
|
fromSourceAttributes =
|
|
|
|
{
|
|
|
|
origSrc,
|
|
|
|
filter,
|
|
|
|
name,
|
|
|
|
}:
|
|
|
|
{
|
|
|
|
_isLibCleanSourceWith = true;
|
|
|
|
inherit origSrc filter name;
|
|
|
|
outPath = builtins.path {
|
|
|
|
inherit filter name;
|
|
|
|
path = origSrc;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
in
|
|
|
|
{
|
|
|
|
|
2024-10-08 09:14:24 +00:00
|
|
|
pathType =
|
|
|
|
lib.warnIf (lib.oldestSupportedReleaseIsAtLeast 2305)
|
2023-04-05 14:10:09 +00:00
|
|
|
"lib.sources.pathType has been moved to lib.filesystem.pathType."
|
|
|
|
lib.filesystem.pathType;
|
|
|
|
|
2024-10-08 09:14:24 +00:00
|
|
|
pathIsDirectory =
|
|
|
|
lib.warnIf (lib.oldestSupportedReleaseIsAtLeast 2305)
|
2023-04-05 14:10:09 +00:00
|
|
|
"lib.sources.pathIsDirectory has been moved to lib.filesystem.pathIsDirectory."
|
|
|
|
lib.filesystem.pathIsDirectory;
|
|
|
|
|
2024-10-08 09:14:24 +00:00
|
|
|
pathIsRegularFile =
|
|
|
|
lib.warnIf (lib.oldestSupportedReleaseIsAtLeast 2305)
|
2023-04-05 14:10:09 +00:00
|
|
|
"lib.sources.pathIsRegularFile has been moved to lib.filesystem.pathIsRegularFile."
|
|
|
|
lib.filesystem.pathIsRegularFile;
|
|
|
|
|
|
|
|
inherit
|
2021-02-05 22:26:46 +00:00
|
|
|
pathIsGitRepo
|
|
|
|
commitIdFromGitRepo
|
|
|
|
|
|
|
|
cleanSource
|
|
|
|
cleanSourceWith
|
|
|
|
cleanSourceFilter
|
|
|
|
pathHasContext
|
|
|
|
canCleanSource
|
|
|
|
|
|
|
|
sourceByRegex
|
|
|
|
sourceFilesBySuffices
|
2021-02-21 11:36:42 +00:00
|
|
|
|
|
|
|
trace
|
2021-02-05 22:26:46 +00:00
|
|
|
;
|
2009-02-09 16:51:03 +00:00
|
|
|
}
|