blob: 619953f00ba16682c5ae28606caa1503a1697536 [file] [log] [blame]
# Copyright 2015-2018 The ChromiumOS Authors
# Distributed under the terms of the GNU General Public License v2
# @ECLASS: cros-go.eclass
# @MAINTAINER:
# The ChromiumOS Authors <chromium-os-dev@chromium.org>
# @BUGREPORTS:
# Please report bugs via https://crbug.com/new (with component "Tools>ChromeOS-Toolchain")
# @VCSURL: https://chromium.googlesource.com/chromiumos/overlays/chromiumos-overlay/+/HEAD/eclass/@ECLASS@
# @BLURB: Eclass for fetching, building, and installing Go packages.
# @DESCRIPTION:
# See http://www.chromium.org/chromium-os/developer-guide/go-in-chromium-os for details.
case ${EAPI} in
[0-6]) die "${ECLASS}: EAPI ${EAPI} not supported" ;;
*) ;;
esac
# @ECLASS-VARIABLE: CROS_GO_SOURCE
# @PRE_INHERIT
# @DESCRIPTION:
# Path to the upstream repository and commit id.
# Go repositories on "github.com" and "*.googlesource.com" are supported.
# The source string contains the path of the git repository containing Go
# packages, and a commit-id (or version tag).
# For example:
# CROS_GO_SOURCE="github.com/golang/glog 44145f04b68cf362d9c4df2182967c2275eaefed"
# will fetch the sources from https://github.com/golang/glog at the
# specified commit-id, and
# CROS_GO_SOURCE="github.com/pkg/errors v0.8.0"
# will fetch the sources from https://github.com/pkg/errors at version
# v0.8.0.
# By default, the import path for Go packages in the repository is the
# same as repository path. This can be overridden by appending a colon
# to the repository path, followed by an alternate import path.
# For example:
# CROS_GO_SOURCE="github.com/go-yaml/yaml:gopkg.in/yaml.v2 v2.2.1"
# will fetch the sources from https://github.com/go-yaml/yaml at version
# v2.2.1, and install the package under "gopkg.in/yaml.v2".
# CROS_GO_SOURCE can contain multiple items when defined as an array:
# CROS_GO_SOURCE=(
# "github.com/golang/glog 44145f04b68cf362d9c4df2182967c2275eaefed"
# "github.com/pkg/errors v0.8.0"
# "github.com/go-yaml/yaml:gopkg.in/yaml.v2 v2.2.1"
# )
# @ECLASS-VARIABLE: CROS_GO_WORKSPACE
# @DESCRIPTION:
# Path to the Go workspace, default is ${S}.
# The Go workspace is searched for packages to build and install.
# If all Go packages in a repository are located under "go/src/":
# CROS_GO_WORKSPACE="${S}/go"
# CROS_GO_WORKSPACE can contain multiple items when defined as an array:
# CROS_GO_WORKSPACE=(
# "${S}"
# "${S}/tast-base"
# )
# @ECLASS-VARIABLE: CROS_GO_BINARIES
# @DESCRIPTION:
# Go programs to build and install.
# Each program is specified by the path of a directory that
# contains a package "main", or a single Go source file which
# must also contain the package "main". For directories, the
# last component of the package path becomes the name of the
# executable. For files, the ".go" suffix is also stripped.
# The executable name can be overridden by appending a colon
# to the package path, followed by an alternate name.
# The install path for an executable can be overridden by
# appending a colon to the package path, followed by the
# desired install path/name for it.
# For example:
# CROS_GO_BINARIES=(
# "golang.org/x/tools/cmd/godoc"
# "golang.org/x/tools/cmd/guru:goguru"
# "golang.org/x/tools/cmd/stringer:/usr/local/bin/gostringer"
# "golang.org/x/tools/cmd/foo.go"
# "golang.org/x/tools/cmd/foo.go:gofoo"
# )
# builds and installs "godoc", "goguru", "gostringer", "foo"
# and "gofoo" binaries.
# Helper function for path and names used in cros-go_src_{compile,install}.
# Returns various views of the information in a CROS_GO_BINARIES entry (a
# "specification" or "spec") separated by colons.
parse_binspec() {
local spec="$1"
# Split spec at colon into source and override.
local source
local override
IFS=: read source override empty <<<"${spec}"
test -z "${empty}" || die "bad CROS_GO_BINARIES entry: \"${spec}\""
local target
local installdir
if [[ -z "${override}" ]] ; then
target="${spec##*/}"
# if there is no override, remove .go suffix (if any).
target="${target%%.go}"
installdir="/usr/bin"
else
# target is the last component of the override path
target="${override##*/}"
installdir="${override%/*}"
fi
# if the source is a single go file, use its full path name
if [[ "${source##*.}" == "go" ]]; then
source="${S}/src/${source}"
fi
# there is no colon in any variable, so colon is a safe separator
echo "${source}:${target}:${installdir}"
}
# @ECLASS-VARIABLE: CROS_GO_VERSION
# @DESCRIPTION:
# Version string to embed in the executable binary.
# The variable main.Version is set to this value at build time.
# For example:
# CROS_GO_VERSION="${PVR}"
# will set main.Version string variable to package version and
# revision (if any) of the ebuild.
# @ECLASS-VARIABLE: CROS_GO_SKIP_DEP_CHECK
# @DESCRIPTION:
# Temporary workaround to allow circular dependencies like:
# google.golang.org/genproto/googleapis/chromeos/uidetection/v1 requires
# google.golang.org/grpc which requires other packages in
# google.golang.org/genproto
# In the past these have been fixed in GOPATH mode by splitting packages
# in various ebuilds like dev-go/genproto vs dev-go/genproto-rpc (etc.)
# and building them sequentially while ignoring upstream module definitions.
# When switching to Go modules, we need to respect upstream module boundaries
# and let the Go modules dependency system properly handle circular deps.
# So in order to merge for eg dev-go/genproto with dev-go/genproto-rpc, we
# need the temporary ability to relax the dep checking on circular deps.
# This variable addition and use is temporary for the GOPATH -> Go modules
# transition and will go away when switching to modules mode.
# For example:
# CROS_GO_SKIP_DEP_CHECK="1"
# @ECLASS-VARIABLE: CROS_GO_PACKAGES
# @DESCRIPTION:
# Go packages to install in /usr/lib/gopath.
# Packages are installed in /usr/lib/gopath such that they
# can be imported later from Go code using the exact paths
# listed here. For example:
# CROS_GO_PACKAGES=(
# "github.com/golang/glog"
# )
# will install package files to
# "/usr/lib/gopath/src/github.com/golang/glog"
# and other Go projects can use the package with
# import "github.com/golang/glog"
# If the last component of a package path is "...", it is
# expanded to include all Go packages under the directory.
# @ECLASS-VARIABLE: CROS_GO_TEST
# @DESCRIPTION:
# Go packages to test.
# Package tests are run with "-short" flag by default.
# Package tests are always built and run locally on host.
# Default is to test all packages in CROS_GO_WORKSPACE(s).
: ${CROS_GO_TEST:=./...}
# @ECLASS-VARIABLE: CROS_GO_VET
# @DESCRIPTION:
# Go packages to check using "go vet".
# As in CROS_GO_PACKAGES, "..." is expanded.
# @ECLASS-VARIABLE: CROS_GO_VET_FLAGS
# @DESCRIPTION:
# Flags to pass to "go vet" if CROS_GO_VET is set.
# See https://golang.org/cmd/vet/ for available flags.
# @FUNCTION: cros-go_out_dir
# @RETURN: an output directory for compiled CROS_GO_BINARIES
cros-go_out_dir() {
cros_go_out="${T}/go_output"
mkdir -p "${cros_go_out}"
echo "${cros_go_out}"
}
inherit toolchain-funcs
BDEPEND="dev-lang/go"
# @FUNCTION: cros-go_get
# @USAGE: <source> [variables to extract]
# @INTERNAL
# @DESCRIPTION:
# Parse source string and extract different components.
# This function parses the string containing upstream
# repository, import path, and commit id information
# (see description of CROS_GO_SOURCE format above).
# It can also be used to construct the name of the
# distfile and a uri for fetching it.
cros-go_get() {
local src commit repopath importpath distfile uri
src="$1"
commit="${src##* }"
repopath="${src%% *}"
importpath="${repopath#*:}"
repopath="${repopath%:*}"
distfile="${repopath//\//-}-${commit}.tar.gz"
uri="https://${repopath}/${commit}.tar.gz"
case "${repopath%%/*}" in
github.com)
uri="https://${repopath}/archive/${commit}.tar.gz" ;;
*.googlesource.com)
uri="https://${repopath}/+archive/${commit}.tar.gz" ;;
*)
die "Unsupported upstream source in ${repopath}" ;;
esac
shift
local arg
for arg in "$@" ; do
case "${arg}" in
commit) printf "%s" "${commit}" ;;
repopath) printf "%s" "${repopath}" ;;
importpath) printf "%s" "${importpath}" ;;
distfile) printf "%s" "${distfile}" ;;
uri) printf "%s" "${uri}" ;;
*) printf "${arg}" ;;
esac
done
}
# @FUNCTION: cros-go_src_uri
# @RETURN: a valid SRC_URI for CROS_GO_SOURCE
# @DESCRIPTION:
# Set the SRC_URI in an ebuild with:
# SRC_URI="$(cros-go_src_uri)"
cros-go_src_uri() {
local src
for src in "${CROS_GO_SOURCE[@]}" ; do
cros-go_get "${src}" uri " -> " distfile "\n"
done
}
# @FUNCTION: cros-go_pkg_nofetch
# @DESCRIPTION:
# Print useful information on how to download a source tarball and
# add it to chromeos-localmirror.
cros-go_pkg_nofetch() {
local src
for src in "${CROS_GO_SOURCE[@]}" ; do
local uri="$(cros-go_get "${src}" uri)"
local distfile="$(cros-go_get "${src}" distfile)"
einfo "Run these commands to add ${distfile} to chromeos-localmirror:"
einfo " wget -O ${distfile} ${uri}"
einfo " gsutil cp -a public-read ${distfile} gs://chromeos-localmirror/distfiles/"
einfo
done
einfo "After all distfiles have been mirrored, update the 'Manifest' file with:"
einfo " ebuild ${EBUILD} manifest"
}
# @FUNCTION: cros-go_src_unpack
# @DESCRIPTION:
# Unpack the source tarball under appropriate location based on
# the desired import path.
cros-go_src_unpack() {
local src
for src in "${CROS_GO_SOURCE[@]}" ; do
local commit="$(cros-go_get "${src}" commit)"
local repopath="$(cros-go_get "${src}" repopath)"
local importpath="$(cros-go_get "${src}" importpath)"
local distfile="$(cros-go_get "${src}" distfile)"
local destdir="${S}/src/${importpath}"
case "${repopath%%/*}" in
github.com)
# Unpacking tarballs from github creates a top level
# directory "projectname-version", so extra logic is
# required to make the contents appear correctly in
# the desired destination directory.
mkdir -p "${destdir%/*}" || die
pushd "${destdir%/*}" >/dev/null || die
unpack "${distfile}"
mv "${repopath##*/}-${commit#v}" "${importpath##*/}" || die
popd >/dev/null || die
;;
*.googlesource.com)
mkdir -p "${destdir}" || die
pushd "${destdir}" >/dev/null || die
unpack "${distfile}"
popd >/dev/null || die
;;
esac
done
}
# @FUNCTION: cros-go_workspace
# @RETURN: Go workspaces, colon separated
# @INTERNAL
# @DESCRIPTION:
# Return the list of workspaces in CROS_GO_WORKSPACE,
# properly formatted for inclusion into GOPATH.
cros-go_workspace() {
if [[ ${#CROS_GO_WORKSPACE[@]} != 0 ]] ; then
( IFS=:; echo "${CROS_GO_WORKSPACE[*]}" )
else
echo "${S}"
fi
}
# @FUNCTION: cros-go_gopath
# @RETURN: a valid GOPATH for CROS_GO_WORKSPACE
# @DESCRIPTION:
# Set the GOPATH in an ebuild with:
# GOPATH="$(cros-go_gopath)"
cros-go_gopath() {
echo "$(cros-go_workspace):${SYSROOT}/usr/lib/gopath"
}
# @FUNCTION: cros_go
# @DESCRIPTION:
# Wrapper function for invoking the Go tool from an ebuild.
# Sets up GOPATH, and uses the appropriate cross-compiler.
cros_go() {
GO111MODULE=${GO111MODULE:-off} GOPATH="$(cros-go_gopath)" $(tc-getGO) "$@" || die
}
# @FUNCTION: go_list
# @DESCRIPTION:
# List all Go packages matching a pattern.
# Only list packages in the current workspace.
go_list() {
GO111MODULE=${GO111MODULE:-off} GOPATH="$(cros-go_gopath)" $(tc-getGO) list "$@" || die
}
# @FUNCTION: go_test
# @DESCRIPTION:
# Wrapper function for building and running unit tests.
# Package tests are always built and run locally on host.
go_test() {
GO111MODULE=${GO111MODULE:-off} GOPATH="$(cros-go_gopath)" $(tc-getBUILD_GO) test "$@" || die
}
# @FUNCTION: go_vet
# @DESCRIPTION:
# Wrapper function for running "go vet".
go_vet() {
# shellcheck disable=SC2154
GO111MODULE="${GO111MODULE:-off}" GOPATH="$(cros-go_gopath)" $(tc-getBUILD_GO) vet \
"${CROS_GO_VET_FLAGS[@]}" "$@" || die
}
# @FUNCTION: go_lint
# @DESCRIPTION:
# Wrapper function for running "golint"
go_lint() {
# shellcheck disable=SC2154
local go_lint_output_base="${CROS_ARTIFACTS_TMP_DIR}/linting_output/go_lint"
mkdir -p "${go_lint_output_base}"
local file_name="${1//'/...'/}"
file_name="${file_name//'/'/-}-$(date +%s)"
GO111MODULE="${GO111MODULE:-off}" GOPATH="$(cros-go_gopath)" golint \
"$@" >> "${go_lint_output_base}/${file_name}.txt" || die
}
# @FUNCTION: cros-go_src_compile
# @DESCRIPTION:
# Build CROS_GO_BINARIES.
cros-go_src_compile() {
out_dir=$(cros-go_out_dir)
local bin
local source
local target
local installdir
for bin in "${CROS_GO_BINARIES[@]}" ; do
einfo "Building \"${bin}\""
IFS=: read source target installdir <<<"$(parse_binspec "${bin}")"
cros_go build -v \
${CROS_GO_VERSION:+"-ldflags=-X main.Version=${CROS_GO_VERSION}"} \
-o "${out_dir}/${target}" \
"${source}"
done
local pkg
for pkg in "${CROS_GO_VET[@]}" ; do
einfo "Vetting \"${pkg}\""
go_vet "${pkg}"
# Enable the option to output to a file so that the chromite build API
# can access Go lints.
if [[ -n "${ENABLE_GO_LINT}" ]]; then
go_lint "${pkg}"
fi
done
}
# @FUNCTION: cros-go_src_test
# @DESCRIPTION:
# Run tests for packages listed in CROS_GO_TEST.
cros-go_src_test() {
local pkglist=( $(go_list "${CROS_GO_TEST[@]}") )
go_test -short "${pkglist[@]}"
}
# @FUNCTION: cros-go_src_install
# @DESCRIPTION:
# Install CROS_GO_BINARIES and CROS_GO_PACKAGES.
cros-go_src_install() {
out_dir=$(cros-go_out_dir)
# Install the compiled binaries.
local bin
local source
local target
local installdir
for bin in "${CROS_GO_BINARIES[@]}" ; do
einfo "Installing \"${bin}\""
IFS=: read source target installdir <<<"$(parse_binspec "${bin}")"
(
# Run in sub-shell so we do not modify env.
exeinto "${installdir}"
doexe "${out_dir}/${target}"
)
done
# `go_list` will try to find the dependencies of the
# packages and fails as they are not available in
# ${GOROOT} and ${GOPATH}. Using `cros_go list` adds
# `/usr/lib/gopath` in the GOPATH env and therefore,
# `go list` can work.
# Install the importable packages in /usr/lib/gopath.
local pkglist=()
if [[ ${#CROS_GO_PACKAGES[@]} != 0 ]] ; then
pkglist=( $(go_list "${CROS_GO_PACKAGES[@]}") )
fi
local pkg
for pkg in "${pkglist[@]}" ; do
einfo "Installing \"${pkg}\""
local pkgdir="$(go_list -f "{{.Dir}}" "${pkg}")"
(
# Run in sub-shell so we do not modify env.
insinto "/usr/lib/gopath/src/${pkg}"
local file
while read -d $'\0' -r file ; do
doins "${file}"
done < <(find "${pkgdir}" -maxdepth 1 ! -type d -print0)
)
done
}
# @FUNCTION: cros-go_pkg_postinst
# @DESCRIPTION:
# Check for missing dependencies of installed packages.
cros-go_pkg_postinst() {
# This only works if we're building and installing from source.
[[ "${MERGE_TYPE}" == "source" ]] || return
# See CROS_GO_SKIP_DEP_CHECK description for details
[[ -n "${CROS_GO_SKIP_DEP_CHECK}" ]] && return
# `go_list` will try to find the dependencies of the
# packages and fails as they are not available in
# ${GOROOT} and ${GOPATH}. Using `cros_go list` adds
# `/usr/lib/gopath` in the GOPATH env and therefore,
# `go list` can work.
# Get the list of packages from the workspace in ${S}.
local pkglist=()
if [[ ${#CROS_GO_PACKAGES[@]} != 0 ]] ; then
pkglist=( $(go_list "${CROS_GO_PACKAGES[@]}") )
fi
# Switch the workspace to where the packages were installed.
local CROS_GO_WORKSPACE="${SYSROOT}/usr/lib/gopath"
# For each installed package, check for missing dependencies.
local pkg
for pkg in "${pkglist[@]}" ; do
if [[ $(go_list -f "{{.Incomplete}}" "${pkg}") == "true" ]] ; then
go_list -f "{{.DepsErrors}}" "${pkg}"
die "Package has missing dependency: \"${pkg}\""
fi
done
}
if [[ ${#CROS_GO_SOURCE[@]} != 0 ]] ; then
EXPORT_FUNCTIONS pkg_nofetch src_unpack
fi
EXPORT_FUNCTIONS src_compile src_test src_install pkg_postinst