biome: 1.9.2 -> 1.9.3 (#349335)
[NixPkgs.git] / pkgs / build-support / go / module.nix
blob3a3ae71de508b15d453f367fd5dfe0028f5a8450
1 { go, cacert, git, lib, stdenv }:
3 { name ? "${args'.pname}-${args'.version}"
4   # The source used to build the derivation.
5 , src
6   # Native build inputs used for the derivation.
7 , nativeBuildInputs ? [ ]
8 , passthru ? { }
9 , patches ? [ ]
11   # A function to override the `goModules` derivation.
12 , overrideModAttrs ? (finalAttrs: previousAttrs: { })
14   # Directory to the `go.mod` and `go.sum` relative to the `src`.
15 , modRoot ? "./"
17   # The SRI hash of the vendored dependencies.
18   # If `vendorHash` is `null`, no dependencies are fetched and
19   # the build relies on the vendor folder within the source.
20 , vendorHash ? throw (
21     if args'?vendorSha256 then
22       "buildGoModule: Expect vendorHash instead of vendorSha256"
23     else
24       "buildGoModule: vendorHash is missing"
25   )
27   # Whether to delete the vendor folder supplied with the source.
28 , deleteVendor ? false
30   # Whether to fetch (go mod download) and proxy the vendor directory.
31   # This is useful if your code depends on c code and go mod tidy does not
32   # include the needed sources to build or if any dependency has case-insensitive
33   # conflicts which will produce platform dependant `vendorHash` checksums.
34 , proxyVendor ? false
36   # We want parallel builds by default.
37 , enableParallelBuilding ? true
39   # Do not enable this without good reason
40   # IE: programs coupled with the compiler.
41 , allowGoReference ? false
43   # Go env. variable to enable CGO.
44 , CGO_ENABLED ? go.CGO_ENABLED
46   # Meta data for the final derivation.
47 , meta ? { }
49   # Not needed with `buildGoModule`.
50 , goPackagePath ? ""
52   # Go linker flags.
53 , ldflags ? [ ]
54   # Go build flags.
55 , GOFLAGS ? [ ]
57   # Needed for buildFlags{,Array} warning
58 , buildFlags ? "" # deprecated
59 , buildFlagsArray ? "" # deprecated
61 , ...
62 }@args':
64 assert goPackagePath != "" -> throw "`goPackagePath` is not needed with `buildGoModule`";
66 let
67   args = removeAttrs args' [ "overrideModAttrs" "vendorSha256" ];
69   GO111MODULE = "on";
70   GOTOOLCHAIN = "local";
73 (stdenv.mkDerivation (finalAttrs:
74   args
75   // {
77   inherit modRoot vendorHash deleteVendor proxyVendor;
78   goModules = if (finalAttrs.vendorHash == null) then "" else
79   (stdenv.mkDerivation {
80     name = "${finalAttrs.name or "${finalAttrs.pname}-${finalAttrs.version}"}-go-modules";
82     nativeBuildInputs = (finalAttrs.nativeBuildInputs or [ ]) ++ [ go git cacert ];
84     inherit (finalAttrs) src modRoot;
85     inherit (go) GOOS GOARCH;
86     inherit GO111MODULE GOTOOLCHAIN;
88     # The following inheritance behavior is not trivial to expect, and some may
89     # argue it's not ideal. Changing it may break vendor hashes in Nixpkgs and
90     # out in the wild. In anycase, it's documented in:
91     # doc/languages-frameworks/go.section.md.
92     prePatch = finalAttrs.prePatch or "";
93     patches = finalAttrs.patches or [ ];
94     patchFlags = finalAttrs.patchFlags or [ ];
95     postPatch = finalAttrs.postPatch or "";
96     preBuild = finalAttrs.preBuild or "";
97     postBuild = finalAttrs.modPostBuild or "";
98     sourceRoot = finalAttrs.sourceRoot or "";
99     setSourceRoot = finalAttrs.setSourceRoot or "";
100     env = finalAttrs.env or { };
102     impureEnvVars = lib.fetchers.proxyImpureEnvVars ++ [
103       "GIT_PROXY_COMMAND"
104       "SOCKS_SERVER"
105       "GOPROXY"
106     ];
108     configurePhase = args.modConfigurePhase or ''
109       runHook preConfigure
110       export GOCACHE=$TMPDIR/go-cache
111       export GOPATH="$TMPDIR/go"
112       cd "$modRoot"
113       runHook postConfigure
114     '';
116     buildPhase = args.modBuildPhase or (''
117       runHook preBuild
118     '' + lib.optionalString finalAttrs.deleteVendor ''
119       if [ ! -d vendor ]; then
120         echo "vendor folder does not exist, 'deleteVendor' is not needed"
121         exit 10
122       else
123         rm -rf vendor
124       fi
125     '' + ''
126       if [ -d vendor ]; then
127         echo "vendor folder exists, please set 'vendorHash = null;' in your expression"
128         exit 10
129       fi
131       export GIT_SSL_CAINFO=$NIX_SSL_CERT_FILE
132       ${if finalAttrs.proxyVendor then ''
133         mkdir -p "''${GOPATH}/pkg/mod/cache/download"
134         go mod download
135       '' else ''
136         if (( "''${NIX_DEBUG:-0}" >= 1 )); then
137           goModVendorFlags+=(-v)
138         fi
139         go mod vendor "''${goModVendorFlags[@]}"
140       ''}
142       mkdir -p vendor
144       runHook postBuild
145     '');
147     installPhase = args.modInstallPhase or ''
148       runHook preInstall
150       ${if finalAttrs.proxyVendor then ''
151         rm -rf "''${GOPATH}/pkg/mod/cache/download/sumdb"
152         cp -r --reflink=auto "''${GOPATH}/pkg/mod/cache/download" $out
153       '' else ''
154         cp -r --reflink=auto vendor $out
155       ''}
157       if ! [ "$(ls -A $out)" ]; then
158         echo "vendor folder is empty, please set 'vendorHash = null;' in your expression"
159         exit 10
160       fi
162       runHook postInstall
163     '';
165     dontFixup = true;
167     outputHashMode = "recursive";
168     outputHash = finalAttrs.vendorHash;
169     # Handle empty `vendorHash`; avoid error:
170     # empty hash requires explicit hash algorithm.
171     outputHashAlgo = if finalAttrs.vendorHash == "" then "sha256" else null;
172     # in case an overlay clears passthru by accident, don't fail evaluation
173   }).overrideAttrs (finalAttrs.passthru.overrideModAttrs or overrideModAttrs);
175     nativeBuildInputs = [ go ] ++ nativeBuildInputs;
177     inherit (go) GOOS GOARCH;
179     GOFLAGS = GOFLAGS
180       ++ lib.warnIf (lib.any (lib.hasPrefix "-mod=") GOFLAGS) "use `proxyVendor` to control Go module/vendor behavior instead of setting `-mod=` in GOFLAGS"
181         (lib.optional (!finalAttrs.proxyVendor) "-mod=vendor")
182       ++ lib.warnIf (builtins.elem "-trimpath" GOFLAGS) "`-trimpath` is added by default to GOFLAGS by buildGoModule when allowGoReference isn't set to true"
183         (lib.optional (!allowGoReference) "-trimpath");
184     inherit CGO_ENABLED enableParallelBuilding GO111MODULE GOTOOLCHAIN;
186     # If not set to an explicit value, set the buildid empty for reproducibility.
187     ldflags = ldflags ++ lib.optional (!lib.any (lib.hasPrefix "-buildid=") ldflags) "-buildid=";
189     configurePhase = args.configurePhase or (''
190       runHook preConfigure
192       export GOCACHE=$TMPDIR/go-cache
193       export GOPATH="$TMPDIR/go"
194       export GOPROXY=off
195       export GOSUMDB=off
196       cd "$modRoot"
197     '' + lib.optionalString (finalAttrs.vendorHash != null) ''
198       ${if finalAttrs.proxyVendor then ''
199         export GOPROXY="file://$goModules"
200       '' else ''
201         rm -rf vendor
202         cp -r --reflink=auto "$goModules" vendor
203       ''}
204     '' + ''
206       # currently pie is only enabled by default in pkgsMusl
207       # this will respect the `hardening{Disable,Enable}` flags if set
208       if [[ $NIX_HARDENING_ENABLE =~ "pie" ]]; then
209         export GOFLAGS="-buildmode=pie $GOFLAGS"
210       fi
212       runHook postConfigure
213     '');
215     buildPhase = args.buildPhase or (
216       lib.warnIf (buildFlags != "" || buildFlagsArray != "")
217         "`buildFlags`/`buildFlagsArray` are deprecated and will be removed in the 24.11 release. Use the `ldflags` and/or `tags` attributes instead of `buildFlags`/`buildFlagsArray`"
218       lib.warnIf (builtins.elem "-buildid=" ldflags)
219         "`-buildid=` is set by default as ldflag by buildGoModule"
220     ''
221       runHook preBuild
223       exclude='\(/_\|examples\|Godeps\|testdata'
224       if [[ -n "$excludedPackages" ]]; then
225         IFS=' ' read -r -a excludedArr <<<$excludedPackages
226         printf -v excludedAlternates '%s\\|' "''${excludedArr[@]}"
227         excludedAlternates=''${excludedAlternates%\\|} # drop final \| added by printf
228         exclude+='\|'"$excludedAlternates"
229       fi
230       exclude+='\)'
232       buildGoDir() {
233         local cmd="$1" dir="$2"
235         declare -ga buildFlagsArray
236         declare -a flags
237         flags+=($buildFlags "''${buildFlagsArray[@]}")
238         flags+=(''${tags:+-tags=''${tags// /,}})
239         flags+=(''${ldflags:+-ldflags="$ldflags"})
240         flags+=("-p" "$NIX_BUILD_CORES")
242         if [ "$cmd" = "test" ]; then
243           flags+=(-vet=off)
244           flags+=($checkFlags)
245         fi
247         local OUT
248         if ! OUT="$(go $cmd "''${flags[@]}" $dir 2>&1)"; then
249           if ! echo "$OUT" | grep -qE '(no( buildable| non-test)?|build constraints exclude all) Go (source )?files'; then
250             echo "$OUT" >&2
251             return 1
252           fi
253         fi
254         if [ -n "$OUT" ]; then
255           echo "$OUT" >&2
256         fi
257         return 0
258       }
260       getGoDirs() {
261         local type;
262         type="$1"
263         if [ -n "$subPackages" ]; then
264           echo "$subPackages" | sed "s,\(^\| \),\1./,g"
265         else
266           find . -type f -name \*$type.go -exec dirname {} \; | grep -v "/vendor/" | sort --unique | grep -v "$exclude"
267         fi
268       }
270       if (( "''${NIX_DEBUG:-0}" >= 1 )); then
271         buildFlagsArray+=(-x)
272       fi
274       if [ -z "$enableParallelBuilding" ]; then
275           export NIX_BUILD_CORES=1
276       fi
277       for pkg in $(getGoDirs ""); do
278         echo "Building subPackage $pkg"
279         buildGoDir install "$pkg"
280       done
281     '' + lib.optionalString (stdenv.hostPlatform != stdenv.buildPlatform) ''
282       # normalize cross-compiled builds w.r.t. native builds
283       (
284         dir=$GOPATH/bin/${go.GOOS}_${go.GOARCH}
285         if [[ -n "$(shopt -s nullglob; echo $dir/*)" ]]; then
286           mv $dir/* $dir/..
287         fi
288         if [[ -d $dir ]]; then
289           rmdir $dir
290         fi
291       )
292     '' + ''
293       runHook postBuild
294     '');
296     doCheck = args.doCheck or true;
297     checkPhase = args.checkPhase or ''
298       runHook preCheck
299       # We do not set trimpath for tests, in case they reference test assets
300       export GOFLAGS=''${GOFLAGS//-trimpath/}
302       for pkg in $(getGoDirs test); do
303         buildGoDir test "$pkg"
304       done
306       runHook postCheck
307     '';
309     installPhase = args.installPhase or ''
310       runHook preInstall
312       mkdir -p $out
313       dir="$GOPATH/bin"
314       [ -e "$dir" ] && cp -r $dir $out
316       runHook postInstall
317     '';
319     strictDeps = true;
321     disallowedReferences = lib.optional (!allowGoReference) go;
323     passthru = {
324       inherit go;
325       # Canonicallize `overrideModAttrs` as an attribute overlay.
326       # `passthru.overrideModAttrs` will be overridden
327       # when users want to override `goModules`.
328       overrideModAttrs = lib.toExtension overrideModAttrs;
329     } // passthru;
331     meta = {
332       # Add default meta information.
333       platforms = go.meta.platforms or lib.platforms.all;
334     } // meta;
335   }