| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							|  |  |  |   # 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 ! ( | 
					
						
							|  |  |  |     # Filter out Subversion and CVS directories. | 
					
						
							|  |  |  |     (type == "directory" && (baseName == ".git" || 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. | 
					
						
							|  |  |  |   # | 
					
						
							|  |  |  |   #     lib.cleanSourceWith f (lib.cleanSourceWith g ./.)     # Succeeds! | 
					
						
							|  |  |  |   #     builtins.filterSource f (builtins.filterSource g ./.) # Fails! | 
					
						
							|  |  |  |   cleanSourceWith = { filter, src }: | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  |     in { | 
					
						
							|  |  |  |       inherit origSrc; | 
					
						
							|  |  |  |       filter = filter'; | 
					
						
							|  |  |  |       outPath = builtins.filterSource filter' origSrc; | 
					
						
							|  |  |  |       _isLibCleanSourceWith = true; | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											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$"]` | 
					
						
							| 
									
										
										
										
											2018-01-02 00:29:20 -05:00
										 |  |  |   sourceByRegex = src: regexes: cleanSourceWith { | 
					
						
							|  |  |  |     filter = (path: type: | 
					
						
							|  |  |  |       let relPath = lib.removePrefix (toString src + "/") (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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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"; | 
					
						
							|  |  |  |         in if lib.pathExists fileName | 
					
						
							|  |  |  |            then | 
					
						
							| 
									
										
										
										
											2016-07-31 21:58:54 +09:00
										 |  |  |              let fileContent = lib.fileContents fileName; | 
					
						
							| 
									
										
										
										
											2016-05-24 23:34:28 +01:00
										 |  |  |                  # Sometimes git stores the commitId directly in the file but | 
					
						
							|  |  |  |                  # sometimes it stores something like: «ref: refs/heads/branch-name» | 
					
						
							| 
									
										
										
										
											2016-07-31 21:58:54 +09:00
										 |  |  |                  matchRef    = match "^ref: (.*)$" fileContent; | 
					
						
							| 
									
										
										
										
											2016-05-24 23:34:28 +01:00
										 |  |  |              in if   isNull matchRef | 
					
						
							| 
									
										
										
										
											2016-07-31 21:58:54 +09:00
										 |  |  |                 then fileContent | 
					
						
							| 
									
										
										
										
											2018-11-07 09:07:42 +00:00
										 |  |  |                 else readCommitFromFile (lib.head matchRef) path | 
					
						
							| 
									
										
										
										
											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: | 
					
						
							|  |  |  |            else if lib.pathExists packedRefsName | 
					
						
							|  |  |  |            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; | 
					
						
							| 
									
										
										
										
											2016-07-27 15:44:26 +01:00
										 |  |  |              in if   isNull matchRef | 
					
						
							|  |  |  |                 then throw ("Could not find " + file + " in " + packedRefsName) | 
					
						
							|  |  |  |                 else lib.head matchRef | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } |