Merge pull request #12957 from owen-mc/go/autobuilder-identify-environment

Go: Add `go-autobuilder --identify-environment`
This commit is contained in:
Owen Mansel-Chan 2023-05-03 16:01:45 +01:00 коммит произвёл GitHub
Родитель f29db40371 841db151f6
Коммит 4de4f35855
Не найден ключ, соответствующий данной подписи
Идентификатор ключа GPG: 4AEE18F83AFDEB23
9 изменённых файлов: 410 добавлений и 66 удалений

Просмотреть файл

@ -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
} }