@trustify-da/trustify-da-javascript-client 0.3.0-ea.ff266a3 → 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 +179 -11
- package/dist/package.json +13 -4
- package/dist/src/analysis.d.ts +16 -0
- package/dist/src/analysis.js +53 -4
- package/dist/src/batch_opts.d.ts +24 -0
- package/dist/src/batch_opts.js +35 -0
- package/dist/src/cli.js +171 -4
- package/dist/src/cyclone_dx_sbom.d.ts +14 -1
- package/dist/src/cyclone_dx_sbom.js +34 -6
- package/dist/src/index.d.ts +134 -2
- package/dist/src/index.js +352 -6
- package/dist/src/license/index.d.ts +2 -2
- package/dist/src/license/index.js +4 -4
- 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.js +9 -2
- package/dist/src/license/project_license.d.ts +1 -6
- package/dist/src/license/project_license.js +4 -81
- package/dist/src/oci_image/utils.js +11 -2
- package/dist/src/provider.d.ts +7 -3
- package/dist/src/provider.js +16 -5
- package/dist/src/providers/base_java.d.ts +5 -9
- package/dist/src/providers/base_java.js +9 -38
- package/dist/src/providers/base_javascript.d.ts +30 -3
- package/dist/src/providers/base_javascript.js +115 -25
- 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 +22 -12
- package/dist/src/providers/golang_gomodules.js +167 -120
- 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 +19 -0
- package/dist/src/providers/java_gradle.js +118 -3
- package/dist/src/providers/java_maven.d.ts +9 -1
- package/dist/src/providers/java_maven.js +103 -10
- 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 +5 -1
- package/dist/src/providers/python_controller.js +8 -4
- package/dist/src/providers/python_pip.d.ts +5 -0
- package/dist/src/providers/python_pip.js +8 -7
- 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.js +4 -3
- 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 +14 -1
- package/dist/src/sbom.js +13 -2
- package/dist/src/tools.d.ts +26 -0
- package/dist/src/tools.js +58 -0
- package/dist/src/workspace.d.ts +70 -0
- package/dist/src/workspace.js +256 -0
- package/package.json +14 -5
- package/dist/src/license/compatibility.d.ts +0 -18
- package/dist/src/license/compatibility.js +0 -45
package/dist/src/tools.d.ts
CHANGED
|
@@ -61,6 +61,32 @@ export function toPurlFromString(strPurl: any): PackageURL | null;
|
|
|
61
61
|
* @param {string} cwd - directory for which to find the root of the git repository.
|
|
62
62
|
*/
|
|
63
63
|
export function getGitRootDir(cwd: string): string | undefined;
|
|
64
|
+
/**
|
|
65
|
+
* Normalize a filesystem path, lowercasing on Windows for case-insensitive comparison.
|
|
66
|
+
*
|
|
67
|
+
* @param {string} thePath
|
|
68
|
+
* @returns {string}
|
|
69
|
+
*/
|
|
70
|
+
export function normalizePath(thePath: string): string;
|
|
71
|
+
/**
|
|
72
|
+
* Walk up from `startDir` to `repoRoot` looking for an executable wrapper script.
|
|
73
|
+
*
|
|
74
|
+
* @param {string} startDir - Absolute directory to start from
|
|
75
|
+
* @param {string} wrapperName - Wrapper filename (e.g. `mvnw`, `gradlew`)
|
|
76
|
+
* @param {string} [repoRoot] - Stop boundary (defaults to git root or filesystem root)
|
|
77
|
+
* @returns {string | undefined}
|
|
78
|
+
*/
|
|
79
|
+
export function traverseForWrapper(startDir: string, wrapperName: string, repoRoot?: string): string | undefined;
|
|
80
|
+
/**
|
|
81
|
+
* Resolve a build-tool binary, preferring a wrapper when configured.
|
|
82
|
+
*
|
|
83
|
+
* @param {string} globalBinary - Global binary name (e.g. `mvn`, `gradle`)
|
|
84
|
+
* @param {string} localWrapper - Wrapper filename (e.g. `mvnw`, `gradlew.bat`)
|
|
85
|
+
* @param {string} startDir - Directory from which to start the wrapper search
|
|
86
|
+
* @param {import('./index.js').Options} [opts={}]
|
|
87
|
+
* @returns {string} Path to the resolved binary
|
|
88
|
+
*/
|
|
89
|
+
export function resolveBinary(globalBinary: string, localWrapper: string, startDir: string, opts?: import("./index.js").Options): string;
|
|
64
90
|
/** this method invokes command string in a process in a synchronous way.
|
|
65
91
|
* @param {string} bin - the command to be invoked
|
|
66
92
|
* @param {Array<string>} args - the args to pass to the binary
|
package/dist/src/tools.js
CHANGED
|
@@ -1,4 +1,6 @@
|
|
|
1
1
|
import { execFileSync } from "child_process";
|
|
2
|
+
import fs from 'node:fs';
|
|
3
|
+
import path from 'node:path';
|
|
2
4
|
import { EOL } from "os";
|
|
3
5
|
import { HttpsProxyAgent } from "https-proxy-agent";
|
|
4
6
|
import { PackageURL } from "packageurl-js";
|
|
@@ -128,6 +130,62 @@ export function getGitRootDir(cwd) {
|
|
|
128
130
|
return undefined;
|
|
129
131
|
}
|
|
130
132
|
}
|
|
133
|
+
/**
|
|
134
|
+
* Normalize a filesystem path, lowercasing on Windows for case-insensitive comparison.
|
|
135
|
+
*
|
|
136
|
+
* @param {string} thePath
|
|
137
|
+
* @returns {string}
|
|
138
|
+
*/
|
|
139
|
+
export function normalizePath(thePath) {
|
|
140
|
+
const normalized = path.resolve(thePath).normalize();
|
|
141
|
+
return process.platform === 'win32' ? normalized.toLowerCase() : normalized;
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* Walk up from `startDir` to `repoRoot` looking for an executable wrapper script.
|
|
145
|
+
*
|
|
146
|
+
* @param {string} startDir - Absolute directory to start from
|
|
147
|
+
* @param {string} wrapperName - Wrapper filename (e.g. `mvnw`, `gradlew`)
|
|
148
|
+
* @param {string} [repoRoot] - Stop boundary (defaults to git root or filesystem root)
|
|
149
|
+
* @returns {string | undefined}
|
|
150
|
+
*/
|
|
151
|
+
export function traverseForWrapper(startDir, wrapperName, repoRoot = undefined) {
|
|
152
|
+
const currentDir = normalizePath(startDir);
|
|
153
|
+
repoRoot = repoRoot || getGitRootDir(currentDir) || path.parse(currentDir).root;
|
|
154
|
+
const wrapperPath = path.join(currentDir, wrapperName);
|
|
155
|
+
try {
|
|
156
|
+
fs.accessSync(wrapperPath, fs.constants.X_OK);
|
|
157
|
+
return wrapperPath;
|
|
158
|
+
}
|
|
159
|
+
catch {
|
|
160
|
+
const rootDir = path.parse(currentDir).root;
|
|
161
|
+
if (currentDir === repoRoot || currentDir === rootDir) {
|
|
162
|
+
return undefined;
|
|
163
|
+
}
|
|
164
|
+
const parentDir = path.dirname(currentDir);
|
|
165
|
+
if (parentDir === currentDir || parentDir === rootDir) {
|
|
166
|
+
return undefined;
|
|
167
|
+
}
|
|
168
|
+
return traverseForWrapper(parentDir, wrapperName, repoRoot);
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
/**
|
|
172
|
+
* Resolve a build-tool binary, preferring a wrapper when configured.
|
|
173
|
+
*
|
|
174
|
+
* @param {string} globalBinary - Global binary name (e.g. `mvn`, `gradle`)
|
|
175
|
+
* @param {string} localWrapper - Wrapper filename (e.g. `mvnw`, `gradlew.bat`)
|
|
176
|
+
* @param {string} startDir - Directory from which to start the wrapper search
|
|
177
|
+
* @param {import('./index.js').Options} [opts={}]
|
|
178
|
+
* @returns {string} Path to the resolved binary
|
|
179
|
+
*/
|
|
180
|
+
export function resolveBinary(globalBinary, localWrapper, startDir, opts = {}) {
|
|
181
|
+
if (getWrapperPreference(globalBinary, opts)) {
|
|
182
|
+
const wrapper = traverseForWrapper(startDir, localWrapper);
|
|
183
|
+
if (wrapper !== undefined) {
|
|
184
|
+
return wrapper;
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
return getCustomPath(globalBinary, opts);
|
|
188
|
+
}
|
|
131
189
|
/** this method invokes command string in a process in a synchronous way.
|
|
132
190
|
* @param {string} bin - the command to be invoked
|
|
133
191
|
* @param {Array<string>} args - the args to pass to the binary
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Resolve ignore globs for workspace discovery: defaults + `TRUSTIFY_DA_WORKSPACE_DISCOVERY_IGNORE` + `opts.workspaceDiscoveryIgnore`.
|
|
3
|
+
* Patterns are fast-glob / micromatch style, relative to the workspace root (forward slashes).
|
|
4
|
+
*
|
|
5
|
+
* @param {{ workspaceDiscoveryIgnore?: string[], TRUSTIFY_DA_WORKSPACE_DISCOVERY_IGNORE?: string, [key: string]: unknown }} [opts={}]
|
|
6
|
+
* @returns {string[]}
|
|
7
|
+
*/
|
|
8
|
+
export function resolveWorkspaceDiscoveryIgnore(opts?: {
|
|
9
|
+
workspaceDiscoveryIgnore?: string[];
|
|
10
|
+
TRUSTIFY_DA_WORKSPACE_DISCOVERY_IGNORE?: string;
|
|
11
|
+
[key: string]: unknown;
|
|
12
|
+
}): string[];
|
|
13
|
+
/**
|
|
14
|
+
* Drop manifest paths whose location matches an ignore pattern (e.g. root unshift, Cargo paths).
|
|
15
|
+
*
|
|
16
|
+
* @param {string[]} manifestPaths
|
|
17
|
+
* @param {string} root
|
|
18
|
+
* @param {string[]} ignorePatterns
|
|
19
|
+
* @returns {string[]}
|
|
20
|
+
*/
|
|
21
|
+
export function filterManifestPathsByDiscoveryIgnore(manifestPaths: string[], root: string, ignorePatterns: string[]): string[];
|
|
22
|
+
/**
|
|
23
|
+
* @typedef {{ valid: true, name: string, version: string } | { valid: false, error: string }} ValidatePackageJsonResult
|
|
24
|
+
*/
|
|
25
|
+
/**
|
|
26
|
+
* Validate a package.json has non-empty `name` and `version` (required for stable SBOM root identity in batch).
|
|
27
|
+
*
|
|
28
|
+
* @param {string} packageJsonPath - Absolute or relative path to package.json
|
|
29
|
+
* @returns {ValidatePackageJsonResult}
|
|
30
|
+
*/
|
|
31
|
+
export function validatePackageJson(packageJsonPath: string): ValidatePackageJsonResult;
|
|
32
|
+
/**
|
|
33
|
+
* Discover all package.json paths in a JS/TS workspace.
|
|
34
|
+
* Reads pnpm-workspace.yaml or package.json workspaces.
|
|
35
|
+
*
|
|
36
|
+
* @param {string} workspaceRoot - Absolute or relative path to workspace root
|
|
37
|
+
* @param {{ workspaceDiscoveryIgnore?: string[], TRUSTIFY_DA_WORKSPACE_DISCOVERY_IGNORE?: string, [key: string]: unknown }} [opts={}] - optional `workspaceDiscoveryIgnore` globs (merged with defaults and env)
|
|
38
|
+
* @returns {Promise<string[]>} Paths to package.json files (absolute)
|
|
39
|
+
*/
|
|
40
|
+
export function discoverWorkspacePackages(workspaceRoot: string, opts?: {
|
|
41
|
+
workspaceDiscoveryIgnore?: string[];
|
|
42
|
+
TRUSTIFY_DA_WORKSPACE_DISCOVERY_IGNORE?: string;
|
|
43
|
+
[key: string]: unknown;
|
|
44
|
+
}): Promise<string[]>;
|
|
45
|
+
/**
|
|
46
|
+
* Convert workspace glob patterns to manifest-file glob patterns,
|
|
47
|
+
* correctly handling negation prefixes.
|
|
48
|
+
*
|
|
49
|
+
* @param {string[]} patterns - Workspace glob patterns (may include negations)
|
|
50
|
+
* @param {string} manifestFileName - e.g. 'package.json' or 'Cargo.toml'
|
|
51
|
+
* @returns {string[]}
|
|
52
|
+
*/
|
|
53
|
+
export function toManifestGlobPatterns(patterns: string[], manifestFileName: string): string[];
|
|
54
|
+
/**
|
|
55
|
+
* Discover all Cargo.toml manifest paths in a Cargo workspace.
|
|
56
|
+
* Uses `cargo metadata` to get workspace members.
|
|
57
|
+
*
|
|
58
|
+
* @param {string} workspaceRoot - Absolute or relative path to workspace root (must contain Cargo.toml and Cargo.lock)
|
|
59
|
+
* @param {import('./index.js').Options} [opts={}]
|
|
60
|
+
* @returns {Promise<string[]>} Paths to Cargo.toml files (absolute)
|
|
61
|
+
*/
|
|
62
|
+
export function discoverWorkspaceCrates(workspaceRoot: string, opts?: import("./index.js").Options): Promise<string[]>;
|
|
63
|
+
export type ValidatePackageJsonResult = {
|
|
64
|
+
valid: true;
|
|
65
|
+
name: string;
|
|
66
|
+
version: string;
|
|
67
|
+
} | {
|
|
68
|
+
valid: false;
|
|
69
|
+
error: string;
|
|
70
|
+
};
|
|
@@ -0,0 +1,256 @@
|
|
|
1
|
+
import fs from 'node:fs';
|
|
2
|
+
import path from 'node:path';
|
|
3
|
+
import fg from 'fast-glob';
|
|
4
|
+
import { load as yamlLoad } from 'js-yaml';
|
|
5
|
+
import micromatch from 'micromatch';
|
|
6
|
+
import { getCustom, getCustomPath, invokeCommand } from './tools.js';
|
|
7
|
+
/** Default paths skipped during JS workspace discovery (merged with user patterns). */
|
|
8
|
+
const DEFAULT_WORKSPACE_DISCOVERY_IGNORE = [
|
|
9
|
+
'**/node_modules/**',
|
|
10
|
+
'**/.git/**',
|
|
11
|
+
];
|
|
12
|
+
/**
|
|
13
|
+
* Resolve ignore globs for workspace discovery: defaults + `TRUSTIFY_DA_WORKSPACE_DISCOVERY_IGNORE` + `opts.workspaceDiscoveryIgnore`.
|
|
14
|
+
* Patterns are fast-glob / micromatch style, relative to the workspace root (forward slashes).
|
|
15
|
+
*
|
|
16
|
+
* @param {{ workspaceDiscoveryIgnore?: string[], TRUSTIFY_DA_WORKSPACE_DISCOVERY_IGNORE?: string, [key: string]: unknown }} [opts={}]
|
|
17
|
+
* @returns {string[]}
|
|
18
|
+
*/
|
|
19
|
+
export function resolveWorkspaceDiscoveryIgnore(opts = {}) {
|
|
20
|
+
const merged = [...DEFAULT_WORKSPACE_DISCOVERY_IGNORE];
|
|
21
|
+
const fromEnv = getCustom('TRUSTIFY_DA_WORKSPACE_DISCOVERY_IGNORE', null, opts);
|
|
22
|
+
if (fromEnv && String(fromEnv).trim()) {
|
|
23
|
+
merged.push(...String(fromEnv).split(',').map(s => s.trim()).filter(Boolean));
|
|
24
|
+
}
|
|
25
|
+
const extra = opts.workspaceDiscoveryIgnore;
|
|
26
|
+
if (Array.isArray(extra)) {
|
|
27
|
+
for (const p of extra) {
|
|
28
|
+
if (typeof p === 'string' && p.trim()) {
|
|
29
|
+
merged.push(p.trim());
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
return [...new Set(merged)];
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* @param {string} root - Workspace root (absolute)
|
|
37
|
+
* @param {string[]} ignorePatterns
|
|
38
|
+
*/
|
|
39
|
+
function buildWorkspaceDiscoveryGlobOptions(root, ignorePatterns) {
|
|
40
|
+
return {
|
|
41
|
+
cwd: root,
|
|
42
|
+
absolute: true,
|
|
43
|
+
onlyFiles: true,
|
|
44
|
+
ignore: ignorePatterns,
|
|
45
|
+
followSymbolicLinks: false,
|
|
46
|
+
};
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* @param {string} filePath - Absolute path
|
|
50
|
+
* @param {string} root - Workspace root (absolute)
|
|
51
|
+
* @returns {string} Relative path with forward slashes
|
|
52
|
+
*/
|
|
53
|
+
function relativePathForGlobMatch(filePath, root) {
|
|
54
|
+
const rel = path.relative(root, filePath);
|
|
55
|
+
return rel.split(path.sep).join('/');
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* Drop manifest paths whose location matches an ignore pattern (e.g. root unshift, Cargo paths).
|
|
59
|
+
*
|
|
60
|
+
* @param {string[]} manifestPaths
|
|
61
|
+
* @param {string} root
|
|
62
|
+
* @param {string[]} ignorePatterns
|
|
63
|
+
* @returns {string[]}
|
|
64
|
+
*/
|
|
65
|
+
export function filterManifestPathsByDiscoveryIgnore(manifestPaths, root, ignorePatterns) {
|
|
66
|
+
if (!ignorePatterns.length) {
|
|
67
|
+
return manifestPaths;
|
|
68
|
+
}
|
|
69
|
+
const resolvedRoot = path.resolve(root);
|
|
70
|
+
return manifestPaths.filter(absPath => {
|
|
71
|
+
const rel = relativePathForGlobMatch(absPath, resolvedRoot);
|
|
72
|
+
if (rel === '') {
|
|
73
|
+
return true;
|
|
74
|
+
}
|
|
75
|
+
return !ignorePatterns.some(pattern => micromatch.isMatch(rel, pattern, { dot: true }));
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* @typedef {{ valid: true, name: string, version: string } | { valid: false, error: string }} ValidatePackageJsonResult
|
|
80
|
+
*/
|
|
81
|
+
/**
|
|
82
|
+
* Validate a package.json has non-empty `name` and `version` (required for stable SBOM root identity in batch).
|
|
83
|
+
*
|
|
84
|
+
* @param {string} packageJsonPath - Absolute or relative path to package.json
|
|
85
|
+
* @returns {ValidatePackageJsonResult}
|
|
86
|
+
*/
|
|
87
|
+
export function validatePackageJson(packageJsonPath) {
|
|
88
|
+
let content;
|
|
89
|
+
try {
|
|
90
|
+
const raw = fs.readFileSync(packageJsonPath, 'utf-8');
|
|
91
|
+
content = JSON.parse(raw);
|
|
92
|
+
}
|
|
93
|
+
catch (err) {
|
|
94
|
+
const msg = err instanceof Error ? err.message : String(err);
|
|
95
|
+
return { valid: false, error: `Invalid package.json: ${msg}` };
|
|
96
|
+
}
|
|
97
|
+
if (!content || typeof content !== 'object') {
|
|
98
|
+
return { valid: false, error: 'package.json must be a JSON object' };
|
|
99
|
+
}
|
|
100
|
+
const name = content.name;
|
|
101
|
+
const version = content.version;
|
|
102
|
+
if (typeof name !== 'string' || !name.trim()) {
|
|
103
|
+
return { valid: false, error: 'Missing or invalid name' };
|
|
104
|
+
}
|
|
105
|
+
if (typeof version !== 'string' || !version.trim()) {
|
|
106
|
+
return { valid: false, error: 'Missing or invalid version' };
|
|
107
|
+
}
|
|
108
|
+
return { valid: true, name: name.trim(), version: version.trim() };
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Discover all package.json paths in a JS/TS workspace.
|
|
112
|
+
* Reads pnpm-workspace.yaml or package.json workspaces.
|
|
113
|
+
*
|
|
114
|
+
* @param {string} workspaceRoot - Absolute or relative path to workspace root
|
|
115
|
+
* @param {{ workspaceDiscoveryIgnore?: string[], TRUSTIFY_DA_WORKSPACE_DISCOVERY_IGNORE?: string, [key: string]: unknown }} [opts={}] - optional `workspaceDiscoveryIgnore` globs (merged with defaults and env)
|
|
116
|
+
* @returns {Promise<string[]>} Paths to package.json files (absolute)
|
|
117
|
+
*/
|
|
118
|
+
export async function discoverWorkspacePackages(workspaceRoot, opts = {}) {
|
|
119
|
+
const root = path.resolve(workspaceRoot);
|
|
120
|
+
const ignorePatterns = resolveWorkspaceDiscoveryIgnore(opts);
|
|
121
|
+
const globOpts = buildWorkspaceDiscoveryGlobOptions(root, ignorePatterns);
|
|
122
|
+
const pnpmWorkspace = path.join(root, 'pnpm-workspace.yaml');
|
|
123
|
+
const packageJson = path.join(root, 'package.json');
|
|
124
|
+
if (fs.existsSync(pnpmWorkspace)) {
|
|
125
|
+
return discoverFromPnpmWorkspace(root, pnpmWorkspace, globOpts, ignorePatterns);
|
|
126
|
+
}
|
|
127
|
+
if (fs.existsSync(packageJson)) {
|
|
128
|
+
return discoverFromPackageJsonWorkspaces(root, packageJson, globOpts, ignorePatterns);
|
|
129
|
+
}
|
|
130
|
+
return [];
|
|
131
|
+
}
|
|
132
|
+
/**
|
|
133
|
+
* @param {string} root
|
|
134
|
+
* @param {string} pnpmWorkspacePath
|
|
135
|
+
* @param {object} globOpts - fast-glob options (cwd, ignore, followSymbolicLinks, …)
|
|
136
|
+
* @param {string[]} ignorePatterns - for post-filter
|
|
137
|
+
* @returns {Promise<string[]>}
|
|
138
|
+
*/
|
|
139
|
+
async function discoverFromPnpmWorkspace(root, pnpmWorkspacePath, globOpts, ignorePatterns) {
|
|
140
|
+
const content = fs.readFileSync(pnpmWorkspacePath, 'utf-8');
|
|
141
|
+
const packages = parsePnpmPackages(content);
|
|
142
|
+
if (packages.length === 0) {
|
|
143
|
+
return [];
|
|
144
|
+
}
|
|
145
|
+
const patterns = toManifestGlobPatterns(packages, 'package.json');
|
|
146
|
+
const manifestPaths = await fg(patterns, globOpts);
|
|
147
|
+
return filterManifestPathsByDiscoveryIgnore(manifestPaths, root, ignorePatterns);
|
|
148
|
+
}
|
|
149
|
+
/**
|
|
150
|
+
* Parse the `packages` array from pnpm-workspace.yaml content.
|
|
151
|
+
* @param {string} content - Raw YAML content
|
|
152
|
+
* @returns {string[]}
|
|
153
|
+
*/
|
|
154
|
+
function parsePnpmPackages(content) {
|
|
155
|
+
let doc;
|
|
156
|
+
try {
|
|
157
|
+
doc = yamlLoad(content);
|
|
158
|
+
}
|
|
159
|
+
catch {
|
|
160
|
+
return [];
|
|
161
|
+
}
|
|
162
|
+
if (!doc || typeof doc !== 'object' || !Array.isArray(doc.packages)) {
|
|
163
|
+
return [];
|
|
164
|
+
}
|
|
165
|
+
return doc.packages.filter(p => typeof p === 'string' && p.trim()).map(p => p.trim());
|
|
166
|
+
}
|
|
167
|
+
/**
|
|
168
|
+
* Convert workspace glob patterns to manifest-file glob patterns,
|
|
169
|
+
* correctly handling negation prefixes.
|
|
170
|
+
*
|
|
171
|
+
* @param {string[]} patterns - Workspace glob patterns (may include negations)
|
|
172
|
+
* @param {string} manifestFileName - e.g. 'package.json' or 'Cargo.toml'
|
|
173
|
+
* @returns {string[]}
|
|
174
|
+
*/
|
|
175
|
+
export function toManifestGlobPatterns(patterns, manifestFileName) {
|
|
176
|
+
return patterns.map(p => {
|
|
177
|
+
if (p.startsWith('!')) {
|
|
178
|
+
return `!${p.slice(1)}/${manifestFileName}`;
|
|
179
|
+
}
|
|
180
|
+
return `${p}/${manifestFileName}`;
|
|
181
|
+
});
|
|
182
|
+
}
|
|
183
|
+
/**
|
|
184
|
+
* @param {string} root
|
|
185
|
+
* @param {string} packageJsonPath
|
|
186
|
+
* @param {object} globOpts
|
|
187
|
+
* @param {string[]} ignorePatterns
|
|
188
|
+
* @returns {Promise<string[]>}
|
|
189
|
+
*/
|
|
190
|
+
async function discoverFromPackageJsonWorkspaces(root, packageJsonPath, globOpts, ignorePatterns) {
|
|
191
|
+
let pkg;
|
|
192
|
+
try {
|
|
193
|
+
pkg = JSON.parse(fs.readFileSync(packageJsonPath, 'utf-8'));
|
|
194
|
+
}
|
|
195
|
+
catch {
|
|
196
|
+
return [];
|
|
197
|
+
}
|
|
198
|
+
const workspaces = pkg.workspaces;
|
|
199
|
+
if (!workspaces) {
|
|
200
|
+
return [];
|
|
201
|
+
}
|
|
202
|
+
const raw = Array.isArray(workspaces) ? workspaces : workspaces.packages || [];
|
|
203
|
+
const patterns = toManifestGlobPatterns(raw.filter(p => typeof p === 'string'), 'package.json');
|
|
204
|
+
if (patterns.length === 0) {
|
|
205
|
+
return [];
|
|
206
|
+
}
|
|
207
|
+
const manifestPaths = await fg(patterns, globOpts);
|
|
208
|
+
const rootPkg = path.join(root, 'package.json');
|
|
209
|
+
if (fs.existsSync(rootPkg) && !manifestPaths.includes(rootPkg)) {
|
|
210
|
+
manifestPaths.unshift(rootPkg);
|
|
211
|
+
}
|
|
212
|
+
return filterManifestPathsByDiscoveryIgnore(manifestPaths, root, ignorePatterns);
|
|
213
|
+
}
|
|
214
|
+
/**
|
|
215
|
+
* Discover all Cargo.toml manifest paths in a Cargo workspace.
|
|
216
|
+
* Uses `cargo metadata` to get workspace members.
|
|
217
|
+
*
|
|
218
|
+
* @param {string} workspaceRoot - Absolute or relative path to workspace root (must contain Cargo.toml and Cargo.lock)
|
|
219
|
+
* @param {import('./index.js').Options} [opts={}]
|
|
220
|
+
* @returns {Promise<string[]>} Paths to Cargo.toml files (absolute)
|
|
221
|
+
*/
|
|
222
|
+
export async function discoverWorkspaceCrates(workspaceRoot, opts = {}) {
|
|
223
|
+
const root = path.resolve(workspaceRoot);
|
|
224
|
+
const cargoToml = path.join(root, 'Cargo.toml');
|
|
225
|
+
const cargoLock = path.join(root, 'Cargo.lock');
|
|
226
|
+
if (!fs.existsSync(cargoToml) || !fs.existsSync(cargoLock)) {
|
|
227
|
+
return [];
|
|
228
|
+
}
|
|
229
|
+
const cargoBin = getCustomPath('cargo', opts);
|
|
230
|
+
let output;
|
|
231
|
+
try {
|
|
232
|
+
output = invokeCommand(cargoBin, ['metadata', '--format-version', '1', '--no-deps'], { cwd: root });
|
|
233
|
+
}
|
|
234
|
+
catch {
|
|
235
|
+
return [];
|
|
236
|
+
}
|
|
237
|
+
let metadata;
|
|
238
|
+
try {
|
|
239
|
+
metadata = JSON.parse(output.toString().trim());
|
|
240
|
+
}
|
|
241
|
+
catch {
|
|
242
|
+
return [];
|
|
243
|
+
}
|
|
244
|
+
const memberIds = new Set(metadata.workspace_members || []);
|
|
245
|
+
const manifestPaths = [];
|
|
246
|
+
for (const pkg of metadata.packages || []) {
|
|
247
|
+
if (memberIds.has(pkg.id) && pkg.manifest_path) {
|
|
248
|
+
const manifestPath = path.resolve(pkg.manifest_path);
|
|
249
|
+
if (fs.existsSync(manifestPath)) {
|
|
250
|
+
manifestPaths.push(manifestPath);
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
const ignorePatterns = resolveWorkspaceDiscoveryIgnore(opts);
|
|
255
|
+
return filterManifestPathsByDiscoveryIgnore(manifestPaths, root, ignorePatterns);
|
|
256
|
+
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@trustify-da/trustify-da-javascript-client",
|
|
3
|
-
"version": "0.3.0-ea.
|
|
3
|
+
"version": "0.3.0-ea.ff694a0",
|
|
4
4
|
"description": "Code-Ready Dependency Analytics JavaScript API.",
|
|
5
5
|
"license": "Apache-2.0",
|
|
6
6
|
"homepage": "https://github.com/guacsec/trustify-da-javascript-client#README.md",
|
|
@@ -38,23 +38,32 @@
|
|
|
38
38
|
"lint": "eslint src test --ext js",
|
|
39
39
|
"lint:fix": "eslint src test --ext js --fix",
|
|
40
40
|
"test": "c8 npm run tests",
|
|
41
|
-
"tests": "mocha --config .mocharc.json
|
|
41
|
+
"tests": "mocha --config .mocharc.json",
|
|
42
42
|
"tests:rep": "mocha --reporter-option maxDiffSize=0 --reporter json > unit-tests-result.json",
|
|
43
|
+
"pretest": "cp node_modules/tree-sitter-requirements/tree-sitter-requirements.wasm src/providers/tree-sitter-requirements.wasm && cp node_modules/tree-sitter-gomod/tree-sitter-gomod.wasm src/providers/tree-sitter-gomod.wasm",
|
|
43
44
|
"precompile": "rm -rf dist",
|
|
44
|
-
"compile": "tsc -p tsconfig.json"
|
|
45
|
+
"compile": "tsc -p tsconfig.json",
|
|
46
|
+
"compile:dev": "tsc -p tsconfig.dev.json",
|
|
47
|
+
"postcompile": "cp node_modules/tree-sitter-requirements/tree-sitter-requirements.wasm dist/src/providers/tree-sitter-requirements.wasm && cp node_modules/tree-sitter-gomod/tree-sitter-gomod.wasm dist/src/providers/tree-sitter-gomod.wasm"
|
|
45
48
|
},
|
|
46
49
|
"dependencies": {
|
|
47
50
|
"@babel/core": "^7.23.2",
|
|
48
51
|
"@cyclonedx/cyclonedx-library": "^6.13.0",
|
|
49
52
|
"eslint-import-resolver-typescript": "^4.4.4",
|
|
50
|
-
"fast-
|
|
53
|
+
"fast-glob": "^3.3.3",
|
|
51
54
|
"fast-xml-parser": "^5.3.4",
|
|
52
55
|
"help": "^3.0.2",
|
|
53
56
|
"https-proxy-agent": "^7.0.6",
|
|
57
|
+
"js-yaml": "^4.1.1",
|
|
58
|
+
"jsonc-parser": "^3.3.1",
|
|
59
|
+
"micromatch": "^4.0.8",
|
|
54
60
|
"node-fetch": "^3.3.2",
|
|
61
|
+
"p-limit": "^4.0.0",
|
|
55
62
|
"packageurl-js": "~1.0.2",
|
|
63
|
+
"smol-toml": "^1.6.0",
|
|
64
|
+
"tree-sitter-gomod": "github:strum355/tree-sitter-go-mod#56326f2ad478892ace58ff247a97d492a3cbcdda",
|
|
56
65
|
"tree-sitter-requirements": "github:Strum355/tree-sitter-requirements#d0261ee76b84253997fe70d7d397e78c006c3801",
|
|
57
|
-
"web-tree-sitter": "^0.26.
|
|
66
|
+
"web-tree-sitter": "^0.26.7",
|
|
58
67
|
"yargs": "^18.0.0"
|
|
59
68
|
},
|
|
60
69
|
"devDependencies": {
|
|
@@ -1,18 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* License compatibility: whether a dependency license is compatible with the project license.
|
|
3
|
-
* Relies on backend-provided license categories.
|
|
4
|
-
*
|
|
5
|
-
* Compatibility is based on restrictiveness hierarchy:
|
|
6
|
-
* PERMISSIVE < WEAK_COPYLEFT < STRONG_COPYLEFT
|
|
7
|
-
*
|
|
8
|
-
* A dependency is compatible if it's equal or less restrictive than the project license.
|
|
9
|
-
* A dependency is incompatible if it's more restrictive than the project license.
|
|
10
|
-
*/
|
|
11
|
-
/**
|
|
12
|
-
* Check if a dependency's license is compatible with the project license based on backend categories.
|
|
13
|
-
*
|
|
14
|
-
* @param {string} [projectCategory] - backend category for project license: PERMISSIVE | WEAK_COPYLEFT | STRONG_COPYLEFT | UNKNOWN
|
|
15
|
-
* @param {string} [dependencyCategory] - backend category for dependency license: PERMISSIVE | WEAK_COPYLEFT | STRONG_COPYLEFT | UNKNOWN
|
|
16
|
-
* @returns {'compatible'|'incompatible'|'unknown'}
|
|
17
|
-
*/
|
|
18
|
-
export function getCompatibility(projectCategory?: string, dependencyCategory?: string): "compatible" | "incompatible" | "unknown";
|
|
@@ -1,45 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* License compatibility: whether a dependency license is compatible with the project license.
|
|
3
|
-
* Relies on backend-provided license categories.
|
|
4
|
-
*
|
|
5
|
-
* Compatibility is based on restrictiveness hierarchy:
|
|
6
|
-
* PERMISSIVE < WEAK_COPYLEFT < STRONG_COPYLEFT
|
|
7
|
-
*
|
|
8
|
-
* A dependency is compatible if it's equal or less restrictive than the project license.
|
|
9
|
-
* A dependency is incompatible if it's more restrictive than the project license.
|
|
10
|
-
*/
|
|
11
|
-
/**
|
|
12
|
-
* Check if a dependency's license is compatible with the project license based on backend categories.
|
|
13
|
-
*
|
|
14
|
-
* @param {string} [projectCategory] - backend category for project license: PERMISSIVE | WEAK_COPYLEFT | STRONG_COPYLEFT | UNKNOWN
|
|
15
|
-
* @param {string} [dependencyCategory] - backend category for dependency license: PERMISSIVE | WEAK_COPYLEFT | STRONG_COPYLEFT | UNKNOWN
|
|
16
|
-
* @returns {'compatible'|'incompatible'|'unknown'}
|
|
17
|
-
*/
|
|
18
|
-
export function getCompatibility(projectCategory, dependencyCategory) {
|
|
19
|
-
if (!projectCategory || !dependencyCategory) {
|
|
20
|
-
return 'unknown';
|
|
21
|
-
}
|
|
22
|
-
const proj = projectCategory.toUpperCase();
|
|
23
|
-
const dep = dependencyCategory.toUpperCase();
|
|
24
|
-
// Unknown categories
|
|
25
|
-
if (proj === 'UNKNOWN' || dep === 'UNKNOWN') {
|
|
26
|
-
return 'unknown';
|
|
27
|
-
}
|
|
28
|
-
// Define restrictiveness levels (higher number = more restrictive)
|
|
29
|
-
const restrictiveness = {
|
|
30
|
-
'PERMISSIVE': 1,
|
|
31
|
-
'WEAK_COPYLEFT': 2,
|
|
32
|
-
'STRONG_COPYLEFT': 3
|
|
33
|
-
};
|
|
34
|
-
const projLevel = restrictiveness[proj];
|
|
35
|
-
const depLevel = restrictiveness[dep];
|
|
36
|
-
if (projLevel === undefined || depLevel === undefined) {
|
|
37
|
-
return 'unknown';
|
|
38
|
-
}
|
|
39
|
-
// Dependency is more restrictive than project → incompatible
|
|
40
|
-
if (depLevel > projLevel) {
|
|
41
|
-
return 'incompatible';
|
|
42
|
-
}
|
|
43
|
-
// Dependency is equal or less restrictive → compatible
|
|
44
|
-
return 'compatible';
|
|
45
|
-
}
|