2019-11-23 16:25:11 +01:00
|
|
|
# Copyright 2019 Google LLC.
|
|
|
|
# SPDX-License-Identifier: Apache-2.0
|
|
|
|
#
|
|
|
|
# buildGo provides Nix functions to build Go packages in the style of Bazel's
|
|
|
|
# rules_go.
|
|
|
|
|
|
|
|
{ pkgs ? import <nixpkgs> {}
|
|
|
|
, ... }:
|
|
|
|
|
|
|
|
let
|
|
|
|
inherit (builtins)
|
|
|
|
attrNames
|
|
|
|
baseNameOf
|
2019-11-25 17:42:49 +01:00
|
|
|
dirOf
|
2019-11-23 16:25:11 +01:00
|
|
|
elemAt
|
|
|
|
filter
|
|
|
|
map
|
|
|
|
match
|
2019-11-24 21:17:17 +01:00
|
|
|
readDir
|
2019-11-25 17:42:49 +01:00
|
|
|
replaceStrings;
|
2019-11-23 16:25:11 +01:00
|
|
|
|
2019-11-25 17:42:49 +01:00
|
|
|
inherit (pkgs) lib go runCommand fetchFromGitHub protobuf symlinkJoin;
|
2019-11-23 16:25:11 +01:00
|
|
|
|
|
|
|
# Helpers for low-level Go compiler invocations
|
|
|
|
spaceOut = lib.concatStringsSep " ";
|
|
|
|
|
|
|
|
includeDepSrc = dep: "-I ${dep}";
|
|
|
|
includeSources = deps: spaceOut (map includeDepSrc deps);
|
|
|
|
|
|
|
|
includeDepLib = dep: "-L ${dep}";
|
|
|
|
includeLibs = deps: spaceOut (map includeDepLib deps);
|
|
|
|
|
|
|
|
srcBasename = src: elemAt (match "([a-z0-9]{32}\-)?(.*\.go)" (baseNameOf src)) 1;
|
|
|
|
srcCopy = path: src: "cp ${src} $out/${path}/${srcBasename src}";
|
|
|
|
srcList = path: srcs: lib.concatStringsSep "\n" (map (srcCopy path) srcs);
|
|
|
|
|
|
|
|
isGoFile = f: (match ".*\.go" f) != null;
|
|
|
|
isGoTest = f: (match ".*_test\.go" f) != null;
|
|
|
|
goFileFilter = k: v: (v == "regular") && (isGoFile k) && (!isGoTest k);
|
|
|
|
goFilesIn = dir:
|
|
|
|
let files = readDir dir;
|
|
|
|
goFiles = filter (f: goFileFilter f files."${f}") (attrNames files);
|
|
|
|
in map (f: dir + "/" + f) goFiles;
|
|
|
|
|
|
|
|
allDeps = deps: lib.unique (lib.flatten (deps ++ (map (d: d.goDeps) deps)));
|
|
|
|
|
2019-11-24 21:41:01 +01:00
|
|
|
xFlags = x_defs: spaceOut (map (k: "-X ${k}=${x_defs."${k}"}") (attrNames x_defs));
|
|
|
|
|
2019-11-25 17:42:49 +01:00
|
|
|
pathToName = p: replaceStrings ["/"] ["_"] (toString p);
|
|
|
|
|
2019-11-23 16:25:11 +01:00
|
|
|
# High-level build functions
|
|
|
|
|
|
|
|
# Build a Go program out of the specified files and dependencies.
|
2019-11-24 21:41:01 +01:00
|
|
|
program = { name, srcs, deps ? [], x_defs ? {} }:
|
2019-11-23 16:25:11 +01:00
|
|
|
let uniqueDeps = allDeps deps;
|
|
|
|
in runCommand name {} ''
|
|
|
|
${go}/bin/go tool compile -o ${name}.a -trimpath=$PWD -trimpath=${go} ${includeSources uniqueDeps} ${spaceOut srcs}
|
|
|
|
mkdir -p $out/bin
|
2019-11-24 21:41:01 +01:00
|
|
|
${go}/bin/go tool link -o $out/bin/${name} -buildid nix ${xFlags x_defs} ${includeLibs uniqueDeps} ${name}.a
|
2019-11-23 16:25:11 +01:00
|
|
|
'';
|
|
|
|
|
|
|
|
# Build a Go library assembled out of the specified files.
|
|
|
|
#
|
|
|
|
# This outputs both the sources and compiled binary, as both are
|
|
|
|
# needed when downstream packages depend on it.
|
|
|
|
package = { name, srcs, deps ? [], path ? name }:
|
|
|
|
let uniqueDeps = allDeps deps;
|
|
|
|
in (runCommand "golib-${name}" {} ''
|
|
|
|
mkdir -p $out/${path}
|
|
|
|
${srcList path (map (s: "${s}") srcs)}
|
|
|
|
${go}/bin/go tool compile -o $out/${path}.a -trimpath=$PWD -trimpath=${go} -p ${path} ${includeSources uniqueDeps} ${spaceOut srcs}
|
|
|
|
'') // { goDeps = uniqueDeps; };
|
2019-11-24 21:17:17 +01:00
|
|
|
|
|
|
|
# Build a Go library out of the specified protobuf definition.
|
|
|
|
proto = { name, proto, path ? name, protocFlags ? "", extraDeps ? [] }: package {
|
|
|
|
inherit name path;
|
|
|
|
deps = [ goProto ] ++ extraDeps;
|
|
|
|
srcs = lib.singleton (runCommand "goproto-${name}.pb.go" {} ''
|
|
|
|
cp ${proto} ${baseNameOf proto}
|
2019-11-25 17:42:49 +01:00
|
|
|
${protobuf}/bin/protoc --plugin=${goProto}/bin/protoc-gen-go \
|
2019-11-24 21:17:17 +01:00
|
|
|
--go_out=${protocFlags}import_path=${baseNameOf path}:. ${baseNameOf proto}
|
|
|
|
mv *.pb.go $out
|
|
|
|
'');
|
|
|
|
};
|
|
|
|
|
2019-11-25 17:42:49 +01:00
|
|
|
# Build an externally defined Go library using `go build` itself.
|
|
|
|
#
|
|
|
|
# Libraries built this way can be included in any standard buildGo
|
|
|
|
# build.
|
|
|
|
#
|
|
|
|
# Contrary to other functions, `src` is expected to point at a
|
|
|
|
# single directory containing the root of the external library.
|
2019-11-26 13:11:43 +01:00
|
|
|
external = { path, src, deps ? [], srcOnly ? false, targets ? [ "..." ] }:
|
2019-11-25 17:42:49 +01:00
|
|
|
let
|
|
|
|
name = pathToName path;
|
|
|
|
uniqueDeps = allDeps deps;
|
|
|
|
srcDir = runCommand "goext-src-${name}" {} ''
|
|
|
|
mkdir -p $out/${dirOf path}
|
|
|
|
cp -r ${src} $out/${dirOf path}/${baseNameOf path}
|
|
|
|
'';
|
|
|
|
gopathSrc = symlinkJoin {
|
|
|
|
name = "gopath-${name}";
|
|
|
|
paths = uniqueDeps ++ [ srcDir ];
|
|
|
|
};
|
|
|
|
gopathPkg = runCommand "goext-pkg-${name}" {} ''
|
|
|
|
mkdir -p gopath $out
|
|
|
|
export GOPATH=$PWD/gopath
|
|
|
|
ln -s ${gopathSrc} gopath/src
|
2019-11-26 13:11:43 +01:00
|
|
|
${go}/bin/go install ${spaceOut (map (t: path + "/" + t) targets)}
|
2019-11-25 17:42:49 +01:00
|
|
|
|
|
|
|
if [[ -d gopath/pkg/linux_amd64 ]]; then
|
|
|
|
echo "Installing Go packages for ${path}"
|
|
|
|
mv gopath/pkg/linux_amd64/* $out
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [[ -d gopath/bin ]]; then
|
|
|
|
echo "Installing Go binaries for ${path}"
|
|
|
|
mv gopath/bin $out/bin
|
|
|
|
fi
|
|
|
|
'';
|
2019-11-26 13:11:43 +01:00
|
|
|
in (if srcOnly then gopathSrc else symlinkJoin {
|
2019-11-25 17:42:49 +01:00
|
|
|
name = "goext-${name}";
|
|
|
|
paths = [ gopathSrc gopathPkg ];
|
2019-11-26 13:11:43 +01:00
|
|
|
}) // { goDeps = uniqueDeps; };
|
2019-11-25 17:42:49 +01:00
|
|
|
|
2019-11-26 13:15:18 +01:00
|
|
|
protoLibs = import ./proto.nix {
|
|
|
|
inherit external;
|
2019-11-24 21:17:17 +01:00
|
|
|
};
|
2019-11-23 16:25:11 +01:00
|
|
|
in {
|
|
|
|
# Only the high-level builder functions are exposed
|
2019-11-25 17:42:49 +01:00
|
|
|
inherit program package proto external;
|
2019-11-23 16:25:11 +01:00
|
|
|
}
|