зеркало из https://github.com/github/codeql.git
Merge pull request #12957 from owen-mc/go/autobuilder-identify-environment
Go: Add `go-autobuilder --identify-environment`
This commit is contained in:
Коммит
4de4f35855
|
@ -3,7 +3,6 @@ package main
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
|
||||||
"log"
|
"log"
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
|
@ -22,28 +21,37 @@ import (
|
||||||
|
|
||||||
func usage() {
|
func usage() {
|
||||||
fmt.Fprintf(os.Stderr,
|
fmt.Fprintf(os.Stderr,
|
||||||
`%s is a wrapper script that installs dependencies and calls the extractor.
|
`%s is a wrapper script that installs dependencies and calls the extractor
|
||||||
|
|
||||||
When LGTM_SRC is not set, the script installs dependencies as described below, and then invokes the
|
Options:
|
||||||
extractor in the working directory.
|
--identify-environment
|
||||||
|
Produce an environment file specifying which Go version should be installed in the environment
|
||||||
|
so that autobuilding will be successful. The location of this file is controlled by the
|
||||||
|
environment variable CODEQL_EXTRACTOR_ENVIRONMENT_JSON, or defaults to 'environment.json' if
|
||||||
|
that is not set.
|
||||||
|
|
||||||
If LGTM_SRC is set, it checks for the presence of the files 'go.mod', 'Gopkg.toml', and
|
Build behavior:
|
||||||
'glide.yaml' to determine how to install dependencies: if a 'Gopkg.toml' file is present, it uses
|
|
||||||
'dep ensure', if there is a 'glide.yaml' it uses 'glide install', and otherwise 'go get'.
|
|
||||||
Additionally, unless a 'go.mod' file is detected, it sets up a temporary GOPATH and moves all
|
|
||||||
source files into a folder corresponding to the package's import path before installing
|
|
||||||
dependencies.
|
|
||||||
|
|
||||||
This behavior can be further customized using environment variables: setting LGTM_INDEX_NEED_GOPATH
|
When LGTM_SRC is not set, the script installs dependencies as described below, and then invokes the
|
||||||
to 'false' disables the GOPATH set-up, CODEQL_EXTRACTOR_GO_BUILD_COMMAND (or alternatively
|
extractor in the working directory.
|
||||||
LGTM_INDEX_BUILD_COMMAND), can be set to a newline-separated list of commands to run in order to
|
|
||||||
install dependencies, and LGTM_INDEX_IMPORT_PATH can be used to override the package import path,
|
|
||||||
which is otherwise inferred from the SEMMLE_REPO_URL or GITHUB_REPOSITORY environment variables.
|
|
||||||
|
|
||||||
In resource-constrained environments, the environment variable CODEQL_EXTRACTOR_GO_MAX_GOROUTINES
|
If LGTM_SRC is set, it checks for the presence of the files 'go.mod', 'Gopkg.toml', and
|
||||||
(or its legacy alias SEMMLE_MAX_GOROUTINES) can be used to limit the number of parallel goroutines
|
'glide.yaml' to determine how to install dependencies: if a 'Gopkg.toml' file is present, it uses
|
||||||
started by the extractor, which reduces CPU and memory requirements. The default value for this
|
'dep ensure', if there is a 'glide.yaml' it uses 'glide install', and otherwise 'go get'.
|
||||||
variable is 32.
|
Additionally, unless a 'go.mod' file is detected, it sets up a temporary GOPATH and moves all
|
||||||
|
source files into a folder corresponding to the package's import path before installing
|
||||||
|
dependencies.
|
||||||
|
|
||||||
|
This behavior can be further customized using environment variables: setting LGTM_INDEX_NEED_GOPATH
|
||||||
|
to 'false' disables the GOPATH set-up, CODEQL_EXTRACTOR_GO_BUILD_COMMAND (or alternatively
|
||||||
|
LGTM_INDEX_BUILD_COMMAND), can be set to a newline-separated list of commands to run in order to
|
||||||
|
install dependencies, and LGTM_INDEX_IMPORT_PATH can be used to override the package import path,
|
||||||
|
which is otherwise inferred from the SEMMLE_REPO_URL or GITHUB_REPOSITORY environment variables.
|
||||||
|
|
||||||
|
In resource-constrained environments, the environment variable CODEQL_EXTRACTOR_GO_MAX_GOROUTINES
|
||||||
|
(or its legacy alias SEMMLE_MAX_GOROUTINES) can be used to limit the number of parallel goroutines
|
||||||
|
started by the extractor, which reduces CPU and memory requirements. The default value for this
|
||||||
|
variable is 32.
|
||||||
`,
|
`,
|
||||||
os.Args[0])
|
os.Args[0])
|
||||||
fmt.Fprintf(os.Stderr, "Usage:\n\n %s\n", os.Args[0])
|
fmt.Fprintf(os.Stderr, "Usage:\n\n %s\n", os.Args[0])
|
||||||
|
@ -92,6 +100,7 @@ func tryBuild(buildFile, cmd string, args ...string) bool {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Returns the import path of the package being built, or "" if it cannot be determined.
|
||||||
func getImportPath() (importpath string) {
|
func getImportPath() (importpath string) {
|
||||||
importpath = os.Getenv("LGTM_INDEX_IMPORT_PATH")
|
importpath = os.Getenv("LGTM_INDEX_IMPORT_PATH")
|
||||||
if importpath == "" {
|
if importpath == "" {
|
||||||
|
@ -116,6 +125,8 @@ func getImportPath() (importpath string) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Returns the import path of the package being built from `repourl`, or "" if it cannot be
|
||||||
|
// determined.
|
||||||
func getImportPathFromRepoURL(repourl string) string {
|
func getImportPathFromRepoURL(repourl string) string {
|
||||||
// check for scp-like URL as in "git@github.com:github/codeql-go.git"
|
// check for scp-like URL as in "git@github.com:github/codeql-go.git"
|
||||||
shorturl := regexp.MustCompile("^([^@]+@)?([^:]+):([^/].*?)(\\.git)?$")
|
shorturl := regexp.MustCompile("^([^@]+@)?([^:]+):([^/].*?)(\\.git)?$")
|
||||||
|
@ -182,6 +193,8 @@ const (
|
||||||
ModVendor
|
ModVendor
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// argsForGoVersion returns the arguments to pass to the Go compiler for the given `ModMode` and
|
||||||
|
// Go version
|
||||||
func (m ModMode) argsForGoVersion(version string) []string {
|
func (m ModMode) argsForGoVersion(version string) []string {
|
||||||
switch m {
|
switch m {
|
||||||
case ModUnset:
|
case ModUnset:
|
||||||
|
@ -221,6 +234,7 @@ func checkVendor() bool {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Returns the directory containing the source code to be analyzed.
|
||||||
func getSourceDir() string {
|
func getSourceDir() string {
|
||||||
srcdir := os.Getenv("LGTM_SRC")
|
srcdir := os.Getenv("LGTM_SRC")
|
||||||
if srcdir != "" {
|
if srcdir != "" {
|
||||||
|
@ -236,6 +250,7 @@ func getSourceDir() string {
|
||||||
return srcdir
|
return srcdir
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Returns the appropriate DependencyInstallerMode for the current project
|
||||||
func getDepMode() DependencyInstallerMode {
|
func getDepMode() DependencyInstallerMode {
|
||||||
if util.FileExists("go.mod") {
|
if util.FileExists("go.mod") {
|
||||||
log.Println("Found go.mod, enabling go modules")
|
log.Println("Found go.mod, enabling go modules")
|
||||||
|
@ -252,6 +267,33 @@ func getDepMode() DependencyInstallerMode {
|
||||||
return GoGetNoModules
|
return GoGetNoModules
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Tries to open `go.mod` and read a go directive, returning the version and whether it was found.
|
||||||
|
func tryReadGoDirective(depMode DependencyInstallerMode) (string, bool) {
|
||||||
|
version := ""
|
||||||
|
found := false
|
||||||
|
if depMode == GoGetWithModules {
|
||||||
|
versionRe := regexp.MustCompile(`(?m)^go[ \t\r]+([0-9]+\.[0-9]+)$`)
|
||||||
|
goMod, err := os.ReadFile("go.mod")
|
||||||
|
if err != nil {
|
||||||
|
log.Println("Failed to read go.mod to check for missing Go version")
|
||||||
|
} else {
|
||||||
|
matches := versionRe.FindSubmatch(goMod)
|
||||||
|
if matches != nil {
|
||||||
|
found = true
|
||||||
|
if len(matches) > 1 {
|
||||||
|
version := string(matches[1])
|
||||||
|
semverVersion := "v" + version
|
||||||
|
if semver.Compare(semverVersion, getEnvGoSemVer()) >= 0 {
|
||||||
|
diagnostics.EmitNewerGoVersionNeeded()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return version, found
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns the appropriate ModMode for the current project
|
||||||
func getModMode(depMode DependencyInstallerMode) ModMode {
|
func getModMode(depMode DependencyInstallerMode) ModMode {
|
||||||
if depMode == GoGetWithModules {
|
if depMode == GoGetWithModules {
|
||||||
// if a vendor/modules.txt file exists, we assume that there are vendored Go dependencies, and
|
// if a vendor/modules.txt file exists, we assume that there are vendored Go dependencies, and
|
||||||
|
@ -265,7 +307,8 @@ func getModMode(depMode DependencyInstallerMode) ModMode {
|
||||||
return ModUnset
|
return ModUnset
|
||||||
}
|
}
|
||||||
|
|
||||||
func fixGoVendorIssues(modMode ModMode, depMode DependencyInstallerMode, goDirectiveFound bool) ModMode {
|
// fixGoVendorIssues fixes issues with go vendor for go version >= 1.14
|
||||||
|
func fixGoVendorIssues(modMode ModMode, depMode DependencyInstallerMode, goModVersionFound bool) ModMode {
|
||||||
if modMode == ModVendor {
|
if modMode == ModVendor {
|
||||||
// fix go vendor issues with go versions >= 1.14 when no go version is specified in the go.mod
|
// fix go vendor issues with go versions >= 1.14 when no go version is specified in the go.mod
|
||||||
// if this is the case, and dependencies were vendored with an old go version (and therefore
|
// if this is the case, and dependencies were vendored with an old go version (and therefore
|
||||||
|
@ -275,9 +318,9 @@ func fixGoVendorIssues(modMode ModMode, depMode DependencyInstallerMode, goDirec
|
||||||
// we work around this by adding an explicit go version of 1.13, which is the last version
|
// we work around this by adding an explicit go version of 1.13, which is the last version
|
||||||
// where this is not an issue
|
// where this is not an issue
|
||||||
if depMode == GoGetWithModules {
|
if depMode == GoGetWithModules {
|
||||||
if !goDirectiveFound {
|
if !goModVersionFound {
|
||||||
// if the go.mod does not contain a version line
|
// if the go.mod does not contain a version line
|
||||||
modulesTxt, err := ioutil.ReadFile("vendor/modules.txt")
|
modulesTxt, err := os.ReadFile("vendor/modules.txt")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println("Failed to read vendor/modules.txt to check for mismatched Go version")
|
log.Println("Failed to read vendor/modules.txt to check for mismatched Go version")
|
||||||
} else if explicitRe := regexp.MustCompile("(?m)^## explicit$"); !explicitRe.Match(modulesTxt) {
|
} else if explicitRe := regexp.MustCompile("(?m)^## explicit$"); !explicitRe.Match(modulesTxt) {
|
||||||
|
@ -294,6 +337,7 @@ func fixGoVendorIssues(modMode ModMode, depMode DependencyInstallerMode, goDirec
|
||||||
return modMode
|
return modMode
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Determines whether the project needs a GOPATH set up
|
||||||
func getNeedGopath(depMode DependencyInstallerMode, importpath string) bool {
|
func getNeedGopath(depMode DependencyInstallerMode, importpath string) bool {
|
||||||
needGopath := true
|
needGopath := true
|
||||||
if depMode == GoGetWithModules {
|
if depMode == GoGetWithModules {
|
||||||
|
@ -316,6 +360,7 @@ func getNeedGopath(depMode DependencyInstallerMode, importpath string) bool {
|
||||||
return needGopath
|
return needGopath
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Try to update `go.mod` and `go.sum` if the go version is >= 1.16.
|
||||||
func tryUpdateGoModAndGoSum(modMode ModMode, depMode DependencyInstallerMode) {
|
func tryUpdateGoModAndGoSum(modMode ModMode, depMode DependencyInstallerMode) {
|
||||||
// Go 1.16 and later won't automatically attempt to update go.mod / go.sum during package loading, so try to update them here:
|
// Go 1.16 and later won't automatically attempt to update go.mod / go.sum during package loading, so try to update them here:
|
||||||
if modMode != ModVendor && depMode == GoGetWithModules && semver.Compare(getEnvGoSemVer(), "v1.16") >= 0 {
|
if modMode != ModVendor && depMode == GoGetWithModules && semver.Compare(getEnvGoSemVer(), "v1.16") >= 0 {
|
||||||
|
@ -361,10 +406,11 @@ type moveGopathInfo struct {
|
||||||
files []string
|
files []string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Moves all files in `srcdir` to a temporary directory with the correct layout to be added to the GOPATH
|
||||||
func moveToTemporaryGopath(srcdir string, importpath string) moveGopathInfo {
|
func moveToTemporaryGopath(srcdir string, importpath string) moveGopathInfo {
|
||||||
// a temporary directory where everything is moved while the correct
|
// a temporary directory where everything is moved while the correct
|
||||||
// directory structure is created.
|
// directory structure is created.
|
||||||
scratch, err := ioutil.TempDir(srcdir, "scratch")
|
scratch, err := os.MkdirTemp(srcdir, "scratch")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to create temporary directory %s in directory %s: %s\n",
|
log.Fatalf("Failed to create temporary directory %s in directory %s: %s\n",
|
||||||
scratch, srcdir, err.Error())
|
scratch, srcdir, err.Error())
|
||||||
|
@ -422,6 +468,8 @@ func moveToTemporaryGopath(srcdir string, importpath string) moveGopathInfo {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Creates a path transformer file in the new directory to ensure paths in the source archive and the snapshot
|
||||||
|
// match the original source location, not the location we moved it to.
|
||||||
func createPathTransformerFile(newdir string) *os.File {
|
func createPathTransformerFile(newdir string) *os.File {
|
||||||
err := os.Chdir(newdir)
|
err := os.Chdir(newdir)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -430,13 +478,14 @@ func createPathTransformerFile(newdir string) *os.File {
|
||||||
|
|
||||||
// set up SEMMLE_PATH_TRANSFORMER to ensure paths in the source archive and the snapshot
|
// set up SEMMLE_PATH_TRANSFORMER to ensure paths in the source archive and the snapshot
|
||||||
// match the original source location, not the location we moved it to
|
// match the original source location, not the location we moved it to
|
||||||
pt, err := ioutil.TempFile("", "path-transformer")
|
pt, err := os.CreateTemp("", "path-transformer")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Unable to create path transformer file: %s.", err.Error())
|
log.Fatalf("Unable to create path transformer file: %s.", err.Error())
|
||||||
}
|
}
|
||||||
return pt
|
return pt
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Writes the path transformer file
|
||||||
func writePathTransformerFile(pt *os.File, realSrc, root, newdir string) {
|
func writePathTransformerFile(pt *os.File, realSrc, root, newdir string) {
|
||||||
_, err := pt.WriteString("#" + realSrc + "\n" + newdir + "//\n")
|
_, err := pt.WriteString("#" + realSrc + "\n" + newdir + "//\n")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -452,6 +501,7 @@ func writePathTransformerFile(pt *os.File, realSrc, root, newdir string) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Adds `root` to GOPATH.
|
||||||
func setGopath(root string) {
|
func setGopath(root string) {
|
||||||
// set/extend GOPATH
|
// set/extend GOPATH
|
||||||
oldGopath := os.Getenv("GOPATH")
|
oldGopath := os.Getenv("GOPATH")
|
||||||
|
@ -471,6 +521,8 @@ func setGopath(root string) {
|
||||||
log.Printf("GOPATH set to %s.\n", newGopath)
|
log.Printf("GOPATH set to %s.\n", newGopath)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Try to build the project without custom commands. If that fails, return a boolean indicating
|
||||||
|
// that we should install dependencies ourselves.
|
||||||
func buildWithoutCustomCommands(modMode ModMode) bool {
|
func buildWithoutCustomCommands(modMode ModMode) bool {
|
||||||
shouldInstallDependencies := false
|
shouldInstallDependencies := false
|
||||||
// try to build the project
|
// try to build the project
|
||||||
|
@ -490,6 +542,7 @@ func buildWithoutCustomCommands(modMode ModMode) bool {
|
||||||
return shouldInstallDependencies
|
return shouldInstallDependencies
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Build the project with custom commands.
|
||||||
func buildWithCustomCommands(inst string) {
|
func buildWithCustomCommands(inst string) {
|
||||||
// write custom build commands into a script, then run it
|
// write custom build commands into a script, then run it
|
||||||
var (
|
var (
|
||||||
|
@ -505,7 +558,7 @@ func buildWithCustomCommands(inst string) {
|
||||||
ext = ".sh"
|
ext = ".sh"
|
||||||
header = "#! /bin/bash\nset -xe +u\n"
|
header = "#! /bin/bash\nset -xe +u\n"
|
||||||
}
|
}
|
||||||
script, err := ioutil.TempFile("", "go-build-command-*"+ext)
|
script, err := os.CreateTemp("", "go-build-command-*"+ext)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Unable to create temporary script holding custom build commands: %s\n", err.Error())
|
log.Fatalf("Unable to create temporary script holding custom build commands: %s\n", err.Error())
|
||||||
}
|
}
|
||||||
|
@ -523,6 +576,7 @@ func buildWithCustomCommands(inst string) {
|
||||||
util.RunCmd(exec.Command(script.Name()))
|
util.RunCmd(exec.Command(script.Name()))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Install dependencies using the given dependency installer mode.
|
||||||
func installDependencies(depMode DependencyInstallerMode) {
|
func installDependencies(depMode DependencyInstallerMode) {
|
||||||
// automatically determine command to install dependencies
|
// automatically determine command to install dependencies
|
||||||
var install *exec.Cmd
|
var install *exec.Cmd
|
||||||
|
@ -574,6 +628,7 @@ func installDependencies(depMode DependencyInstallerMode) {
|
||||||
util.RunCmd(install)
|
util.RunCmd(install)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Run the extractor.
|
||||||
func extract(depMode DependencyInstallerMode, modMode ModMode) {
|
func extract(depMode DependencyInstallerMode, modMode ModMode) {
|
||||||
extractor, err := util.GetExtractorPath()
|
extractor, err := util.GetExtractorPath()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -601,12 +656,8 @@ func extract(depMode DependencyInstallerMode, modMode ModMode) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func main() {
|
// Build the project and run the extractor.
|
||||||
if len(os.Args) > 1 {
|
func installDependenciesAndBuild() {
|
||||||
usage()
|
|
||||||
os.Exit(2)
|
|
||||||
}
|
|
||||||
|
|
||||||
log.Printf("Autobuilder was built with %s, environment has %s\n", runtime.Version(), getEnvGoVersion())
|
log.Printf("Autobuilder was built with %s, environment has %s\n", runtime.Version(), getEnvGoVersion())
|
||||||
|
|
||||||
srcdir := getSourceDir()
|
srcdir := getSourceDir()
|
||||||
|
@ -617,31 +668,14 @@ func main() {
|
||||||
// determine how to install dependencies and whether a GOPATH needs to be set up before
|
// determine how to install dependencies and whether a GOPATH needs to be set up before
|
||||||
// extraction
|
// extraction
|
||||||
depMode := getDepMode()
|
depMode := getDepMode()
|
||||||
goDirectiveFound := false
|
|
||||||
if _, present := os.LookupEnv("GO111MODULE"); !present {
|
if _, present := os.LookupEnv("GO111MODULE"); !present {
|
||||||
os.Setenv("GO111MODULE", "auto")
|
os.Setenv("GO111MODULE", "auto")
|
||||||
}
|
}
|
||||||
if depMode == GoGetWithModules {
|
|
||||||
versionRe := regexp.MustCompile(`(?m)^go[ \t\r]+([0-9]+\.[0-9]+)$`)
|
_, goModVersionFound := tryReadGoDirective(depMode)
|
||||||
goMod, err := ioutil.ReadFile("go.mod")
|
|
||||||
if err != nil {
|
|
||||||
log.Println("Failed to read go.mod to check for missing Go version")
|
|
||||||
} else {
|
|
||||||
matches := versionRe.FindSubmatch(goMod)
|
|
||||||
if matches != nil {
|
|
||||||
goDirectiveFound = true
|
|
||||||
if len(matches) > 1 {
|
|
||||||
goDirectiveVersion := "v" + string(matches[1])
|
|
||||||
if semver.Compare(goDirectiveVersion, getEnvGoSemVer()) >= 0 {
|
|
||||||
diagnostics.EmitNewerGoVersionNeeded()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
modMode := getModMode(depMode)
|
modMode := getModMode(depMode)
|
||||||
modMode = fixGoVendorIssues(modMode, depMode, goDirectiveFound)
|
modMode = fixGoVendorIssues(modMode, depMode, goModVersionFound)
|
||||||
|
|
||||||
tryUpdateGoModAndGoSum(modMode, depMode)
|
tryUpdateGoModAndGoSum(modMode, depMode)
|
||||||
|
|
||||||
|
@ -692,3 +726,186 @@ func main() {
|
||||||
|
|
||||||
extract(depMode, modMode)
|
extract(depMode, modMode)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const minGoVersion = "1.11"
|
||||||
|
const maxGoVersion = "1.20"
|
||||||
|
|
||||||
|
// Check if `version` is lower than `minGoVersion` or higher than `maxGoVersion`. Note that for
|
||||||
|
// this comparison we ignore the patch part of the version, so 1.20.1 and 1.20 are considered
|
||||||
|
// equal.
|
||||||
|
func outsideSupportedRange(version string) bool {
|
||||||
|
short := semver.MajorMinor("v" + version)
|
||||||
|
return semver.Compare(short, "v"+minGoVersion) < 0 || semver.Compare(short, "v"+maxGoVersion) > 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if `v.goModVersion` or `v.goEnvVersion` are outside of the supported range. If so, emit
|
||||||
|
// a diagnostic and return an empty version to indicate that we should not attempt to install a
|
||||||
|
// different version of Go.
|
||||||
|
func checkForUnsupportedVersions(v versionInfo) (msg, version string) {
|
||||||
|
if v.goModVersionFound && outsideSupportedRange(v.goModVersion) {
|
||||||
|
msg = "The version of Go found in the `go.mod` file (" + v.goModVersion +
|
||||||
|
") is outside of the supported range (" + minGoVersion + "-" + maxGoVersion + ")."
|
||||||
|
version = ""
|
||||||
|
diagnostics.EmitUnsupportedVersionGoMod(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
if v.goEnVersionFound && outsideSupportedRange(v.goEnvVersion) {
|
||||||
|
msg = "The version of Go installed in the environment (" + v.goEnvVersion +
|
||||||
|
") is outside of the supported range (" + minGoVersion + "-" + maxGoVersion + ")."
|
||||||
|
version = ""
|
||||||
|
diagnostics.EmitUnsupportedVersionEnvironment(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
return msg, version
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if either `v.goEnVersionFound` or `v.goModVersionFound` are false. If so, emit
|
||||||
|
// a diagnostic and return the version to install, or the empty string if we should not attempt to
|
||||||
|
// install a version of Go. We assume that `checkForUnsupportedVersions` has already been
|
||||||
|
// called, so any versions that are found are within the supported range.
|
||||||
|
func checkForVersionsNotFound(v versionInfo) (msg, version string) {
|
||||||
|
if !v.goEnVersionFound && !v.goModVersionFound {
|
||||||
|
msg = "No version of Go installed and no `go.mod` file found. Writing an environment " +
|
||||||
|
"file specifying the maximum supported version of Go (" + maxGoVersion + ")."
|
||||||
|
version = maxGoVersion
|
||||||
|
diagnostics.EmitNoGoModAndNoGoEnv(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !v.goEnVersionFound && v.goModVersionFound {
|
||||||
|
msg = "No version of Go installed. Writing an environment file specifying the version " +
|
||||||
|
"of Go found in the `go.mod` file (" + v.goModVersion + ")."
|
||||||
|
version = v.goModVersion
|
||||||
|
diagnostics.EmitNoGoEnv(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
if v.goEnVersionFound && !v.goModVersionFound {
|
||||||
|
msg = "No `go.mod` file found. Version " + v.goEnvVersion + " installed in the environment."
|
||||||
|
version = ""
|
||||||
|
diagnostics.EmitNoGoMod(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
return msg, version
|
||||||
|
}
|
||||||
|
|
||||||
|
// Compare `v.goModVersion` and `v.goEnvVersion`. emit a diagnostic and return the version to
|
||||||
|
// install, or the empty string if we should not attempt to install a version of Go. We assume that
|
||||||
|
// `checkForUnsupportedVersions` and `checkForVersionsNotFound` have already been called, so both
|
||||||
|
// versions are found and are within the supported range.
|
||||||
|
func compareVersions(v versionInfo) (msg, version string) {
|
||||||
|
if semver.Compare("v"+v.goModVersion, "v"+v.goEnvVersion) > 0 {
|
||||||
|
msg = "The version of Go installed in the environment (" + v.goEnvVersion +
|
||||||
|
") is lower than the version found in the `go.mod` file (" + v.goModVersion +
|
||||||
|
").\nWriting an environment file specifying the version of Go from the `go.mod` " +
|
||||||
|
"file (" + v.goModVersion + ")."
|
||||||
|
version = v.goModVersion
|
||||||
|
diagnostics.EmitVersionGoModHigherVersionEnvironment(msg)
|
||||||
|
} else {
|
||||||
|
msg = "The version of Go installed in the environment (" + v.goEnvVersion +
|
||||||
|
") is high enough for the version found in the `go.mod` file (" + v.goModVersion + ")."
|
||||||
|
version = ""
|
||||||
|
diagnostics.EmitVersionGoModNotHigherVersionEnvironment(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
return msg, version
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check the versions of Go found in the environment and in the `go.mod` file, and return a
|
||||||
|
// version to install. If the version is the empty string then no installation is required.
|
||||||
|
func getVersionToInstall(v versionInfo) (msg, version string) {
|
||||||
|
msg, version = checkForUnsupportedVersions(v)
|
||||||
|
if msg != "" {
|
||||||
|
return msg, version
|
||||||
|
}
|
||||||
|
|
||||||
|
msg, version = checkForVersionsNotFound(v)
|
||||||
|
if msg != "" {
|
||||||
|
return msg, version
|
||||||
|
}
|
||||||
|
|
||||||
|
msg, version = compareVersions(v)
|
||||||
|
return msg, version
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write an environment file to the current directory. If `version` is the empty string then
|
||||||
|
// write an empty environment file, otherwise write an environment file specifying the version
|
||||||
|
// of Go to install. The path to the environment file is specified by the
|
||||||
|
// CODEQL_EXTRACTOR_ENVIRONMENT_JSON environment variable, or defaults to `environment.json`.
|
||||||
|
func writeEnvironmentFile(version string) {
|
||||||
|
var content string
|
||||||
|
if version == "" {
|
||||||
|
content = `{ "include": [] }`
|
||||||
|
} else {
|
||||||
|
content = `{ "include": [ { "go": { "version": "` + version + `" } } ] }`
|
||||||
|
}
|
||||||
|
|
||||||
|
filename, ok := os.LookupEnv("CODEQL_EXTRACTOR_ENVIRONMENT_JSON")
|
||||||
|
if !ok {
|
||||||
|
filename = "environment.json"
|
||||||
|
}
|
||||||
|
|
||||||
|
targetFile, err := os.Create(filename)
|
||||||
|
if err != nil {
|
||||||
|
log.Println("Failed to create environment file " + filename + ": ")
|
||||||
|
log.Println(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
if err := targetFile.Close(); err != nil {
|
||||||
|
log.Println("Failed to close environment file " + filename + ":")
|
||||||
|
log.Println(err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
_, err = targetFile.WriteString(content)
|
||||||
|
if err != nil {
|
||||||
|
log.Println("Failed to write to environment file " + filename + ": ")
|
||||||
|
log.Println(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type versionInfo struct {
|
||||||
|
goModVersion string // The version of Go found in the go directive in the `go.mod` file.
|
||||||
|
goModVersionFound bool // Whether a `go` directive was found in the `go.mod` file.
|
||||||
|
goEnvVersion string // The version of Go found in the environment.
|
||||||
|
goEnVersionFound bool // Whether an installation of Go was found in the environment.
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v versionInfo) String() string {
|
||||||
|
return fmt.Sprintf(
|
||||||
|
"go.mod version: %s, go.mod directive found: %t, go env version: %s, go installation found: %t",
|
||||||
|
v.goModVersion, v.goModVersionFound, v.goEnvVersion, v.goEnVersionFound)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if Go is installed in the environment.
|
||||||
|
func isGoInstalled() bool {
|
||||||
|
_, err := exec.LookPath("go")
|
||||||
|
return err == nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the version of Go to install and write it to an environment file.
|
||||||
|
func identifyEnvironment() {
|
||||||
|
var v versionInfo
|
||||||
|
depMode := getDepMode()
|
||||||
|
v.goModVersion, v.goModVersionFound = tryReadGoDirective(depMode)
|
||||||
|
|
||||||
|
v.goEnVersionFound = isGoInstalled()
|
||||||
|
if v.goEnVersionFound {
|
||||||
|
v.goEnvVersion = getEnvGoVersion()[2:]
|
||||||
|
}
|
||||||
|
|
||||||
|
msg, versionToInstall := getVersionToInstall(v)
|
||||||
|
log.Println(msg)
|
||||||
|
|
||||||
|
writeEnvironmentFile(versionToInstall)
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
if len(os.Args) == 1 {
|
||||||
|
installDependenciesAndBuild()
|
||||||
|
} else if len(os.Args) == 2 && os.Args[1] == "--identify-environment" {
|
||||||
|
identifyEnvironment()
|
||||||
|
} else {
|
||||||
|
usage()
|
||||||
|
os.Exit(2)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -33,3 +33,56 @@ func TestParseGoVersion(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestGetVersionToInstall(t *testing.T) {
|
||||||
|
tests := map[versionInfo]string{
|
||||||
|
// checkForUnsupportedVersions()
|
||||||
|
|
||||||
|
// go.mod version below minGoVersion
|
||||||
|
{"0.0", true, "1.20.3", true}: "",
|
||||||
|
{"0.0", true, "9999.0", true}: "",
|
||||||
|
{"0.0", true, "1.2.2", true}: "",
|
||||||
|
{"0.0", true, "", false}: "",
|
||||||
|
// go.mod version above maxGoVersion
|
||||||
|
{"9999.0", true, "1.20.3", true}: "",
|
||||||
|
{"9999.0", true, "9999.0.1", true}: "",
|
||||||
|
{"9999.0", true, "1.1", true}: "",
|
||||||
|
{"9999.0", true, "", false}: "",
|
||||||
|
// Go installation found with version below minGoVersion
|
||||||
|
{"1.20", true, "1.2.2", true}: "",
|
||||||
|
{"1.11", true, "1.2.2", true}: "",
|
||||||
|
{"", false, "1.2.2", true}: "",
|
||||||
|
// Go installation found with version above maxGoVersion
|
||||||
|
{"1.20", true, "9999.0.1", true}: "",
|
||||||
|
{"1.11", true, "9999.0.1", true}: "",
|
||||||
|
{"", false, "9999.0.1", true}: "",
|
||||||
|
|
||||||
|
// checkForVersionsNotFound()
|
||||||
|
|
||||||
|
// Go installation not found, go.mod version in supported range
|
||||||
|
{"1.20", true, "", false}: "1.20",
|
||||||
|
{"1.11", true, "", false}: "1.11",
|
||||||
|
// Go installation not found, go.mod not found
|
||||||
|
{"", false, "", false}: maxGoVersion,
|
||||||
|
// Go installation found with version in supported range, go.mod not found
|
||||||
|
{"", false, "1.11.13", true}: "",
|
||||||
|
{"", false, "1.20.3", true}: "",
|
||||||
|
|
||||||
|
// compareVersions()
|
||||||
|
|
||||||
|
// Go installation found with version in supported range, go.mod version in supported range and go.mod version > go installation version
|
||||||
|
{"1.20", true, "1.11.13", true}: "1.20",
|
||||||
|
{"1.20", true, "1.12", true}: "1.20",
|
||||||
|
// Go installation found with version in supported range, go.mod version in supported range and go.mod version <= go installation version
|
||||||
|
// (Note comparisons ignore the patch version)
|
||||||
|
{"1.11", true, "1.20", true}: "",
|
||||||
|
{"1.11", true, "1.20.3", true}: "",
|
||||||
|
{"1.20", true, "1.20.3", true}: "",
|
||||||
|
}
|
||||||
|
for input, expected := range tests {
|
||||||
|
_, actual := getVersionToInstall(input)
|
||||||
|
if actual != expected {
|
||||||
|
t.Errorf("Expected getVersionToInstall(\"%s\") to be \"%s\", but got \"%s\".", input, expected, actual)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -2,7 +2,6 @@ package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"regexp"
|
"regexp"
|
||||||
|
@ -22,7 +21,7 @@ func main() {
|
||||||
buildSteps := os.Args[2]
|
buildSteps := os.Args[2]
|
||||||
|
|
||||||
haveRepo := false
|
haveRepo := false
|
||||||
content, err := ioutil.ReadFile(vars)
|
content, err := os.ReadFile(vars)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
@ -34,7 +33,7 @@ func main() {
|
||||||
additionalVars += "SEMMLE_REPO_URL=${repository}\n"
|
additionalVars += "SEMMLE_REPO_URL=${repository}\n"
|
||||||
}
|
}
|
||||||
content = append(content, []byte(additionalVars)...)
|
content = append(content, []byte(additionalVars)...)
|
||||||
err = ioutil.WriteFile(vars, content, 0644)
|
err = os.WriteFile(vars, content, 0644)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
@ -47,7 +46,7 @@ func main() {
|
||||||
<build export="%s">${semmle_dist}/language-packs/go/tools/platform/${semmle_platform}/bin/go-autobuilder</build>
|
<build export="%s">${semmle_dist}/language-packs/go/tools/platform/${semmle_platform}/bin/go-autobuilder</build>
|
||||||
</autoupdate>
|
</autoupdate>
|
||||||
`, export))
|
`, export))
|
||||||
err = ioutil.WriteFile(buildSteps, content, 0644)
|
err = os.WriteFile(buildSteps, content, 0644)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,6 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"go/scanner"
|
"go/scanner"
|
||||||
"go/token"
|
"go/token"
|
||||||
"io/ioutil"
|
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"strings"
|
"strings"
|
||||||
|
@ -20,7 +19,7 @@ func main() {
|
||||||
defer csv.Flush()
|
defer csv.Flush()
|
||||||
|
|
||||||
for _, fileName := range flag.Args() {
|
for _, fileName := range flag.Args() {
|
||||||
src, err := ioutil.ReadFile(fileName)
|
src, err := os.ReadFile(fileName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Unable to read file %s.", fileName)
|
log.Fatalf("Unable to read file %s.", fileName)
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,6 +30,7 @@ type visibilityStruct struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
var fullVisibility *visibilityStruct = &visibilityStruct{true, true, true}
|
var fullVisibility *visibilityStruct = &visibilityStruct{true, true, true}
|
||||||
|
var telemetryOnly *visibilityStruct = &visibilityStruct{false, false, true}
|
||||||
|
|
||||||
type locationStruct struct {
|
type locationStruct struct {
|
||||||
File string `json:"file,omitempty"`
|
File string `json:"file,omitempty"`
|
||||||
|
@ -192,3 +193,80 @@ func EmitRelativeImportPaths() {
|
||||||
noLocation,
|
noLocation,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func EmitUnsupportedVersionGoMod(msg string) {
|
||||||
|
emitDiagnostic(
|
||||||
|
"go/autobuilder/env-unsupported-version-in-go-mod",
|
||||||
|
"Unsupported Go version in `go.mod` file",
|
||||||
|
msg,
|
||||||
|
severityNote,
|
||||||
|
telemetryOnly,
|
||||||
|
noLocation,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func EmitUnsupportedVersionEnvironment(msg string) {
|
||||||
|
emitDiagnostic(
|
||||||
|
"go/autobuilder/env-unsupported-version-in-environment",
|
||||||
|
"Unsupported Go version in environment",
|
||||||
|
msg,
|
||||||
|
severityNote,
|
||||||
|
telemetryOnly,
|
||||||
|
noLocation,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func EmitNoGoModAndNoGoEnv(msg string) {
|
||||||
|
emitDiagnostic(
|
||||||
|
"go/autobuilder/env-no-go-mod-and-no-go-env",
|
||||||
|
"No `go.mod` file found and no Go version in environment",
|
||||||
|
msg,
|
||||||
|
severityNote,
|
||||||
|
telemetryOnly,
|
||||||
|
noLocation,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func EmitNoGoEnv(msg string) {
|
||||||
|
emitDiagnostic(
|
||||||
|
"go/autobuilder/env-no-go-env",
|
||||||
|
"No Go version in environment",
|
||||||
|
msg,
|
||||||
|
severityNote,
|
||||||
|
telemetryOnly,
|
||||||
|
noLocation,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func EmitNoGoMod(msg string) {
|
||||||
|
emitDiagnostic(
|
||||||
|
"go/autobuilder/env-no-go-mod",
|
||||||
|
"No `go.mod` file found",
|
||||||
|
msg,
|
||||||
|
severityNote,
|
||||||
|
telemetryOnly,
|
||||||
|
noLocation,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func EmitVersionGoModHigherVersionEnvironment(msg string) {
|
||||||
|
emitDiagnostic(
|
||||||
|
"go/autobuilder/env-version-go-mod-higher-than-go-env",
|
||||||
|
"The Go version in `go.mod` file is higher than the Go version in environment",
|
||||||
|
msg,
|
||||||
|
severityNote,
|
||||||
|
telemetryOnly,
|
||||||
|
noLocation,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func EmitVersionGoModNotHigherVersionEnvironment(msg string) {
|
||||||
|
emitDiagnostic(
|
||||||
|
"go/autobuilder/env-version-go-mod-lower-than-or-equal-to-go-env",
|
||||||
|
"The Go version in `go.mod` file is lower than or equal to the Go version in environment",
|
||||||
|
msg,
|
||||||
|
severityNote,
|
||||||
|
telemetryOnly,
|
||||||
|
noLocation,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
|
@ -10,7 +10,6 @@ import (
|
||||||
"go/token"
|
"go/token"
|
||||||
"go/types"
|
"go/types"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
@ -1807,7 +1806,7 @@ func extractNumLines(tw *trap.Writer, fileName string, ast *ast.File) {
|
||||||
|
|
||||||
// count lines of code by tokenizing
|
// count lines of code by tokenizing
|
||||||
linesOfCode := 0
|
linesOfCode := 0
|
||||||
src, err := ioutil.ReadFile(fileName)
|
src, err := os.ReadFile(fileName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Unable to read file %s.", fileName)
|
log.Fatalf("Unable to read file %s.", fileName)
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,13 +2,14 @@ package extractor
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"golang.org/x/mod/modfile"
|
"io"
|
||||||
"io/ioutil"
|
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"golang.org/x/mod/modfile"
|
||||||
|
|
||||||
"github.com/github/codeql-go/extractor/dbscheme"
|
"github.com/github/codeql-go/extractor/dbscheme"
|
||||||
"github.com/github/codeql-go/extractor/srcarchive"
|
"github.com/github/codeql-go/extractor/srcarchive"
|
||||||
"github.com/github/codeql-go/extractor/trap"
|
"github.com/github/codeql-go/extractor/trap"
|
||||||
|
@ -45,7 +46,7 @@ func (extraction *Extraction) extractGoMod(path string) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("failed to open go.mod file %s: %s", path, err.Error())
|
return fmt.Errorf("failed to open go.mod file %s: %s", path, err.Error())
|
||||||
}
|
}
|
||||||
data, err := ioutil.ReadAll(file)
|
data, err := io.ReadAll(file)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("failed to read go.mod file %s: %s", path, err.Error())
|
return fmt.Errorf("failed to read go.mod file %s: %s", path, err.Error())
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,13 +1,12 @@
|
||||||
package srcarchive
|
package srcarchive
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"io/ioutil"
|
|
||||||
"os"
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
func mkProjectLayout(projectLayoutSource string, t *testing.T) (*ProjectLayout, error) {
|
func mkProjectLayout(projectLayoutSource string, t *testing.T) (*ProjectLayout, error) {
|
||||||
pt, err := ioutil.TempFile("", "path-transformer")
|
pt, err := os.CreateTemp("", "path-transformer")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("Unable to create temporary file for project layout: %s", err.Error())
|
t.Fatalf("Unable to create temporary file for project layout: %s", err.Error())
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,7 +7,6 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"go/ast"
|
"go/ast"
|
||||||
"go/types"
|
"go/types"
|
||||||
"io/ioutil"
|
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"unicode/utf8"
|
"unicode/utf8"
|
||||||
|
@ -51,7 +50,7 @@ func NewWriter(path string, pkg *packages.Package) (*Writer, error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
tmpFile, err := ioutil.TempFile(trapFileDir, filepath.Base(trapFilePath))
|
tmpFile, err := os.CreateTemp(trapFileDir, filepath.Base(trapFilePath))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
Загрузка…
Ссылка в новой задаче