@trustify-da/trustify-da-javascript-client 0.3.0-ea.fbdacbb → 0.3.0-ea.ff694a0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +191 -11
- package/dist/package.json +24 -11
- package/dist/src/analysis.d.ts +21 -5
- package/dist/src/analysis.js +74 -80
- package/dist/src/batch_opts.d.ts +24 -0
- package/dist/src/batch_opts.js +35 -0
- package/dist/src/cli.js +241 -8
- package/dist/src/cyclone_dx_sbom.d.ts +17 -3
- package/dist/src/cyclone_dx_sbom.js +48 -8
- package/dist/src/index.d.ts +197 -11
- package/dist/src/index.js +356 -8
- package/dist/src/license/index.d.ts +28 -0
- package/dist/src/license/index.js +100 -0
- package/dist/src/license/license_utils.d.ts +40 -0
- package/dist/src/license/license_utils.js +134 -0
- package/dist/src/license/licenses_api.d.ts +34 -0
- package/dist/src/license/licenses_api.js +98 -0
- package/dist/src/license/project_license.d.ts +20 -0
- package/dist/src/license/project_license.js +62 -0
- package/dist/src/oci_image/images.d.ts +4 -5
- package/dist/src/oci_image/utils.d.ts +4 -4
- package/dist/src/oci_image/utils.js +11 -2
- package/dist/src/provider.d.ts +18 -5
- package/dist/src/provider.js +31 -5
- package/dist/src/providers/base_java.d.ts +8 -14
- package/dist/src/providers/base_java.js +9 -38
- package/dist/src/providers/base_javascript.d.ts +40 -7
- package/dist/src/providers/base_javascript.js +138 -24
- package/dist/src/providers/base_pyproject.d.ts +158 -0
- package/dist/src/providers/base_pyproject.js +322 -0
- package/dist/src/providers/golang_gomodules.d.ts +30 -13
- package/dist/src/providers/golang_gomodules.js +176 -121
- package/dist/src/providers/gomod_parser.d.ts +4 -0
- package/dist/src/providers/gomod_parser.js +16 -0
- package/dist/src/providers/java_gradle.d.ts +28 -3
- package/dist/src/providers/java_gradle.js +128 -4
- package/dist/src/providers/java_gradle_groovy.d.ts +1 -1
- package/dist/src/providers/java_gradle_kotlin.d.ts +1 -1
- package/dist/src/providers/java_maven.d.ts +20 -5
- package/dist/src/providers/java_maven.js +126 -6
- package/dist/src/providers/javascript_bun.d.ts +10 -0
- package/dist/src/providers/javascript_bun.js +100 -0
- package/dist/src/providers/javascript_npm.d.ts +1 -0
- package/dist/src/providers/javascript_npm.js +21 -0
- package/dist/src/providers/javascript_pnpm.d.ts +1 -1
- package/dist/src/providers/javascript_pnpm.js +8 -4
- package/dist/src/providers/manifest.d.ts +2 -0
- package/dist/src/providers/manifest.js +22 -4
- package/dist/src/providers/marker_evaluator.d.ts +14 -0
- package/dist/src/providers/marker_evaluator.js +191 -0
- package/dist/src/providers/processors/yarn_berry_processor.js +88 -5
- package/dist/src/providers/python_controller.d.ts +10 -3
- package/dist/src/providers/python_controller.js +61 -59
- package/dist/src/providers/python_pip.d.ts +16 -4
- package/dist/src/providers/python_pip.js +51 -58
- package/dist/src/providers/python_pip_pyproject.d.ts +61 -0
- package/dist/src/providers/python_pip_pyproject.js +146 -0
- package/dist/src/providers/python_poetry.d.ts +75 -0
- package/dist/src/providers/python_poetry.js +238 -0
- package/dist/src/providers/python_uv.d.ts +55 -0
- package/dist/src/providers/python_uv.js +227 -0
- package/dist/src/providers/requirements_parser.d.ts +6 -0
- package/dist/src/providers/requirements_parser.js +24 -0
- package/dist/src/providers/rust_cargo.d.ts +53 -0
- package/dist/src/providers/rust_cargo.js +614 -0
- package/dist/src/providers/tree-sitter-gomod.wasm +0 -0
- package/dist/src/providers/tree-sitter-requirements.wasm +0 -0
- package/dist/src/sbom.d.ts +17 -2
- package/dist/src/sbom.js +16 -4
- package/dist/src/tools.d.ts +48 -6
- package/dist/src/tools.js +114 -1
- package/dist/src/workspace.d.ts +70 -0
- package/dist/src/workspace.js +256 -0
- package/package.json +25 -12
|
@@ -1,7 +1,6 @@
|
|
|
1
|
-
import fs from 'node:fs';
|
|
2
1
|
import path from 'node:path';
|
|
3
2
|
import { PackageURL } from 'packageurl-js';
|
|
4
|
-
import { getCustomPath,
|
|
3
|
+
import { getCustomPath, getWrapperPreference, invokeCommand, traverseForWrapper } from "../tools.js";
|
|
5
4
|
/** @typedef {import('../provider').Provider} */
|
|
6
5
|
/** @typedef {import('../provider').Provided} Provided */
|
|
7
6
|
/** @typedef {{name: string, version: string}} Package */
|
|
@@ -26,6 +25,13 @@ export default class Base_Java {
|
|
|
26
25
|
this.globalBinary = globalBinary;
|
|
27
26
|
this.localWrapper = localWrapper;
|
|
28
27
|
}
|
|
28
|
+
/**
|
|
29
|
+
* Returns the package manager name (e.g. mvn, gradle)
|
|
30
|
+
* @returns {string}
|
|
31
|
+
*/
|
|
32
|
+
packageManagerName() {
|
|
33
|
+
return this.globalBinary;
|
|
34
|
+
}
|
|
29
35
|
/**
|
|
30
36
|
* Recursively populates the SBOM instance with the parsed graph
|
|
31
37
|
* @param {string} src - Source dependency to start the calculations from
|
|
@@ -131,7 +137,7 @@ export default class Base_Java {
|
|
|
131
137
|
const toolPath = getCustomPath(this.globalBinary, opts);
|
|
132
138
|
const useWrapper = getWrapperPreference(this.globalBinary, opts);
|
|
133
139
|
if (useWrapper) {
|
|
134
|
-
const wrapper = this.
|
|
140
|
+
const wrapper = traverseForWrapper(manifestDir, this.localWrapper);
|
|
135
141
|
if (wrapper !== undefined) {
|
|
136
142
|
try {
|
|
137
143
|
this._invokeCommand(wrapper, ['--version'], { cwd: manifestDir });
|
|
@@ -156,39 +162,4 @@ export default class Base_Java {
|
|
|
156
162
|
}
|
|
157
163
|
return toolPath;
|
|
158
164
|
}
|
|
159
|
-
/**
|
|
160
|
-
*
|
|
161
|
-
* @param {string} startingManifest - the path of the manifest from which to start searching for the wrapper
|
|
162
|
-
* @param {string} repoRoot - the root of the repository at which point to stop searching for mvnw, derived via git if unset and then fallsback
|
|
163
|
-
* to the root of the drive the manifest is on (assumes absolute path is given)
|
|
164
|
-
* @returns {string|undefined}
|
|
165
|
-
*/
|
|
166
|
-
traverseForWrapper(startingManifest, repoRoot = undefined) {
|
|
167
|
-
const normalizedManifest = this.normalizePath(startingManifest);
|
|
168
|
-
const currentDir = this.normalizePath(path.dirname(normalizedManifest));
|
|
169
|
-
repoRoot = repoRoot || getGitRootDir(currentDir) || path.parse(normalizedManifest).root;
|
|
170
|
-
const wrapperPath = path.join(currentDir, this.localWrapper);
|
|
171
|
-
try {
|
|
172
|
-
fs.accessSync(wrapperPath, fs.constants.X_OK);
|
|
173
|
-
return wrapperPath;
|
|
174
|
-
}
|
|
175
|
-
catch (error) {
|
|
176
|
-
if (error.code === 'ENOENT') {
|
|
177
|
-
const rootDir = path.parse(currentDir).root;
|
|
178
|
-
if (currentDir === repoRoot || currentDir === rootDir) {
|
|
179
|
-
return undefined;
|
|
180
|
-
}
|
|
181
|
-
const parentDir = path.dirname(currentDir);
|
|
182
|
-
if (parentDir === currentDir || parentDir === rootDir) {
|
|
183
|
-
return undefined;
|
|
184
|
-
}
|
|
185
|
-
return this.traverseForWrapper(path.join(parentDir, path.basename(normalizedManifest)), repoRoot);
|
|
186
|
-
}
|
|
187
|
-
throw new Error(`failure searching for ${this.localWrapper}`, { cause: error });
|
|
188
|
-
}
|
|
189
|
-
}
|
|
190
|
-
normalizePath(thePath) {
|
|
191
|
-
const normalized = path.resolve(thePath).normalize();
|
|
192
|
-
return process.platform === 'win32' ? normalized.toLowerCase() : normalized;
|
|
193
|
-
}
|
|
194
165
|
}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
|
-
|
|
2
|
-
/** @typedef {import('../provider
|
|
1
|
+
export type purlType = import("../provider").Provider;
|
|
2
|
+
/** @typedef {import('../provider').Provider} */
|
|
3
|
+
/** @typedef {import('../provider').Provided} Provided */
|
|
3
4
|
/**
|
|
4
5
|
* The ecosystem identifier for JavaScript/npm packages
|
|
5
6
|
* @type {string}
|
|
@@ -46,6 +47,11 @@ export default class Base_javascript {
|
|
|
46
47
|
*/
|
|
47
48
|
protected _cmdName(): string;
|
|
48
49
|
/**
|
|
50
|
+
* Returns the package manager name (e.g. npm, yarn, pnpm, bun)
|
|
51
|
+
* @returns {string} The package manager name
|
|
52
|
+
*/
|
|
53
|
+
packageManagerName(): string;
|
|
54
|
+
/**
|
|
49
55
|
* Returns the command arguments for listing dependencies
|
|
50
56
|
* @returns {Array<string>} The command arguments
|
|
51
57
|
* @abstract
|
|
@@ -66,11 +72,26 @@ export default class Base_javascript {
|
|
|
66
72
|
*/
|
|
67
73
|
isSupported(manifestName: string): boolean;
|
|
68
74
|
/**
|
|
69
|
-
*
|
|
75
|
+
* Walks up the directory tree from manifestDir looking for the lock file.
|
|
76
|
+
* Stops when the lock file is found, when a package.json with a "workspaces"
|
|
77
|
+
* field is encountered without a lock file (workspace root boundary), or
|
|
78
|
+
* when the filesystem root is reached.
|
|
79
|
+
*
|
|
80
|
+
* When TRUSTIFY_DA_WORKSPACE_DIR is set, checks only that directory (no walk-up).
|
|
81
|
+
*
|
|
82
|
+
* @param {string} manifestDir - The directory to start searching from
|
|
83
|
+
* @param {Object} [opts={}] - optional; may contain TRUSTIFY_DA_WORKSPACE_DIR
|
|
84
|
+
* @returns {string|null} The directory containing the lock file, or null
|
|
85
|
+
* @protected
|
|
86
|
+
*/
|
|
87
|
+
protected _isWorkspaceRoot(dir: any): string | null;
|
|
88
|
+
_findLockFileDir(manifestDir: any, opts?: {}): string | null;
|
|
89
|
+
/**
|
|
70
90
|
* @param {string} manifestDir - The base directory where the manifest is located
|
|
91
|
+
* @param {Object} [opts={}] - optional; may contain TRUSTIFY_DA_WORKSPACE_DIR
|
|
71
92
|
* @returns {boolean} True if the lock file exists
|
|
72
93
|
*/
|
|
73
|
-
validateLockFile(manifestDir: string): boolean;
|
|
94
|
+
validateLockFile(manifestDir: string, opts?: any): boolean;
|
|
74
95
|
/**
|
|
75
96
|
* Provides content and content type for stack analysis
|
|
76
97
|
* @param {string} manifestPath - The manifest path or name
|
|
@@ -85,13 +106,20 @@ export default class Base_javascript {
|
|
|
85
106
|
* @returns {Provided} The provided data for component analysis
|
|
86
107
|
*/
|
|
87
108
|
provideComponent(manifestPath: string, opts?: any): Provided;
|
|
109
|
+
/**
|
|
110
|
+
* Read license from manifest (package.json). Reused by npm, pnpm, yarn.
|
|
111
|
+
* @param {string} manifestPath - path to package.json
|
|
112
|
+
* @returns {string|null}
|
|
113
|
+
*/
|
|
114
|
+
readLicenseFromManifest(manifestPath: string): string | null;
|
|
88
115
|
/**
|
|
89
116
|
* Builds the dependency tree for the project
|
|
90
117
|
* @param {boolean} includeTransitive - Whether to include transitive dependencies
|
|
118
|
+
* @param {Object} [opts={}] - Configuration options; when `TRUSTIFY_DA_WORKSPACE_DIR` is set, commands run from workspace root
|
|
91
119
|
* @returns {Object} The dependency tree
|
|
92
120
|
* @protected
|
|
93
121
|
*/
|
|
94
|
-
protected _buildDependencyTree(includeTransitive: boolean): any;
|
|
122
|
+
protected _buildDependencyTree(includeTransitive: boolean, opts?: any): any;
|
|
95
123
|
/**
|
|
96
124
|
* Recursively builds the Sbom from the JSON that npm listing returns
|
|
97
125
|
* @param {Sbom} sbom - The SBOM object to add dependencies to
|
|
@@ -113,6 +141,12 @@ export default class Base_javascript {
|
|
|
113
141
|
*/
|
|
114
142
|
protected _version(): string;
|
|
115
143
|
/**
|
|
144
|
+
* Creates or updates the lock file for the package manager
|
|
145
|
+
* @param {string} manifestDir - Directory containing the manifest file
|
|
146
|
+
* @protected
|
|
147
|
+
*/
|
|
148
|
+
protected _createLockFile(manifestDir: string): void;
|
|
149
|
+
/**
|
|
116
150
|
* Parses the dependency tree output
|
|
117
151
|
* @param {string} output - The output to parse
|
|
118
152
|
* @returns {string} The parsed output
|
|
@@ -121,7 +155,6 @@ export default class Base_javascript {
|
|
|
121
155
|
protected _parseDepTreeOutput(output: string): string;
|
|
122
156
|
#private;
|
|
123
157
|
}
|
|
124
|
-
export type
|
|
125
|
-
export type Provided = import('../provider.js').Provided;
|
|
158
|
+
export type Provided = import("../provider").Provided;
|
|
126
159
|
import Manifest from './manifest.js';
|
|
127
160
|
import Sbom from '../sbom.js';
|
|
@@ -1,11 +1,12 @@
|
|
|
1
1
|
import fs from 'node:fs';
|
|
2
2
|
import os from "node:os";
|
|
3
3
|
import path from 'node:path';
|
|
4
|
+
import { getLicense } from '../license/license_utils.js';
|
|
4
5
|
import Sbom from '../sbom.js';
|
|
5
|
-
import { getCustom, getCustomPath, invokeCommand, toPurl, toPurlFromString } from
|
|
6
|
+
import { getCustom, getCustomPath, invokeCommand, toPurl, toPurlFromString } from '../tools.js';
|
|
6
7
|
import Manifest from './manifest.js';
|
|
7
|
-
/** @typedef {import('../provider
|
|
8
|
-
/** @typedef {import('../provider
|
|
8
|
+
/** @typedef {import('../provider').Provider} */
|
|
9
|
+
/** @typedef {import('../provider').Provided} Provided */
|
|
9
10
|
/**
|
|
10
11
|
* The ecosystem identifier for JavaScript/npm packages
|
|
11
12
|
* @type {string}
|
|
@@ -70,6 +71,13 @@ export default class Base_javascript {
|
|
|
70
71
|
throw new TypeError("_cmdName must be implemented");
|
|
71
72
|
}
|
|
72
73
|
/**
|
|
74
|
+
* Returns the package manager name (e.g. npm, yarn, pnpm, bun)
|
|
75
|
+
* @returns {string} The package manager name
|
|
76
|
+
*/
|
|
77
|
+
packageManagerName() {
|
|
78
|
+
return this._cmdName();
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
73
81
|
* Returns the command arguments for listing dependencies
|
|
74
82
|
* @returns {Array<string>} The command arguments
|
|
75
83
|
* @abstract
|
|
@@ -96,13 +104,63 @@ export default class Base_javascript {
|
|
|
96
104
|
return 'package.json' === manifestName;
|
|
97
105
|
}
|
|
98
106
|
/**
|
|
99
|
-
*
|
|
107
|
+
* Walks up the directory tree from manifestDir looking for the lock file.
|
|
108
|
+
* Stops when the lock file is found, when a package.json with a "workspaces"
|
|
109
|
+
* field is encountered without a lock file (workspace root boundary), or
|
|
110
|
+
* when the filesystem root is reached.
|
|
111
|
+
*
|
|
112
|
+
* When TRUSTIFY_DA_WORKSPACE_DIR is set, checks only that directory (no walk-up).
|
|
113
|
+
*
|
|
114
|
+
* @param {string} manifestDir - The directory to start searching from
|
|
115
|
+
* @param {Object} [opts={}] - optional; may contain TRUSTIFY_DA_WORKSPACE_DIR
|
|
116
|
+
* @returns {string|null} The directory containing the lock file, or null
|
|
117
|
+
* @protected
|
|
118
|
+
*/
|
|
119
|
+
_isWorkspaceRoot(dir) {
|
|
120
|
+
if (fs.existsSync(path.join(dir, 'pnpm-workspace.yaml'))) {
|
|
121
|
+
return true;
|
|
122
|
+
}
|
|
123
|
+
const pkgJsonPath = path.join(dir, 'package.json');
|
|
124
|
+
if (fs.existsSync(pkgJsonPath)) {
|
|
125
|
+
try {
|
|
126
|
+
const content = JSON.parse(fs.readFileSync(pkgJsonPath, 'utf-8'));
|
|
127
|
+
if (content.workspaces) {
|
|
128
|
+
return true;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
catch (_) {
|
|
132
|
+
// ignore parse errors
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
return false;
|
|
136
|
+
}
|
|
137
|
+
_findLockFileDir(manifestDir, opts = {}) {
|
|
138
|
+
const workspaceDir = getCustom('TRUSTIFY_DA_WORKSPACE_DIR', null, opts);
|
|
139
|
+
if (workspaceDir) {
|
|
140
|
+
const dir = path.resolve(workspaceDir);
|
|
141
|
+
return fs.existsSync(path.join(dir, this._lockFileName())) ? dir : null;
|
|
142
|
+
}
|
|
143
|
+
let dir = path.resolve(manifestDir);
|
|
144
|
+
let parent = dir;
|
|
145
|
+
do {
|
|
146
|
+
dir = parent;
|
|
147
|
+
if (fs.existsSync(path.join(dir, this._lockFileName()))) {
|
|
148
|
+
return dir;
|
|
149
|
+
}
|
|
150
|
+
if (this._isWorkspaceRoot(dir)) {
|
|
151
|
+
return null;
|
|
152
|
+
}
|
|
153
|
+
parent = path.dirname(dir);
|
|
154
|
+
} while (parent !== dir);
|
|
155
|
+
return null;
|
|
156
|
+
}
|
|
157
|
+
/**
|
|
100
158
|
* @param {string} manifestDir - The base directory where the manifest is located
|
|
159
|
+
* @param {Object} [opts={}] - optional; may contain TRUSTIFY_DA_WORKSPACE_DIR
|
|
101
160
|
* @returns {boolean} True if the lock file exists
|
|
102
161
|
*/
|
|
103
|
-
validateLockFile(manifestDir) {
|
|
104
|
-
|
|
105
|
-
return fs.existsSync(lock);
|
|
162
|
+
validateLockFile(manifestDir, opts = {}) {
|
|
163
|
+
return this._findLockFileDir(manifestDir, opts) !== null;
|
|
106
164
|
}
|
|
107
165
|
/**
|
|
108
166
|
* Provides content and content type for stack analysis
|
|
@@ -132,17 +190,42 @@ export default class Base_javascript {
|
|
|
132
190
|
contentType: 'application/vnd.cyclonedx+json'
|
|
133
191
|
};
|
|
134
192
|
}
|
|
193
|
+
/**
|
|
194
|
+
* Read license from manifest (package.json). Reused by npm, pnpm, yarn.
|
|
195
|
+
* @param {string} manifestPath - path to package.json
|
|
196
|
+
* @returns {string|null}
|
|
197
|
+
*/
|
|
198
|
+
readLicenseFromManifest(manifestPath) {
|
|
199
|
+
let manifestLicense;
|
|
200
|
+
try {
|
|
201
|
+
const content = JSON.parse(fs.readFileSync(manifestPath, 'utf-8'));
|
|
202
|
+
if (typeof content.license === 'string') {
|
|
203
|
+
manifestLicense = content.license.trim() || null;
|
|
204
|
+
}
|
|
205
|
+
else if (Array.isArray(content.licenses) && content.licenses.length > 0) {
|
|
206
|
+
const first = content.licenses[0];
|
|
207
|
+
const name = first.type || first.name;
|
|
208
|
+
manifestLicense = (typeof name === 'string' ? name.trim() : null);
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
catch {
|
|
212
|
+
manifestLicense = null;
|
|
213
|
+
}
|
|
214
|
+
return getLicense(manifestLicense, manifestPath);
|
|
215
|
+
}
|
|
135
216
|
/**
|
|
136
217
|
* Builds the dependency tree for the project
|
|
137
218
|
* @param {boolean} includeTransitive - Whether to include transitive dependencies
|
|
219
|
+
* @param {Object} [opts={}] - Configuration options; when `TRUSTIFY_DA_WORKSPACE_DIR` is set, commands run from workspace root
|
|
138
220
|
* @returns {Object} The dependency tree
|
|
139
221
|
* @protected
|
|
140
222
|
*/
|
|
141
|
-
_buildDependencyTree(includeTransitive) {
|
|
223
|
+
_buildDependencyTree(includeTransitive, opts = {}) {
|
|
142
224
|
this._version();
|
|
143
|
-
|
|
144
|
-
this
|
|
145
|
-
|
|
225
|
+
const manifestDir = path.dirname(this.#manifest.manifestPath);
|
|
226
|
+
const cmdDir = this._findLockFileDir(manifestDir, opts) || manifestDir;
|
|
227
|
+
this._createLockFile(cmdDir);
|
|
228
|
+
let output = this.#executeListCmd(includeTransitive, cmdDir);
|
|
146
229
|
output = this._parseDepTreeOutput(output);
|
|
147
230
|
return JSON.parse(output);
|
|
148
231
|
}
|
|
@@ -153,14 +236,38 @@ export default class Base_javascript {
|
|
|
153
236
|
* @private
|
|
154
237
|
*/
|
|
155
238
|
#getSBOM(opts = {}) {
|
|
156
|
-
const depsObject = this._buildDependencyTree(true);
|
|
239
|
+
const depsObject = this._buildDependencyTree(true, opts);
|
|
157
240
|
let mainComponent = toPurl(purlType, this.#manifest.name, this.#manifest.version);
|
|
241
|
+
const license = this.readLicenseFromManifest(this.#manifest.manifestPath);
|
|
158
242
|
let sbom = new Sbom();
|
|
159
|
-
sbom.addRoot(mainComponent);
|
|
243
|
+
sbom.addRoot(mainComponent, license);
|
|
160
244
|
this._addDependenciesToSbom(sbom, depsObject);
|
|
245
|
+
this.#ensurePeerAndOptionalDeps(sbom);
|
|
161
246
|
sbom.filterIgnoredDeps(this.#manifest.ignored);
|
|
162
247
|
return sbom.getAsJsonString(opts);
|
|
163
248
|
}
|
|
249
|
+
/**
|
|
250
|
+
* Ensures peer and optional dependencies declared in the manifest are
|
|
251
|
+
* present in the SBOM, even when the package manager does not resolve them
|
|
252
|
+
* (e.g. yarn does not include peer deps in its dependency listing).
|
|
253
|
+
* @param {Sbom} sbom - The SBOM to supplement
|
|
254
|
+
* @private
|
|
255
|
+
*/
|
|
256
|
+
#ensurePeerAndOptionalDeps(sbom) {
|
|
257
|
+
const rootPurl = toPurl(purlType, this.#manifest.name, this.#manifest.version);
|
|
258
|
+
const depSources = [this.#manifest.peerDependencies, this.#manifest.optionalDependencies];
|
|
259
|
+
for (const source of depSources) {
|
|
260
|
+
for (const [name, version] of Object.entries(source)) {
|
|
261
|
+
// Build the purl prefix for exact matching (e.g. "pkg:npm/minimist@"
|
|
262
|
+
// or "pkg:npm/%40hapi/joi@") to avoid substring false positives
|
|
263
|
+
const probe = toPurl(purlType, name, version);
|
|
264
|
+
const purlPrefix = probe.toString().replace(/@[^@]*$/, '@');
|
|
265
|
+
if (!sbom.checkDependsOnByPurlPrefix(rootPurl, purlPrefix)) {
|
|
266
|
+
sbom.addDependency(rootPurl, probe);
|
|
267
|
+
}
|
|
268
|
+
}
|
|
269
|
+
}
|
|
270
|
+
}
|
|
164
271
|
/**
|
|
165
272
|
* Recursively builds the Sbom from the JSON that npm listing returns
|
|
166
273
|
* @param {Sbom} sbom - The SBOM object to add dependencies to
|
|
@@ -168,7 +275,10 @@ export default class Base_javascript {
|
|
|
168
275
|
* @protected
|
|
169
276
|
*/
|
|
170
277
|
_addDependenciesToSbom(sbom, depTree) {
|
|
171
|
-
const dependencies =
|
|
278
|
+
const dependencies = {
|
|
279
|
+
...depTree["dependencies"],
|
|
280
|
+
...depTree["optionalDependencies"],
|
|
281
|
+
};
|
|
172
282
|
Object.entries(dependencies)
|
|
173
283
|
.forEach(entry => {
|
|
174
284
|
const [name, artifact] = entry;
|
|
@@ -204,10 +314,11 @@ export default class Base_javascript {
|
|
|
204
314
|
* @private
|
|
205
315
|
*/
|
|
206
316
|
#getDirectDependencySbom(opts = {}) {
|
|
207
|
-
const depTree = this._buildDependencyTree(false);
|
|
317
|
+
const depTree = this._buildDependencyTree(false, opts);
|
|
208
318
|
let mainComponent = toPurl(purlType, this.#manifest.name, this.#manifest.version);
|
|
319
|
+
const license = this.readLicenseFromManifest(this.#manifest.manifestPath);
|
|
209
320
|
let sbom = new Sbom();
|
|
210
|
-
sbom.addRoot(mainComponent);
|
|
321
|
+
sbom.addRoot(mainComponent, license);
|
|
211
322
|
const rootDeps = this._getRootDependencies(depTree);
|
|
212
323
|
const sortedDepsKeys = Array
|
|
213
324
|
.from(rootDeps.keys())
|
|
@@ -217,6 +328,7 @@ export default class Base_javascript {
|
|
|
217
328
|
const rootPurl = toPurlFromString(sbom.getRoot().purl);
|
|
218
329
|
sbom.addDependency(rootPurl, rootDeps.get(key));
|
|
219
330
|
}
|
|
331
|
+
this.#ensurePeerAndOptionalDeps(sbom);
|
|
220
332
|
sbom.filterIgnoredDeps(this.#manifest.ignored);
|
|
221
333
|
return sbom.getAsJsonString(opts);
|
|
222
334
|
}
|
|
@@ -227,10 +339,14 @@ export default class Base_javascript {
|
|
|
227
339
|
* @protected
|
|
228
340
|
*/
|
|
229
341
|
_getRootDependencies(depTree) {
|
|
230
|
-
|
|
342
|
+
const allDeps = {
|
|
343
|
+
...depTree.dependencies,
|
|
344
|
+
...depTree.optionalDependencies,
|
|
345
|
+
};
|
|
346
|
+
if (Object.keys(allDeps).length === 0) {
|
|
231
347
|
return new Map();
|
|
232
348
|
}
|
|
233
|
-
return new Map(Object.entries(
|
|
349
|
+
return new Map(Object.entries(allDeps).map(([key, value]) => [key, toPurl(purlType, key, value.version)]));
|
|
234
350
|
}
|
|
235
351
|
/**
|
|
236
352
|
* Executes the list command to get dependencies
|
|
@@ -241,7 +357,7 @@ export default class Base_javascript {
|
|
|
241
357
|
*/
|
|
242
358
|
#executeListCmd(includeTransitive, manifestDir) {
|
|
243
359
|
const listArgs = this._listCmdArgs(includeTransitive, manifestDir);
|
|
244
|
-
return this.#invokeCommand(listArgs);
|
|
360
|
+
return this.#invokeCommand(listArgs, { cwd: manifestDir });
|
|
245
361
|
}
|
|
246
362
|
/**
|
|
247
363
|
* Gets the version of the package manager
|
|
@@ -254,19 +370,17 @@ export default class Base_javascript {
|
|
|
254
370
|
/**
|
|
255
371
|
* Creates or updates the lock file for the package manager
|
|
256
372
|
* @param {string} manifestDir - Directory containing the manifest file
|
|
257
|
-
* @
|
|
373
|
+
* @protected
|
|
258
374
|
*/
|
|
259
|
-
|
|
375
|
+
_createLockFile(manifestDir) {
|
|
260
376
|
const originalDir = process.cwd();
|
|
261
377
|
const isWindows = os.platform() === 'win32';
|
|
262
378
|
if (isWindows) {
|
|
263
|
-
// On Windows, --prefix flag doesn't work as expected
|
|
264
|
-
// Instead of installing from the prefix folder, it installs from current working directory
|
|
265
379
|
process.chdir(manifestDir);
|
|
266
380
|
}
|
|
267
381
|
try {
|
|
268
382
|
const args = this._updateLockFileCmdArgs(manifestDir);
|
|
269
|
-
this.#invokeCommand(args);
|
|
383
|
+
this.#invokeCommand(args, { cwd: manifestDir });
|
|
270
384
|
}
|
|
271
385
|
finally {
|
|
272
386
|
if (isWindows) {
|
|
@@ -0,0 +1,158 @@
|
|
|
1
|
+
/** @typedef {{name: string, version: string, children: string[], hashes?: Array<{alg: string, content: string}>}} GraphEntry */
|
|
2
|
+
/** @typedef {{name: string, version: string, dependencies: DepTreeEntry[]}} DepTreeEntry */
|
|
3
|
+
/** @typedef {{directDeps: string[], graph: Map<string, GraphEntry>}} DependencyData */
|
|
4
|
+
/** @typedef {{ecosystem: string, content: string, contentType: string}} Provided */
|
|
5
|
+
export default class Base_pyproject {
|
|
6
|
+
/**
|
|
7
|
+
* @param {string} manifestName
|
|
8
|
+
* @returns {boolean}
|
|
9
|
+
*/
|
|
10
|
+
isSupported(manifestName: string): boolean;
|
|
11
|
+
/**
|
|
12
|
+
* @param {string} manifestDir
|
|
13
|
+
* @param {Object} [opts={}]
|
|
14
|
+
* @returns {boolean}
|
|
15
|
+
*/
|
|
16
|
+
validateLockFile(manifestDir: string, opts?: any): boolean;
|
|
17
|
+
/**
|
|
18
|
+
* Walk up from manifestDir to find the directory containing the lock file.
|
|
19
|
+
* Follows the same pattern as Base_javascript._findLockFileDir().
|
|
20
|
+
* @param {string} manifestDir
|
|
21
|
+
* @param {Object} [opts={}]
|
|
22
|
+
* @returns {string|null}
|
|
23
|
+
* @protected
|
|
24
|
+
*/
|
|
25
|
+
protected _findLockFileDir(manifestDir: string, opts?: any): string | null;
|
|
26
|
+
/**
|
|
27
|
+
* Detect workspace root boundaries.
|
|
28
|
+
* Currently only uv has native workspace support ([tool.uv.workspace] in pyproject.toml).
|
|
29
|
+
* Poetry has no workspace/monorepo support (python-poetry/poetry#2270), so each
|
|
30
|
+
* poetry project is treated independently — see Python_poetry._findLockFileDir().
|
|
31
|
+
* @param {string} dir
|
|
32
|
+
* @returns {boolean}
|
|
33
|
+
* @protected
|
|
34
|
+
*/
|
|
35
|
+
protected _isWorkspaceRoot(dir: string): boolean;
|
|
36
|
+
/**
|
|
37
|
+
* Read project license from pyproject.toml, with fallback to LICENSE file.
|
|
38
|
+
* @param {string} manifestPath
|
|
39
|
+
* @returns {string|null}
|
|
40
|
+
*/
|
|
41
|
+
readLicenseFromManifest(manifestPath: string): string | null;
|
|
42
|
+
/**
|
|
43
|
+
* @param {string} manifest - path to pyproject.toml
|
|
44
|
+
* @param {Object} [opts={}]
|
|
45
|
+
* @returns {Promise<Provided>}
|
|
46
|
+
*/
|
|
47
|
+
provideStack(manifest: string, opts?: any): Promise<Provided>;
|
|
48
|
+
/**
|
|
49
|
+
* @param {string} manifest - path to pyproject.toml
|
|
50
|
+
* @param {Object} [opts={}]
|
|
51
|
+
* @returns {Promise<Provided>}
|
|
52
|
+
*/
|
|
53
|
+
provideComponent(manifest: string, opts?: any): Promise<Provided>;
|
|
54
|
+
/**
|
|
55
|
+
* @returns {string}
|
|
56
|
+
* @protected
|
|
57
|
+
*/
|
|
58
|
+
protected _lockFileName(): string;
|
|
59
|
+
/**
|
|
60
|
+
* @returns {string}
|
|
61
|
+
* @protected
|
|
62
|
+
*/
|
|
63
|
+
protected _cmdName(): string;
|
|
64
|
+
/**
|
|
65
|
+
* Returns the package manager name (e.g. pip, poetry, uv)
|
|
66
|
+
* @returns {string}
|
|
67
|
+
*/
|
|
68
|
+
packageManagerName(): string;
|
|
69
|
+
/**
|
|
70
|
+
* Resolve dependencies using the tool-specific command and parser.
|
|
71
|
+
*
|
|
72
|
+
* @param {string} manifestDir - directory containing the target pyproject.toml
|
|
73
|
+
* @param {string} workspaceDir - workspace root (where the lock file lives);
|
|
74
|
+
* only used by providers that need workspace-level resolution (e.g. uv)
|
|
75
|
+
* @param {object} parsed - parsed pyproject.toml
|
|
76
|
+
* @param {Object} opts
|
|
77
|
+
* @returns {Promise<DependencyData>}
|
|
78
|
+
* @protected
|
|
79
|
+
*/
|
|
80
|
+
protected _getDependencyData(manifestDir: string, workspaceDir: string, parsed: object, opts: any): Promise<DependencyData>;
|
|
81
|
+
/**
|
|
82
|
+
* Canonicalize a Python package name per PEP 503.
|
|
83
|
+
* @param {string} name
|
|
84
|
+
* @returns {string}
|
|
85
|
+
* @protected
|
|
86
|
+
*/
|
|
87
|
+
protected _canonicalize(name: string): string;
|
|
88
|
+
/**
|
|
89
|
+
* Get the project name from pyproject.toml.
|
|
90
|
+
* @param {object} parsed
|
|
91
|
+
* @returns {string|null}
|
|
92
|
+
* @protected
|
|
93
|
+
*/
|
|
94
|
+
protected _getProjectName(parsed: object): string | null;
|
|
95
|
+
/**
|
|
96
|
+
* Get the project version from pyproject.toml.
|
|
97
|
+
* @param {object} parsed
|
|
98
|
+
* @returns {string|null}
|
|
99
|
+
* @protected
|
|
100
|
+
*/
|
|
101
|
+
protected _getProjectVersion(parsed: object): string | null;
|
|
102
|
+
/**
|
|
103
|
+
* Scan raw pyproject.toml text for dependencies with ignore markers.
|
|
104
|
+
* @param {string} manifestPath
|
|
105
|
+
* @returns {Set<string>}
|
|
106
|
+
* @protected
|
|
107
|
+
*/
|
|
108
|
+
protected _getIgnoredDeps(manifestPath: string): Set<string>;
|
|
109
|
+
/**
|
|
110
|
+
* Compute the set of graph nodes reachable from direct deps, excluding ignored.
|
|
111
|
+
* @param {Map<string, GraphEntry>} graph
|
|
112
|
+
* @param {string[]} directDeps
|
|
113
|
+
* @param {Set<string>} ignoredDeps
|
|
114
|
+
* @returns {Set<string>}
|
|
115
|
+
* @protected
|
|
116
|
+
*/
|
|
117
|
+
protected _reachableNodes(graph: Map<string, GraphEntry>, directDeps: string[], ignoredDeps: Set<string>): Set<string>;
|
|
118
|
+
/**
|
|
119
|
+
* @param {string} name
|
|
120
|
+
* @param {string} version
|
|
121
|
+
* @returns {PackageURL}
|
|
122
|
+
* @protected
|
|
123
|
+
*/
|
|
124
|
+
protected _toPurl(name: string, version: string): PackageURL;
|
|
125
|
+
/**
|
|
126
|
+
* Create SBOM json string for a pyproject.toml project.
|
|
127
|
+
* @param {string} manifest - path to pyproject.toml
|
|
128
|
+
* @param {Object} opts
|
|
129
|
+
* @param {boolean} includeTransitive
|
|
130
|
+
* @returns {Promise<string>}
|
|
131
|
+
* @private
|
|
132
|
+
*/
|
|
133
|
+
private _createSbom;
|
|
134
|
+
}
|
|
135
|
+
export type GraphEntry = {
|
|
136
|
+
name: string;
|
|
137
|
+
version: string;
|
|
138
|
+
children: string[];
|
|
139
|
+
hashes?: Array<{
|
|
140
|
+
alg: string;
|
|
141
|
+
content: string;
|
|
142
|
+
}>;
|
|
143
|
+
};
|
|
144
|
+
export type DepTreeEntry = {
|
|
145
|
+
name: string;
|
|
146
|
+
version: string;
|
|
147
|
+
dependencies: DepTreeEntry[];
|
|
148
|
+
};
|
|
149
|
+
export type DependencyData = {
|
|
150
|
+
directDeps: string[];
|
|
151
|
+
graph: Map<string, GraphEntry>;
|
|
152
|
+
};
|
|
153
|
+
export type Provided = {
|
|
154
|
+
ecosystem: string;
|
|
155
|
+
content: string;
|
|
156
|
+
contentType: string;
|
|
157
|
+
};
|
|
158
|
+
import { PackageURL } from 'packageurl-js';
|