Skip to content

Commit

Permalink
Correctly encode PURLs in SBOM output
Browse files Browse the repository at this point in the history
  • Loading branch information
eoftedal committed Aug 20, 2024
1 parent fe36a0d commit 6fdcbbb
Show file tree
Hide file tree
Showing 7 changed files with 38 additions and 7 deletions.
6 changes: 6 additions & 0 deletions node/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,11 @@
# Changelog

## [5.2.2]

### Bugfix

- Encode purls correctly in SBOM

## [5.2.1]

### Bugfix
Expand Down
2 changes: 1 addition & 1 deletion node/lib/retire.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
*/

var exports = exports || {};
exports.version = '5.2.1';
exports.version = '5.2.2';

function isDefined(o) {
return typeof o !== 'undefined';
Expand Down
4 changes: 2 additions & 2 deletions node/package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion node/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
"author": "Erlend Oftedal <[email protected]>",
"name": "retire",
"description": "Retire is a tool for detecting use of vulnerable libraries",
"version": "5.2.1",
"version": "5.2.2",
"license": "Apache-2.0",
"repository": {
"type": "git",
Expand Down
16 changes: 16 additions & 0 deletions node/spec/tests/purl.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
import { should } from 'chai';
import 'mocha';
should();
import { generatePURL } from '../../lib/reporters/utils';

describe('purl encoding', () => {
it('should not touch a simple string', () => {
generatePURL({ component: 'jquery', version: '1.2.3' }).should.equal('pkg:npm/[email protected]');
});
it('should encode @ in package scopes', () => {
generatePURL({ component: '@angular/core', version: '1.2.3' }).should.equal('pkg:npm/%40angular/[email protected]');
});
it('should not doulbe encode', () => {
generatePURL({ component: '%40angular/core', version: '1.2.3' }).should.equal('pkg:npm/%40angular/[email protected]');
});
});
2 changes: 1 addition & 1 deletion node/src/repo.ts
Original file line number Diff line number Diff line change
Expand Up @@ -104,7 +104,7 @@ export function validateRepository(
bowername: z.array(z.string().regex(/^[a-z0-9.-]+$/i)).optional(),
basePurl: z
.string()
.regex(/^pkg:[a-z0-9/]+$/i)
.regex(/^pkg:[a-z0-9%.-/]+$/i)
.optional(),
npmname: z
.string()
Expand Down
13 changes: 11 additions & 2 deletions node/src/reporters/utils.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,18 @@
import { Component } from '../types';

function encodePURLchars(str: string): string {
return str.replace(
/[^A-Za-z0-9.+/=-%]/g,
(match) => '%' + ('0' + match.charCodeAt(0).toString(16).toUpperCase()).slice(-2),
);
}

export function generatePURL(component: Component): string {
if (component.basePurl) {
return component.basePurl + '@' + component.version;
const [pType, ...rest] = component.basePurl.split(':');
const pathElements = rest.join(':').split('/').map(encodePURLchars).join('/');
return `${pType}:${pathElements}@${encodePURLchars(component.version)}`;
}
const compName = component.npmname || component.component;
return `pkg:npm/${compName}@${component.version}`;
return `pkg:npm/${encodePURLchars(compName)}@${encodePURLchars(component.version)}`;
}

0 comments on commit 6fdcbbb

Please sign in to comment.