@trustify-da/trustify-da-javascript-client 0.3.0-ea.8adb67b → 0.3.0-ea.8eab29b

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.
Files changed (70) hide show
  1. package/README.md +191 -11
  2. package/dist/package.json +19 -6
  3. package/dist/src/analysis.d.ts +16 -0
  4. package/dist/src/analysis.js +74 -80
  5. package/dist/src/batch_opts.d.ts +24 -0
  6. package/dist/src/batch_opts.js +35 -0
  7. package/dist/src/cli.js +241 -8
  8. package/dist/src/cyclone_dx_sbom.d.ts +17 -2
  9. package/dist/src/cyclone_dx_sbom.js +56 -8
  10. package/dist/src/index.d.ts +138 -4
  11. package/dist/src/index.js +356 -8
  12. package/dist/src/license/index.d.ts +28 -0
  13. package/dist/src/license/index.js +100 -0
  14. package/dist/src/license/license_utils.d.ts +40 -0
  15. package/dist/src/license/license_utils.js +134 -0
  16. package/dist/src/license/licenses_api.d.ts +34 -0
  17. package/dist/src/license/licenses_api.js +98 -0
  18. package/dist/src/license/project_license.d.ts +20 -0
  19. package/dist/src/license/project_license.js +62 -0
  20. package/dist/src/oci_image/utils.js +11 -2
  21. package/dist/src/provider.d.ts +18 -5
  22. package/dist/src/provider.js +31 -5
  23. package/dist/src/providers/base_java.d.ts +5 -9
  24. package/dist/src/providers/base_java.js +9 -38
  25. package/dist/src/providers/base_javascript.d.ts +40 -7
  26. package/dist/src/providers/base_javascript.js +138 -24
  27. package/dist/src/providers/base_pyproject.d.ts +158 -0
  28. package/dist/src/providers/base_pyproject.js +322 -0
  29. package/dist/src/providers/golang_gomodules.d.ts +29 -12
  30. package/dist/src/providers/golang_gomodules.js +176 -121
  31. package/dist/src/providers/gomod_parser.d.ts +4 -0
  32. package/dist/src/providers/gomod_parser.js +16 -0
  33. package/dist/src/providers/java_gradle.d.ts +25 -0
  34. package/dist/src/providers/java_gradle.js +128 -4
  35. package/dist/src/providers/java_maven.d.ts +16 -1
  36. package/dist/src/providers/java_maven.js +125 -5
  37. package/dist/src/providers/javascript_bun.d.ts +10 -0
  38. package/dist/src/providers/javascript_bun.js +100 -0
  39. package/dist/src/providers/javascript_npm.d.ts +1 -0
  40. package/dist/src/providers/javascript_npm.js +21 -0
  41. package/dist/src/providers/javascript_pnpm.d.ts +1 -1
  42. package/dist/src/providers/javascript_pnpm.js +8 -4
  43. package/dist/src/providers/manifest.d.ts +2 -0
  44. package/dist/src/providers/manifest.js +22 -4
  45. package/dist/src/providers/marker_evaluator.d.ts +14 -0
  46. package/dist/src/providers/marker_evaluator.js +191 -0
  47. package/dist/src/providers/processors/yarn_berry_processor.js +88 -5
  48. package/dist/src/providers/python_controller.d.ts +10 -3
  49. package/dist/src/providers/python_controller.js +118 -62
  50. package/dist/src/providers/python_pip.d.ts +16 -4
  51. package/dist/src/providers/python_pip.js +51 -58
  52. package/dist/src/providers/python_pip_pyproject.d.ts +61 -0
  53. package/dist/src/providers/python_pip_pyproject.js +146 -0
  54. package/dist/src/providers/python_poetry.d.ts +75 -0
  55. package/dist/src/providers/python_poetry.js +238 -0
  56. package/dist/src/providers/python_uv.d.ts +55 -0
  57. package/dist/src/providers/python_uv.js +227 -0
  58. package/dist/src/providers/requirements_parser.d.ts +6 -0
  59. package/dist/src/providers/requirements_parser.js +24 -0
  60. package/dist/src/providers/rust_cargo.d.ts +56 -0
  61. package/dist/src/providers/rust_cargo.js +641 -0
  62. package/dist/src/providers/tree-sitter-gomod.wasm +0 -0
  63. package/dist/src/providers/tree-sitter-requirements.wasm +0 -0
  64. package/dist/src/sbom.d.ts +17 -2
  65. package/dist/src/sbom.js +16 -4
  66. package/dist/src/tools.d.ts +44 -0
  67. package/dist/src/tools.js +114 -1
  68. package/dist/src/workspace.d.ts +70 -0
  69. package/dist/src/workspace.js +256 -0
  70. package/package.json +20 -7
@@ -1,5 +1,6 @@
1
- /** @typedef {import('../provider.js').Provider} Provider */
2
- /** @typedef {import('../provider.js').Provided} Provided */
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
- * Checks if a required lock file exists in the same path as the manifest
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 Provider = import("../provider.js").Provider;
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 "../tools.js";
6
+ import { getCustom, getCustomPath, invokeCommand, toPurl, toPurlFromString } from '../tools.js';
6
7
  import Manifest from './manifest.js';
7
- /** @typedef {import('../provider.js').Provider} Provider */
8
- /** @typedef {import('../provider.js').Provided} Provided */
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
- * Checks if a required lock file exists in the same path as the manifest
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
- const lock = path.join(manifestDir, this._lockFileName());
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
- let manifestDir = path.dirname(this.#manifest.manifestPath);
144
- this.#createLockFile(manifestDir);
145
- let output = this.#executeListCmd(includeTransitive, manifestDir);
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 = depTree["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
- if (!depTree.dependencies) {
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(depTree.dependencies).map(([key, value]) => [key, toPurl(purlType, key, value.version)]));
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
- * @private
373
+ * @protected
258
374
  */
259
- #createLockFile(manifestDir) {
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';