2019-06-04 18:50:29 +02:00
|
|
|
{ lib }:
|
|
|
|
|
2020-03-30 09:45:25 +02:00
|
|
|
# This Nix expression allows filtering a local directory by
|
|
|
|
# specifying dirRootsToInclude, dirsToExclude and filesToInclude.
|
|
|
|
# It also filters out symlinks to result folders created by nix-build,
|
|
|
|
# as well as backup/swap/generated files.
|
2019-06-04 18:50:29 +02:00
|
|
|
|
|
|
|
let
|
2019-07-29 10:33:35 +02:00
|
|
|
inherit (lib)
|
2020-03-30 09:45:25 +02:00
|
|
|
any range flatten length sublist cleanSourceFilter
|
|
|
|
splitString hasPrefix removePrefix concatStringsSep;
|
|
|
|
inherit (builtins) map match;
|
2019-06-04 18:50:29 +02:00
|
|
|
|
2019-11-25 10:17:28 +01:00
|
|
|
mkFilter = {
|
2020-03-30 09:45:25 +02:00
|
|
|
# primary path under which all files are included, unless excluded
|
|
|
|
root,
|
|
|
|
# list of regex expressions to match files to include/exclude
|
2020-04-08 15:06:40 +02:00
|
|
|
include ? [ ],
|
|
|
|
exclude ? [ ], # has precedence over include
|
|
|
|
# by default we ignore Version Control System files
|
|
|
|
ignoreVCS ? true,
|
|
|
|
}:
|
2019-06-04 18:50:29 +02:00
|
|
|
let
|
2020-03-30 09:45:25 +02:00
|
|
|
# removes superfluous slashes from the path
|
2019-12-13 12:54:28 +01:00
|
|
|
cleanRoot = "${toString (/. + root)}/";
|
2020-03-30 09:45:25 +02:00
|
|
|
in path: type:
|
|
|
|
let
|
|
|
|
# unpack path: "x/y/.*" => ["x" "x/y" "x/y/.*"]
|
|
|
|
unpackPath = path:
|
|
|
|
let
|
|
|
|
tokens = splitString "/" path;
|
|
|
|
perms = range 1 (length tokens);
|
|
|
|
subPaths = builtins.map (x: sublist 0 x tokens) perms;
|
|
|
|
in builtins.map (x: concatStringsSep "/" x) subPaths;
|
|
|
|
# accept subdirs from regexes
|
|
|
|
includeSubdirs = regexes: flatten (map unpackPath regexes);
|
|
|
|
# checks all regexes in a list against str
|
|
|
|
matchesRegexes = str: regexes: (map (r: (match r str)) regexes);
|
|
|
|
# match returns empty list on match
|
|
|
|
isMatch = x: x == [ ];
|
|
|
|
# path relative to search root
|
|
|
|
relPath = removePrefix cleanRoot path;
|
|
|
|
# check if any of the regexes match the relative path
|
|
|
|
checkRegexes = regexes: any isMatch (matchesRegexes relPath regexes);
|
|
|
|
|
|
|
|
# main check methods
|
|
|
|
isRootSubdir = hasPrefix cleanRoot path;
|
|
|
|
isIncluded = checkRegexes (includeSubdirs include);
|
|
|
|
isExcluded = checkRegexes exclude;
|
2020-04-08 15:06:40 +02:00
|
|
|
isVCS = ignoreVCS && !cleanSourceFilter path type;
|
2019-06-04 18:50:29 +02:00
|
|
|
|
2020-03-30 09:45:25 +02:00
|
|
|
in
|
|
|
|
if !isRootSubdir then
|
|
|
|
# everything outside of root is excluded
|
|
|
|
false
|
2020-04-08 15:06:40 +02:00
|
|
|
else if isExcluded || isVCS then
|
2020-03-30 09:45:25 +02:00
|
|
|
# isExcluded has precedence over isIncluded
|
|
|
|
false
|
|
|
|
else
|
|
|
|
isIncluded;
|
2019-06-04 18:50:29 +02:00
|
|
|
|
|
|
|
in mkFilter
|