@trustify-da/trustify-da-javascript-client 0.3.0-ea.e12bc82 → 0.3.0-ea.e645720
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 +23 -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 +352 -7
- 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 +17 -5
- package/dist/src/provider.js +29 -5
- package/dist/src/providers/base_java.d.ts +3 -14
- package/dist/src/providers/base_java.js +2 -38
- package/dist/src/providers/base_javascript.d.ts +29 -7
- package/dist/src/providers/base_javascript.js +129 -22
- package/dist/src/providers/base_pyproject.d.ts +153 -0
- package/dist/src/providers/base_pyproject.js +315 -0
- package/dist/src/providers/golang_gomodules.d.ts +29 -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_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 +15 -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 +144 -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 +52 -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 +24 -12
|
@@ -1,10 +1,12 @@
|
|
|
1
1
|
import fs from 'node:fs';
|
|
2
2
|
import path from 'node:path';
|
|
3
|
-
import { EOL } from "os";
|
|
4
3
|
import { PackageURL } from 'packageurl-js';
|
|
4
|
+
import { readLicenseFile } from '../license/license_utils.js';
|
|
5
5
|
import Sbom from '../sbom.js';
|
|
6
6
|
import { getCustom, getCustomPath, invokeCommand } from "../tools.js";
|
|
7
|
-
|
|
7
|
+
import { filterManifestPathsByDiscoveryIgnore, resolveWorkspaceDiscoveryIgnore } from '../workspace.js';
|
|
8
|
+
import { getParser, getRequireQuery } from './gomod_parser.js';
|
|
9
|
+
export default { isSupported, validateLockFile, provideComponent, provideStack, readLicenseFromManifest };
|
|
8
10
|
/** @typedef {import('../provider').Provider} */
|
|
9
11
|
/** @typedef {import('../provider').Provided} Provided */
|
|
10
12
|
/** @typedef {{name: string, version: string}} Package */
|
|
@@ -12,43 +14,50 @@ export default { isSupported, validateLockFile, provideComponent, provideStack }
|
|
|
12
14
|
/**
|
|
13
15
|
* @type {string} ecosystem for npm-npm is 'maven'
|
|
14
16
|
* @private
|
|
15
|
-
|
|
17
|
+
*/
|
|
16
18
|
const ecosystem = 'golang';
|
|
17
19
|
const defaultMainModuleVersion = "v0.0.0";
|
|
18
20
|
/**
|
|
19
|
-
* @param {string} manifestName
|
|
20
|
-
* @returns {boolean}
|
|
21
|
-
|
|
21
|
+
* @param {string} manifestName the subject manifest name-type
|
|
22
|
+
* @returns {boolean} return true if `pom.xml` is the manifest name-type
|
|
23
|
+
*/
|
|
22
24
|
function isSupported(manifestName) {
|
|
23
25
|
return 'go.mod' === manifestName;
|
|
24
26
|
}
|
|
25
27
|
/**
|
|
26
|
-
*
|
|
28
|
+
* Go modules have no standard license field in go.mod
|
|
29
|
+
* @param {string} manifestPath path to go.mod
|
|
30
|
+
* @returns {string|null}
|
|
31
|
+
*/
|
|
32
|
+
// eslint-disable-next-line no-unused-vars
|
|
33
|
+
function readLicenseFromManifest(manifestPath) { return readLicenseFile(manifestPath); }
|
|
34
|
+
/**
|
|
35
|
+
* @param {string} manifestDir the directory where the manifest lies
|
|
27
36
|
*/
|
|
28
37
|
function validateLockFile() { return true; }
|
|
29
38
|
/**
|
|
30
39
|
* Provide content and content type for maven-maven stack analysis.
|
|
31
|
-
* @param {string} manifest
|
|
32
|
-
* @param {{}} [opts={}]
|
|
33
|
-
* @returns {Provided}
|
|
40
|
+
* @param {string} manifest the manifest path or name
|
|
41
|
+
* @param {{}} [opts={}] optional various options to pass along the application
|
|
42
|
+
* @returns {Promise<Provided>}
|
|
34
43
|
*/
|
|
35
|
-
function provideStack(manifest, opts = {}) {
|
|
44
|
+
async function provideStack(manifest, opts = {}) {
|
|
36
45
|
return {
|
|
37
46
|
ecosystem,
|
|
38
|
-
content: getSBOM(manifest, opts, true),
|
|
47
|
+
content: await getSBOM(manifest, opts, true),
|
|
39
48
|
contentType: 'application/vnd.cyclonedx+json'
|
|
40
49
|
};
|
|
41
50
|
}
|
|
42
51
|
/**
|
|
43
52
|
* Provide content and content type for maven-maven component analysis.
|
|
44
|
-
* @param {string} manifest
|
|
45
|
-
* @param {{}} [opts={}]
|
|
46
|
-
* @returns {Provided}
|
|
53
|
+
* @param {string} manifest path to go.mod for component report
|
|
54
|
+
* @param {{}} [opts={}] optional various options to pass along the application
|
|
55
|
+
* @returns {Promise<Provided>}
|
|
47
56
|
*/
|
|
48
|
-
function provideComponent(manifest, opts = {}) {
|
|
57
|
+
async function provideComponent(manifest, opts = {}) {
|
|
49
58
|
return {
|
|
50
59
|
ecosystem,
|
|
51
|
-
content: getSBOM(manifest, opts, false),
|
|
60
|
+
content: await getSBOM(manifest, opts, false),
|
|
52
61
|
contentType: 'application/vnd.cyclonedx+json'
|
|
53
62
|
};
|
|
54
63
|
}
|
|
@@ -69,50 +78,52 @@ function getChildVertexFromEdge(edge) {
|
|
|
69
78
|
return edge.split(" ")[1];
|
|
70
79
|
}
|
|
71
80
|
/**
|
|
72
|
-
*
|
|
73
|
-
*
|
|
74
|
-
*
|
|
81
|
+
* Check whether a require_spec has a valid exhortignore marker.
|
|
82
|
+
* For direct dependencies: `//exhortignore` or `// exhortignore`
|
|
83
|
+
* For indirect dependencies: `// indirect; exhortignore` (semicolon-separated)
|
|
84
|
+
* @param {import('web-tree-sitter').SyntaxNode} specNode
|
|
85
|
+
* @return {boolean}
|
|
75
86
|
*/
|
|
76
|
-
function
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
87
|
+
function hasExhortIgnore(specNode) {
|
|
88
|
+
// Ideally this would be the following tree-sitter query instead, but for some
|
|
89
|
+
// reason it throws an error here but not in the playground.
|
|
90
|
+
// (require_spec) ((module_path) @path (version) (comment) @comment (#match? @comment "^//.*exhortignore"))
|
|
91
|
+
// QueryError: Bad pattern structure at offset 53: '(comment) @comment (#match? @comment "^//.*exhortignore")) @spec'...
|
|
92
|
+
let comments = specNode.children.filter(c => c.type === 'comment');
|
|
93
|
+
for (let comment of comments) {
|
|
94
|
+
let text = comment.text;
|
|
95
|
+
if (/^\/\/\s*indirect;\s*exhortignore/.test(text)) {
|
|
96
|
+
return true;
|
|
97
|
+
}
|
|
98
|
+
if (/^\/\/\s*exhortignore/.test(text)) {
|
|
99
|
+
return true;
|
|
88
100
|
}
|
|
89
101
|
}
|
|
90
|
-
return
|
|
91
|
-
}
|
|
92
|
-
/**
|
|
93
|
-
* extract package name from go.mod line that contains exhortignore comment.
|
|
94
|
-
* @param line a row contains exhortignore as part of a comment
|
|
95
|
-
* @return {string} the full package name + group/namespace + version
|
|
96
|
-
* @private
|
|
97
|
-
*/
|
|
98
|
-
function extractPackageName(line) {
|
|
99
|
-
let trimmedRow = line.trim();
|
|
100
|
-
let firstRemarkNotationOccurrence = trimmedRow.indexOf("//");
|
|
101
|
-
return trimmedRow.substring(0, firstRemarkNotationOccurrence).trim();
|
|
102
|
+
return false;
|
|
102
103
|
}
|
|
103
104
|
/**
|
|
104
105
|
*
|
|
105
|
-
* @param {string
|
|
106
|
-
* @
|
|
106
|
+
* @param {string} manifestContent go.mod file contents
|
|
107
|
+
* @param {import('web-tree-sitter').Parser} parser
|
|
108
|
+
* @param {import('web-tree-sitter').Query} requireQuery
|
|
109
|
+
* @return {PackageURL[]} list of ignored dependencies
|
|
107
110
|
*/
|
|
108
|
-
function getIgnoredDeps(
|
|
109
|
-
let
|
|
110
|
-
|
|
111
|
-
|
|
111
|
+
function getIgnoredDeps(manifestContent, parser, requireQuery) {
|
|
112
|
+
let tree = parser.parse(manifestContent);
|
|
113
|
+
return requireQuery.matches(tree.rootNode)
|
|
114
|
+
.filter(match => {
|
|
115
|
+
let specNode = match.captures.find(c => c.name === 'spec').node;
|
|
116
|
+
return hasExhortIgnore(specNode);
|
|
117
|
+
})
|
|
118
|
+
.map(match => {
|
|
119
|
+
let name = match.captures.find(c => c.name === 'name').node.text;
|
|
120
|
+
let version = match.captures.find(c => c.name === 'version').node.text;
|
|
121
|
+
return toPurl(`${name} ${version}`, /[ ]{1,3}/);
|
|
122
|
+
});
|
|
112
123
|
}
|
|
113
124
|
/**
|
|
114
125
|
*
|
|
115
|
-
* @param {[
|
|
126
|
+
* @param {PackageURL[]} allIgnoredDeps list of purls of all dependencies that should be ignored
|
|
116
127
|
* @param {PackageURL} purl object to be checked if needed to be ignored
|
|
117
128
|
* @return {boolean}
|
|
118
129
|
*/
|
|
@@ -131,59 +142,29 @@ function enforceRemovingIgnoredDepsInCaseOfAutomaticVersionUpdate(ignoredDeps, s
|
|
|
131
142
|
}
|
|
132
143
|
/**
|
|
133
144
|
*
|
|
134
|
-
* @param {
|
|
135
|
-
* @param {
|
|
136
|
-
* @
|
|
145
|
+
* @param {string} manifestContent go.mod file contents
|
|
146
|
+
* @param {import('web-tree-sitter').Parser} parser
|
|
147
|
+
* @param {import('web-tree-sitter').Query} requireQuery
|
|
148
|
+
* @return {string[]} all dependencies from go.mod file as "name version" strings
|
|
137
149
|
*/
|
|
138
|
-
function collectAllDepsFromManifest(
|
|
139
|
-
let
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
while (requirePositionObject.index > -1) {
|
|
146
|
-
let depsInsideRequirementsBlock = currentSegmentOfGoMod.substring(requirePositionObject.index + requirePositionObject.startingOffeset).trim();
|
|
147
|
-
let endOfBlockIndex = depsInsideRequirementsBlock.indexOf(")");
|
|
148
|
-
let currentIndex = 0;
|
|
149
|
-
while (currentIndex < endOfBlockIndex) {
|
|
150
|
-
let endOfLinePosition = depsInsideRequirementsBlock.indexOf(EOL, currentIndex);
|
|
151
|
-
let dependency = depsInsideRequirementsBlock.substring(currentIndex, endOfLinePosition);
|
|
152
|
-
result.push(dependency.trim());
|
|
153
|
-
currentIndex = endOfLinePosition + 1;
|
|
154
|
-
}
|
|
155
|
-
currentSegmentOfGoMod = currentSegmentOfGoMod.substring(endOfBlockIndex + 1).trim();
|
|
156
|
-
requirePositionObject = decideRequireBlockIndex(currentSegmentOfGoMod);
|
|
157
|
-
}
|
|
158
|
-
function decideRequireBlockIndex(goMod) {
|
|
159
|
-
let object = {};
|
|
160
|
-
let index = goMod.indexOf("require(");
|
|
161
|
-
object.startingOffeset = "require(".length;
|
|
162
|
-
if (index === -1) {
|
|
163
|
-
index = goMod.indexOf("require (");
|
|
164
|
-
object.startingOffeset = "require (".length;
|
|
165
|
-
if (index === -1) {
|
|
166
|
-
index = goMod.indexOf("require (");
|
|
167
|
-
object.startingOffeset = "require (".length;
|
|
168
|
-
}
|
|
169
|
-
}
|
|
170
|
-
object.index = index;
|
|
171
|
-
return object;
|
|
172
|
-
}
|
|
173
|
-
return result;
|
|
150
|
+
function collectAllDepsFromManifest(manifestContent, parser, requireQuery) {
|
|
151
|
+
let tree = parser.parse(manifestContent);
|
|
152
|
+
return requireQuery.matches(tree.rootNode).map(match => {
|
|
153
|
+
let name = match.captures.find(c => c.name === 'name').node.text;
|
|
154
|
+
let version = match.captures.find(c => c.name === 'version').node.text;
|
|
155
|
+
return `${name} ${version}`;
|
|
156
|
+
});
|
|
174
157
|
}
|
|
175
158
|
/**
|
|
176
159
|
*
|
|
177
160
|
* @param {string} rootElementName the rootElementName element of go mod graph, to compare only direct deps from go mod graph against go.mod manifest
|
|
178
|
-
* @param{[
|
|
179
|
-
* @param {string
|
|
161
|
+
* @param {string[]} goModGraphOutputRows the goModGraphOutputRows from go mod graph' output
|
|
162
|
+
* @param {string} manifestContent go.mod file contents
|
|
180
163
|
* @private
|
|
181
164
|
*/
|
|
182
|
-
function performManifestVersionsCheck(rootElementName, goModGraphOutputRows,
|
|
183
|
-
let goMod = fs.readFileSync(manifest).toString().trim();
|
|
184
|
-
let lines = goMod.split(getLineSeparatorGolang());
|
|
165
|
+
function performManifestVersionsCheck(rootElementName, goModGraphOutputRows, manifestContent, parser, requireQuery) {
|
|
185
166
|
let comparisonLines = goModGraphOutputRows.filter((line) => line.startsWith(rootElementName)).map((line) => getChildVertexFromEdge(line));
|
|
186
|
-
let manifestDeps = collectAllDepsFromManifest(
|
|
167
|
+
let manifestDeps = collectAllDepsFromManifest(manifestContent, parser, requireQuery);
|
|
187
168
|
try {
|
|
188
169
|
comparisonLines.forEach((dependency) => {
|
|
189
170
|
let parts = dependency.split("@");
|
|
@@ -195,7 +176,7 @@ function performManifestVersionsCheck(rootElementName, goModGraphOutputRows, man
|
|
|
195
176
|
let currentVersion = components[1];
|
|
196
177
|
if (currentDepName === depName) {
|
|
197
178
|
if (currentVersion !== version) {
|
|
198
|
-
throw new Error(`
|
|
179
|
+
throw new Error(`version mismatch for dependency "${depName}", manifest version=${currentVersion}, installed version=${version}, if you want to allow version mismatch for analysis between installed and requested packages, set environment variable/setting MATCH_MANIFEST_VERSIONS=false`);
|
|
199
180
|
}
|
|
200
181
|
}
|
|
201
182
|
});
|
|
@@ -211,10 +192,10 @@ function performManifestVersionsCheck(rootElementName, goModGraphOutputRows, man
|
|
|
211
192
|
* @param {string} manifest - path for go.mod
|
|
212
193
|
* @param {{}} [opts={}] - optional various options to pass along the application
|
|
213
194
|
* @param {boolean} includeTransitive - whether the sbom should contain transitive dependencies of the main module or not.
|
|
214
|
-
* @returns {string} the SBOM json content
|
|
195
|
+
* @returns {Promise<string>} the SBOM json content
|
|
215
196
|
* @private
|
|
216
197
|
*/
|
|
217
|
-
function getSBOM(manifest, opts = {}, includeTransitive) {
|
|
198
|
+
async function getSBOM(manifest, opts = {}, includeTransitive) {
|
|
218
199
|
// get custom goBin path
|
|
219
200
|
let goBin = getCustomPath('go', opts);
|
|
220
201
|
// verify goBin is accessible
|
|
@@ -240,17 +221,29 @@ function getSBOM(manifest, opts = {}, includeTransitive) {
|
|
|
240
221
|
catch (error) {
|
|
241
222
|
throw new Error('failed to determine root module name', { cause: error });
|
|
242
223
|
}
|
|
243
|
-
let
|
|
224
|
+
let manifestContent = fs.readFileSync(manifest).toString();
|
|
225
|
+
let [parser, requireQuery] = await Promise.all([getParser(), getRequireQuery()]);
|
|
226
|
+
let ignoredDeps = getIgnoredDeps(manifestContent, parser, requireQuery);
|
|
244
227
|
let allIgnoredDeps = ignoredDeps.map((dep) => dep.toString());
|
|
245
228
|
let sbom = new Sbom();
|
|
246
229
|
let rows = goGraphOutput.split(getLineSeparatorGolang()).filter(line => !line.includes(' go@'));
|
|
247
|
-
let root =
|
|
230
|
+
let root = goModEditOutput['Module']['Path'];
|
|
231
|
+
// Build set of direct dependency paths from go mod edit -json
|
|
232
|
+
let directDepPaths = new Set();
|
|
233
|
+
if (goModEditOutput['Require']) {
|
|
234
|
+
goModEditOutput['Require'].forEach(req => {
|
|
235
|
+
if (!req['Indirect']) {
|
|
236
|
+
directDepPaths.add(req['Path']);
|
|
237
|
+
}
|
|
238
|
+
});
|
|
239
|
+
}
|
|
248
240
|
let matchManifestVersions = getCustom("MATCH_MANIFEST_VERSIONS", "false", opts);
|
|
249
241
|
if (matchManifestVersions === "true") {
|
|
250
|
-
performManifestVersionsCheck(root, rows,
|
|
242
|
+
performManifestVersionsCheck(root, rows, manifestContent, parser, requireQuery);
|
|
251
243
|
}
|
|
252
244
|
const mainModule = toPurl(root, "@");
|
|
253
|
-
|
|
245
|
+
const license = readLicenseFromManifest(manifest);
|
|
246
|
+
sbom.addRoot(mainModule, license);
|
|
254
247
|
const exhortGoMvsLogicEnabled = getCustom("TRUSTIFY_DA_GO_MVS_LOGIC_ENABLED", "true", opts);
|
|
255
248
|
if (includeTransitive && exhortGoMvsLogicEnabled === "true") {
|
|
256
249
|
rows = getFinalPackagesVersionsForModule(rows, manifest, goBin);
|
|
@@ -264,7 +257,11 @@ function getSBOM(manifest, opts = {}, includeTransitive) {
|
|
|
264
257
|
currentParent = getParentVertexFromEdge(row);
|
|
265
258
|
source = toPurl(currentParent, "@");
|
|
266
259
|
}
|
|
267
|
-
let
|
|
260
|
+
let child = getChildVertexFromEdge(row);
|
|
261
|
+
let target = toPurl(child, "@");
|
|
262
|
+
if (getParentVertexFromEdge(row) === root && !directDepPaths.has(getPackageName(child))) {
|
|
263
|
+
return;
|
|
264
|
+
}
|
|
268
265
|
sbom.addDependency(source, target);
|
|
269
266
|
});
|
|
270
267
|
// at the end, filter out all ignored dependencies including versions.
|
|
@@ -274,10 +271,12 @@ function getSBOM(manifest, opts = {}, includeTransitive) {
|
|
|
274
271
|
else {
|
|
275
272
|
let directDependencies = rows.filter(row => row.startsWith(root));
|
|
276
273
|
directDependencies.forEach(pair => {
|
|
277
|
-
let
|
|
278
|
-
let
|
|
279
|
-
if (dependencyNotIgnored(ignoredDeps,
|
|
280
|
-
|
|
274
|
+
let child = getChildVertexFromEdge(pair);
|
|
275
|
+
let target = toPurl(child, "@");
|
|
276
|
+
if (dependencyNotIgnored(ignoredDeps, target)) {
|
|
277
|
+
if (directDepPaths.has(getPackageName(child))) {
|
|
278
|
+
sbom.addDependency(mainModule, target);
|
|
279
|
+
}
|
|
281
280
|
}
|
|
282
281
|
});
|
|
283
282
|
enforceRemovingIgnoredDepsInCaseOfAutomaticVersionUpdate(ignoredDeps, sbom);
|
|
@@ -287,7 +286,7 @@ function getSBOM(manifest, opts = {}, includeTransitive) {
|
|
|
287
286
|
/**
|
|
288
287
|
* Utility function for creating Purl String
|
|
289
288
|
|
|
290
|
-
* @param {string
|
|
289
|
+
* @param {string} dependency the name of the artifact, can include a namespace(group) or not - namespace/artifactName.
|
|
291
290
|
* @param {RegExp} delimiter delimiter between name of dependency and version
|
|
292
291
|
* @private
|
|
293
292
|
* @returns {PackageURL|null} PackageUrl Object ready to be used in SBOM
|
|
@@ -312,12 +311,12 @@ function toPurl(dependency, delimiter) {
|
|
|
312
311
|
}
|
|
313
312
|
return pkg;
|
|
314
313
|
}
|
|
315
|
-
/** This function gets rows from go mod graph
|
|
314
|
+
/** This function gets rows from go mod graph, and go.mod graph, and selecting for all
|
|
316
315
|
* packages the has more than one minor the final versions as selected by golang MVS algorithm.
|
|
317
|
-
* @param {[
|
|
316
|
+
* @param {string[]} rows all the rows from go modules dependency tree
|
|
318
317
|
* @param {string} manifestPath the path of the go.mod file
|
|
319
318
|
* @param {string} path to go binary
|
|
320
|
-
* @return {[
|
|
319
|
+
* @return {string[]} rows that contains final versions.
|
|
321
320
|
*/
|
|
322
321
|
function getFinalPackagesVersionsForModule(rows, manifestPath, goBin) {
|
|
323
322
|
let manifestDir = path.dirname(manifestPath);
|
|
@@ -330,13 +329,7 @@ function getFinalPackagesVersionsForModule(rows, manifestPath, goBin) {
|
|
|
330
329
|
catch (error) {
|
|
331
330
|
throw new Error('failed to list all modules', { cause: error });
|
|
332
331
|
}
|
|
333
|
-
let finalVersionModules =
|
|
334
|
-
finalVersionsForAllModules.split(getLineSeparatorGolang()).filter(string => string.trim() !== "")
|
|
335
|
-
.filter(string => string.trim().split(" ").length === 2)
|
|
336
|
-
.forEach((dependency) => {
|
|
337
|
-
let dep = dependency.split(" ");
|
|
338
|
-
finalVersionModules[dep[0]] = dep[1];
|
|
339
|
-
});
|
|
332
|
+
let finalVersionModules = parseModuleVersions(finalVersionsForAllModules);
|
|
340
333
|
let finalVersionModulesArray = new Array();
|
|
341
334
|
rows.filter(string => string.trim() !== "").forEach(module => {
|
|
342
335
|
let child = getChildVertexFromEdge(module);
|
|
@@ -372,7 +365,7 @@ function getFinalPackagesVersionsForModule(rows, manifestPath, goBin) {
|
|
|
372
365
|
/**
|
|
373
366
|
*
|
|
374
367
|
* @param {string} fullPackage - full package with its name and version
|
|
375
|
-
* @return
|
|
368
|
+
* @return {string} package name only
|
|
376
369
|
* @private
|
|
377
370
|
*/
|
|
378
371
|
function getPackageName(fullPackage) {
|
|
@@ -390,14 +383,76 @@ function isSpecialGoModule(moduleName) {
|
|
|
390
383
|
/**
|
|
391
384
|
*
|
|
392
385
|
* @param {string} fullPackage - full package with its name and version
|
|
393
|
-
* @return
|
|
386
|
+
* @return {string|undefined} package version only
|
|
394
387
|
* @private
|
|
395
388
|
*/
|
|
396
389
|
function getVersionOfPackage(fullPackage) {
|
|
397
390
|
let parts = fullPackage.split("@");
|
|
398
391
|
return parts.length > 1 ? parts[1] : undefined;
|
|
399
392
|
}
|
|
393
|
+
function parseModuleVersions(goListOutput) {
|
|
394
|
+
let modules = {};
|
|
395
|
+
goListOutput.split(getLineSeparatorGolang()).filter(string => string.trim() !== "")
|
|
396
|
+
.forEach((line) => {
|
|
397
|
+
let parts = line.trim().split(" ");
|
|
398
|
+
if (parts.length === 2) {
|
|
399
|
+
modules[parts[0]] = parts[1];
|
|
400
|
+
}
|
|
401
|
+
else if (parts.length >= 4 && parts[2] === "=>") {
|
|
402
|
+
modules[parts[0]] = parts[parts.length - 1];
|
|
403
|
+
}
|
|
404
|
+
});
|
|
405
|
+
return modules;
|
|
406
|
+
}
|
|
400
407
|
function getLineSeparatorGolang() {
|
|
401
408
|
let reg = /\n|\r\n/;
|
|
402
409
|
return reg;
|
|
403
410
|
}
|
|
411
|
+
/**
|
|
412
|
+
* Discover all go.mod manifest paths in a Go workspace.
|
|
413
|
+
* Uses `go work edit -json` to get workspace members.
|
|
414
|
+
*
|
|
415
|
+
* @param {string} workspaceRoot - Absolute or relative path to workspace root (must contain go.work)
|
|
416
|
+
* @param {import('../index.js').Options} [opts={}]
|
|
417
|
+
* @returns {Promise<string[]>} Paths to go.mod files (absolute)
|
|
418
|
+
*/
|
|
419
|
+
export async function discoverGoWorkspaceModules(workspaceRoot, opts = {}) {
|
|
420
|
+
const root = path.resolve(workspaceRoot);
|
|
421
|
+
const goWork = path.join(root, 'go.work');
|
|
422
|
+
if (!fs.existsSync(goWork)) {
|
|
423
|
+
return [];
|
|
424
|
+
}
|
|
425
|
+
const goBin = getCustomPath('go', opts);
|
|
426
|
+
let output;
|
|
427
|
+
try {
|
|
428
|
+
output = invokeCommand(goBin, ['work', 'edit', '-json', goWork], { cwd: root });
|
|
429
|
+
}
|
|
430
|
+
catch {
|
|
431
|
+
return [];
|
|
432
|
+
}
|
|
433
|
+
let workspace;
|
|
434
|
+
try {
|
|
435
|
+
workspace = JSON.parse(output.toString().trim());
|
|
436
|
+
}
|
|
437
|
+
catch {
|
|
438
|
+
return [];
|
|
439
|
+
}
|
|
440
|
+
const useEntries = workspace.Use || [];
|
|
441
|
+
if (useEntries.length === 0) {
|
|
442
|
+
return [];
|
|
443
|
+
}
|
|
444
|
+
const manifestPaths = [];
|
|
445
|
+
for (const entry of useEntries) {
|
|
446
|
+
const diskPath = entry.DiskPath;
|
|
447
|
+
if (!diskPath) {
|
|
448
|
+
continue;
|
|
449
|
+
}
|
|
450
|
+
const moduleDir = path.resolve(root, diskPath);
|
|
451
|
+
const goMod = path.join(moduleDir, 'go.mod');
|
|
452
|
+
if (fs.existsSync(goMod)) {
|
|
453
|
+
manifestPaths.push(goMod);
|
|
454
|
+
}
|
|
455
|
+
}
|
|
456
|
+
const ignorePatterns = resolveWorkspaceDiscoveryIgnore(opts);
|
|
457
|
+
return filterManifestPathsByDiscoveryIgnore(manifestPaths, root, ignorePatterns);
|
|
458
|
+
}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { readFile } from 'node:fs/promises';
|
|
2
|
+
import { Language, Parser, Query } from 'web-tree-sitter';
|
|
3
|
+
const wasmUrl = new URL('./tree-sitter-gomod.wasm', import.meta.url);
|
|
4
|
+
async function init() {
|
|
5
|
+
await Parser.init();
|
|
6
|
+
const wasmBytes = new Uint8Array(await readFile(wasmUrl));
|
|
7
|
+
return await Language.load(wasmBytes);
|
|
8
|
+
}
|
|
9
|
+
export async function getParser() {
|
|
10
|
+
const language = await init();
|
|
11
|
+
return new Parser().setLanguage(language);
|
|
12
|
+
}
|
|
13
|
+
export async function getRequireQuery() {
|
|
14
|
+
const language = await init();
|
|
15
|
+
return new Query(language, '(require_spec (module_path) @name (version) @version) @spec');
|
|
16
|
+
}
|
|
@@ -1,3 +1,22 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Discover all build.gradle[.kts] manifest paths in a Gradle multi-project build.
|
|
3
|
+
* Uses a custom init script to get structured project listing.
|
|
4
|
+
*
|
|
5
|
+
* @param {string} workspaceRoot - Absolute or relative path to workspace root (must contain settings.gradle[.kts])
|
|
6
|
+
* @param {import('../index.js').Options} [opts={}]
|
|
7
|
+
* @returns {Promise<string[]>} Paths to build.gradle[.kts] files (absolute)
|
|
8
|
+
*/
|
|
9
|
+
export function discoverGradleSubprojects(workspaceRoot: string, opts?: import("../index.js").Options): Promise<string[]>;
|
|
10
|
+
/**
|
|
11
|
+
* Parse the structured output from the Gradle init script.
|
|
12
|
+
*
|
|
13
|
+
* @param {string} raw - Raw stdout from gradle
|
|
14
|
+
* @returns {{ path: string, dir: string }[]}
|
|
15
|
+
*/
|
|
16
|
+
export function parseGradleInitScriptOutput(raw: string): {
|
|
17
|
+
path: string;
|
|
18
|
+
dir: string;
|
|
19
|
+
}[];
|
|
1
20
|
/**
|
|
2
21
|
* This class provides common functionality for Groovy and Kotlin DSL files.
|
|
3
22
|
*/
|
|
@@ -15,21 +34,27 @@ export default class Java_gradle extends Base_java {
|
|
|
15
34
|
* @param {string} manifestDir - the directory where the manifest lies
|
|
16
35
|
*/
|
|
17
36
|
validateLockFile(): boolean;
|
|
37
|
+
/**
|
|
38
|
+
* Gradle manifests (build.gradle, build.gradle.kts) have no standard license field.
|
|
39
|
+
* @param {string} manifestPath - path to manifest
|
|
40
|
+
* @returns {null}
|
|
41
|
+
*/
|
|
42
|
+
readLicenseFromManifest(manifestPath: string): null;
|
|
18
43
|
/**
|
|
19
44
|
* Provide content and content type for stack analysis.
|
|
20
45
|
* @param {string} manifest - the manifest path or name
|
|
21
46
|
* @param {{}} [opts={}] - optional various options to pass along the application
|
|
22
47
|
* @returns {Provided}
|
|
23
48
|
*/
|
|
24
|
-
provideStack(manifest: string, opts?: {}
|
|
49
|
+
provideStack(manifest: string, opts?: {}): Provided;
|
|
25
50
|
/**
|
|
26
51
|
* Provide content and content type for maven-maven component analysis.
|
|
27
52
|
* @param {string} manifest - path to pom.xml for component report
|
|
28
53
|
* @param {{}} [opts={}] - optional various options to pass along the application
|
|
29
54
|
* @returns {Provided}
|
|
30
55
|
*/
|
|
31
|
-
provideComponent(manifest: string, opts?: {}
|
|
56
|
+
provideComponent(manifest: string, opts?: {}): Provided;
|
|
32
57
|
#private;
|
|
33
58
|
}
|
|
34
|
-
export type Provided = import(
|
|
59
|
+
export type Provided = import("../provider.js").Provided;
|
|
35
60
|
import Base_java from "./base_java.js";
|