Skip to content

Commit a2f9d58

Browse files
authored
feat: add purls flag to eol scan (#141)
This flag allows eol scan to directly consume a list of purls to send to our api. Users can still generate an sbom or use an existing sbom to get a list of purls. This feature allows `scan eol` to be useable by clients who want to derive a list of purls using a method besides cyclonedx sboms. This feature also allows `scan eol` to `dog food` the output of `report purls -s`.
1 parent de81b24 commit a2f9d58

File tree

5 files changed

+134
-7
lines changed

5 files changed

+134
-7
lines changed

README.md

+4-1
Original file line numberDiff line numberDiff line change
@@ -122,13 +122,14 @@ Scan a given sbom for EOL data
122122

123123
```
124124
USAGE
125-
$ hd scan eol [--json] [-f <value>] [-d <value>] [-s] [-a] [-c]
125+
$ hd scan eol [--json] [-f <value>] [-p <value>] [-d <value>] [-s] [-a] [-c]
126126
127127
FLAGS
128128
-a, --all Show all components (default is EOL and LTS only)
129129
-c, --getCustomerSupport Get Never-Ending Support for End-of-Life components
130130
-d, --dir=<value> The directory to scan in order to create a cyclonedx sbom
131131
-f, --file=<value> The file path of an existing cyclonedx sbom to scan for EOL
132+
-p, --purls=<value> The file path of a list of purls to scan for EOL
132133
-s, --save Save the generated SBOM as nes.sbom.json in the scanned directory
133134
134135
GLOBAL FLAGS
@@ -142,6 +143,8 @@ EXAMPLES
142143
143144
$ hd scan eol --file=path/to/sbom.json
144145
146+
$ hd scan eol --purls=path/to/purls.json
147+
145148
$ hd scan eol -a --dir=./my-project
146149
```
147150

src/commands/report/purls.ts

+1-3
Original file line numberDiff line numberDiff line change
@@ -83,9 +83,7 @@ export default class ReportPurls extends Command {
8383
}
8484

8585
// Return wrapped object with metadata
86-
return {
87-
purls,
88-
};
86+
return { purls };
8987
} catch (error) {
9088
this.error(`Failed to generate PURLs: ${getErrorMessage(error)}`);
9189
}

src/commands/scan/eol.ts

