2020-10-20 09:49:26 +02:00
|
|
|
package whitesource
|
|
|
|
|
|
|
|
import (
|
2021-02-03 15:52:48 +02:00
|
|
|
"encoding/json"
|
2020-10-20 09:49:26 +02:00
|
|
|
"fmt"
|
2021-02-03 15:52:48 +02:00
|
|
|
"path/filepath"
|
|
|
|
|
|
|
|
"github.com/SAP/jenkins-library/pkg/log"
|
|
|
|
"github.com/pkg/errors"
|
2020-10-20 09:49:26 +02:00
|
|
|
)
|
|
|
|
|
2021-02-03 15:52:48 +02:00
|
|
|
const jvmTarGz = "jvm.tar.gz"
|
|
|
|
const jvmDir = "./jvm"
|
|
|
|
|
2020-10-20 09:49:26 +02:00
|
|
|
// ExecuteUAScan executes a scan with the Whitesource Unified Agent.
|
|
|
|
func (s *Scan) ExecuteUAScan(config *ScanOptions, utils Utils) error {
|
2021-02-03 15:52:48 +02:00
|
|
|
if config.BuildTool != "mta" {
|
|
|
|
return s.ExecuteUAScanInPath(config, utils, ".")
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Entry().Infof("Executing WhiteSource UA scan for MTA project")
|
|
|
|
pomExists, _ := utils.FileExists("pom.xml")
|
|
|
|
if pomExists {
|
|
|
|
mavenConfig := *config
|
|
|
|
mavenConfig.BuildTool = "maven"
|
|
|
|
if err := s.ExecuteUAScanInPath(&mavenConfig, utils, "."); err != nil {
|
|
|
|
return errors.Wrap(err, "failed to run scan for maven modules of mta")
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// ToDo: only warning message?
|
|
|
|
//log.Entry().Warning("MTA project does not contain a pom.xml in the root. Scan results might be incomplete")
|
|
|
|
return fmt.Errorf("mta project does not contain an aggregator pom.xml in the root - this is mandatory")
|
|
|
|
}
|
|
|
|
|
|
|
|
packageJSONFiles, err := utils.FindPackageJSONFiles(config)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "failed to find package.json files")
|
|
|
|
}
|
|
|
|
if len(packageJSONFiles) > 0 {
|
|
|
|
npmConfig := *config
|
|
|
|
npmConfig.BuildTool = "npm"
|
|
|
|
for _, packageJSONFile := range packageJSONFiles {
|
|
|
|
// we only need the path here
|
|
|
|
modulePath, _ := filepath.Split(packageJSONFile)
|
|
|
|
projectName, err := getProjectNameFromPackageJSON(packageJSONFile, utils)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrapf(err, "failed retrieve project name")
|
|
|
|
}
|
|
|
|
npmConfig.ProjectName = projectName
|
|
|
|
// ToDo: likely needs to be refactored, AggregateProjectName should only be available if we want to force aggregation?
|
|
|
|
s.AggregateProjectName = projectName
|
|
|
|
if err := s.ExecuteUAScanInPath(&npmConfig, utils, modulePath); err != nil {
|
|
|
|
return errors.Wrapf(err, "failed to run scan for npm module %v", modulePath)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_ = removeJre(filepath.Join(jvmDir, "bin", "java"), utils)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExecuteUAScanInPath executes a scan with the Whitesource Unified Agent in a dedicated scanPath.
|
|
|
|
func (s *Scan) ExecuteUAScanInPath(config *ScanOptions, utils Utils, scanPath string) error {
|
2020-10-20 09:49:26 +02:00
|
|
|
// Download the unified agent jar file if one does not exist
|
2021-02-03 15:52:48 +02:00
|
|
|
err := downloadAgent(config, utils)
|
|
|
|
if err != nil {
|
2020-10-20 09:49:26 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-02-03 15:52:48 +02:00
|
|
|
// Download JRE in case none is available
|
|
|
|
javaPath, err := downloadJre(config, utils)
|
|
|
|
if err != nil {
|
2020-10-20 09:49:26 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-02-03 15:52:48 +02:00
|
|
|
// ToDo: Check if Download of Docker/container image should be done here instead of in cmd/whitesourceExecuteScan.go
|
|
|
|
|
|
|
|
// ToDo: check if this is required
|
2020-11-10 10:09:51 +02:00
|
|
|
if err := s.AppendScannedProject(s.AggregateProjectName); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-02-03 15:52:48 +02:00
|
|
|
configPath, err := config.RewriteUAConfigurationFile(utils)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(scanPath) == 0 {
|
|
|
|
scanPath = "."
|
|
|
|
}
|
|
|
|
|
|
|
|
// ToDo: remove parameters which are added to UA config via RewriteUAConfigurationFile()
|
|
|
|
// let the scanner resolve project name on its own?
|
|
|
|
err = utils.RunExecutable(javaPath, "-jar", config.AgentFileName, "-d", scanPath, "-c", configPath,
|
2020-10-20 09:49:26 +02:00
|
|
|
"-apiKey", config.OrgToken, "-userKey", config.UserToken, "-project", s.AggregateProjectName,
|
2021-02-03 15:52:48 +02:00
|
|
|
"-product", config.ProductName, "-productVersion", s.ProductVersion, "-wss.url", config.AgentURL)
|
|
|
|
|
|
|
|
if err := removeJre(javaPath, utils); err != nil {
|
|
|
|
log.Entry().Warning(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
if err := removeJre(javaPath, utils); err != nil {
|
|
|
|
log.Entry().Warning(err)
|
|
|
|
}
|
|
|
|
exitCode := utils.GetExitCode()
|
|
|
|
log.Entry().Infof("WhiteSource scan failed with exit code %v", exitCode)
|
|
|
|
evaluateExitCode(exitCode)
|
|
|
|
return errors.Wrapf(err, "failed to execute WhiteSource scan with exit code %v", exitCode)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func evaluateExitCode(exitCode int) {
|
|
|
|
switch exitCode {
|
|
|
|
case 255:
|
|
|
|
log.Entry().Info("General error has occurred.")
|
|
|
|
log.SetErrorCategory(log.ErrorUndefined)
|
|
|
|
case 254:
|
|
|
|
log.Entry().Info("Whitesource found one or multiple policy violations.")
|
|
|
|
log.SetErrorCategory(log.ErrorCompliance)
|
|
|
|
case 253:
|
|
|
|
log.Entry().Info("The local scan client failed to execute the scan.")
|
|
|
|
log.SetErrorCategory(log.ErrorUndefined)
|
|
|
|
case 252:
|
|
|
|
log.Entry().Info("There was a failure in the connection to the WhiteSource servers.")
|
|
|
|
log.SetErrorCategory(log.ErrorInfrastructure)
|
|
|
|
case 251:
|
|
|
|
log.Entry().Info("The server failed to analyze the scan.")
|
|
|
|
log.SetErrorCategory(log.ErrorService)
|
|
|
|
case 250:
|
|
|
|
log.Entry().Info("One of the package manager's prerequisite steps (e.g. npm install) failed.")
|
|
|
|
log.SetErrorCategory(log.ErrorCustom)
|
|
|
|
default:
|
|
|
|
log.Entry().Info("Whitesource scan failed with unknown error code")
|
|
|
|
log.SetErrorCategory(log.ErrorUndefined)
|
|
|
|
}
|
2020-10-20 09:49:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// downloadAgent downloads the unified agent jar file if one does not exist
|
|
|
|
func downloadAgent(config *ScanOptions, utils Utils) error {
|
|
|
|
agentFile := config.AgentFileName
|
|
|
|
exists, err := utils.FileExists(agentFile)
|
|
|
|
if err != nil {
|
2021-02-03 15:52:48 +02:00
|
|
|
return errors.Wrapf(err, "failed to check if file '%s' exists", agentFile)
|
2020-10-20 09:49:26 +02:00
|
|
|
}
|
|
|
|
if !exists {
|
|
|
|
err := utils.DownloadFile(config.AgentDownloadURL, agentFile, nil, nil)
|
|
|
|
if err != nil {
|
2021-02-03 15:52:48 +02:00
|
|
|
return errors.Wrapf(err, "failed to download unified agent from URL '%s' to file '%s'", config.AgentDownloadURL, agentFile)
|
2020-10-20 09:49:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-02-03 15:52:48 +02:00
|
|
|
// downloadJre downloads the a JRE in case no java command can be executed
|
|
|
|
func downloadJre(config *ScanOptions, utils Utils) (string, error) {
|
|
|
|
// cater for multiple executions
|
|
|
|
if exists, _ := utils.FileExists(filepath.Join(jvmDir, "bin", "java")); exists {
|
|
|
|
return filepath.Join(jvmDir, "bin", "java"), nil
|
2020-10-20 09:49:26 +02:00
|
|
|
}
|
2021-02-03 15:52:48 +02:00
|
|
|
err := utils.RunExecutable("java", "-version")
|
|
|
|
javaPath := "java"
|
|
|
|
if err != nil {
|
|
|
|
log.Entry().Infof("No Java installation found, downloading JVM from %v", config.JreDownloadURL)
|
|
|
|
err := utils.DownloadFile(config.JreDownloadURL, jvmTarGz, nil, nil)
|
|
|
|
if err != nil {
|
|
|
|
return "", errors.Wrapf(err, "failed to download jre from URL '%s'", config.JreDownloadURL)
|
|
|
|
}
|
2020-10-20 09:49:26 +02:00
|
|
|
|
2021-02-03 15:52:48 +02:00
|
|
|
// ToDo: replace tar call with go library call
|
|
|
|
err = utils.MkdirAll(jvmDir, 0755)
|
|
|
|
|
|
|
|
err = utils.RunExecutable("tar", fmt.Sprintf("--directory=%v", jvmDir), "--strip-components=1", "-xzf", jvmTarGz)
|
|
|
|
if err != nil {
|
|
|
|
return "", errors.Wrapf(err, "failed to extract %v", jvmTarGz)
|
|
|
|
}
|
|
|
|
log.Entry().Info("Java successfully installed")
|
|
|
|
javaPath = filepath.Join(jvmDir, "bin", "java")
|
2020-10-20 09:49:26 +02:00
|
|
|
}
|
2021-02-03 15:52:48 +02:00
|
|
|
return javaPath, nil
|
|
|
|
}
|
2020-10-20 09:49:26 +02:00
|
|
|
|
2021-02-03 15:52:48 +02:00
|
|
|
func removeJre(javaPath string, utils Utils) error {
|
|
|
|
if javaPath == "java" {
|
|
|
|
return nil
|
2020-10-20 09:49:26 +02:00
|
|
|
}
|
2021-02-03 15:52:48 +02:00
|
|
|
if err := utils.RemoveAll(jvmDir); err != nil {
|
|
|
|
return fmt.Errorf("failed to remove downloaded and extracted jvm from %v", jvmDir)
|
|
|
|
}
|
|
|
|
log.Entry().Debugf("Java successfully removed from %v", jvmDir)
|
|
|
|
if err := utils.FileRemove(jvmTarGz); err != nil {
|
|
|
|
return fmt.Errorf("failed to remove downloaded %v", jvmTarGz)
|
|
|
|
}
|
|
|
|
log.Entry().Debugf("%v successfully removed", jvmTarGz)
|
|
|
|
return nil
|
|
|
|
}
|
2020-10-20 09:49:26 +02:00
|
|
|
|
2021-02-03 15:52:48 +02:00
|
|
|
func getProjectNameFromPackageJSON(packageJSONPath string, utils Utils) (string, error) {
|
|
|
|
fileContents, err := utils.FileRead(packageJSONPath)
|
|
|
|
if err != nil {
|
|
|
|
return "", errors.Wrapf(err, "failed to read file %v", packageJSONPath)
|
2020-10-20 09:49:26 +02:00
|
|
|
}
|
2021-02-03 15:52:48 +02:00
|
|
|
var packageJSON = make(map[string]interface{})
|
|
|
|
if err := json.Unmarshal(fileContents, &packageJSON); err != nil {
|
|
|
|
return "", errors.Wrapf(err, "failed to read file content of %v", packageJSONPath)
|
2020-10-20 09:49:26 +02:00
|
|
|
}
|
|
|
|
|
2021-02-03 15:52:48 +02:00
|
|
|
projectNameEntry, exists := packageJSON["name"]
|
|
|
|
if !exists {
|
|
|
|
return "", fmt.Errorf("the file '%s' must configure a name", packageJSONPath)
|
2020-10-20 09:49:26 +02:00
|
|
|
}
|
|
|
|
|
2021-02-03 15:52:48 +02:00
|
|
|
projectName, isString := projectNameEntry.(string)
|
|
|
|
if !isString {
|
|
|
|
return "", fmt.Errorf("the file '%s' must configure a name as string", packageJSONPath)
|
2020-10-20 09:49:26 +02:00
|
|
|
}
|
|
|
|
|
2021-02-03 15:52:48 +02:00
|
|
|
return projectName, nil
|
2020-10-20 09:49:26 +02:00
|
|
|
}
|