| 
									
										
										
										
											2009-02-09 16:51:03 +00:00
										 |  |  | # Functions for copying sources to the Nix store. | 
					
						
							| 
									
										
										
										
											2017-07-28 20:05:35 -04:00
										 |  |  | { lib }: | 
					
						
							| 
									
										
										
										
											2009-02-09 16:51:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | rec { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-02 16:03:35 +01:00
										 |  |  |   # Returns the type of a path: regular (for file), symlink, or directory | 
					
						
							|  |  |  |   pathType = p: with builtins; getAttr (baseNameOf p) (readDir (dirOf p)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # Returns true if the path exists and is a directory, false otherwise | 
					
						
							|  |  |  |   pathIsDirectory = p: if builtins.pathExists p then (pathType p) == "directory" else false; | 
					
						
							| 
									
										
										
										
											2009-02-09 16:51:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-10 21:24:31 +01:00
										 |  |  |   # Returns true if the path exists and is a regular file, false otherwise | 
					
						
							|  |  |  |   pathIsRegularFile = p: if builtins.pathExists p then (pathType p) == "regular" else false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-09 16:51:03 +00:00
										 |  |  |   # Bring in a path as a source, filtering out all Subversion and CVS | 
					
						
							|  |  |  |   # directories, as well as backup files (*~). | 
					
						
							| 
									
										
										
										
											2016-11-17 22:21:18 -05:00
										 |  |  |   cleanSourceFilter = name: type: let baseName = baseNameOf (toString name); in ! ( | 
					
						
							| 
									
										
										
										
											2019-04-02 18:01:07 +02:00
										 |  |  |     # Filter out version control software files/directories | 
					
						
							|  |  |  |     (baseName == ".git" || type == "directory" && (baseName == ".svn" || baseName == "CVS" || baseName == ".hg")) || | 
					
						
							| 
									
										
										
										
											2017-09-12 14:50:12 +03:00
										 |  |  |     # Filter out editor backup / swap files. | 
					
						
							| 
									
										
										
										
											2016-11-17 22:21:18 -05:00
										 |  |  |     lib.hasSuffix "~" baseName || | 
					
						
							| 
									
										
										
										
											2017-09-12 14:50:12 +03:00
										 |  |  |     builtins.match "^\\.sw[a-z]$" baseName != null || | 
					
						
							|  |  |  |     builtins.match "^\\..*\\.sw[a-z]$" baseName != null || | 
					
						
							| 
									
										
										
										
											2017-08-29 20:27:04 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-17 22:21:18 -05:00
										 |  |  |     # Filter out generates files. | 
					
						
							|  |  |  |     lib.hasSuffix ".o" baseName || | 
					
						
							|  |  |  |     lib.hasSuffix ".so" baseName || | 
					
						
							|  |  |  |     # Filter out nix-build result symlinks | 
					
						
							|  |  |  |     (type == "symlink" && lib.hasPrefix "result" baseName) | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-30 18:13:16 +01:00
										 |  |  |   # Filters a source tree removing version control files and directories using cleanSourceWith | 
					
						
							|  |  |  |   # | 
					
						
							|  |  |  |   # Example: | 
					
						
							|  |  |  |   #          cleanSource ./. | 
					
						
							| 
									
										
										
										
											2018-01-02 00:29:20 -05:00
										 |  |  |   cleanSource = src: cleanSourceWith { filter = cleanSourceFilter; inherit src; }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   # 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. | 
					
						
							|  |  |  |   # | 
					
						
							| 
									
										
										
										
											2019-09-03 10:36:57 +02:00
										 |  |  |   #     lib.cleanSourceWith { | 
					
						
							|  |  |  |   #       filter = f; | 
					
						
							|  |  |  |   #       src = lib.cleanSourceWith { | 
					
						
							|  |  |  |   #         filter = g; | 
					
						
							|  |  |  |   #         src = ./.; | 
					
						
							|  |  |  |   #       }; | 
					
						
							|  |  |  |   #     } | 
					
						
							|  |  |  |   #     # Succeeds! | 
					
						
							|  |  |  |   # | 
					
						
							|  |  |  |   #     builtins.filterSource f (builtins.filterSource g ./.) | 
					
						
							|  |  |  |   #     # Fails! | 
					
						
							|  |  |  |   # | 
					
						
							|  |  |  |   # Parameters: | 
					
						
							|  |  |  |   # | 
					
						
							|  |  |  |   #   src:      A path or cleanSourceWith result to filter and/or rename. | 
					
						
							|  |  |  |   # | 
					
						
							|  |  |  |   #   filter:   A function (path -> type -> bool) | 
					
						
							|  |  |  |   #             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 | 
					
						
							|  |  |  |   # | 
					
						
							|  |  |  |   #   name:     Optional name to use as part of the store path. | 
					
						
							|  |  |  |   #             This defaults `src.name` or otherwise `baseNameOf src`. | 
					
						
							|  |  |  |   #             We recommend setting `name` whenever `src` is syntactically `./.`. | 
					
						
							|  |  |  |   #             Otherwise, you depend on `./.`'s name in the parent directory, | 
					
						
							|  |  |  |   #             which can cause inconsistent names, defeating caching. | 
					
						
							|  |  |  |   # | 
					
						
							|  |  |  |   cleanSourceWith = { filter ? _path: _type: true, src, name ? null }: | 
					
						
							| 
									
										
										
										
											2018-01-02 00:29:20 -05:00
										 |  |  |     let | 
					
						
							|  |  |  |       isFiltered = src ? _isLibCleanSourceWith; | 
					
						
							|  |  |  |       origSrc = if isFiltered then src.origSrc else src; | 
					
						
							|  |  |  |       filter' = if isFiltered then name: type: filter name type && src.filter name type else filter; | 
					
						
							| 
									
										
										
										
											2019-09-03 10:36:57 +02:00
										 |  |  |       name' = if name != null then name else if isFiltered then src.name else baseNameOf src; | 
					
						
							| 
									
										
										
										
											2018-01-02 00:29:20 -05:00
										 |  |  |     in { | 
					
						
							|  |  |  |       inherit origSrc; | 
					
						
							|  |  |  |       filter = filter'; | 
					
						
							| 
									
										
										
										
											2019-09-03 10:36:57 +02:00
										 |  |  |       outPath = builtins.path { filter = filter'; path = origSrc; name = name'; }; | 
					
						
							| 
									
										
										
										
											2018-01-02 00:29:20 -05:00
										 |  |  |       _isLibCleanSourceWith = true; | 
					
						
							| 
									
										
										
										
											2019-09-03 10:36:57 +02:00
										 |  |  |       name = name'; | 
					
						
							| 
									
										
										
										
											2018-01-02 00:29:20 -05:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2009-02-09 16:51:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-17 18:56:39 +00:00
										 |  |  |   # Filter sources by a list of regular expressions. | 
					
						
							|  |  |  |   # | 
					
						
							|  |  |  |   # E.g. `src = sourceByRegex ./my-subproject [".*\.py$" "^database.sql$"]` | 
					
						
							| 
									
										
										
										
											2019-07-19 16:23:11 +02: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); | 
					
						
							|  |  |  |         in lib.any (re: builtins.match re relPath != null) regexes); | 
					
						
							|  |  |  |       inherit src; | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2009-02-09 16:51:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   # Get all files ending with the specified suffices from the given | 
					
						
							| 
									
										
										
										
											2014-08-25 14:33:17 +02:00
										 |  |  |   # directory or its descendants.  E.g. `sourceFilesBySuffices ./dir | 
					
						
							|  |  |  |   # [".xml" ".c"]'. | 
					
						
							| 
									
										
										
										
											2009-02-09 16:51:03 +00:00
										 |  |  |   sourceFilesBySuffices = path: exts: | 
					
						
							|  |  |  |     let filter = name: type: | 
					
						
							|  |  |  |       let base = baseNameOf (toString name); | 
					
						
							| 
									
										
										
										
											2014-08-25 14:33:17 +02:00
										 |  |  |       in type == "directory" || lib.any (ext: lib.hasSuffix ext base) exts; | 
					
						
							| 
									
										
										
										
											2018-01-02 00:29:20 -05:00
										 |  |  |     in cleanSourceWith { inherit filter; src = path; }; | 
					
						
							| 
									
										
										
										
											2009-02-09 16:51:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-20 00:44:07 +01:00
										 |  |  |   pathIsGitRepo = path: (builtins.tryEval (commitIdFromGitRepo path)).success; | 
					
						
							| 
									
										
										
										
											2016-06-02 16:03:35 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-24 23:34:28 +01:00
										 |  |  |   # Get the commit id of a git repo | 
					
						
							|  |  |  |   # Example: commitIdFromGitRepo <nixpkgs/.git> | 
					
						
							|  |  |  |   commitIdFromGitRepo = | 
					
						
							| 
									
										
										
										
											2018-11-07 09:07:42 +00:00
										 |  |  |     let readCommitFromFile = file: path: | 
					
						
							| 
									
										
										
										
											2016-05-24 23:34:28 +01:00
										 |  |  |       with builtins; | 
					
						
							|  |  |  |         let fileName       = toString path + "/" + file; | 
					
						
							|  |  |  |             packedRefsName = toString path + "/packed-refs"; | 
					
						
							| 
									
										
										
										
											2020-01-14 21:08:38 +01:00
										 |  |  |             absolutePath   = base: path: | 
					
						
							|  |  |  |               if lib.hasPrefix "/" path | 
					
						
							|  |  |  |               then path | 
					
						
							|  |  |  |               else toString (/. + "${base}/${path}"); | 
					
						
							| 
									
										
										
										
											2020-01-10 21:24:31 +01: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 throw ("File contains no gitdir reference: " + path) | 
					
						
							|  |  |  |                 else | 
					
						
							| 
									
										
										
										
											2020-01-14 21:08:38 +01:00
										 |  |  |                   let gitDir     = absolutePath (dirOf path) (lib.head m); | 
					
						
							| 
									
										
										
										
											2020-01-10 21:24:31 +01:00
										 |  |  |                       commonDir' = if pathIsRegularFile "${gitDir}/commondir" | 
					
						
							|  |  |  |                                    then lib.fileContents "${gitDir}/commondir" | 
					
						
							|  |  |  |                                    else gitDir; | 
					
						
							| 
									
										
										
										
											2020-01-14 21:08:38 +01:00
										 |  |  |                       commonDir  = absolutePath gitDir commonDir'; | 
					
						
							| 
									
										
										
										
											2020-01-10 21:24:31 +01:00
										 |  |  |                       refFile    = lib.removePrefix "${commonDir}/" "${gitDir}/${file}"; | 
					
						
							|  |  |  |                   in readCommitFromFile refFile commonDir | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |            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 23:34:28 +01:00
										 |  |  |            then | 
					
						
							| 
									
										
										
										
											2016-07-31 21:58:54 +09:00
										 |  |  |              let fileContent = lib.fileContents fileName; | 
					
						
							|  |  |  |                  matchRef    = match "^ref: (.*)$" fileContent; | 
					
						
							| 
									
										
										
										
											2020-01-10 21:24:31 +01:00
										 |  |  |              in if  matchRef == null | 
					
						
							| 
									
										
										
										
											2016-07-31 21:58:54 +09:00
										 |  |  |                 then fileContent | 
					
						
							| 
									
										
										
										
											2018-11-07 09:07:42 +00:00
										 |  |  |                 else readCommitFromFile (lib.head matchRef) path | 
					
						
							| 
									
										
										
										
											2020-01-10 21:24:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |            else if pathIsRegularFile packedRefsName | 
					
						
							| 
									
										
										
										
											2016-05-24 23:34:28 +01: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 15:44:26 +01:00
										 |  |  |              let fileContent = readFile packedRefsName; | 
					
						
							| 
									
										
										
										
											2016-08-08 11:43:39 +01:00
										 |  |  |                  matchRef    = match (".*\n([^\n ]*) " + file + "\n.*") fileContent; | 
					
						
							| 
									
										
										
										
											2020-01-10 21:24:31 +01:00
										 |  |  |              in if  matchRef == null | 
					
						
							| 
									
										
										
										
											2016-07-27 15:44:26 +01:00
										 |  |  |                 then throw ("Could not find " + file + " in " + packedRefsName) | 
					
						
							|  |  |  |                 else lib.head matchRef | 
					
						
							| 
									
										
										
										
											2020-01-10 21:24:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-24 23:34:28 +01:00
										 |  |  |            else throw ("Not a .git directory: " + path); | 
					
						
							| 
									
										
										
										
											2018-11-07 09:07:42 +00:00
										 |  |  |     in readCommitFromFile "HEAD"; | 
					
						
							| 
									
										
										
										
											2018-01-11 10:17:56 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   pathHasContext = builtins.hasContext or (lib.hasPrefix builtins.storeDir); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   canCleanSource = src: src ? _isLibCleanSourceWith || !(pathHasContext (toString src)); | 
					
						
							| 
									
										
										
										
											2009-02-09 16:51:03 +00:00
										 |  |  | } |