@@ -19,27 +19,29 @@ Copyright (c) OWASP Foundation. All Rights Reserved.
19
19
20
20
// import submodules so to prevent load of unused not-tree-shakable dependencies - like 'AJV'
21
21
import type { FromNodePackageJson as PJB } from '@cyclonedx/cyclonedx-library/Builders'
22
- import { ComponentType , ExternalReferenceType , LicenseAcknowledgement } from '@cyclonedx/cyclonedx-library/Enums'
22
+ import { AttachmentEncoding , ComponentType , ExternalReferenceType , LicenseAcknowledgement } from '@cyclonedx/cyclonedx-library/Enums'
23
23
import type { FromNodePackageJson as PJF } from '@cyclonedx/cyclonedx-library/Factories'
24
- import { Bom , Component , ExternalReference , type License , Property , Tool } from '@cyclonedx/cyclonedx-library/Models'
24
+ import { Attachment , Bom , Component , ComponentEvidence , ExternalReference , type License , NamedLicense , Property , Tool } from '@cyclonedx/cyclonedx-library/Models'
25
25
import { BomUtility } from '@cyclonedx/cyclonedx-library/Utils'
26
26
import { Cache , type FetchOptions , type Locator , type LocatorHash , type Package , type Project , structUtils , ThrowReport , type Workspace , YarnVersion } from '@yarnpkg/core'
27
27
import { ppath } from '@yarnpkg/fslib'
28
28
import { gitUtils as YarnPluginGitUtils } from '@yarnpkg/plugin-git'
29
29
import normalizePackageData from 'normalize-package-data'
30
30
31
31
import { getBuildtimeInfo } from './_buildtimeInfo'
32
- import { isString , tryRemoveSecretsFromUrl , trySanitizeGitUrl } from './_helpers'
32
+ import { getMimeForLicenseFile , isString , tryRemoveSecretsFromUrl , trySanitizeGitUrl } from './_helpers'
33
33
import { wsAnchoredPackage } from './_yarnCompat'
34
34
import { PropertyNames , PropertyValueBool } from './properties'
35
35
36
36
type ManifestFetcher = ( pkg : Package ) => Promise < any >
37
+ type LicenseEvidenceFetcher = ( pkg : Package ) => AsyncGenerator < License >
37
38
38
39
interface BomBuilderOptions {
39
40
omitDevDependencies ?: BomBuilder [ 'omitDevDependencies' ]
40
41
metaComponentType ?: BomBuilder [ 'metaComponentType' ]
41
42
reproducible ?: BomBuilder [ 'reproducible' ]
42
43
shortPURLs ?: BomBuilder [ 'shortPURLs' ]
44
+ gatherLicenseTexts ?: BomBuilder [ 'gatherLicenseTexts' ]
43
45
}
44
46
45
47
export class BomBuilder {
@@ -51,6 +53,7 @@ export class BomBuilder {
51
53
metaComponentType : ComponentType
52
54
reproducible : boolean
53
55
shortPURLs : boolean
56
+ gatherLicenseTexts : boolean
54
57
55
58
console : Console
56
59
@@ -69,13 +72,15 @@ export class BomBuilder {
69
72
this . metaComponentType = options . metaComponentType ?? ComponentType . Application
70
73
this . reproducible = options . reproducible ?? false
71
74
this . shortPURLs = options . shortPURLs ?? false
75
+ this . gatherLicenseTexts = options . gatherLicenseTexts ?? false
72
76
73
77
this . console = console_
74
78
}
75
79
76
80
async buildFromWorkspace ( workspace : Workspace ) : Promise < Bom > {
77
81
// @TODO make switch to disable load from fs
78
82
const fetchManifest : ManifestFetcher = await this . makeManifestFetcher ( workspace . project )
83
+ const fetchLicenseEvidences : LicenseEvidenceFetcher = await this . makeLicenseEvidenceFetcher ( workspace . project )
79
84
80
85
const setLicensesDeclared = function ( license : License ) : void {
81
86
license . acknowledgement = LicenseAcknowledgement . Declared
@@ -118,7 +123,8 @@ export class BomBuilder {
118
123
}
119
124
for await ( const component of this . gatherDependencies (
120
125
rootComponent , rootPackage ,
121
- workspace . project , fetchManifest
126
+ workspace . project ,
127
+ fetchManifest , fetchLicenseEvidences
122
128
) ) {
123
129
component . licenses . forEach ( setLicensesDeclared )
124
130
@@ -162,33 +168,90 @@ export class BomBuilder {
162
168
}
163
169
}
164
170
171
+ readonly #LICENSE_FILENAME_PATTERN = / ^ (?: U N ) ? L I C E N [ C S ] E | .\. L I C E N [ C S ] E $ | ^ N O T I C E $ / i
172
+
173
+ private async makeLicenseEvidenceFetcher ( project : Project ) : Promise < LicenseEvidenceFetcher > {
174
+ const fetcher = project . configuration . makeFetcher ( )
175
+ const fetcherOptions : FetchOptions = {
176
+ project,
177
+ fetcher,
178
+ cache : await Cache . find ( project . configuration ) ,
179
+ checksums : project . storedChecksums ,
180
+ report : new ThrowReport ( ) ,
181
+ cacheOptions : { skipIntegrityCheck : true }
182
+ }
183
+ const LICENSE_FILENAME_PATTERN = this . #LICENSE_FILENAME_PATTERN
184
+ return async function * ( pkg : Package ) : AsyncGenerator < License > {
185
+ const { packageFs, prefixPath, releaseFs } = await fetcher . fetch ( pkg , fetcherOptions )
186
+ try {
187
+ // option `withFileTypes:true` is not supported and causes crashes
188
+ const files = packageFs . readdirSync ( prefixPath )
189
+ for ( const file of files ) {
190
+ if ( ! LICENSE_FILENAME_PATTERN . test ( file ) ) {
191
+ continue
192
+ }
193
+
194
+ const contentType = getMimeForLicenseFile ( file )
195
+ if ( contentType === undefined ) {
196
+ continue
197
+ }
198
+
199
+ const fp = ppath . join ( prefixPath , file )
200
+ yield new NamedLicense (
201
+ `file: ${ file } ` ,
202
+ {
203
+ text : new Attachment (
204
+ packageFs . readFileSync ( fp ) . toString ( 'base64' ) ,
205
+ {
206
+ contentType,
207
+ encoding : AttachmentEncoding . Base64
208
+ }
209
+ )
210
+ } )
211
+ }
212
+ } finally {
213
+ if ( releaseFs !== undefined ) {
214
+ releaseFs ( )
215
+ }
216
+ }
217
+ }
218
+ }
219
+
165
220
private async makeComponentFromPackage (
166
221
pkg : Package ,
167
222
fetchManifest : ManifestFetcher ,
223
+ fetchLicenseEvidence : LicenseEvidenceFetcher ,
168
224
type ?: ComponentType | undefined
169
225
) : Promise < Component | false | undefined > {
170
- const data = await fetchManifest ( pkg )
226
+ const manifest = await fetchManifest ( pkg )
171
227
// the data in the manifest might be incomplete, so lets set the properties that yarn discovered and fixed
172
228
/* eslint-disable-next-line @typescript-eslint/strict-boolean-expressions */
173
- data . name = pkg . scope ? `@${ pkg . scope } /${ pkg . name } ` : pkg . name
174
- data . version = pkg . version
175
- return this . makeComponent ( pkg , data , type )
229
+ manifest . name = pkg . scope ? `@${ pkg . scope } /${ pkg . name } ` : pkg . name
230
+ manifest . version = pkg . version
231
+ const component = this . makeComponent ( pkg , manifest , type )
232
+ if ( this . gatherLicenseTexts && component instanceof Component ) {
233
+ component . evidence = new ComponentEvidence ( )
234
+ for await ( const le of fetchLicenseEvidence ( pkg ) ) {
235
+ component . evidence . licenses . add ( le )
236
+ }
237
+ }
238
+ return component
176
239
}
177
240
178
- private makeComponent ( locator : Locator , data : any , type ?: ComponentType | undefined ) : Component | false | undefined {
241
+ private makeComponent ( locator : Locator , manifest : any , type ?: ComponentType | undefined ) : Component | false | undefined {
179
242
// work with a deep copy, because `normalizePackageData()` might modify the data
180
- const dataC = structuredClonePolyfill ( data )
181
- normalizePackageData ( dataC as normalizePackageData . Input )
243
+ const manifestC = structuredClonePolyfill ( manifest )
244
+ normalizePackageData ( manifestC as normalizePackageData . Input )
182
245
// region fix normalizations
183
- if ( isString ( data . version ) ) {
246
+ if ( isString ( manifest . version ) ) {
184
247
// allow non-SemVer strings
185
- dataC . version = data . version . trim ( )
248
+ manifestC . version = manifest . version . trim ( )
186
249
}
187
250
// endregion fix normalizations
188
251
189
252
// work with a deep copy, because `normalizePackageData()` might modify the data
190
253
const component = this . componentBuilder . makeComponent (
191
- dataC as normalizePackageData . Package , type )
254
+ manifestC as normalizePackageData . Package , type )
192
255
if ( component === undefined ) {
193
256
this . console . debug ( 'DEBUG | skip broken component: %j' , locator )
194
257
return undefined
@@ -296,7 +359,8 @@ export class BomBuilder {
296
359
async * gatherDependencies (
297
360
component : Component , pkg : Package ,
298
361
project : Project ,
299
- fetchManifest : ManifestFetcher
362
+ fetchManifest : ManifestFetcher ,
363
+ fetchLicenseEvidences : LicenseEvidenceFetcher
300
364
) : AsyncGenerator < Component > {
301
365
// ATTENTION: multiple packages may have the same `identHash`, but the `locatorHash` is unique.
302
366
const knownComponents = new Map < LocatorHash , Component > ( [ [ pkg . locatorHash , component ] ] )
@@ -308,7 +372,8 @@ export class BomBuilder {
308
372
let depComponent = knownComponents . get ( depPkg . locatorHash )
309
373
if ( depComponent === undefined ) {
310
374
const _depIDN = structUtils . prettyLocatorNoColors ( depPkg )
311
- const _depC = await this . makeComponentFromPackage ( depPkg , fetchManifest )
375
+ const _depC = await this . makeComponentFromPackage ( depPkg ,
376
+ fetchManifest , fetchLicenseEvidences )
312
377
if ( _depC === false ) {
313
378
// shall be skipped
314
379
this . console . debug ( 'DEBUG | skip impossible component %j' , _depIDN )
0 commit comments