2017-07-19 23:17:00 +02:00
|
|
|
package testutils
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"go/build"
|
|
|
|
"io/ioutil"
|
|
|
|
"log"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
"strings"
|
|
|
|
|
2020-04-01 22:18:39 +02:00
|
|
|
"github.com/securego/gosec/v2"
|
2019-04-25 09:25:32 +02:00
|
|
|
"golang.org/x/tools/go/packages"
|
2017-07-19 23:17:00 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type buildObj struct {
|
2019-04-25 09:25:32 +02:00
|
|
|
pkg *build.Package
|
|
|
|
config *packages.Config
|
|
|
|
pkgs []*packages.Package
|
2017-07-19 23:17:00 +02:00
|
|
|
}
|
|
|
|
|
2017-12-13 09:39:00 +02:00
|
|
|
// TestPackage is a mock package for testing purposes
|
2017-07-19 23:17:00 +02:00
|
|
|
type TestPackage struct {
|
|
|
|
Path string
|
|
|
|
Files map[string]string
|
2020-05-18 09:35:14 +02:00
|
|
|
onDisk bool
|
2017-07-19 23:17:00 +02:00
|
|
|
build *buildObj
|
|
|
|
}
|
|
|
|
|
2017-12-13 09:39:00 +02:00
|
|
|
// NewTestPackage will create a new and empty package. Must call Close() to cleanup
|
2018-10-11 14:45:31 +02:00
|
|
|
// auxiliary files
|
2017-07-19 23:17:00 +02:00
|
|
|
func NewTestPackage() *TestPackage {
|
2019-04-25 09:25:32 +02:00
|
|
|
workingDir, err := ioutil.TempDir("", "gosecs_test")
|
2017-07-19 23:17:00 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return &TestPackage{
|
|
|
|
Path: workingDir,
|
|
|
|
Files: make(map[string]string),
|
2020-05-18 09:35:14 +02:00
|
|
|
onDisk: false,
|
2017-07-19 23:17:00 +02:00
|
|
|
build: nil,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddFile inserts the filename and contents into the package contents
|
|
|
|
func (p *TestPackage) AddFile(filename, content string) {
|
|
|
|
p.Files[path.Join(p.Path, filename)] = content
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *TestPackage) write() error {
|
2020-05-18 09:35:14 +02:00
|
|
|
if p.onDisk {
|
2017-07-19 23:17:00 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
for filename, content := range p.Files {
|
2021-08-18 13:16:21 +02:00
|
|
|
if e := ioutil.WriteFile(filename, []byte(content), 0o644); e != nil {
|
2017-07-19 23:17:00 +02:00
|
|
|
return e
|
2021-12-15 20:31:14 +02:00
|
|
|
} //#nosec G306
|
2017-07-19 23:17:00 +02:00
|
|
|
}
|
2020-05-18 09:35:14 +02:00
|
|
|
p.onDisk = true
|
2017-07-19 23:17:00 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build ensures all files are persisted to disk and built
|
|
|
|
func (p *TestPackage) Build() error {
|
|
|
|
if p.build != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if err := p.write(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
basePackage, err := build.Default.ImportDir(p.Path, build.ImportComment)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2017-12-13 09:39:00 +02:00
|
|
|
var packageFiles []string
|
2017-07-19 23:17:00 +02:00
|
|
|
for _, filename := range basePackage.GoFiles {
|
|
|
|
packageFiles = append(packageFiles, path.Join(p.Path, filename))
|
|
|
|
}
|
|
|
|
|
2019-04-25 09:25:32 +02:00
|
|
|
conf := &packages.Config{
|
2019-10-02 14:05:14 +02:00
|
|
|
Mode: gosec.LoadMode,
|
2019-04-25 09:25:32 +02:00
|
|
|
Tests: false,
|
|
|
|
}
|
|
|
|
pkgs, err := packages.Load(conf, packageFiles...)
|
2017-07-19 23:17:00 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
p.build = &buildObj{
|
2019-04-25 09:25:32 +02:00
|
|
|
pkg: basePackage,
|
|
|
|
config: conf,
|
|
|
|
pkgs: pkgs,
|
2017-07-19 23:17:00 +02:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateContext builds a context out of supplied package context
|
2018-07-19 18:42:25 +02:00
|
|
|
func (p *TestPackage) CreateContext(filename string) *gosec.Context {
|
2017-07-19 23:17:00 +02:00
|
|
|
if err := p.Build(); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-04-25 09:25:32 +02:00
|
|
|
for _, pkg := range p.build.pkgs {
|
|
|
|
for _, file := range pkg.Syntax {
|
|
|
|
pkgFile := pkg.Fset.File(file.Pos()).Name()
|
2017-07-19 23:17:00 +02:00
|
|
|
strip := fmt.Sprintf("%s%c", p.Path, os.PathSeparator)
|
|
|
|
pkgFile = strings.TrimPrefix(pkgFile, strip)
|
|
|
|
if pkgFile == filename {
|
2018-07-19 18:42:25 +02:00
|
|
|
ctx := &gosec.Context{
|
2020-01-06 10:55:52 +02:00
|
|
|
FileSet: pkg.Fset,
|
|
|
|
Root: file,
|
|
|
|
Config: gosec.NewConfig(),
|
|
|
|
Info: pkg.TypesInfo,
|
|
|
|
Pkg: pkg.Types,
|
|
|
|
Imports: gosec.NewImportTracker(),
|
|
|
|
PassedValues: make(map[string]interface{}),
|
2017-07-19 23:17:00 +02:00
|
|
|
}
|
|
|
|
ctx.Imports.TrackPackages(ctx.Pkg.Imports()...)
|
|
|
|
return ctx
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close will delete the package and all files in that directory
|
|
|
|
func (p *TestPackage) Close() {
|
2020-05-18 09:35:14 +02:00
|
|
|
if p.onDisk {
|
2018-02-07 15:07:24 +02:00
|
|
|
err := os.RemoveAll(p.Path)
|
2018-03-09 07:36:31 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
2017-07-19 23:17:00 +02:00
|
|
|
}
|
|
|
|
}
|
2019-04-29 18:32:39 +02:00
|
|
|
|
|
|
|
// Pkgs returns the current built packages
|
|
|
|
func (p *TestPackage) Pkgs() []*packages.Package {
|
|
|
|
if p.build != nil {
|
|
|
|
return p.build.pkgs
|
|
|
|
}
|
|
|
|
return []*packages.Package{}
|
|
|
|
}
|
2021-01-01 21:30:45 +02:00
|
|
|
|
|
|
|
// PrintErrors prints to os.Stderr the accumulated errors of built packages
|
|
|
|
func (p *TestPackage) PrintErrors() int {
|
|
|
|
return packages.PrintErrors(p.Pkgs())
|
|
|
|
}
|