Skip to content

Commit bfbf25d

Browse files
authored
add PackageUrlFactory (#69)
Signed-off-by: Jan Kowalleck <[email protected]>
1 parent 45d8fbc commit bfbf25d

8 files changed

+158
-31
lines changed

src/builders/fromPackageJson.node.ts

+6-27
Original file line numberDiff line numberDiff line change
@@ -17,10 +17,7 @@ SPDX-License-Identifier: Apache-2.0
1717
Copyright (c) OWASP Foundation. All Rights Reserved.
1818
*/
1919

20-
import { PackageURL } from 'packageurl-js'
21-
2220
import * as Enums from '../enums'
23-
import { ExternalReferenceType } from '../enums'
2421
import * as Models from '../models'
2522
import * as Factories from '../factories/index.node'
2623
import { PackageJson, splitNameGroup } from '../helpers/packageJson'
@@ -56,7 +53,10 @@ export class ComponentBuilder {
5653
readonly #extRefFactory: Factories.FromPackageJson.ExternalReferenceFactory
5754
readonly #licenseFactory: Factories.LicenseFactory
5855

59-
constructor (extRefFactory: Factories.FromPackageJson.ExternalReferenceFactory, licenseFactory: Factories.LicenseFactory) {
56+
constructor (
57+
extRefFactory: Factories.FromPackageJson.ExternalReferenceFactory,
58+
licenseFactory: Factories.LicenseFactory
59+
) {
6060
this.#extRefFactory = extRefFactory
6161
this.#licenseFactory = licenseFactory
6262
}
@@ -67,7 +67,7 @@ export class ComponentBuilder {
6767
}
6868

6969
const [name, group] = splitNameGroup(data.name)
70-
if (name.length === 0) {
70+
if (name.length <= 0) {
7171
return undefined
7272
}
7373

@@ -101,28 +101,7 @@ export class ComponentBuilder {
101101
? []
102102
: [license]
103103
),
104-
version,
105-
purl: this.#makePurl(name, group, version, externalReferences)
104+
version
106105
})
107106
}
108-
109-
#makePurl (
110-
name: string,
111-
group: string | undefined,
112-
version: string | undefined,
113-
externalReferences: Models.ExternalReference[]
114-
): PackageURL {
115-
const qualifiers: { [key: string]: string } = {}
116-
const subpath = undefined
117-
118-
const vcsUrl = externalReferences.filter(
119-
({ type }) => type === ExternalReferenceType.VCS
120-
)[0]?.url.toString()
121-
if (vcsUrl !== undefined) {
122-
qualifiers.vcs_url = vcsUrl
123-
}
124-
125-
return new PackageURL(
126-
'npm', group, name, version, qualifiers, subpath)
127-
}
128107
}

src/factories/index.common.ts

+1
Original file line numberDiff line numberDiff line change
@@ -20,3 +20,4 @@ Copyright (c) OWASP Foundation. All Rights Reserved.
2020
// not everything is public, yet
2121

2222
export * from './license'
23+
export * from './packageUrl'

src/factories/packageUrl.ts

+55
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
/*!
2+
This file is part of CycloneDX JavaScript Library.
3+
4+
Licensed under the Apache License, Version 2.0 (the "License");
5+
you may not use this file except in compliance with the License.
6+
You may obtain a copy of the License at
7+
8+
http://www.apache.org/licenses/LICENSE-2.0
9+
10+
Unless required by applicable law or agreed to in writing, software
11+
distributed under the License is distributed on an "AS IS" BASIS,
12+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
See the License for the specific language governing permissions and
14+
limitations under the License.
15+
16+
SPDX-License-Identifier: Apache-2.0
17+
Copyright (c) OWASP Foundation. All Rights Reserved.
18+
*/
19+
20+
import { Component } from '../models'
21+
import { PackageURL } from 'packageurl-js'
22+
import { ExternalReferenceType } from '../enums'
23+
24+
export class PackageUrlFactory {
25+
readonly #type: string
26+
27+
constructor (type: PackageURL['type']) {
28+
this.#type = type
29+
}
30+
31+
makeFromComponent (component: Component): PackageURL | undefined {
32+
const qualifiers: { [key: string]: string } = {}
33+
let subpath: string | undefined
34+
35+
for (const e of component.externalReferences) {
36+
if (e.type === ExternalReferenceType.VCS) {
37+
[qualifiers.vcs_url, subpath] = e.url.toString().split('#', 2)
38+
break
39+
}
40+
}
41+
42+
try {
43+
return new PackageURL(
44+
this.#type,
45+
component.group,
46+
component.name,
47+
component.version,
48+
qualifiers,
49+
subpath
50+
)
51+
} catch {
52+
return undefined
53+
}
54+
}
55+
}

tests/integration/Builders.FromPackageJson.ComponentBuilder.test.js

