mirror of
https://github.com/SAP/jenkins-library.git
synced 2025-01-20 05:19:40 +02:00
a1988f6808
* Add GH issue creation + SARIF * Code cleanup * Fix fmt, add debug * Code enhancements * Fix * Added debug info * Rework UA log scan * Fix code * read UA version * Fix nil reference * Extraction * Credentials * Issue creation * Error handling * Fix issue creation * query escape * Query escape 2 * Revert * Test avoid update * HTTP client * Add support for custom TLS certs * Fix code * Fix code 2 * Fix code 3 * Disable cert check * Fix auth * Remove implicit trust * Skip verification * Fix * Fix client * Fix HTTP auth * Fix trusted certs * Trim version * Code * Add token * Added token handling to client * Fix token * Cleanup * Fix token * Token rework * Fix code * Kick out oauth client * Kick out oauth client * Transport wrapping * Token * Simplification * Refactor * Variation * Check * Fix * Debug * Switch client * Variation * Debug * Switch to cert check * Add debug * Parse self * Cleanup * Update resources/metadata/whitesourceExecuteScan.yaml * Add debug * Expose subjects * Patch * Debug * Debug2 * Debug3 * Fix logging response body * Cleanup * Cleanup * Fix request body logging * Cleanup import * Fix import cycle * Cleanup * Fix fmt * Fix NopCloser reference * Regenerate * Reintroduce * Fix test * Fix tests * Correction * Fix error * Code fix * Fix tests * Add tests * Fix code climate issues * Code climate * Code climate again * Code climate again * Fix fmt * Fix fmt 2 Co-authored-by: Oliver Nocon <33484802+OliverNocon@users.noreply.github.com>
140 lines
4.9 KiB
Go
140 lines
4.9 KiB
Go
package whitesource
|
|
|
|
import (
|
|
"fmt"
|
|
"sort"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/SAP/jenkins-library/pkg/log"
|
|
"github.com/SAP/jenkins-library/pkg/piperutils"
|
|
)
|
|
|
|
// Scan stores information about scanned WhiteSource projects (modules).
|
|
type Scan struct {
|
|
// AggregateProjectName stores the name of the WhiteSource project where scans shall be aggregated.
|
|
// It does not include the ProductVersion.
|
|
AggregateProjectName string
|
|
// ProductVersion is the global version that is used across all Projects (modules) during the scan.
|
|
ProductVersion string
|
|
scannedProjects map[string]Project
|
|
scanTimes map[string]time.Time
|
|
AgentName string
|
|
AgentVersion string
|
|
}
|
|
|
|
func (s *Scan) init() {
|
|
if s.scannedProjects == nil {
|
|
s.scannedProjects = make(map[string]Project)
|
|
}
|
|
if s.scanTimes == nil {
|
|
s.scanTimes = make(map[string]time.Time)
|
|
}
|
|
}
|
|
|
|
func (s *Scan) versionSuffix() string {
|
|
return " - " + s.ProductVersion
|
|
}
|
|
|
|
// AppendScannedProject checks that no Project with the same name is already contained in the list of scanned projects,
|
|
// and appends a new Project with the given name. The global product version is appended to the name.
|
|
func (s *Scan) AppendScannedProject(projectName string) error {
|
|
if len(projectName) == 0 {
|
|
return fmt.Errorf("projectName must not be empty")
|
|
}
|
|
if strings.HasSuffix(projectName, s.versionSuffix()) {
|
|
return fmt.Errorf("projectName is not expected to include the product version already")
|
|
}
|
|
return s.AppendScannedProjectVersion(projectName + s.versionSuffix())
|
|
}
|
|
|
|
// AppendScannedProjectVersion checks that no Project with the same name is already contained in the list of scanned
|
|
// projects, and appends a new Project with the given name (which is expected to include the product version).
|
|
func (s *Scan) AppendScannedProjectVersion(projectName string) error {
|
|
if !strings.HasSuffix(projectName, s.versionSuffix()) {
|
|
return fmt.Errorf("projectName is expected to include the product version")
|
|
}
|
|
if len(projectName) == len(s.versionSuffix()) {
|
|
return fmt.Errorf("projectName consists only of the product version")
|
|
}
|
|
s.init()
|
|
_, exists := s.scannedProjects[projectName]
|
|
if exists {
|
|
log.Entry().Errorf("A module with the name '%s' was already scanned. "+
|
|
"Your project's modules must have unique names.", projectName)
|
|
return fmt.Errorf("project with name '%s' was already scanned", projectName)
|
|
}
|
|
s.scannedProjects[projectName] = Project{Name: projectName}
|
|
s.scanTimes[projectName] = time.Now()
|
|
return nil
|
|
}
|
|
|
|
// ProjectByName returns a WhiteSource Project previously established via AppendScannedProject().
|
|
func (s *Scan) ProjectByName(projectName string) (Project, bool) {
|
|
project, exists := s.scannedProjects[projectName]
|
|
return project, exists
|
|
}
|
|
|
|
// ScannedProjects returns the WhiteSource projects that have been added via AppendScannedProject() as a slice.
|
|
func (s *Scan) ScannedProjects() []Project {
|
|
var projects []Project
|
|
for _, project := range s.scannedProjects {
|
|
projects = append(projects, project)
|
|
}
|
|
return projects
|
|
}
|
|
|
|
// ScannedProjectNames returns a sorted list of all scanned project names
|
|
func (s *Scan) ScannedProjectNames() []string {
|
|
projectNames := []string{}
|
|
for _, project := range s.ScannedProjects() {
|
|
projectNames = append(projectNames, project.Name)
|
|
}
|
|
// Sorting helps the list become stable across pipeline runs (and in the unit tests),
|
|
// as the order in which we travers map keys is not deterministic.
|
|
sort.Strings(projectNames)
|
|
return projectNames
|
|
}
|
|
|
|
// ScanTime returns the time at which the respective WhiteSource Project was scanned, or the the
|
|
// zero value of time.Time, if AppendScannedProject() was not called with that name.
|
|
func (s *Scan) ScanTime(projectName string) time.Time {
|
|
if s.scanTimes == nil {
|
|
return time.Time{}
|
|
}
|
|
return s.scanTimes[projectName]
|
|
}
|
|
|
|
type whitesource interface {
|
|
GetProjectsMetaInfo(productToken string) ([]Project, error)
|
|
GetProjectRiskReport(projectToken string) ([]byte, error)
|
|
GetProjectVulnerabilityReport(projectToken string, format string) ([]byte, error)
|
|
}
|
|
|
|
// UpdateProjects pulls the current backend metadata for all WhiteSource projects in the product with
|
|
// the given productToken, and updates all scanned projects with the obtained information.
|
|
func (s *Scan) UpdateProjects(productToken string, sys whitesource) error {
|
|
s.init()
|
|
projects, err := sys.GetProjectsMetaInfo(productToken)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to retrieve WhiteSource projects meta info: %w", err)
|
|
}
|
|
|
|
var projectsToUpdate []string
|
|
for projectName := range s.scannedProjects {
|
|
projectsToUpdate = append(projectsToUpdate, projectName)
|
|
}
|
|
|
|
for _, project := range projects {
|
|
_, exists := s.scannedProjects[project.Name]
|
|
if exists {
|
|
s.scannedProjects[project.Name] = project
|
|
projectsToUpdate, _ = piperutils.RemoveAll(projectsToUpdate, project.Name)
|
|
}
|
|
}
|
|
if len(projectsToUpdate) != 0 {
|
|
log.Entry().Warnf("Could not fetch metadata for projects %v", projectsToUpdate)
|
|
}
|
|
return nil
|
|
}
|