mirror of
https://github.com/anchore/syft.git
synced 2025-11-17 16:33:21 +01:00
chore: stop re-exporting wfn.Attributes (#2534)
* chore: stop re-exporting wfn.Attributes Previously, Syft re-exported wfn.Attributes from the nvdtools package as a member of the Package struct. However, Syft doesn't own this struct, and so after Syft 1.0, might be forced to bump a semver major version due to a breaking change in wfn.Attributes. Rather than incur this risk going into 1.0, instead replace Syft's use of wfn.Attributes with Syft's own cpe.CPE type. That type has some pass-through calls to wfn.Attributes, but hides the dependency from the rest of the application. Signed-off-by: Will Murphy <will.murphy@anchore.com> * chore: make cpe.CPE type a Stringer Previously, the cpe.CPE type was an alias for wfn.Attributes from nvdtools. Now that it is a type we control, make the String method take the CPE as a receiver, rather than as a normal parameter, so that Syft's cpe.CPE type implements Stringer. Signed-off-by: Will Murphy <will.murphy@anchore.com> --------- Signed-off-by: Will Murphy <will.murphy@anchore.com>
This commit is contained in:
parent
0fe13888d5
commit
878df69330
@ -2,13 +2,11 @@ package cpe
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"sort"
|
"sort"
|
||||||
|
|
||||||
"github.com/facebookincubator/nvdtools/wfn"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var _ sort.Interface = (*BySpecificity)(nil)
|
var _ sort.Interface = (*BySpecificity)(nil)
|
||||||
|
|
||||||
type BySpecificity []wfn.Attributes
|
type BySpecificity []CPE
|
||||||
|
|
||||||
func (c BySpecificity) Len() int { return len(c) }
|
func (c BySpecificity) Len() int { return len(c) }
|
||||||
|
|
||||||
@ -35,17 +33,17 @@ func (c BySpecificity) Less(i, j int) bool {
|
|||||||
return c[i].BindToFmtString() < c[j].BindToFmtString()
|
return c[i].BindToFmtString() < c[j].BindToFmtString()
|
||||||
}
|
}
|
||||||
|
|
||||||
func countFieldLength(cpe wfn.Attributes) int {
|
func countFieldLength(cpe CPE) int {
|
||||||
return len(cpe.Part + cpe.Vendor + cpe.Product + cpe.Version + cpe.TargetSW)
|
return len(cpe.Part + cpe.Vendor + cpe.Product + cpe.Version + cpe.TargetSW)
|
||||||
}
|
}
|
||||||
|
|
||||||
func weightedCountForSpecifiedFields(cpe wfn.Attributes) int {
|
func weightedCountForSpecifiedFields(cpe CPE) int {
|
||||||
checksForSpecifiedField := []func(cpe wfn.Attributes) (bool, int){
|
checksForSpecifiedField := []func(cpe CPE) (bool, int){
|
||||||
func(cpe wfn.Attributes) (bool, int) { return cpe.Part != "", 2 },
|
func(cpe CPE) (bool, int) { return cpe.Part != "", 2 },
|
||||||
func(cpe wfn.Attributes) (bool, int) { return cpe.Vendor != "", 3 },
|
func(cpe CPE) (bool, int) { return cpe.Vendor != "", 3 },
|
||||||
func(cpe wfn.Attributes) (bool, int) { return cpe.Product != "", 4 },
|
func(cpe CPE) (bool, int) { return cpe.Product != "", 4 },
|
||||||
func(cpe wfn.Attributes) (bool, int) { return cpe.Version != "", 1 },
|
func(cpe CPE) (bool, int) { return cpe.Version != "", 1 },
|
||||||
func(cpe wfn.Attributes) (bool, int) { return cpe.TargetSW != "", 1 },
|
func(cpe CPE) (bool, int) { return cpe.TargetSW != "", 1 },
|
||||||
}
|
}
|
||||||
|
|
||||||
weightedCount := 0
|
weightedCount := 0
|
||||||
|
|||||||
@ -8,7 +8,35 @@ import (
|
|||||||
"github.com/facebookincubator/nvdtools/wfn"
|
"github.com/facebookincubator/nvdtools/wfn"
|
||||||
)
|
)
|
||||||
|
|
||||||
type CPE = wfn.Attributes
|
type CPE struct {
|
||||||
|
Part string
|
||||||
|
Vendor string
|
||||||
|
Product string
|
||||||
|
Version string
|
||||||
|
Update string
|
||||||
|
Edition string
|
||||||
|
SWEdition string
|
||||||
|
TargetSW string
|
||||||
|
TargetHW string
|
||||||
|
Other string
|
||||||
|
Language string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c CPE) asAttributes() wfn.Attributes {
|
||||||
|
return wfn.Attributes(c)
|
||||||
|
}
|
||||||
|
|
||||||
|
func fromAttributes(a wfn.Attributes) CPE {
|
||||||
|
return CPE(a)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c CPE) BindToFmtString() string {
|
||||||
|
return c.asAttributes().BindToFmtString()
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewWithAny() CPE {
|
||||||
|
return fromAttributes(*(wfn.NewAttributesWithAny()))
|
||||||
|
}
|
||||||
|
|
||||||
const (
|
const (
|
||||||
allowedCPEPunctuation = "-!\"#$%&'()+,./:;<=>@[]^`{|}~"
|
allowedCPEPunctuation = "-!\"#$%&'()+,./:;<=>@[]^`{|}~"
|
||||||
@ -34,7 +62,7 @@ func New(cpeStr string) (CPE, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// ensure that this CPE can be validated after being fully sanitized
|
// ensure that this CPE can be validated after being fully sanitized
|
||||||
if ValidateString(String(c)) != nil {
|
if ValidateString(c.String()) != nil {
|
||||||
return CPE{}, err
|
return CPE{}, err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -71,20 +99,22 @@ func newWithoutValidation(cpeStr string) (CPE, error) {
|
|||||||
return CPE{}, fmt.Errorf("failed to parse CPE=%q", cpeStr)
|
return CPE{}, fmt.Errorf("failed to parse CPE=%q", cpeStr)
|
||||||
}
|
}
|
||||||
|
|
||||||
// we need to compare the raw data since we are constructing CPEs in other locations
|
syftCPE := fromAttributes(*value)
|
||||||
value.Vendor = normalizeField(value.Vendor)
|
|
||||||
value.Product = normalizeField(value.Product)
|
|
||||||
value.Language = normalizeField(value.Language)
|
|
||||||
value.Version = normalizeField(value.Version)
|
|
||||||
value.TargetSW = normalizeField(value.TargetSW)
|
|
||||||
value.Part = normalizeField(value.Part)
|
|
||||||
value.Edition = normalizeField(value.Edition)
|
|
||||||
value.Other = normalizeField(value.Other)
|
|
||||||
value.SWEdition = normalizeField(value.SWEdition)
|
|
||||||
value.TargetHW = normalizeField(value.TargetHW)
|
|
||||||
value.Update = normalizeField(value.Update)
|
|
||||||
|
|
||||||
return *value, nil
|
// we need to compare the raw data since we are constructing CPEs in other locations
|
||||||
|
syftCPE.Vendor = normalizeField(syftCPE.Vendor)
|
||||||
|
syftCPE.Product = normalizeField(syftCPE.Product)
|
||||||
|
syftCPE.Language = normalizeField(syftCPE.Language)
|
||||||
|
syftCPE.Version = normalizeField(syftCPE.Version)
|
||||||
|
syftCPE.TargetSW = normalizeField(syftCPE.TargetSW)
|
||||||
|
syftCPE.Part = normalizeField(syftCPE.Part)
|
||||||
|
syftCPE.Edition = normalizeField(syftCPE.Edition)
|
||||||
|
syftCPE.Other = normalizeField(syftCPE.Other)
|
||||||
|
syftCPE.SWEdition = normalizeField(syftCPE.SWEdition)
|
||||||
|
syftCPE.TargetHW = normalizeField(syftCPE.TargetHW)
|
||||||
|
syftCPE.Update = normalizeField(syftCPE.Update)
|
||||||
|
|
||||||
|
return syftCPE, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func normalizeField(field string) string {
|
func normalizeField(field string) string {
|
||||||
@ -112,7 +142,7 @@ func stripSlashes(s string) string {
|
|||||||
return sb.String()
|
return sb.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
func String(c CPE) string {
|
func (c CPE) String() string {
|
||||||
output := CPE{}
|
output := CPE{}
|
||||||
output.Vendor = sanitize(c.Vendor)
|
output.Vendor = sanitize(c.Vendor)
|
||||||
output.Product = sanitize(c.Product)
|
output.Product = sanitize(c.Product)
|
||||||
|
|||||||
@ -3,6 +3,7 @@ package cpe
|
|||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"github.com/google/go-cmp/cmp"
|
||||||
"os"
|
"os"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
@ -41,8 +42,8 @@ func Test_New(t *testing.T) {
|
|||||||
t.Fatalf("got an error while creating CPE: %+v", err)
|
t.Fatalf("got an error while creating CPE: %+v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if String(actual) != String(test.expected) {
|
if d := cmp.Diff(actual, test.expected); d != "" {
|
||||||
t.Errorf("mismatched entries:\n\texpected:%+v\n\t actual:%+v\n", String(test.expected), String(actual))
|
t.Errorf("CPE mismatch (-want +got):\n%s", d)
|
||||||
}
|
}
|
||||||
|
|
||||||
})
|
})
|
||||||
@ -98,7 +99,7 @@ func Test_CPEParser(t *testing.T) {
|
|||||||
assert.Equal(t, c1, c2)
|
assert.Equal(t, c1, c2)
|
||||||
assert.Equal(t, c1, test.WFN)
|
assert.Equal(t, c1, test.WFN)
|
||||||
assert.Equal(t, c2, test.WFN)
|
assert.Equal(t, c2, test.WFN)
|
||||||
assert.Equal(t, String(test.WFN), test.CPEString)
|
assert.Equal(t, test.WFN.String(), test.CPEString)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -164,12 +165,12 @@ func Test_InvalidCPE(t *testing.T) {
|
|||||||
if test.expectedErr {
|
if test.expectedErr {
|
||||||
assert.Error(t, err)
|
assert.Error(t, err)
|
||||||
if t.Failed() {
|
if t.Failed() {
|
||||||
t.Logf("got CPE: %q details: %+v", String(c), c)
|
t.Logf("got CPE: %q details: %+v", c, c)
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
assert.Equal(t, test.expected, String(c))
|
assert.Equal(t, test.expected, c.String())
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -215,13 +216,13 @@ func Test_RoundTrip(t *testing.T) {
|
|||||||
for _, test := range tests {
|
for _, test := range tests {
|
||||||
t.Run(test.name, func(t *testing.T) {
|
t.Run(test.name, func(t *testing.T) {
|
||||||
// CPE string must be preserved through a round trip
|
// CPE string must be preserved through a round trip
|
||||||
assert.Equal(t, test.cpe, String(Must(test.cpe)))
|
assert.Equal(t, test.cpe, Must(test.cpe).String())
|
||||||
// The parsed CPE must be the same after a round trip
|
// The parsed CPE must be the same after a round trip
|
||||||
assert.Equal(t, Must(test.cpe), Must(String(Must(test.cpe))))
|
assert.Equal(t, Must(test.cpe), Must(Must(test.cpe).String()))
|
||||||
// The test case parsed CPE must be the same after parsing the input string
|
// The test case parsed CPE must be the same after parsing the input string
|
||||||
assert.Equal(t, test.parsedCPE, Must(test.cpe))
|
assert.Equal(t, test.parsedCPE, Must(test.cpe))
|
||||||
// The test case parsed CPE must produce the same string as the input cpe
|
// The test case parsed CPE must produce the same string as the input cpe
|
||||||
assert.Equal(t, String(test.parsedCPE), test.cpe)
|
assert.Equal(t, test.parsedCPE.String(), test.cpe)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -12,7 +12,7 @@ func encodeSingleCPE(p pkg.Package) string {
|
|||||||
// Since the CPEs in a package are sorted by specificity
|
// Since the CPEs in a package are sorted by specificity
|
||||||
// we can extract the first CPE as the one to output in cyclonedx
|
// we can extract the first CPE as the one to output in cyclonedx
|
||||||
if len(p.CPEs) > 0 {
|
if len(p.CPEs) > 0 {
|
||||||
return cpe.String(p.CPEs[0])
|
return p.CPEs[0].String()
|
||||||
}
|
}
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
@ -25,7 +25,7 @@ func encodeCPEs(p pkg.Package) (out []cyclonedx.Property) {
|
|||||||
}
|
}
|
||||||
out = append(out, cyclonedx.Property{
|
out = append(out, cyclonedx.Property{
|
||||||
Name: "syft:cpe23",
|
Name: "syft:cpe23",
|
||||||
Value: cpe.String(c),
|
Value: c.String(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
|
|||||||
@ -107,7 +107,7 @@ func formatCPE(cpeString string) string {
|
|||||||
log.Debugf("skipping invalid CPE: %s", cpeString)
|
log.Debugf("skipping invalid CPE: %s", cpeString)
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
return cpe.String(c)
|
return c.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewBomDescriptor returns a new BomDescriptor tailored for the current time and "syft" tool details.
|
// NewBomDescriptor returns a new BomDescriptor tailored for the current time and "syft" tool details.
|
||||||
|
|||||||
@ -1,7 +1,6 @@
|
|||||||
package spdxhelpers
|
package spdxhelpers
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/anchore/syft/syft/cpe"
|
|
||||||
"github.com/anchore/syft/syft/pkg"
|
"github.com/anchore/syft/syft/pkg"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -11,7 +10,7 @@ func ExternalRefs(p pkg.Package) (externalRefs []ExternalRef) {
|
|||||||
for _, c := range p.CPEs {
|
for _, c := range p.CPEs {
|
||||||
externalRefs = append(externalRefs, ExternalRef{
|
externalRefs = append(externalRefs, ExternalRef{
|
||||||
ReferenceCategory: SecurityReferenceCategory,
|
ReferenceCategory: SecurityReferenceCategory,
|
||||||
ReferenceLocator: cpe.String(c),
|
ReferenceLocator: c.String(),
|
||||||
ReferenceType: Cpe23ExternalRefType,
|
ReferenceType: Cpe23ExternalRefType,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|||||||
@ -27,7 +27,7 @@ func Test_ExternalRefs(t *testing.T) {
|
|||||||
expected: []ExternalRef{
|
expected: []ExternalRef{
|
||||||
{
|
{
|
||||||
ReferenceCategory: SecurityReferenceCategory,
|
ReferenceCategory: SecurityReferenceCategory,
|
||||||
ReferenceLocator: cpe.String(testCPE),
|
ReferenceLocator: testCPE.String(),
|
||||||
ReferenceType: Cpe23ExternalRefType,
|
ReferenceType: Cpe23ExternalRefType,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@ -9,7 +9,6 @@ import (
|
|||||||
"github.com/anchore/syft/internal"
|
"github.com/anchore/syft/internal"
|
||||||
"github.com/anchore/syft/internal/log"
|
"github.com/anchore/syft/internal/log"
|
||||||
"github.com/anchore/syft/syft/artifact"
|
"github.com/anchore/syft/syft/artifact"
|
||||||
"github.com/anchore/syft/syft/cpe"
|
|
||||||
"github.com/anchore/syft/syft/file"
|
"github.com/anchore/syft/syft/file"
|
||||||
"github.com/anchore/syft/syft/format/syftjson/model"
|
"github.com/anchore/syft/syft/format/syftjson/model"
|
||||||
"github.com/anchore/syft/syft/internal/packagemetadata"
|
"github.com/anchore/syft/syft/internal/packagemetadata"
|
||||||
@ -232,7 +231,7 @@ func toLicenseModel(pkgLicenses []pkg.License) (modelLicenses []model.License) {
|
|||||||
func toPackageModel(p pkg.Package, cfg EncoderConfig) model.Package {
|
func toPackageModel(p pkg.Package, cfg EncoderConfig) model.Package {
|
||||||
var cpes = make([]string, len(p.CPEs))
|
var cpes = make([]string, len(p.CPEs))
|
||||||
for i, c := range p.CPEs {
|
for i, c := range p.CPEs {
|
||||||
cpes[i] = cpe.String(c)
|
cpes[i] = c.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
// we want to make sure all catalogers are
|
// we want to make sure all catalogers are
|
||||||
|
|||||||
@ -79,7 +79,7 @@ func Test_ClassifierCPEs(t *testing.T) {
|
|||||||
|
|
||||||
var cpes []string
|
var cpes []string
|
||||||
for _, c := range p.CPEs {
|
for _, c := range p.CPEs {
|
||||||
cpes = append(cpes, cpe.String(c))
|
cpes = append(cpes, c.String())
|
||||||
}
|
}
|
||||||
require.Equal(t, test.cpes, cpes)
|
require.Equal(t, test.cpes, cpes)
|
||||||
})
|
})
|
||||||
|
|||||||
@ -36,7 +36,7 @@ cpeLoop:
|
|||||||
}
|
}
|
||||||
|
|
||||||
func disallowNonParseableCPEs(c cpe.CPE, _ pkg.Package) bool {
|
func disallowNonParseableCPEs(c cpe.CPE, _ pkg.Package) bool {
|
||||||
v := cpe.String(c)
|
v := c.String()
|
||||||
_, err := cpe.New(v)
|
_, err := cpe.New(v)
|
||||||
|
|
||||||
cannotParse := err != nil
|
cannotParse := err != nil
|
||||||
|
|||||||
@ -23,14 +23,14 @@ import (
|
|||||||
// the CPE database, so they will be preferred over other candidates:
|
// the CPE database, so they will be preferred over other candidates:
|
||||||
var knownVendors = strset.New("apache")
|
var knownVendors = strset.New("apache")
|
||||||
|
|
||||||
func newCPE(product, vendor, version, targetSW string) *wfn.Attributes {
|
func newCPE(product, vendor, version, targetSW string) *cpe.CPE {
|
||||||
c := *(wfn.NewAttributesWithAny())
|
c := cpe.NewWithAny()
|
||||||
c.Part = "a"
|
c.Part = "a"
|
||||||
c.Product = product
|
c.Product = product
|
||||||
c.Vendor = vendor
|
c.Vendor = vendor
|
||||||
c.Version = version
|
c.Version = version
|
||||||
c.TargetSW = targetSW
|
c.TargetSW = targetSW
|
||||||
if cpe.ValidateString(cpe.String(c)) != nil {
|
if cpe.ValidateString(c.String()) != nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
return &c
|
return &c
|
||||||
|
|||||||
@ -10,7 +10,6 @@ import (
|
|||||||
"github.com/scylladb/go-set/strset"
|
"github.com/scylladb/go-set/strset"
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
|
|
||||||
"github.com/anchore/syft/syft/cpe"
|
|
||||||
"github.com/anchore/syft/syft/pkg"
|
"github.com/anchore/syft/syft/pkg"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -717,7 +716,7 @@ func TestGeneratePackageCPEs(t *testing.T) {
|
|||||||
expectedCpeSet := set.NewStringSet(test.expected...)
|
expectedCpeSet := set.NewStringSet(test.expected...)
|
||||||
actualCpeSet := set.NewStringSet()
|
actualCpeSet := set.NewStringSet()
|
||||||
for _, a := range actual {
|
for _, a := range actual {
|
||||||
actualCpeSet.Add(cpe.String(a))
|
actualCpeSet.Add(a.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
extra := strset.Difference(actualCpeSet, expectedCpeSet).List()
|
extra := strset.Difference(actualCpeSet, expectedCpeSet).List()
|
||||||
|
|||||||
@ -5,7 +5,6 @@ import (
|
|||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
|
|
||||||
"github.com/anchore/syft/syft/cpe"
|
|
||||||
"github.com/anchore/syft/syft/pkg/cataloger/internal/pkgtest"
|
"github.com/anchore/syft/syft/pkg/cataloger/internal/pkgtest"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -82,7 +81,7 @@ func Test_Binary_Cataloger_Stdlib_Cpe(t *testing.T) {
|
|||||||
t.Run(tc.name, func(t *testing.T) {
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
got, err := generateStdlibCpe(tc.candidate)
|
got, err := generateStdlibCpe(tc.candidate)
|
||||||
assert.NoError(t, err, "expected no err; got %v", err)
|
assert.NoError(t, err, "expected no err; got %v", err)
|
||||||
assert.Equal(t, cpe.String(got), tc.want)
|
assert.Equal(t, got.String(), tc.want)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user