-3
Original file line numberDiff line numberDiff line change
@@ -21,8 +21,6 @@ Copyright (c) OWASP Foundation. All Rights Reserved.
2121
const assert = require('assert')
2222
const { suite, test } = require('mocha')
2323

24-
const { PackageURL } = require('packageurl-js')
25-
2624
const {
2725
Enums,
2826
Models,
@@ -60,7 +58,6 @@ suite('Builders.FromPackageJson.ComponentBuilder', () => {
6058
externalReferences: new Models.ExternalReferenceRepository([`FAKE REFERENCES ${salt}`]),
6159
licenses: new Models.LicenseRepository([`FAKE LICENSE ${salt}`]),
6260
group: '@foo',
63-
purl: new PackageURL('npm', '@foo', 'bar', `1.33.7-alpha.23.${salt}`, {}, undefined),
6461
version: `1.33.7-alpha.23.${salt}`
6562
}
6663
)

tests/unit/Factories.LicenseFactory.spec.js tests/integration/Factories.LicenseFactory.test.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@ const {
2626
Models: { LicenseExpression, NamedLicense, SpdxLicense }
2727
} = require('../../')
2828

29-
suite('LicenseFactory', () => {
29+
suite('Factories.LicenseFactory', () => {
3030
test('makeFromString() -> LicenseExpression', () => {
3131
const sut = new LicenseFactory()
3232

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,95 @@
1+
'use strict'
2+
/*!
3+
This file is part of CycloneDX JavaScript Library.
4+
5+
Licensed under the Apache License, Version 2.0 (the "License");
6+
you may not use this file except in compliance with the License.
7+
You may obtain a copy of the License at
8+
9+
http://www.apache.org/licenses/LICENSE-2.0
10+
11+
Unless required by applicable law or agreed to in writing, software
12+
distributed under the License is distributed on an "AS IS" BASIS,
13+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
See the License for the specific language governing permissions and
15+
limitations under the License.
16+
17+
SPDX-License-Identifier: Apache-2.0
18+
Copyright (c) OWASP Foundation. All Rights Reserved.
19+
*/
20+
21+
const assert = require('assert')
22+
const { suite, test } = require('mocha')
23+
24+
const { PackageURL } = require('packageurl-js')
25+
26+
const {
27+
Enums,
28+
Models,
29+
Factories: { PackageUrlFactory }
30+
} = require('../../')
31+
32+
suite('Factories.PackageUrlFactory', () => {
33+
const salt = Math.random()
34+
35+
const sut = new PackageUrlFactory('testing')
36+
37+
suite('makeFromComponent', () => {
38+
test('no-name-no-purl', () => {
39+
const component = new Models.Component(
40+
Enums.ComponentType.Library,
41+
''
42+
)
43+
const expected = undefined
44+
const actual = sut.makeFromComponent(component)
45+
assert.strictEqual(actual, expected)
46+
})
47+
48+
test('name-group-version', () => {
49+
const component = new Models.Component(
50+
Enums.ComponentType.Library,
51+
`name-${salt}`,
52+
{
53+
group: `@group-${salt}`,
54+
version: `v1+${salt}`,
55+
externalReferences: new Models.ExternalReferenceRepository([
56+
new Models.ExternalReference('https://foo.bar', Enums.ExternalReferenceType.Website)
57+
])
58+
}
59+
)
60+
const expected = new PackageURL('testing', `@group-${salt}`, `name-${salt}`, `v1+${salt}`, {}, undefined)
61+
const actual = sut.makeFromComponent(component)
62+
assert.deepStrictEqual(actual, expected)
63+
})
64+
65+
test('vcs-url without subpath', () => {
66+
const component = new Models.Component(
67+
Enums.ComponentType.Library,
68+
`name-${salt}`,
69+
{
70+
externalReferences: new Models.ExternalReferenceRepository([
71+
new Models.ExternalReference('git://foo.bar', Enums.ExternalReferenceType.VCS)
72+
])
73+
}
74+
)
75+
const expected = new PackageURL('testing', undefined, `name-${salt}`, undefined, { vcs_url: 'git://foo.bar' }, undefined)
76+
const actual = sut.makeFromComponent(component)
77+
assert.deepStrictEqual(actual, expected)
78+
})
79+
80+
test('vcs-url with subpath', () => {
81+
const component = new Models.Component(
82+
Enums.ComponentType.Library,
83+
`name-${salt}`,
84+
{
85+
externalReferences: new Models.ExternalReferenceRepository([
86+
new Models.ExternalReference('git://foo.bar#sub/path', Enums.ExternalReferenceType.VCS)
87+
])
88+
}
89+
)
90+
const expected = new PackageURL('testing', undefined, `name-${salt}`, undefined, { vcs_url: 'git://foo.bar' }, 'sub/path')
91+
const actual = sut.makeFromComponent(component)
92+
assert.deepStrictEqual(actual, expected)
93+
})
94+
})
95+
})

0 commit comments

Comments
 (0)