+28-2
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import type { ScanResult, ScanResultComponent } from '../../api/types/nes.types.
55
import type { Sbom } from '../../service/eol/cdx.svc.ts';
66
import { getErrorMessage, isErrnoException } from '../../service/error.svc.ts';
77
import { extractPurls } from '../../service/purls.svc.ts';
8+
import { parsePurlsFile } from '../../service/purls.svc.ts';
89
import { createStatusDisplay } from '../../ui/eol.ui.ts';
910
import { INDICATORS, STATUS_COLORS } from '../../ui/shared.us.ts';
1011
import ScanSbom from './sbom.ts';
@@ -15,13 +16,18 @@ export default class ScanEol extends Command {
1516
static override examples = [
1617
'<%= config.bin %> <%= command.id %> --dir=./my-project',
1718
'<%= config.bin %> <%= command.id %> --file=path/to/sbom.json',
19+
'<%= config.bin %> <%= command.id %> --purls=path/to/purls.json',
1820
'<%= config.bin %> <%= command.id %> -a --dir=./my-project',
1921
];
2022
static override flags = {
2123
file: Flags.string({
2224
char: 'f',
2325
description: 'The file path of an existing cyclonedx sbom to scan for EOL',
2426
}),
27+
purls: Flags.string({
28+
char: 'p',
29+
description: 'The file path of a list of purls to scan for EOL',
30+
}),
2531
dir: Flags.string({
2632
char: 'd',
2733
description: 'The directory to scan in order to create a cyclonedx sbom',
@@ -50,8 +56,7 @@ export default class ScanEol extends Command {
5056
this.log(ux.colorize('yellow', 'Never-Ending Support is on the way. Please stay tuned for this feature.'));
5157
}
5258

53-
const sbom = await ScanSbom.loadSbom(flags, this.config);
54-
const scan = await this.scanSbom(sbom);
59+
const scan = await this.getScan(flags, this.config);
5560

5661
ux.action.stop('\nScan completed');
5762

@@ -70,6 +75,27 @@ export default class ScanEol extends Command {
7075
return { components: filteredComponents };
7176
}
7277

78+
private async getScan(flags: Record<string, string>, config: Command['config']): Promise<ScanResult> {
79+
if (flags.purls) {
80+
const purls = this.getPurlsFromFile(flags.purls);
81+
return submitScan(purls);
82+
}
83+
84+
const sbom = await ScanSbom.loadSbom(flags, config);
85+
const scan = this.scanSbom(sbom);
86+
87+
return scan;
88+
}
89+
90+
private getPurlsFromFile(filePath: string): string[] {
91+
try {
92+
const purlsFileString = fs.readFileSync(filePath, 'utf8');
93+
return parsePurlsFile(purlsFileString);
94+
} catch (error) {
95+
this.error(`Failed to read purls file. ${getErrorMessage(error)}`);
96+
}
97+
}
98+
7399
private async scanSbom(sbom: Sbom): Promise<ScanResult> {
74100
let scan: ScanResult;
75101
let purls: string[];

src/service/purls.svc.ts

+30
Original file line numberDiff line numberDiff line change
@@ -31,3 +31,33 @@ export async function extractPurls(sbom: Sbom): Promise<string[]> {
3131
const { components: comps } = sbom;
3232
return comps.map((c) => c.purl) ?? [];
3333
}
34+
35+
/**
36+
* Parse a purls file in either JSON or text format, including the format of
37+
* nes.purls.json - { purls: [ 'pkg:npm/express@4.18.2', 'pkg:npm/[email protected]' ] }
38+
* or a text file with one purl per line.
39+
*/
40+
export function parsePurlsFile(purlsFileString: string): string[] {
41+
try {
42+
const parsed = JSON.parse(purlsFileString);
43+
44+
if (parsed && Array.isArray(parsed.purls)) {
45+
return parsed.purls;
46+
}
47+
48+
if (Array.isArray(parsed)) {
49+
return parsed;
50+
}
51+
} catch {
52+
const lines = purlsFileString
53+
.split('\n')
54+
.map((line) => line.trim())
55+
.filter((line) => line.length > 0 && line.startsWith('pkg:'));
56+
57+
if (lines.length > 0) {
58+
return lines;
59+
}
60+
}
61+
62+
throw new Error('Invalid purls file: must be either JSON with purls array or text file with one purl per line');
63+
}

test/service/purls.svc.test.ts

+71-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import assert from 'node:assert';
22
import { describe, it } from 'node:test';
3-
import { formatCsvValue, getPurlOutput } from '../../src/service/purls.svc.ts';
3+
import { formatCsvValue, getPurlOutput, parsePurlsFile } from '../../src/service/purls.svc.ts';
44

55
describe('getPurlOutput', () => {
66
describe('json output', () => {
@@ -49,3 +49,73 @@ describe('formatCsvValue', () => {
4949
assert.strictEqual(formatCsvValue(''), '');
5050
});
5151
});
52+
53+
describe('parsePurlsFile', () => {
54+
describe('JSON format', () => {
55+
it('should parse nes.purls.json format', () => {
56+
const input = JSON.stringify({
57+
purls: ['pkg:npm/@apollo/[email protected]', 'pkg:npm/[email protected]'],
58+
});
59+
const result = parsePurlsFile(input);
60+
assert.deepStrictEqual(result, ['pkg:npm/@apollo/[email protected]', 'pkg:npm/[email protected]']);
61+
});
62+
63+
it('should parse direct JSON array', () => {
64+
const input = JSON.stringify(['pkg:npm/[email protected]', 'pkg:npm/[email protected]']);
65+
const result = parsePurlsFile(input);
66+
assert.deepStrictEqual(result, ['pkg:npm/[email protected]', 'pkg:npm/[email protected]']);
67+
});
68+
});
69+
70+
describe('text format', () => {
71+
it('should parse text file with one purl per line', () => {
72+
const input = `pkg:npm/[email protected]
73+
74+
const result = parsePurlsFile(input);
75+
assert.deepStrictEqual(result, ['pkg:npm/[email protected]', 'pkg:npm/[email protected]']);
76+
});
77+
78+
it('should handle empty lines and whitespace', () => {
79+
const input = `
80+
81+
82+
83+
`;
84+
const result = parsePurlsFile(input);
85+
assert.deepStrictEqual(result, ['pkg:npm/[email protected]', 'pkg:npm/[email protected]']);
86+
});
87+
88+
it('should filter out invalid lines', () => {
89+
const input = `
90+
not-a-purl
91+
92+
also-not-a-purl
93+
94+
`;
95+
const result = parsePurlsFile(input);
96+
assert.deepStrictEqual(result, ['pkg:npm/[email protected]', 'pkg:npm/[email protected]']);
97+
});
98+
});
99+
100+
describe('error handling', () => {
101+
it('should throw error for invalid JSON', () => {
102+
const input = '{ invalid json }';
103+
assert.throws(() => parsePurlsFile(input), {
104+
message: 'Invalid purls file: must be either JSON with purls array or text file with one purl per line',
105+
});
106+
});
107+
108+
it('should throw error for empty file', () => {
109+
assert.throws(() => parsePurlsFile(''), {
110+
message: 'Invalid purls file: must be either JSON with purls array or text file with one purl per line',
111+
});
112+
});
113+
114+
it('should throw error for file with no valid purls', () => {
115+
const input = 'not-a-purl\nstill-not-a-purl';
116+
assert.throws(() => parsePurlsFile(input), {
117+
message: 'Invalid purls file: must be either JSON with purls array or text file with one purl per line',
118+
});
119+
});
120+
});
121+
});

0 commit comments

Comments
 (0)