prscan 1.0.0
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/.vscode/launch.json +14 -0
- package/README.MD +32 -0
- package/dist/bot/lark.d.ts +2 -0
- package/dist/bot/lark.d.ts.map +1 -0
- package/dist/bot/lark.js +156 -0
- package/dist/bot/lark.js.map +1 -0
- package/dist/cli/cli.d.ts +2 -0
- package/dist/cli/cli.d.ts.map +1 -0
- package/dist/cli/cli.js +77 -0
- package/dist/cli/cli.js.map +1 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +46 -0
- package/dist/index.js.map +1 -0
- package/dist/report/index.d.ts +7 -0
- package/dist/report/index.d.ts.map +1 -0
- package/dist/report/index.js +45 -0
- package/dist/report/index.js.map +1 -0
- package/dist/tool/prscan.d.ts +72 -0
- package/dist/tool/prscan.d.ts.map +1 -0
- package/dist/tool/prscan.js +477 -0
- package/dist/tool/prscan.js.map +1 -0
- package/dist/util/analyze.d.ts +4 -0
- package/dist/util/analyze.d.ts.map +1 -0
- package/dist/util/analyze.js +213 -0
- package/dist/util/analyze.js.map +1 -0
- package/dist/util/archive.d.ts +34 -0
- package/dist/util/archive.d.ts.map +1 -0
- package/dist/util/archive.js +110 -0
- package/dist/util/archive.js.map +1 -0
- package/dist/util/memory-archive.d.ts +37 -0
- package/dist/util/memory-archive.d.ts.map +1 -0
- package/dist/util/memory-archive.js +128 -0
- package/dist/util/memory-archive.js.map +1 -0
- package/dist/util/npm.d.ts +46 -0
- package/dist/util/npm.d.ts.map +1 -0
- package/dist/util/npm.js +35 -0
- package/dist/util/npm.js.map +1 -0
- package/dist/util/parse.d.ts +18 -0
- package/dist/util/parse.d.ts.map +1 -0
- package/dist/util/parse.js +92 -0
- package/dist/util/parse.js.map +1 -0
- package/dist/util/proxy.d.ts +45 -0
- package/dist/util/proxy.d.ts.map +1 -0
- package/dist/util/proxy.js +143 -0
- package/dist/util/proxy.js.map +1 -0
- package/dist/util/repo.d.ts +103 -0
- package/dist/util/repo.d.ts.map +1 -0
- package/dist/util/repo.js +170 -0
- package/dist/util/repo.js.map +1 -0
- package/package.json +35 -0
- package/report.png +0 -0
- package/src/bot/lark.ts +184 -0
- package/src/cli/cli.ts +80 -0
- package/src/index.ts +67 -0
- package/src/report/index.ts +50 -0
- package/src/tool/prscan.ts +634 -0
- package/src/util/analyze.ts +248 -0
- package/src/util/memory-archive.ts +184 -0
- package/src/util/npm.ts +100 -0
- package/src/util/parse.ts +103 -0
- package/src/util/repo.ts +224 -0
- package/tsconfig.json +43 -0
|
@@ -0,0 +1,213 @@
|
|
|
1
|
+
import * as t from "@babel/types";
|
|
2
|
+
import { parse } from "@babel/parser";
|
|
3
|
+
import _traverse, {} from "@babel/traverse";
|
|
4
|
+
// 使用 require 方式导入以获得正确的类型
|
|
5
|
+
const traverse = _traverse.default;
|
|
6
|
+
export const ECMAGlobals = [
|
|
7
|
+
"AggregateError",
|
|
8
|
+
"Array",
|
|
9
|
+
"ArrayBuffer",
|
|
10
|
+
"Atomics",
|
|
11
|
+
"BigInt",
|
|
12
|
+
"BigInt64Array",
|
|
13
|
+
"BigUint64Array",
|
|
14
|
+
"Boolean",
|
|
15
|
+
"DataView",
|
|
16
|
+
"Date",
|
|
17
|
+
"Error",
|
|
18
|
+
"EvalError",
|
|
19
|
+
"FinalizationRegistry",
|
|
20
|
+
"Float16Array",
|
|
21
|
+
"Float32Array",
|
|
22
|
+
"Float64Array",
|
|
23
|
+
"Function",
|
|
24
|
+
"Infinity",
|
|
25
|
+
"Int16Array",
|
|
26
|
+
"Int32Array",
|
|
27
|
+
"Int8Array",
|
|
28
|
+
"Intl",
|
|
29
|
+
"Iterator",
|
|
30
|
+
"JSON",
|
|
31
|
+
"Map",
|
|
32
|
+
"Math",
|
|
33
|
+
"NaN",
|
|
34
|
+
"Number",
|
|
35
|
+
"Object",
|
|
36
|
+
"Promise",
|
|
37
|
+
"Proxy",
|
|
38
|
+
"RangeError",
|
|
39
|
+
"ReferenceError",
|
|
40
|
+
"Reflect",
|
|
41
|
+
"RegExp",
|
|
42
|
+
"Set",
|
|
43
|
+
"SharedArrayBuffer",
|
|
44
|
+
"String",
|
|
45
|
+
"Symbol",
|
|
46
|
+
"SyntaxError",
|
|
47
|
+
"TypeError",
|
|
48
|
+
"Uint16Array",
|
|
49
|
+
"Uint32Array",
|
|
50
|
+
"Uint8Array",
|
|
51
|
+
"Uint8ClampedArray",
|
|
52
|
+
"URIError",
|
|
53
|
+
"WeakMap",
|
|
54
|
+
"WeakRef",
|
|
55
|
+
"WeakSet",
|
|
56
|
+
"decodeURI",
|
|
57
|
+
"decodeURIComponent",
|
|
58
|
+
"encodeURI",
|
|
59
|
+
"encodeURIComponent",
|
|
60
|
+
"escape",
|
|
61
|
+
// "eval",
|
|
62
|
+
"globalThis",
|
|
63
|
+
"isFinite",
|
|
64
|
+
"isNaN",
|
|
65
|
+
"parseFloat",
|
|
66
|
+
"parseInt",
|
|
67
|
+
"undefined",
|
|
68
|
+
"unescape",
|
|
69
|
+
];
|
|
70
|
+
class GVPool {
|
|
71
|
+
globals = new Map();
|
|
72
|
+
add(name, perm) {
|
|
73
|
+
// 不管控除eval以外的ECMA全局变量的读权限
|
|
74
|
+
if (ECMAGlobals.includes(name) && perm === "r") {
|
|
75
|
+
return;
|
|
76
|
+
}
|
|
77
|
+
if (this.globals.has(name)) {
|
|
78
|
+
if (perm === "rw") {
|
|
79
|
+
this.globals.set(name, "rw");
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
else {
|
|
83
|
+
this.globals.set(name, perm);
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
export function analyzeGlobals(code) {
|
|
88
|
+
const pool = new GVPool();
|
|
89
|
+
const ast = parse(code, {
|
|
90
|
+
sourceType: "unambiguous",
|
|
91
|
+
});
|
|
92
|
+
traverse(ast, {
|
|
93
|
+
MemberExpression(path) {
|
|
94
|
+
let expr = [];
|
|
95
|
+
let node = path.node;
|
|
96
|
+
let flag = true;
|
|
97
|
+
while (t.isMemberExpression(node)) {
|
|
98
|
+
if (t.isIdentifier(node.property) && !node.computed) {
|
|
99
|
+
expr.unshift(node.property.name);
|
|
100
|
+
}
|
|
101
|
+
else if (t.isStringLiteral(node.property)) {
|
|
102
|
+
expr.unshift(node.property.value);
|
|
103
|
+
}
|
|
104
|
+
else {
|
|
105
|
+
flag = false;
|
|
106
|
+
break;
|
|
107
|
+
}
|
|
108
|
+
if (t.isIdentifier(node.object)) {
|
|
109
|
+
expr.unshift(node.object.name);
|
|
110
|
+
break;
|
|
111
|
+
}
|
|
112
|
+
else if (t.isMemberExpression(node.object)) {
|
|
113
|
+
node = node.object;
|
|
114
|
+
}
|
|
115
|
+
else {
|
|
116
|
+
flag = false;
|
|
117
|
+
break;
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
if (!flag) {
|
|
121
|
+
return;
|
|
122
|
+
}
|
|
123
|
+
// 处理 __webpack_require__.g.xxx 的情况, __webpack_require__.g 就是 globalThis
|
|
124
|
+
if (expr.length >= 2 &&
|
|
125
|
+
expr[0] === "__webpack_require__" &&
|
|
126
|
+
expr[1] === "g") {
|
|
127
|
+
expr = expr.slice(2);
|
|
128
|
+
}
|
|
129
|
+
const globalAlias = ["globalThis", "self", "window"];
|
|
130
|
+
let mustBeGlobal = false;
|
|
131
|
+
while (expr.length > 0 &&
|
|
132
|
+
globalAlias.includes(expr[0]) &&
|
|
133
|
+
!path.scope.hasBinding(expr[0], { noGlobals: true })) {
|
|
134
|
+
expr = expr.slice(1);
|
|
135
|
+
// 如果是globalThis/self/window开头的, 则一定是全局变量
|
|
136
|
+
mustBeGlobal = true;
|
|
137
|
+
}
|
|
138
|
+
if (expr.length > 0 &&
|
|
139
|
+
(mustBeGlobal ||
|
|
140
|
+
!path.scope.hasBinding(expr[0], { noGlobals: true })) &&
|
|
141
|
+
expr[0] !== "arguments") {
|
|
142
|
+
pool.add(expr[0], t.isAssignmentExpression(path.parent) &&
|
|
143
|
+
path.parent.left === path.node &&
|
|
144
|
+
expr.length === 1
|
|
145
|
+
? "rw"
|
|
146
|
+
: "r");
|
|
147
|
+
}
|
|
148
|
+
path.skip();
|
|
149
|
+
},
|
|
150
|
+
Identifier(path) {
|
|
151
|
+
if (!path.scope.hasBinding(path.node.name, { noGlobals: true })) {
|
|
152
|
+
// 排除一些不算真正变量读取的场景
|
|
153
|
+
if ((t.isMemberExpression(path.parent) ||
|
|
154
|
+
t.isOptionalMemberExpression(path.parent)) &&
|
|
155
|
+
path.parent.property === path.node) {
|
|
156
|
+
return;
|
|
157
|
+
}
|
|
158
|
+
else if (t.isObjectProperty(path.parent) &&
|
|
159
|
+
path.parent.key === path.node) {
|
|
160
|
+
return;
|
|
161
|
+
}
|
|
162
|
+
else if ((t.isLabeledStatement(path.parent) ||
|
|
163
|
+
t.isBreakStatement(path.parent) ||
|
|
164
|
+
t.isContinueStatement(path.parent)) &&
|
|
165
|
+
path.parent.label === path.node) {
|
|
166
|
+
return;
|
|
167
|
+
}
|
|
168
|
+
else if ((t.isFunctionDeclaration(path.parent) ||
|
|
169
|
+
t.isFunctionExpression(path.parent)) &&
|
|
170
|
+
path.parent.id === path.node) {
|
|
171
|
+
return;
|
|
172
|
+
}
|
|
173
|
+
else if (t.isCatchClause(path.parent) &&
|
|
174
|
+
path.parent.param === path.node) {
|
|
175
|
+
return;
|
|
176
|
+
}
|
|
177
|
+
else if (t.isFunction(path.parent) &&
|
|
178
|
+
(path.parent.params.includes(path.node) ||
|
|
179
|
+
path.parent.id === path.node ||
|
|
180
|
+
path.parent.kind === path.node.name ||
|
|
181
|
+
path.parent.key === path.node)) {
|
|
182
|
+
return;
|
|
183
|
+
}
|
|
184
|
+
else if (t.isMetaProperty(path.parent)) {
|
|
185
|
+
return;
|
|
186
|
+
}
|
|
187
|
+
else if (t.isClassProperty(path.parent) &&
|
|
188
|
+
path.parent.key === path.node) {
|
|
189
|
+
return;
|
|
190
|
+
// globalAlias默认可访问
|
|
191
|
+
}
|
|
192
|
+
else if (path.node.name === "globalThis" ||
|
|
193
|
+
path.node.name === "window" ||
|
|
194
|
+
path.node.name === "self") {
|
|
195
|
+
return;
|
|
196
|
+
}
|
|
197
|
+
else if (path.node.name === "arguments") {
|
|
198
|
+
return;
|
|
199
|
+
}
|
|
200
|
+
pool.add(path.node.name, t.isAssignmentExpression(path.parent) &&
|
|
201
|
+
path.parent.left === path.node
|
|
202
|
+
? "rw"
|
|
203
|
+
: "r");
|
|
204
|
+
}
|
|
205
|
+
},
|
|
206
|
+
});
|
|
207
|
+
const obj = Object.create(null);
|
|
208
|
+
for (const [name, perm] of pool.globals.entries()) {
|
|
209
|
+
obj[name] = perm;
|
|
210
|
+
}
|
|
211
|
+
return obj;
|
|
212
|
+
}
|
|
213
|
+
//# sourceMappingURL=analyze.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"analyze.js","sourceRoot":"","sources":["../../src/util/analyze.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
export interface ExtractOptions {
|
|
2
|
+
/** 解压到的目标目录 */
|
|
3
|
+
targetDir: string;
|
|
4
|
+
/** 文件过滤器,返回true表示解压该文件 */
|
|
5
|
+
filter?: (path: string, entry: any) => boolean;
|
|
6
|
+
/** 是否覆盖已存在的文件 */
|
|
7
|
+
overwrite?: boolean;
|
|
8
|
+
/** 是否创建目标目录 */
|
|
9
|
+
createDir?: boolean;
|
|
10
|
+
}
|
|
11
|
+
export interface ArchiveEntry {
|
|
12
|
+
path: string;
|
|
13
|
+
type: 'file' | 'directory' | 'symlink' | 'other';
|
|
14
|
+
size: number;
|
|
15
|
+
mode: number;
|
|
16
|
+
mtime: Date;
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* 解压 .tar.gz 文件
|
|
20
|
+
*/
|
|
21
|
+
export declare function extractTarGz(archivePath: string, options: ExtractOptions): Promise<string[]>;
|
|
22
|
+
/**
|
|
23
|
+
* 列出压缩包内容而不解压
|
|
24
|
+
*/
|
|
25
|
+
export declare function listTarGz(archivePath: string): Promise<ArchiveEntry[]>;
|
|
26
|
+
/**
|
|
27
|
+
* 解压单个文件
|
|
28
|
+
*/
|
|
29
|
+
export declare function extractSingleFile(archivePath: string, filePath: string, targetPath: string): Promise<boolean>;
|
|
30
|
+
/**
|
|
31
|
+
* 解压并过滤特定类型文件
|
|
32
|
+
*/
|
|
33
|
+
export declare function extractFilesByExtension(archivePath: string, targetDir: string, extensions: string[]): Promise<string[]>;
|
|
34
|
+
//# sourceMappingURL=archive.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"archive.d.ts","sourceRoot":"","sources":["../../src/util/archive.ts"],"names":[],"mappings":"AAIA,MAAM,WAAW,cAAc;IAC3B,eAAe;IACf,SAAS,EAAE,MAAM,CAAC;IAClB,0BAA0B;IAC1B,MAAM,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,GAAG,KAAK,OAAO,CAAC;IAC/C,iBAAiB;IACjB,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,eAAe;IACf,SAAS,CAAC,EAAE,OAAO,CAAC;CACvB;AAED,MAAM,WAAW,YAAY;IACzB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,GAAG,WAAW,GAAG,SAAS,GAAG,OAAO,CAAC;IACjD,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,IAAI,CAAC;CACf;AAED;;GAEG;AACH,wBAAsB,YAAY,CAC9B,WAAW,EAAE,MAAM,EACnB,OAAO,EAAE,cAAc,GACxB,OAAO,CAAC,MAAM,EAAE,CAAC,CAoCnB;AAED;;GAEG;AACH,wBAAsB,SAAS,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,YAAY,EAAE,CAAC,CAiB5E;AAED;;GAEG;AACH,wBAAsB,iBAAiB,CACnC,WAAW,EAAE,MAAM,EACnB,QAAQ,EAAE,MAAM,EAChB,UAAU,EAAE,MAAM,GACnB,OAAO,CAAC,OAAO,CAAC,CAuBlB;AAED;;GAEG;AACH,wBAAsB,uBAAuB,CACzC,WAAW,EAAE,MAAM,EACnB,SAAS,EAAE,MAAM,EACjB,UAAU,EAAE,MAAM,EAAE,GACrB,OAAO,CAAC,MAAM,EAAE,CAAC,CAWnB"}
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
import tar from 'tar';
|
|
2
|
+
import { mkdir, access } from 'fs/promises';
|
|
3
|
+
import { dirname } from 'path';
|
|
4
|
+
/**
|
|
5
|
+
* 解压 .tar.gz 文件
|
|
6
|
+
*/
|
|
7
|
+
export async function extractTarGz(archivePath, options) {
|
|
8
|
+
const { targetDir, filter, overwrite = true, createDir = true } = options;
|
|
9
|
+
// 创建目标目录
|
|
10
|
+
if (createDir) {
|
|
11
|
+
await mkdir(targetDir, { recursive: true });
|
|
12
|
+
}
|
|
13
|
+
// 检查目标目录是否存在
|
|
14
|
+
try {
|
|
15
|
+
await access(targetDir);
|
|
16
|
+
}
|
|
17
|
+
catch {
|
|
18
|
+
throw new Error(`Target directory does not exist: ${targetDir}`);
|
|
19
|
+
}
|
|
20
|
+
const extractedFiles = [];
|
|
21
|
+
await tar.extract({
|
|
22
|
+
file: archivePath,
|
|
23
|
+
cwd: targetDir,
|
|
24
|
+
filter: filter ? (path, entry) => {
|
|
25
|
+
const shouldExtract = filter(path, entry);
|
|
26
|
+
if (shouldExtract) {
|
|
27
|
+
extractedFiles.push(path);
|
|
28
|
+
}
|
|
29
|
+
return shouldExtract;
|
|
30
|
+
} : undefined,
|
|
31
|
+
newer: !overwrite, // 如果不覆盖,只解压更新的文件
|
|
32
|
+
onentry: (entry) => {
|
|
33
|
+
if (!filter) {
|
|
34
|
+
extractedFiles.push(entry.path);
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
});
|
|
38
|
+
return extractedFiles;
|
|
39
|
+
}
|
|
40
|
+
/**
|
|
41
|
+
* 列出压缩包内容而不解压
|
|
42
|
+
*/
|
|
43
|
+
export async function listTarGz(archivePath) {
|
|
44
|
+
const entries = [];
|
|
45
|
+
await tar.list({
|
|
46
|
+
file: archivePath,
|
|
47
|
+
onentry: (entry) => {
|
|
48
|
+
entries.push({
|
|
49
|
+
path: entry.path,
|
|
50
|
+
type: getEntryType(entry.type),
|
|
51
|
+
size: entry.size || 0,
|
|
52
|
+
mode: entry.mode || 0,
|
|
53
|
+
mtime: entry.mtime || new Date()
|
|
54
|
+
});
|
|
55
|
+
}
|
|
56
|
+
});
|
|
57
|
+
return entries;
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* 解压单个文件
|
|
61
|
+
*/
|
|
62
|
+
export async function extractSingleFile(archivePath, filePath, targetPath) {
|
|
63
|
+
let found = false;
|
|
64
|
+
// 确保目标目录存在
|
|
65
|
+
await mkdir(dirname(targetPath), { recursive: true });
|
|
66
|
+
await tar.extract({
|
|
67
|
+
file: archivePath,
|
|
68
|
+
filter: (path) => {
|
|
69
|
+
if (path === filePath) {
|
|
70
|
+
found = true;
|
|
71
|
+
return true;
|
|
72
|
+
}
|
|
73
|
+
return false;
|
|
74
|
+
},
|
|
75
|
+
strip: 0,
|
|
76
|
+
C: dirname(targetPath),
|
|
77
|
+
transform: {
|
|
78
|
+
[filePath]: targetPath
|
|
79
|
+
}
|
|
80
|
+
});
|
|
81
|
+
return found;
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* 解压并过滤特定类型文件
|
|
85
|
+
*/
|
|
86
|
+
export async function extractFilesByExtension(archivePath, targetDir, extensions) {
|
|
87
|
+
const normalizedExts = extensions.map(ext => ext.startsWith('.') ? ext : `.${ext}`);
|
|
88
|
+
return extractTarGz(archivePath, {
|
|
89
|
+
targetDir,
|
|
90
|
+
filter: (path) => {
|
|
91
|
+
return normalizedExts.some(ext => path.endsWith(ext));
|
|
92
|
+
}
|
|
93
|
+
});
|
|
94
|
+
}
|
|
95
|
+
function getEntryType(type) {
|
|
96
|
+
switch (type) {
|
|
97
|
+
case 'File':
|
|
98
|
+
case 'file':
|
|
99
|
+
return 'file';
|
|
100
|
+
case 'Directory':
|
|
101
|
+
case 'directory':
|
|
102
|
+
return 'directory';
|
|
103
|
+
case 'SymbolicLink':
|
|
104
|
+
case 'symlink':
|
|
105
|
+
return 'symlink';
|
|
106
|
+
default:
|
|
107
|
+
return 'other';
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
//# sourceMappingURL=archive.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"archive.js","sourceRoot":"","sources":["../../src/util/archive.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
export interface ExtractedFile {
|
|
2
|
+
path: string;
|
|
3
|
+
content: Buffer;
|
|
4
|
+
size: number;
|
|
5
|
+
type: 'file' | 'directory' | 'symlink';
|
|
6
|
+
mode: number;
|
|
7
|
+
mtime: Date;
|
|
8
|
+
}
|
|
9
|
+
export interface ExtractionOptions {
|
|
10
|
+
/** 只解压匹配的文件路径 */
|
|
11
|
+
filter?: (path: string) => boolean;
|
|
12
|
+
/** 最大文件大小限制 (bytes) */
|
|
13
|
+
maxFileSize?: number;
|
|
14
|
+
/** 最大总解压大小限制 (bytes) */
|
|
15
|
+
maxTotalSize?: number;
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* 在内存中解压 .tar.gz Buffer
|
|
19
|
+
*/
|
|
20
|
+
export declare function extractTarGzFromBuffer(tarGzBuffer: Buffer, options?: ExtractionOptions): Promise<ExtractedFile[]>;
|
|
21
|
+
/**
|
|
22
|
+
* 从URL下载并在内存中解压
|
|
23
|
+
*/
|
|
24
|
+
export declare function extractTarGzFromUrl(url: string, options?: ExtractionOptions): Promise<ExtractedFile[]>;
|
|
25
|
+
/**
|
|
26
|
+
* 获取特定文件内容
|
|
27
|
+
*/
|
|
28
|
+
export declare function getFileFromTarGz(tarGzBuffer: Buffer, filePath: string): Promise<Buffer | null>;
|
|
29
|
+
/**
|
|
30
|
+
* 只获取文件列表,不提取内容
|
|
31
|
+
*/
|
|
32
|
+
export declare function listTarGzContents(tarGzBuffer: Buffer): Promise<string[]>;
|
|
33
|
+
/**
|
|
34
|
+
* 按文件扩展名过滤提取
|
|
35
|
+
*/
|
|
36
|
+
export declare function extractByExtensions(tarGzBuffer: Buffer, extensions: string[]): Promise<ExtractedFile[]>;
|
|
37
|
+
//# sourceMappingURL=memory-archive.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"memory-archive.d.ts","sourceRoot":"","sources":["../../src/util/memory-archive.ts"],"names":[],"mappings":"AAKA,MAAM,WAAW,aAAa;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,GAAG,WAAW,GAAG,SAAS,CAAC;IACvC,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,IAAI,CAAC;CACf;AAED,MAAM,WAAW,iBAAiB;IAC9B,iBAAiB;IACjB,MAAM,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,OAAO,CAAC;IACnC,uBAAuB;IACvB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,wBAAwB;IACxB,YAAY,CAAC,EAAE,MAAM,CAAC;CACzB;AAED;;GAEG;AACH,wBAAsB,sBAAsB,CACxC,WAAW,EAAE,MAAM,EACnB,OAAO,GAAE,iBAAsB,GAChC,OAAO,CAAC,aAAa,EAAE,CAAC,CA6E1B;AAGD;;GAEG;AACH,wBAAsB,mBAAmB,CACrC,GAAG,EAAE,MAAM,EACX,OAAO,GAAE,iBAAsB,GAChC,OAAO,CAAC,aAAa,EAAE,CAAC,CAQ1B;AAED;;GAEG;AACH,wBAAsB,gBAAgB,CAClC,WAAW,EAAE,MAAM,EACnB,QAAQ,EAAE,MAAM,GACjB,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAMxB;AAED;;GAEG;AACH,wBAAsB,iBAAiB,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,CAsB9E;AAED;;GAEG;AACH,wBAAsB,mBAAmB,CACrC,WAAW,EAAE,MAAM,EACnB,UAAU,EAAE,MAAM,EAAE,GACrB,OAAO,CAAC,aAAa,EAAE,CAAC,CAW1B"}
|
|
@@ -0,0 +1,128 @@
|
|
|
1
|
+
import tarStream from 'tar-stream';
|
|
2
|
+
import gunzip from 'gunzip-maybe';
|
|
3
|
+
import { Readable } from 'stream';
|
|
4
|
+
import got from 'got';
|
|
5
|
+
/**
|
|
6
|
+
* 在内存中解压 .tar.gz Buffer
|
|
7
|
+
*/
|
|
8
|
+
export async function extractTarGzFromBuffer(tarGzBuffer, options = {}) {
|
|
9
|
+
const { filter, maxFileSize = 50 * 1024 * 1024, maxTotalSize = 500 * 1024 * 1024 } = options;
|
|
10
|
+
const files = [];
|
|
11
|
+
let totalSize = 0;
|
|
12
|
+
return new Promise((resolve, reject) => {
|
|
13
|
+
const extract = tarStream.extract();
|
|
14
|
+
extract.on('entry', (header, stream, next) => {
|
|
15
|
+
// 应用过滤器
|
|
16
|
+
if (filter && !filter(header.name)) {
|
|
17
|
+
stream.on('end', next);
|
|
18
|
+
stream.resume();
|
|
19
|
+
return;
|
|
20
|
+
}
|
|
21
|
+
// 检查文件大小限制
|
|
22
|
+
if (header.size && header.size > maxFileSize) {
|
|
23
|
+
stream.on('end', next);
|
|
24
|
+
stream.resume();
|
|
25
|
+
console.warn(`File ${header.name} exceeds size limit, skipping`);
|
|
26
|
+
return;
|
|
27
|
+
}
|
|
28
|
+
const chunks = [];
|
|
29
|
+
let fileSize = 0;
|
|
30
|
+
stream.on('data', (chunk) => {
|
|
31
|
+
fileSize += chunk.length;
|
|
32
|
+
totalSize += chunk.length;
|
|
33
|
+
// 检查总大小限制
|
|
34
|
+
if (totalSize > maxTotalSize) {
|
|
35
|
+
reject(new Error('Total extraction size limit exceeded'));
|
|
36
|
+
return;
|
|
37
|
+
}
|
|
38
|
+
chunks.push(chunk);
|
|
39
|
+
});
|
|
40
|
+
stream.on('end', () => {
|
|
41
|
+
if (header.type === 'file') {
|
|
42
|
+
files.push({
|
|
43
|
+
path: header.name,
|
|
44
|
+
content: Buffer.concat(chunks),
|
|
45
|
+
size: fileSize,
|
|
46
|
+
type: 'file',
|
|
47
|
+
mode: header.mode || 0,
|
|
48
|
+
mtime: header.mtime || new Date()
|
|
49
|
+
});
|
|
50
|
+
}
|
|
51
|
+
else if (header.type === 'directory') {
|
|
52
|
+
files.push({
|
|
53
|
+
path: header.name,
|
|
54
|
+
content: Buffer.alloc(0),
|
|
55
|
+
size: 0,
|
|
56
|
+
type: 'directory',
|
|
57
|
+
mode: header.mode || 0,
|
|
58
|
+
mtime: header.mtime || new Date()
|
|
59
|
+
});
|
|
60
|
+
}
|
|
61
|
+
next();
|
|
62
|
+
});
|
|
63
|
+
stream.on('error', reject);
|
|
64
|
+
});
|
|
65
|
+
extract.on('finish', () => {
|
|
66
|
+
resolve(files);
|
|
67
|
+
});
|
|
68
|
+
extract.on('error', reject);
|
|
69
|
+
// 处理压缩流
|
|
70
|
+
Readable.from(tarGzBuffer)
|
|
71
|
+
.pipe(gunzip())
|
|
72
|
+
.pipe(extract);
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* 从URL下载并在内存中解压
|
|
77
|
+
*/
|
|
78
|
+
export async function extractTarGzFromUrl(url, options = {}) {
|
|
79
|
+
const response = await got(url);
|
|
80
|
+
if (response.statusCode !== 200) {
|
|
81
|
+
throw new Error(`Failed to download: ${response.statusMessage}`);
|
|
82
|
+
}
|
|
83
|
+
const buffer = response.rawBody;
|
|
84
|
+
return extractTarGzFromBuffer(buffer, options);
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* 获取特定文件内容
|
|
88
|
+
*/
|
|
89
|
+
export async function getFileFromTarGz(tarGzBuffer, filePath) {
|
|
90
|
+
const files = await extractTarGzFromBuffer(tarGzBuffer, {
|
|
91
|
+
filter: (path) => path === filePath
|
|
92
|
+
});
|
|
93
|
+
return files.length > 0 ? files[0].content : null;
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* 只获取文件列表,不提取内容
|
|
97
|
+
*/
|
|
98
|
+
export async function listTarGzContents(tarGzBuffer) {
|
|
99
|
+
const fileList = [];
|
|
100
|
+
return new Promise((resolve, reject) => {
|
|
101
|
+
const extract = tarStream.extract();
|
|
102
|
+
extract.on('entry', (header, stream, next) => {
|
|
103
|
+
fileList.push(header.name);
|
|
104
|
+
stream.on('end', next);
|
|
105
|
+
stream.resume(); // 跳过内容读取
|
|
106
|
+
});
|
|
107
|
+
extract.on('finish', () => {
|
|
108
|
+
resolve(fileList);
|
|
109
|
+
});
|
|
110
|
+
extract.on('error', reject);
|
|
111
|
+
Readable.from(tarGzBuffer)
|
|
112
|
+
.pipe(gunzip())
|
|
113
|
+
.pipe(extract);
|
|
114
|
+
});
|
|
115
|
+
}
|
|
116
|
+
/**
|
|
117
|
+
* 按文件扩展名过滤提取
|
|
118
|
+
*/
|
|
119
|
+
export async function extractByExtensions(tarGzBuffer, extensions) {
|
|
120
|
+
const normalizedExts = extensions.map(ext => ext.startsWith('.') ? ext.toLowerCase() : `.${ext.toLowerCase()}`);
|
|
121
|
+
return extractTarGzFromBuffer(tarGzBuffer, {
|
|
122
|
+
filter: (path) => {
|
|
123
|
+
const ext = path.toLowerCase().substring(path.lastIndexOf('.'));
|
|
124
|
+
return normalizedExts.includes(ext);
|
|
125
|
+
}
|
|
126
|
+
});
|
|
127
|
+
}
|
|
128
|
+
//# sourceMappingURL=memory-archive.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"memory-archive.js","sourceRoot":"","sources":["../../src/util/memory-archive.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
export interface NpmPackageInfo {
|
|
2
|
+
_id: string;
|
|
3
|
+
_rev: string;
|
|
4
|
+
bugs: {
|
|
5
|
+
url: string;
|
|
6
|
+
};
|
|
7
|
+
description: string;
|
|
8
|
+
"dist-tags": {
|
|
9
|
+
latest: string;
|
|
10
|
+
beta: string;
|
|
11
|
+
};
|
|
12
|
+
homepage: string;
|
|
13
|
+
keywords: string[];
|
|
14
|
+
license: string;
|
|
15
|
+
name: string;
|
|
16
|
+
repository: {
|
|
17
|
+
type: string;
|
|
18
|
+
url: string;
|
|
19
|
+
};
|
|
20
|
+
time: Record<string, string>;
|
|
21
|
+
versions: Record<string, {
|
|
22
|
+
_id: string;
|
|
23
|
+
version: string;
|
|
24
|
+
name: string;
|
|
25
|
+
license: string;
|
|
26
|
+
keywords: string[];
|
|
27
|
+
homepage: string;
|
|
28
|
+
description: string;
|
|
29
|
+
dist: {
|
|
30
|
+
fileCount: number;
|
|
31
|
+
unpackedSize: number;
|
|
32
|
+
tarball: string;
|
|
33
|
+
shasum: string;
|
|
34
|
+
integrity: string;
|
|
35
|
+
};
|
|
36
|
+
}>;
|
|
37
|
+
}
|
|
38
|
+
export interface NpmDownloadStats {
|
|
39
|
+
downloads: number;
|
|
40
|
+
start: string;
|
|
41
|
+
end: string;
|
|
42
|
+
package: string;
|
|
43
|
+
}
|
|
44
|
+
export declare function getNpmPackageInfo(packageName: string): Promise<NpmPackageInfo | null>;
|
|
45
|
+
export declare function getNpmPackageDownloadStats(packageName: string, period?: "last-day" | "last-week" | "last-month" | "last-year"): Promise<NpmDownloadStats | null>;
|
|
46
|
+
//# sourceMappingURL=npm.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"npm.d.ts","sourceRoot":"","sources":["../../src/util/npm.ts"],"names":[],"mappings":"AAEA,MAAM,WAAW,cAAc;IAC3B,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE;QACF,GAAG,EAAE,MAAM,CAAC;KACf,CAAC;IACF,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE;QACT,MAAM,EAAE,MAAM,CAAC;QACf,IAAI,EAAE,MAAM,CAAC;KAChB,CAAC;IACF,QAAQ,EAAE,MAAM,CAAC;IACjB,QAAQ,EAAE,MAAM,EAAE,CAAC;IACnB,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,MAAM,CAAC;IACb,UAAU,EAAE;QACR,IAAI,EAAE,MAAM,CAAC;QACb,GAAG,EAAE,MAAM,CAAC;KACf,CAAC;IACF,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IAC7B,QAAQ,EAAE,MAAM,CACZ,MAAM,EACN;QACI,GAAG,EAAE,MAAM,CAAC;QACZ,OAAO,EAAE,MAAM,CAAC;QAChB,IAAI,EAAE,MAAM,CAAC;QACb,OAAO,EAAE,MAAM,CAAC;QAChB,QAAQ,EAAE,MAAM,EAAE,CAAC;QACnB,QAAQ,EAAE,MAAM,CAAC;QACjB,WAAW,EAAE,MAAM,CAAC;QACpB,IAAI,EAAE;YACF,SAAS,EAAE,MAAM,CAAC;YAClB,YAAY,EAAE,MAAM,CAAC;YACrB,OAAO,EAAE,MAAM,CAAC;YAChB,MAAM,EAAE,MAAM,CAAC;YACf,SAAS,EAAE,MAAM,CAAC;SACrB,CAAC;KACL,CACJ,CAAC;CACL;AAED,MAAM,WAAW,gBAAgB;IAC7B,SAAS,EAAE,MAAM,CAAC;IAClB,KAAK,EAAE,MAAM,CAAC;IACd,GAAG,EAAE,MAAM,CAAC;IACZ,OAAO,EAAE,MAAM,CAAC;CACnB;AAID,wBAAsB,iBAAiB,CACnC,WAAW,EAAE,MAAM,GACpB,OAAO,CAAC,cAAc,GAAG,IAAI,CAAC,CAoBhC;AAED,wBAAsB,0BAA0B,CAC5C,WAAW,EAAE,MAAM,EACnB,MAAM,GAAE,UAAU,GAAG,WAAW,GAAG,YAAY,GAAG,WAA0B,GAC7E,OAAO,CAAC,gBAAgB,GAAG,IAAI,CAAC,CAoBlC"}
|
package/dist/util/npm.js
ADDED
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import got from "got";
|
|
2
|
+
const maxRetries = 3;
|
|
3
|
+
export async function getNpmPackageInfo(packageName) {
|
|
4
|
+
for (let i = 0; i < maxRetries; i++) {
|
|
5
|
+
try {
|
|
6
|
+
const response = await got(`https://registry.npmjs.org/${packageName}`);
|
|
7
|
+
if (response.statusCode !== 200) {
|
|
8
|
+
throw new Error(`HTTP ${response.statusCode}: ${response.statusMessage}`);
|
|
9
|
+
}
|
|
10
|
+
const packageData = JSON.parse(response.body);
|
|
11
|
+
return packageData;
|
|
12
|
+
}
|
|
13
|
+
catch (error) {
|
|
14
|
+
continue;
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
return null;
|
|
18
|
+
}
|
|
19
|
+
export async function getNpmPackageDownloadStats(packageName, period = "last-month") {
|
|
20
|
+
for (let i = 0; i < maxRetries; i++) {
|
|
21
|
+
try {
|
|
22
|
+
const response = await got(`https://api.npmjs.org/downloads/point/${period}/${packageName}`);
|
|
23
|
+
if (response.statusCode !== 200) {
|
|
24
|
+
throw new Error(`HTTP ${response.statusCode}: ${response.statusMessage}`);
|
|
25
|
+
}
|
|
26
|
+
const data = JSON.parse(response.body);
|
|
27
|
+
return data || null;
|
|
28
|
+
}
|
|
29
|
+
catch (error) {
|
|
30
|
+
continue;
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
return null;
|
|
34
|
+
}
|
|
35
|
+
//# sourceMappingURL=npm.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"npm.js","sourceRoot":"","sources":["../../src/util/npm.ts"],"names":[],"mappings":"AAAA,OAAO,GAAG,MAAM,KAAK,CAAC;AAkDtB,MAAM,UAAU,GAAG,CAAC,CAAC;AAErB,MAAM,CAAC,KAAK,UAAU,iBAAiB,CACnC,WAAmB;IAEnB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,UAAU,EAAE,CAAC,EAAE,EAAE,CAAC;QAClC,IAAI,CAAC;YACD,MAAM,QAAQ,GAAG,MAAM,GAAG,CACtB,8BAA8B,WAAW,EAAE,CAC9C,CAAC;YAEF,IAAI,QAAQ,CAAC,UAAU,KAAK,GAAG,EAAE,CAAC;gBAC9B,MAAM,IAAI,KAAK,CACX,QAAQ,QAAQ,CAAC,UAAU,KAAK,QAAQ,CAAC,aAAa,EAAE,CAC3D,CAAC;YACN,CAAC;YAED,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAmB,CAAC;YAChE,OAAO,WAAW,CAAC;QACvB,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACb,SAAS;QACb,CAAC;IACL,CAAC;IACD,OAAO,IAAI,CAAC;AAChB,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,0BAA0B,CAC5C,WAAmB,EACnB,SAAgE,YAAY;IAE5E,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,UAAU,EAAE,CAAC,EAAE,EAAE,CAAC;QAClC,IAAI,CAAC;YACD,MAAM,QAAQ,GAAG,MAAM,GAAG,CACtB,yCAAyC,MAAM,IAAI,WAAW,EAAE,CACnE,CAAC;YAEF,IAAI,QAAQ,CAAC,UAAU,KAAK,GAAG,EAAE,CAAC;gBAC9B,MAAM,IAAI,KAAK,CACX,QAAQ,QAAQ,CAAC,UAAU,KAAK,QAAQ,CAAC,aAAa,EAAE,CAC3D,CAAC;YACN,CAAC;YAED,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAqB,CAAC;YAC3D,OAAO,IAAI,IAAI,IAAI,CAAC;QACxB,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACb,SAAS;QACb,CAAC;IACL,CAAC;IACD,OAAO,IAAI,CAAC;AAChB,CAAC"}
|