mirror of
https://github.com/anchore/syft.git
synced 2025-11-17 08:23:15 +01:00
* improve struct and field comments and incorporate into json schema Signed-off-by: Alex Goodman <wagoodman@users.noreply.github.com> * address review feedback Signed-off-by: Alex Goodman <wagoodman@users.noreply.github.com> --------- Signed-off-by: Alex Goodman <wagoodman@users.noreply.github.com>
263 lines
10 KiB
Go
263 lines
10 KiB
Go
package pkg
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"sort"
|
|
"strings"
|
|
|
|
"github.com/anchore/syft/internal"
|
|
"github.com/anchore/syft/syft/file"
|
|
)
|
|
|
|
var jenkinsPluginPomPropertiesGroupIDs = []string{
|
|
"io.jenkins.plugins",
|
|
"org.jenkins.plugins",
|
|
"org.jenkins-ci.plugins",
|
|
"io.jenkins-ci.plugins",
|
|
"com.cloudbees.jenkins.plugins",
|
|
}
|
|
|
|
// JavaVMInstallation represents a Java Virtual Machine installation discovered on the system with its release information and file list.
|
|
type JavaVMInstallation struct {
|
|
// Release is JVM release information and version details
|
|
Release JavaVMRelease `json:"release"`
|
|
|
|
// Files are the list of files that are part of this JVM installation
|
|
Files []string `json:"files"`
|
|
}
|
|
|
|
func (m JavaVMInstallation) OwnedFiles() []string {
|
|
return m.Files
|
|
}
|
|
|
|
// JavaVMRelease represents JVM version and build information extracted from the release file in a Java installation.
|
|
type JavaVMRelease struct {
|
|
// Implementor is extracted with the `java.vendor` JVM property
|
|
Implementor string `mapstructure:"IMPLEMENTOR,omitempty" json:"implementor,omitempty"`
|
|
|
|
// ImplementorVersion is extracted with the `java.vendor.version` JVM property
|
|
ImplementorVersion string `mapstructure:"IMPLEMENTOR_VERSION,omitempty" json:"implementorVersion,omitempty"`
|
|
|
|
// JavaRuntimeVersion is extracted from the 'java.runtime.version' JVM property
|
|
JavaRuntimeVersion string `mapstructure:"JAVA_RUNTIME_VERSION,omitempty" json:"javaRuntimeVersion,omitempty"`
|
|
|
|
// JavaVersion matches that from `java -version` command output
|
|
JavaVersion string `mapstructure:"JAVA_VERSION,omitempty" json:"javaVersion,omitempty"`
|
|
|
|
// JavaVersionDate is extracted from the 'java.version.date' JVM property
|
|
JavaVersionDate string `mapstructure:"JAVA_VERSION_DATE,omitempty" json:"javaVersionDate,omitempty"`
|
|
|
|
// Libc can either be 'glibc' or 'musl'
|
|
Libc string `mapstructure:"LIBC,omitempty" json:"libc,omitempty"`
|
|
|
|
// Modules is a list of JVM modules that are packaged
|
|
Modules []string `mapstructure:"MODULES,omitempty" json:"modules,omitempty"`
|
|
|
|
// OsArch is the target CPU architecture
|
|
OsArch string `mapstructure:"OS_ARCH,omitempty" json:"osArch,omitempty"`
|
|
|
|
// OsName is the name of the target runtime operating system environment
|
|
OsName string `mapstructure:"OS_NAME,omitempty" json:"osName,omitempty"`
|
|
|
|
// OsVersion is the version of the target runtime operating system environment
|
|
OsVersion string `mapstructure:"OS_VERSION,omitempty" json:"osVersion,omitempty"`
|
|
|
|
// Source refers to the origin repository of OpenJDK source
|
|
Source string `mapstructure:"SOURCE,omitempty" json:"source,omitempty"`
|
|
|
|
// BuildSource Git SHA of the build repository
|
|
BuildSource string `mapstructure:"BUILD_SOURCE,omitempty" json:"buildSource,omitempty"`
|
|
|
|
// BuildSourceRepo refers to rhe repository URL for the build source
|
|
BuildSourceRepo string `mapstructure:"BUILD_SOURCE_REPO,omitempty" json:"buildSourceRepo,omitempty"`
|
|
|
|
// SourceRepo refers to the OpenJDK repository URL
|
|
SourceRepo string `mapstructure:"SOURCE_REPO,omitempty" json:"sourceRepo,omitempty"`
|
|
|
|
// FullVersion is extracted from the 'java.runtime.version' JVM property
|
|
FullVersion string `mapstructure:"FULL_VERSION,omitempty" json:"fullVersion,omitempty"`
|
|
|
|
// SemanticVersion is derived from the OpenJDK version
|
|
SemanticVersion string `mapstructure:"SEMANTIC_VERSION,omitempty" json:"semanticVersion,omitempty"`
|
|
|
|
// BuildInfo contains additional build information
|
|
BuildInfo string `mapstructure:"BUILD_INFO,omitempty" json:"buildInfo,omitempty"`
|
|
|
|
// JvmVariant specifies the JVM variant (e.g., Hotspot or OpenJ9)
|
|
JvmVariant string `mapstructure:"JVM_VARIANT,omitempty" json:"jvmVariant,omitempty"`
|
|
|
|
// JvmVersion is extracted from the 'java.vm.version' JVM property
|
|
JvmVersion string `mapstructure:"JVM_VERSION,omitempty" json:"jvmVersion,omitempty"`
|
|
|
|
// ImageType can be 'JDK' or 'JRE'
|
|
ImageType string `mapstructure:"IMAGE_TYPE,omitempty" json:"imageType,omitempty"`
|
|
|
|
// BuildType can be 'commercial' (used in some older oracle JDK distributions)
|
|
BuildType string `mapstructure:"BUILD_TYPE,omitempty" json:"buildType,omitempty"`
|
|
}
|
|
|
|
// JavaArchive encapsulates all Java ecosystem metadata for a package as well as an (optional) parent relationship.
|
|
type JavaArchive struct {
|
|
// VirtualPath is path within the archive hierarchy, where nested entries are delimited with ':' (for nested JARs)
|
|
VirtualPath string `json:"virtualPath" cyclonedx:"virtualPath"`
|
|
|
|
// Manifest is parsed META-INF/MANIFEST.MF contents
|
|
Manifest *JavaManifest `mapstructure:"Manifest" json:"manifest,omitempty"`
|
|
|
|
// PomProperties is parsed pom.properties file contents
|
|
PomProperties *JavaPomProperties `mapstructure:"PomProperties" json:"pomProperties,omitempty" cyclonedx:"-"`
|
|
|
|
// PomProject is parsed pom.xml file contents
|
|
PomProject *JavaPomProject `mapstructure:"PomProject" json:"pomProject,omitempty"`
|
|
|
|
// ArchiveDigests is cryptographic hashes of the archive file
|
|
ArchiveDigests []file.Digest `hash:"ignore" json:"digest,omitempty"`
|
|
|
|
// Parent is reference to parent package (for nested archives)
|
|
Parent *Package `hash:"ignore" json:"-"`
|
|
}
|
|
|
|
// JavaPomProperties represents the fields of interest extracted from a Java archive's pom.properties file.
|
|
type JavaPomProperties struct {
|
|
// Path is path to the pom.properties file within the archive
|
|
Path string `mapstructure:"path" json:"path"`
|
|
|
|
// Name is the project name
|
|
Name string `mapstructure:"name" json:"name"`
|
|
|
|
// GroupID is Maven group identifier uniquely identifying the project across all projects (follows reversed domain name convention like com.company.project)
|
|
GroupID string `mapstructure:"groupId" json:"groupId" cyclonedx:"groupID"`
|
|
|
|
// ArtifactID is Maven artifact identifier, the name of the jar/artifact (unique within the groupId scope)
|
|
ArtifactID string `mapstructure:"artifactId" json:"artifactId" cyclonedx:"artifactID"`
|
|
|
|
// Version is artifact version
|
|
Version string `mapstructure:"version" json:"version"`
|
|
|
|
// Scope is dependency scope determining when dependency is available (compile=default all phases, test=test compilation/execution only, runtime=runtime and test not compile, provided=expected from JDK or container)
|
|
Scope string `mapstructure:"scope" json:"scope,omitempty"`
|
|
|
|
// Extra is additional custom properties not in standard Maven coordinates
|
|
Extra map[string]string `mapstructure:",remain" json:"extraFields,omitempty"`
|
|
}
|
|
|
|
// JavaPomProject represents fields of interest extracted from a Java archive's pom.xml file. See https://maven.apache.org/ref/3.6.3/maven-model/maven.html for more details.
|
|
type JavaPomProject struct {
|
|
// Path is path to the pom.xml file within the archive
|
|
Path string `json:"path"`
|
|
|
|
// Parent is the parent POM reference for inheritance (child POMs inherit configuration from parent)
|
|
Parent *JavaPomParent `json:"parent,omitempty"`
|
|
|
|
// GroupID is Maven group identifier (reversed domain name like org.apache.maven)
|
|
GroupID string `json:"groupId"`
|
|
|
|
// ArtifactID is Maven artifact identifier (project name)
|
|
ArtifactID string `json:"artifactId"`
|
|
|
|
// Version is project version (together with groupId and artifactId forms Maven coordinates groupId:artifactId:version)
|
|
Version string `json:"version"`
|
|
|
|
// Name is a human-readable project name (displayed in Maven-generated documentation)
|
|
Name string `json:"name"`
|
|
|
|
// Description is detailed project description
|
|
Description string `json:"description,omitempty"`
|
|
|
|
// URL is the project URL (typically project website or repository)
|
|
URL string `json:"url,omitempty"`
|
|
}
|
|
|
|
// JavaPomParent contains the fields within the <parent> tag in a pom.xml file
|
|
type JavaPomParent struct {
|
|
// GroupID is the parent Maven group identifier
|
|
GroupID string `json:"groupId"`
|
|
|
|
// ArtifactID is the parent Maven artifact identifier
|
|
ArtifactID string `json:"artifactId"`
|
|
|
|
// Version is the parent version (child inherits configuration from this specific version of parent POM)
|
|
Version string `json:"version"`
|
|
}
|
|
|
|
// PkgTypeIndicated returns the package Type indicated by the data contained in the JavaPomProperties.
|
|
func (p JavaPomProperties) PkgTypeIndicated() Type {
|
|
if internal.HasAnyOfPrefixes(p.GroupID, jenkinsPluginPomPropertiesGroupIDs...) || strings.Contains(p.GroupID, ".jenkins.plugin") {
|
|
return JenkinsPluginPkg
|
|
}
|
|
|
|
return JavaPkg
|
|
}
|
|
|
|
// JavaManifest represents the fields of interest extracted from a Java archive's META-INF/MANIFEST.MF file.
|
|
type JavaManifest struct {
|
|
// Main is main manifest attributes as key-value pairs
|
|
Main KeyValues `json:"main,omitempty"`
|
|
|
|
// Sections are the named sections from the manifest (e.g. per-entry attributes)
|
|
Sections []KeyValues `json:"sections,omitempty"`
|
|
}
|
|
|
|
type unmarshalJavaManifest JavaManifest
|
|
|
|
type legacyJavaManifest struct {
|
|
Main map[string]string `json:"main"`
|
|
NamedSections map[string]map[string]string `json:"namedSections"`
|
|
}
|
|
|
|
func (m *JavaManifest) UnmarshalJSON(b []byte) error {
|
|
var either map[string]any
|
|
err := json.Unmarshal(b, &either)
|
|
if err != nil {
|
|
return fmt.Errorf("could not unmarshal java manifest: %w", err)
|
|
}
|
|
if _, ok := either["namedSections"]; ok {
|
|
var lm legacyJavaManifest
|
|
if err = json.Unmarshal(b, &lm); err != nil {
|
|
return fmt.Errorf("could not unmarshal java manifest: %w", err)
|
|
}
|
|
*m = lm.toNewManifest()
|
|
return nil
|
|
}
|
|
var jm unmarshalJavaManifest
|
|
err = json.Unmarshal(b, &jm)
|
|
if err != nil {
|
|
return fmt.Errorf("could not unmarshal java manifest: %w", err)
|
|
}
|
|
*m = JavaManifest(jm)
|
|
return nil
|
|
}
|
|
|
|
func (lm legacyJavaManifest) toNewManifest() JavaManifest {
|
|
var result JavaManifest
|
|
result.Main = keyValuesFromMap(lm.Main)
|
|
var sectionNames []string
|
|
for k := range lm.NamedSections {
|
|
sectionNames = append(sectionNames, k)
|
|
}
|
|
sort.Strings(sectionNames)
|
|
var sections []KeyValues
|
|
for _, name := range sectionNames {
|
|
section := KeyValues{
|
|
KeyValue{
|
|
Key: "Name",
|
|
Value: name,
|
|
},
|
|
}
|
|
section = append(section, keyValuesFromMap(lm.NamedSections[name])...)
|
|
sections = append(sections, section)
|
|
}
|
|
result.Sections = sections
|
|
return result
|
|
}
|
|
|
|
func (m JavaManifest) Section(name string) KeyValues {
|
|
for _, section := range m.Sections {
|
|
if sectionName, ok := section.Get("Name"); ok && sectionName == name {
|
|
return section
|
|
}
|
|
}
|
|
return nil
|
|
}
|