forked from CycloneDX/cdxgen
-
Notifications
You must be signed in to change notification settings - Fork 0
/
piptree.js
148 lines (134 loc) · 4.2 KB
/
piptree.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
/**
* The idea behind this plugin came from the excellent pipdeptree package
* https://github.com/tox-dev/pipdeptree
*
* We use the internal pip api to construct the dependency tree for modern python + pip environments
*/
import {
existsSync,
mkdtempSync,
readFileSync,
rmSync,
writeFileSync
} from "node:fs";
import { join } from "node:path";
import { tmpdir } from "node:os";
import { spawnSync } from "node:child_process";
const PIP_TREE_PLUGIN_CONTENT = `
import importlib.metadata as importlib_metadata
import json
import sys
from pip._internal.metadata import pkg_resources
def frozen_req_from_dist(dist):
try:
from pip._internal.operations.freeze import FrozenRequirement
except ImportError:
from pip import FrozenRequirement
try:
from pip._internal import metadata
dist = metadata.pkg_resources.Distribution(dist)
try:
fr = FrozenRequirement.from_dist(dist)
except TypeError:
fr = FrozenRequirement.from_dist(dist, [])
return str(fr).strip()
except ImportError:
pass
def get_installed_distributions():
dists = pkg_resources.Environment.from_paths(None).iter_installed_distributions(
local_only=False,
skip=(),
user_only=False,
)
return [d._dist for d in dists]
def find_deps(idx, visited, reqs, traverse_count):
freqs = []
for r in reqs:
d = idx.get(r.key)
if not d:
continue
r.project_name = d.project_name if d is not None else r.project_name
if len(visited) > 100 or visited.get(r.project_name, 0) > 5:
return freqs
specs = sorted(r.specs, reverse=True)
specs_str = ",".join(["".join(sp) for sp in specs]) if specs else ""
dreqs = d.requires()
freqs.append(
{
"name": r.project_name,
"version": importlib_metadata.version(r.key),
"versionSpecifiers": specs_str,
"dependencies": find_deps(idx, visited, dreqs, traverse_count + 1) if dreqs and traverse_count < 200 else [],
}
)
visited[r.project_name] = visited.get(r.project_name, 0) + 1
return freqs
def main(argv):
out_file = "piptree.json" if len(argv) < 2 else argv[-1]
tree = []
pkgs = get_installed_distributions()
idx = {p.key: p for p in pkgs}
visited = {}
traverse_count = 0
for p in pkgs:
fr = frozen_req_from_dist(p)
if not fr.startswith('# Editable'):
tmpA = fr.split("==")
else:
fr = p.key
tmpA = [fr,p.version]
name = tmpA[0]
if name.startswith("-e"):
name = name.split("#egg=")[-1].split(" ")[0].split("&")[0]
version = "latest"
if len(tmpA) == 2:
version = tmpA[1]
tree.append(
{
"name": name.split(" ")[0],
"version": version,
"dependencies": find_deps(idx, visited, p.requires(), traverse_count + 1),
}
)
all_deps = {}
for t in tree:
for d in t["dependencies"]:
all_deps[d["name"]] = True
trimmed_tree = [
t for t in tree if t["name"] not in all_deps
]
with open(out_file, mode="w", encoding="utf-8") as fp:
json.dump(trimmed_tree, fp)
if __name__ == "__main__":
main(sys.argv)
`;
/**
* Execute the piptree plugin and return the generated tree as json object
*/
export const getTreeWithPlugin = (env, python_cmd, basePath) => {
let tree = [];
const tempDir = mkdtempSync(join(tmpdir(), "cdxgen-piptree-"));
const pipPlugin = join(tempDir, "piptree.py");
const pipTreeJson = join(tempDir, "piptree.json");
const pipPluginArgs = [pipPlugin, pipTreeJson];
writeFileSync(pipPlugin, PIP_TREE_PLUGIN_CONTENT);
const result = spawnSync(python_cmd, pipPluginArgs, {
cwd: basePath,
encoding: "utf-8",
env
});
if (result.status !== 0 || result.error) {
console.log(result.stdout, result.stderr);
}
if (existsSync(pipTreeJson)) {
tree = JSON.parse(
readFileSync(pipTreeJson, {
encoding: "utf-8"
})
);
}
if (rmSync) {
rmSync(tempDir, { recursive: true, force: true });
}
return tree;
};