@monodog/utils 1.0.0 → 1.0.2
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/dist/helpers.d.ts +93 -0
- package/dist/helpers.js +478 -0
- package/dist/helpers.js.map +1 -0
- package/helpers.d.ts +63 -48
- package/helpers.js +254 -248
- package/helpers.ts +1 -1
- package/package.json +1 -2
- package/tsconfig.json +1 -1
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
export interface PackageInfo {
|
|
2
|
+
name: string;
|
|
3
|
+
version: string;
|
|
4
|
+
type: string;
|
|
5
|
+
path: string;
|
|
6
|
+
dependencies: Record<string, string>;
|
|
7
|
+
devDependencies: Record<string, string>;
|
|
8
|
+
peerDependencies: Record<string, string>;
|
|
9
|
+
scripts: Record<string, string>;
|
|
10
|
+
maintainers: string[];
|
|
11
|
+
description?: string;
|
|
12
|
+
license?: string;
|
|
13
|
+
repository?: Record<string, string>;
|
|
14
|
+
}
|
|
15
|
+
export interface DependencyInfo {
|
|
16
|
+
name: string;
|
|
17
|
+
version: string;
|
|
18
|
+
type: 'dependency' | 'devDependency' | 'peerDependency';
|
|
19
|
+
latest?: string;
|
|
20
|
+
status?: 'up-to-date' | 'outdated' | 'major-update' | 'unknown';
|
|
21
|
+
outdated?: boolean;
|
|
22
|
+
}
|
|
23
|
+
export interface PackageHealth {
|
|
24
|
+
buildStatus: 'success' | 'failed' | 'running' | 'unknown';
|
|
25
|
+
testCoverage: number;
|
|
26
|
+
lintStatus: 'pass' | 'fail' | 'unknown';
|
|
27
|
+
securityAudit: 'pass' | 'fail' | 'unknown';
|
|
28
|
+
overallScore: number;
|
|
29
|
+
}
|
|
30
|
+
export interface MonorepoStats {
|
|
31
|
+
totalPackages: number;
|
|
32
|
+
apps: number;
|
|
33
|
+
libraries: number;
|
|
34
|
+
tools: number;
|
|
35
|
+
healthyPackages: number;
|
|
36
|
+
warningPackages: number;
|
|
37
|
+
errorPackages: number;
|
|
38
|
+
outdatedDependencies: number;
|
|
39
|
+
totalDependencies: number;
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Scans the monorepo and returns information about all packages
|
|
43
|
+
*/
|
|
44
|
+
declare function scanMonorepo(rootDir: string): PackageInfo[];
|
|
45
|
+
/**
|
|
46
|
+
* Analyzes dependencies and determines their status
|
|
47
|
+
*/
|
|
48
|
+
/**
|
|
49
|
+
* Calculates package health score based on various metrics
|
|
50
|
+
*/
|
|
51
|
+
declare function calculatePackageHealth(buildStatus: PackageHealth['buildStatus'], testCoverage: number, lintStatus: PackageHealth['lintStatus'], securityAudit: PackageHealth['securityAudit']): PackageHealth;
|
|
52
|
+
/**
|
|
53
|
+
* Generates comprehensive monorepo statistics
|
|
54
|
+
*/
|
|
55
|
+
declare function generateMonorepoStats(packages: PackageInfo[]): MonorepoStats;
|
|
56
|
+
/**
|
|
57
|
+
* Finds circular dependencies in the monorepo
|
|
58
|
+
*/
|
|
59
|
+
declare function findCircularDependencies(packages: PackageInfo[]): string[][];
|
|
60
|
+
/**
|
|
61
|
+
* Generates a dependency graph for visualization
|
|
62
|
+
*/
|
|
63
|
+
declare function generateDependencyGraph(packages: PackageInfo[]): {
|
|
64
|
+
nodes: {
|
|
65
|
+
label: string;
|
|
66
|
+
type: string;
|
|
67
|
+
version: string;
|
|
68
|
+
dependencies: number;
|
|
69
|
+
}[];
|
|
70
|
+
edges: {
|
|
71
|
+
from: string;
|
|
72
|
+
to: string;
|
|
73
|
+
type: string;
|
|
74
|
+
}[];
|
|
75
|
+
};
|
|
76
|
+
/**
|
|
77
|
+
* Checks if a package has outdated dependencies
|
|
78
|
+
*/
|
|
79
|
+
declare function checkOutdatedDependencies(packageInfo: PackageInfo): DependencyInfo[];
|
|
80
|
+
/**
|
|
81
|
+
* Formats version numbers for comparison
|
|
82
|
+
*/
|
|
83
|
+
/**
|
|
84
|
+
* Compares two version strings
|
|
85
|
+
*/
|
|
86
|
+
/**
|
|
87
|
+
* Gets package size information
|
|
88
|
+
*/
|
|
89
|
+
declare function getPackageSize(packagePath: string): {
|
|
90
|
+
size: number;
|
|
91
|
+
files: number;
|
|
92
|
+
};
|
|
93
|
+
export { scanMonorepo, generateMonorepoStats, findCircularDependencies, generateDependencyGraph, checkOutdatedDependencies, getPackageSize, calculatePackageHealth, };
|
package/dist/helpers.js
ADDED
|
@@ -0,0 +1,478 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __values = (this && this.__values) || function(o) {
|
|
36
|
+
var s = typeof Symbol === "function" && Symbol.iterator, m = s && o[s], i = 0;
|
|
37
|
+
if (m) return m.call(o);
|
|
38
|
+
if (o && typeof o.length === "number") return {
|
|
39
|
+
next: function () {
|
|
40
|
+
if (o && i >= o.length) o = void 0;
|
|
41
|
+
return { value: o && o[i++], done: !o };
|
|
42
|
+
}
|
|
43
|
+
};
|
|
44
|
+
throw new TypeError(s ? "Object is not iterable." : "Symbol.iterator is not defined.");
|
|
45
|
+
};
|
|
46
|
+
var __read = (this && this.__read) || function (o, n) {
|
|
47
|
+
var m = typeof Symbol === "function" && o[Symbol.iterator];
|
|
48
|
+
if (!m) return o;
|
|
49
|
+
var i = m.call(o), r, ar = [], e;
|
|
50
|
+
try {
|
|
51
|
+
while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);
|
|
52
|
+
}
|
|
53
|
+
catch (error) { e = { error: error }; }
|
|
54
|
+
finally {
|
|
55
|
+
try {
|
|
56
|
+
if (r && !r.done && (m = i["return"])) m.call(i);
|
|
57
|
+
}
|
|
58
|
+
finally { if (e) throw e.error; }
|
|
59
|
+
}
|
|
60
|
+
return ar;
|
|
61
|
+
};
|
|
62
|
+
var __spreadArray = (this && this.__spreadArray) || function (to, from, pack) {
|
|
63
|
+
if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {
|
|
64
|
+
if (ar || !(i in from)) {
|
|
65
|
+
if (!ar) ar = Array.prototype.slice.call(from, 0, i);
|
|
66
|
+
ar[i] = from[i];
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
return to.concat(ar || Array.prototype.slice.call(from));
|
|
70
|
+
};
|
|
71
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
72
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
73
|
+
};
|
|
74
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
75
|
+
exports.scanMonorepo = scanMonorepo;
|
|
76
|
+
exports.generateMonorepoStats = generateMonorepoStats;
|
|
77
|
+
exports.findCircularDependencies = findCircularDependencies;
|
|
78
|
+
exports.generateDependencyGraph = generateDependencyGraph;
|
|
79
|
+
exports.checkOutdatedDependencies = checkOutdatedDependencies;
|
|
80
|
+
exports.getPackageSize = getPackageSize;
|
|
81
|
+
exports.calculatePackageHealth = calculatePackageHealth;
|
|
82
|
+
// import { Package } from '@prisma/client';
|
|
83
|
+
var fs = __importStar(require("fs"));
|
|
84
|
+
var path_1 = __importDefault(require("path"));
|
|
85
|
+
/**
|
|
86
|
+
* Scans the monorepo and returns information about all packages
|
|
87
|
+
*/
|
|
88
|
+
function scanMonorepo(rootDir) {
|
|
89
|
+
var e_1, _a, e_2, _b, e_3, _c;
|
|
90
|
+
var packages = [];
|
|
91
|
+
console.log('rootDir', rootDir);
|
|
92
|
+
// Scan packages directory
|
|
93
|
+
var packagesDir = path_1.default.join(rootDir, 'packages');
|
|
94
|
+
if (fs.existsSync(packagesDir)) {
|
|
95
|
+
var packageDirs = fs
|
|
96
|
+
.readdirSync(packagesDir, { withFileTypes: true })
|
|
97
|
+
.filter(function (dirent) { return dirent.isDirectory(); })
|
|
98
|
+
.map(function (dirent) { return dirent.name; });
|
|
99
|
+
try {
|
|
100
|
+
for (var packageDirs_1 = __values(packageDirs), packageDirs_1_1 = packageDirs_1.next(); !packageDirs_1_1.done; packageDirs_1_1 = packageDirs_1.next()) {
|
|
101
|
+
var packageName = packageDirs_1_1.value;
|
|
102
|
+
var packagePath = path_1.default.join(packagesDir, packageName);
|
|
103
|
+
var packageInfo = parsePackageInfo(packagePath, packageName);
|
|
104
|
+
if (packageInfo) {
|
|
105
|
+
packages.push(packageInfo);
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
|
110
|
+
finally {
|
|
111
|
+
try {
|
|
112
|
+
if (packageDirs_1_1 && !packageDirs_1_1.done && (_a = packageDirs_1.return)) _a.call(packageDirs_1);
|
|
113
|
+
}
|
|
114
|
+
finally { if (e_1) throw e_1.error; }
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
// Scan apps directory
|
|
118
|
+
var appsDir = path_1.default.join(rootDir, 'apps');
|
|
119
|
+
if (fs.existsSync(appsDir)) {
|
|
120
|
+
var appDirs = fs
|
|
121
|
+
.readdirSync(appsDir, { withFileTypes: true })
|
|
122
|
+
.filter(function (dirent) { return dirent.isDirectory(); })
|
|
123
|
+
.map(function (dirent) { return dirent.name; });
|
|
124
|
+
try {
|
|
125
|
+
for (var appDirs_1 = __values(appDirs), appDirs_1_1 = appDirs_1.next(); !appDirs_1_1.done; appDirs_1_1 = appDirs_1.next()) {
|
|
126
|
+
var appName = appDirs_1_1.value;
|
|
127
|
+
var appPath = path_1.default.join(appsDir, appName);
|
|
128
|
+
var appInfo = parsePackageInfo(appPath, appName, 'app');
|
|
129
|
+
if (appInfo) {
|
|
130
|
+
packages.push(appInfo);
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
catch (e_2_1) { e_2 = { error: e_2_1 }; }
|
|
135
|
+
finally {
|
|
136
|
+
try {
|
|
137
|
+
if (appDirs_1_1 && !appDirs_1_1.done && (_b = appDirs_1.return)) _b.call(appDirs_1);
|
|
138
|
+
}
|
|
139
|
+
finally { if (e_2) throw e_2.error; }
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
// Scan libs directory
|
|
143
|
+
var libsDir = path_1.default.join(rootDir, 'libs');
|
|
144
|
+
if (fs.existsSync(libsDir)) {
|
|
145
|
+
var libDirs = fs
|
|
146
|
+
.readdirSync(libsDir, { withFileTypes: true })
|
|
147
|
+
.filter(function (dirent) { return dirent.isDirectory(); })
|
|
148
|
+
.map(function (dirent) { return dirent.name; });
|
|
149
|
+
try {
|
|
150
|
+
for (var libDirs_1 = __values(libDirs), libDirs_1_1 = libDirs_1.next(); !libDirs_1_1.done; libDirs_1_1 = libDirs_1.next()) {
|
|
151
|
+
var libName = libDirs_1_1.value;
|
|
152
|
+
var libPath = path_1.default.join(libsDir, libName);
|
|
153
|
+
var libInfo = parsePackageInfo(libPath, libName, 'lib');
|
|
154
|
+
if (libInfo) {
|
|
155
|
+
packages.push(libInfo);
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
catch (e_3_1) { e_3 = { error: e_3_1 }; }
|
|
160
|
+
finally {
|
|
161
|
+
try {
|
|
162
|
+
if (libDirs_1_1 && !libDirs_1_1.done && (_c = libDirs_1.return)) _c.call(libDirs_1);
|
|
163
|
+
}
|
|
164
|
+
finally { if (e_3) throw e_3.error; }
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
return packages;
|
|
168
|
+
}
|
|
169
|
+
/*** Parses package.json and determines package type */
|
|
170
|
+
function parsePackageInfo(packagePath, packageName, forcedType) {
|
|
171
|
+
var packageJsonPath = path_1.default.join(packagePath, 'package.json');
|
|
172
|
+
if (!fs.existsSync(packageJsonPath)) {
|
|
173
|
+
return null;
|
|
174
|
+
}
|
|
175
|
+
try {
|
|
176
|
+
var packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8'));
|
|
177
|
+
// Determine package type
|
|
178
|
+
var packageType = 'lib';
|
|
179
|
+
if (forcedType) {
|
|
180
|
+
packageType = forcedType;
|
|
181
|
+
}
|
|
182
|
+
else if (packageJson.scripts && packageJson.scripts.start) {
|
|
183
|
+
packageType = 'app';
|
|
184
|
+
}
|
|
185
|
+
else if (packageJson.keywords && packageJson.keywords.includes('tool')) {
|
|
186
|
+
packageType = 'tool';
|
|
187
|
+
}
|
|
188
|
+
return {
|
|
189
|
+
name: packageJson.name || packageName,
|
|
190
|
+
version: packageJson.version || '0.0.0',
|
|
191
|
+
type: packageType,
|
|
192
|
+
path: packagePath,
|
|
193
|
+
dependencies: packageJson.dependencies || {},
|
|
194
|
+
devDependencies: packageJson.devDependencies || {},
|
|
195
|
+
peerDependencies: packageJson.peerDependencies || {},
|
|
196
|
+
scripts: packageJson.scripts || {},
|
|
197
|
+
maintainers: packageJson.maintainers || [],
|
|
198
|
+
description: packageJson.description,
|
|
199
|
+
license: packageJson.license,
|
|
200
|
+
repository: packageJson.repository || {},
|
|
201
|
+
};
|
|
202
|
+
}
|
|
203
|
+
catch (error) {
|
|
204
|
+
console.error("Error parsing package.json for ".concat(packageName, ":"), error);
|
|
205
|
+
return null;
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
/**
|
|
209
|
+
* Analyzes dependencies and determines their status
|
|
210
|
+
*/
|
|
211
|
+
// function analyzeDependencies(
|
|
212
|
+
// dependencies: Record<string, string>,
|
|
213
|
+
// type: 'production' | 'development' = 'production'
|
|
214
|
+
// ): DependencyInfo[] {
|
|
215
|
+
// return Object.entries(dependencies).map(([name, version]) => ({
|
|
216
|
+
// name,
|
|
217
|
+
// currentVersion: version,
|
|
218
|
+
// status: 'unknown', // Would be determined by npm registry check
|
|
219
|
+
// type,
|
|
220
|
+
// }));
|
|
221
|
+
// }
|
|
222
|
+
/**
|
|
223
|
+
* Calculates package health score based on various metrics
|
|
224
|
+
*/
|
|
225
|
+
function calculatePackageHealth(buildStatus, testCoverage, lintStatus, securityAudit) {
|
|
226
|
+
var score = 0;
|
|
227
|
+
// Build status (30 points)
|
|
228
|
+
switch (buildStatus) {
|
|
229
|
+
case 'success':
|
|
230
|
+
score += 30;
|
|
231
|
+
break;
|
|
232
|
+
case 'running':
|
|
233
|
+
score += 15;
|
|
234
|
+
break;
|
|
235
|
+
case 'failed':
|
|
236
|
+
score += 0;
|
|
237
|
+
break;
|
|
238
|
+
default:
|
|
239
|
+
score += 10;
|
|
240
|
+
}
|
|
241
|
+
// Test coverage (25 points)
|
|
242
|
+
score += Math.min(25, (testCoverage / 100) * 25);
|
|
243
|
+
// Lint status (25 points)
|
|
244
|
+
switch (lintStatus) {
|
|
245
|
+
case 'pass':
|
|
246
|
+
score += 25;
|
|
247
|
+
break;
|
|
248
|
+
case 'fail':
|
|
249
|
+
score += 0;
|
|
250
|
+
break;
|
|
251
|
+
default:
|
|
252
|
+
score += 10;
|
|
253
|
+
}
|
|
254
|
+
// Security audit (20 points)
|
|
255
|
+
switch (securityAudit) {
|
|
256
|
+
case 'pass':
|
|
257
|
+
score += 20;
|
|
258
|
+
break;
|
|
259
|
+
case 'fail':
|
|
260
|
+
score += 0;
|
|
261
|
+
break;
|
|
262
|
+
default:
|
|
263
|
+
score += 10;
|
|
264
|
+
}
|
|
265
|
+
return {
|
|
266
|
+
buildStatus: buildStatus,
|
|
267
|
+
testCoverage: testCoverage,
|
|
268
|
+
lintStatus: lintStatus,
|
|
269
|
+
securityAudit: securityAudit,
|
|
270
|
+
overallScore: Math.round(score),
|
|
271
|
+
};
|
|
272
|
+
}
|
|
273
|
+
/**
|
|
274
|
+
* Generates comprehensive monorepo statistics
|
|
275
|
+
*/
|
|
276
|
+
function generateMonorepoStats(packages) {
|
|
277
|
+
var stats = {
|
|
278
|
+
totalPackages: packages.length,
|
|
279
|
+
apps: packages.filter(function (p) { return p.type === 'app'; }).length,
|
|
280
|
+
libraries: packages.filter(function (p) { return p.type === 'lib'; }).length,
|
|
281
|
+
tools: packages.filter(function (p) { return p.type === 'tool'; }).length,
|
|
282
|
+
healthyPackages: 0,
|
|
283
|
+
warningPackages: 0,
|
|
284
|
+
errorPackages: 0,
|
|
285
|
+
outdatedDependencies: 0,
|
|
286
|
+
totalDependencies: 0,
|
|
287
|
+
};
|
|
288
|
+
// Calculate dependency counts
|
|
289
|
+
packages.forEach(function (pkg) {
|
|
290
|
+
var _a;
|
|
291
|
+
stats.totalDependencies += Object.keys(pkg.dependencies).length;
|
|
292
|
+
stats.totalDependencies += Object.keys(pkg.devDependencies).length;
|
|
293
|
+
stats.totalDependencies += Object.keys((_a = pkg.peerDependencies) !== null && _a !== void 0 ? _a : {}).length;
|
|
294
|
+
});
|
|
295
|
+
return stats;
|
|
296
|
+
}
|
|
297
|
+
/**
|
|
298
|
+
* Finds circular dependencies in the monorepo
|
|
299
|
+
*/
|
|
300
|
+
function findCircularDependencies(packages) {
|
|
301
|
+
var e_4, _a;
|
|
302
|
+
var graph = new Map();
|
|
303
|
+
var visited = new Set();
|
|
304
|
+
var recursionStack = new Set();
|
|
305
|
+
var circularDeps = [];
|
|
306
|
+
// Build dependency graph
|
|
307
|
+
packages.forEach(function (pkg) {
|
|
308
|
+
graph.set(pkg.name, Object.keys(pkg.dependencies));
|
|
309
|
+
});
|
|
310
|
+
function dfs(node, path) {
|
|
311
|
+
var e_5, _a;
|
|
312
|
+
if (recursionStack.has(node)) {
|
|
313
|
+
var cycleStart = path.indexOf(node);
|
|
314
|
+
circularDeps.push(path.slice(cycleStart));
|
|
315
|
+
return;
|
|
316
|
+
}
|
|
317
|
+
if (visited.has(node)) {
|
|
318
|
+
return;
|
|
319
|
+
}
|
|
320
|
+
visited.add(node);
|
|
321
|
+
recursionStack.add(node);
|
|
322
|
+
path.push(node);
|
|
323
|
+
var dependencies = graph.get(node) || [];
|
|
324
|
+
try {
|
|
325
|
+
for (var dependencies_1 = __values(dependencies), dependencies_1_1 = dependencies_1.next(); !dependencies_1_1.done; dependencies_1_1 = dependencies_1.next()) {
|
|
326
|
+
var dep = dependencies_1_1.value;
|
|
327
|
+
if (graph.has(dep)) {
|
|
328
|
+
dfs(dep, __spreadArray([], __read(path), false));
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
}
|
|
332
|
+
catch (e_5_1) { e_5 = { error: e_5_1 }; }
|
|
333
|
+
finally {
|
|
334
|
+
try {
|
|
335
|
+
if (dependencies_1_1 && !dependencies_1_1.done && (_a = dependencies_1.return)) _a.call(dependencies_1);
|
|
336
|
+
}
|
|
337
|
+
finally { if (e_5) throw e_5.error; }
|
|
338
|
+
}
|
|
339
|
+
recursionStack.delete(node);
|
|
340
|
+
}
|
|
341
|
+
try {
|
|
342
|
+
for (var _b = __values(graph.keys()), _c = _b.next(); !_c.done; _c = _b.next()) {
|
|
343
|
+
var node = _c.value;
|
|
344
|
+
if (!visited.has(node)) {
|
|
345
|
+
dfs(node, []);
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
catch (e_4_1) { e_4 = { error: e_4_1 }; }
|
|
350
|
+
finally {
|
|
351
|
+
try {
|
|
352
|
+
if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
|
|
353
|
+
}
|
|
354
|
+
finally { if (e_4) throw e_4.error; }
|
|
355
|
+
}
|
|
356
|
+
return circularDeps;
|
|
357
|
+
}
|
|
358
|
+
/**
|
|
359
|
+
* Generates a dependency graph for visualization
|
|
360
|
+
*/
|
|
361
|
+
function generateDependencyGraph(packages) {
|
|
362
|
+
var nodes = packages.map(function (pkg) { return ({
|
|
363
|
+
// id: pkg.name,
|
|
364
|
+
label: pkg.name,
|
|
365
|
+
type: pkg.type,
|
|
366
|
+
version: pkg.version,
|
|
367
|
+
dependencies: Object.keys(pkg.dependencies).length,
|
|
368
|
+
}); });
|
|
369
|
+
var edges = [];
|
|
370
|
+
packages.forEach(function (pkg) {
|
|
371
|
+
Object.keys(pkg.dependencies).forEach(function (depName) {
|
|
372
|
+
// Only include internal dependencies
|
|
373
|
+
if (packages.some(function (p) { return p.name === depName; })) {
|
|
374
|
+
edges.push({
|
|
375
|
+
from: pkg.name,
|
|
376
|
+
to: depName,
|
|
377
|
+
type: 'internal',
|
|
378
|
+
});
|
|
379
|
+
}
|
|
380
|
+
});
|
|
381
|
+
});
|
|
382
|
+
return { nodes: nodes, edges: edges };
|
|
383
|
+
}
|
|
384
|
+
/**
|
|
385
|
+
* Checks if a package has outdated dependencies
|
|
386
|
+
*/
|
|
387
|
+
function checkOutdatedDependencies(packageInfo) {
|
|
388
|
+
var outdated = [];
|
|
389
|
+
// This would typically involve checking against npm registry
|
|
390
|
+
// For now, we'll simulate with some basic checks
|
|
391
|
+
Object.entries(packageInfo.dependencies).forEach(function (_a) {
|
|
392
|
+
var _b = __read(_a, 2), name = _b[0], version = _b[1];
|
|
393
|
+
if (version.startsWith('^') || version.startsWith('~')) {
|
|
394
|
+
// Could be outdated, would need registry check
|
|
395
|
+
outdated.push({
|
|
396
|
+
name: name,
|
|
397
|
+
version: version,
|
|
398
|
+
status: 'unknown',
|
|
399
|
+
type: 'dependency',
|
|
400
|
+
});
|
|
401
|
+
}
|
|
402
|
+
});
|
|
403
|
+
return outdated;
|
|
404
|
+
}
|
|
405
|
+
/**
|
|
406
|
+
* Formats version numbers for comparison
|
|
407
|
+
*/
|
|
408
|
+
// function parseVersion(version: string): number[] {
|
|
409
|
+
// return version
|
|
410
|
+
// .replace(/^[^0-9]*/, '')
|
|
411
|
+
// .split('.')
|
|
412
|
+
// .map(Number);
|
|
413
|
+
// }
|
|
414
|
+
/**
|
|
415
|
+
* Compares two version strings
|
|
416
|
+
*/
|
|
417
|
+
// function compareVersions(v1: string, v2: string): number {
|
|
418
|
+
// const parsed1 = parseVersion(v1);
|
|
419
|
+
// const parsed2 = parseVersion(v2);
|
|
420
|
+
// for (let i = 0; i < Math.max(parsed1.length, parsed2.length); i++) {
|
|
421
|
+
// const num1 = parsed1[i] || 0;
|
|
422
|
+
// const num2 = parsed2[i] || 0;
|
|
423
|
+
// if (num1 > num2) return 1;
|
|
424
|
+
// if (num1 < num2) return -1;
|
|
425
|
+
// }
|
|
426
|
+
// return 0;
|
|
427
|
+
// }
|
|
428
|
+
/**
|
|
429
|
+
* Gets package size information
|
|
430
|
+
*/
|
|
431
|
+
function getPackageSize(packagePath) {
|
|
432
|
+
try {
|
|
433
|
+
var totalSize_1 = 0;
|
|
434
|
+
var fileCount_1 = 0;
|
|
435
|
+
var calculateSize_1 = function (dirPath) {
|
|
436
|
+
var e_6, _a;
|
|
437
|
+
var items = fs.readdirSync(dirPath, { withFileTypes: true });
|
|
438
|
+
try {
|
|
439
|
+
for (var items_1 = __values(items), items_1_1 = items_1.next(); !items_1_1.done; items_1_1 = items_1.next()) {
|
|
440
|
+
var item = items_1_1.value;
|
|
441
|
+
var fullPath = path_1.default.join(dirPath, item.name);
|
|
442
|
+
if (item.isDirectory()) {
|
|
443
|
+
// Skip node_modules and other build artifacts
|
|
444
|
+
if (!['node_modules', 'dist', 'build', '.git'].includes(item.name)) {
|
|
445
|
+
calculateSize_1(fullPath);
|
|
446
|
+
}
|
|
447
|
+
}
|
|
448
|
+
else {
|
|
449
|
+
try {
|
|
450
|
+
var stats = fs.statSync(fullPath);
|
|
451
|
+
totalSize_1 += stats.size;
|
|
452
|
+
fileCount_1++;
|
|
453
|
+
}
|
|
454
|
+
catch (error) {
|
|
455
|
+
// Skip files we can't read
|
|
456
|
+
}
|
|
457
|
+
}
|
|
458
|
+
}
|
|
459
|
+
}
|
|
460
|
+
catch (e_6_1) { e_6 = { error: e_6_1 }; }
|
|
461
|
+
finally {
|
|
462
|
+
try {
|
|
463
|
+
if (items_1_1 && !items_1_1.done && (_a = items_1.return)) _a.call(items_1);
|
|
464
|
+
}
|
|
465
|
+
finally { if (e_6) throw e_6.error; }
|
|
466
|
+
}
|
|
467
|
+
};
|
|
468
|
+
calculateSize_1(packagePath);
|
|
469
|
+
return {
|
|
470
|
+
size: totalSize_1,
|
|
471
|
+
files: fileCount_1,
|
|
472
|
+
};
|
|
473
|
+
}
|
|
474
|
+
catch (error) {
|
|
475
|
+
return { size: 0, files: 0 };
|
|
476
|
+
}
|
|
477
|
+
}
|
|
478
|
+
//# sourceMappingURL=helpers.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"helpers.js","sourceRoot":"","sources":["../helpers.ts"],"names":[],"mappings":"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"}
|
package/helpers.d.ts
CHANGED
|
@@ -1,42 +1,42 @@
|
|
|
1
1
|
export interface PackageInfo {
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
2
|
+
name: string;
|
|
3
|
+
version: string;
|
|
4
|
+
type: string;
|
|
5
|
+
path: string;
|
|
6
|
+
dependencies: Record<string, string>;
|
|
7
|
+
devDependencies: Record<string, string>;
|
|
8
|
+
peerDependencies: Record<string, string>;
|
|
9
|
+
scripts: Record<string, string>;
|
|
10
|
+
maintainers: string[];
|
|
11
|
+
description?: string;
|
|
12
|
+
license?: string;
|
|
13
|
+
repository?: Record<string, string>;
|
|
14
14
|
}
|
|
15
15
|
export interface DependencyInfo {
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
16
|
+
name: string;
|
|
17
|
+
version: string;
|
|
18
|
+
type: 'dependency' | 'devDependency' | 'peerDependency';
|
|
19
|
+
latest?: string;
|
|
20
|
+
status?: 'up-to-date' | 'outdated' | 'major-update' | 'unknown';
|
|
21
|
+
outdated?: boolean;
|
|
22
22
|
}
|
|
23
23
|
export interface PackageHealth {
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
24
|
+
buildStatus: 'success' | 'failed' | 'running' | 'unknown';
|
|
25
|
+
testCoverage: number;
|
|
26
|
+
lintStatus: 'pass' | 'fail' | 'unknown';
|
|
27
|
+
securityAudit: 'pass' | 'fail' | 'unknown';
|
|
28
|
+
overallScore: number;
|
|
29
29
|
}
|
|
30
30
|
export interface MonorepoStats {
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
31
|
+
totalPackages: number;
|
|
32
|
+
apps: number;
|
|
33
|
+
libraries: number;
|
|
34
|
+
tools: number;
|
|
35
|
+
healthyPackages: number;
|
|
36
|
+
warningPackages: number;
|
|
37
|
+
errorPackages: number;
|
|
38
|
+
outdatedDependencies: number;
|
|
39
|
+
totalDependencies: number;
|
|
40
40
|
}
|
|
41
41
|
/**
|
|
42
42
|
* Scans the monorepo and returns information about all packages
|
|
@@ -48,7 +48,12 @@ declare function scanMonorepo(rootDir: string): PackageInfo[];
|
|
|
48
48
|
/**
|
|
49
49
|
* Calculates package health score based on various metrics
|
|
50
50
|
*/
|
|
51
|
-
declare function calculatePackageHealth(
|
|
51
|
+
declare function calculatePackageHealth(
|
|
52
|
+
buildStatus: PackageHealth['buildStatus'],
|
|
53
|
+
testCoverage: number,
|
|
54
|
+
lintStatus: PackageHealth['lintStatus'],
|
|
55
|
+
securityAudit: PackageHealth['securityAudit']
|
|
56
|
+
): PackageHealth;
|
|
52
57
|
/**
|
|
53
58
|
* Generates comprehensive monorepo statistics
|
|
54
59
|
*/
|
|
@@ -61,27 +66,37 @@ declare function findCircularDependencies(packages: PackageInfo[]): string[][];
|
|
|
61
66
|
* Generates a dependency graph for visualization
|
|
62
67
|
*/
|
|
63
68
|
declare function generateDependencyGraph(packages: PackageInfo[]): {
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
69
|
+
nodes: {
|
|
70
|
+
label: string;
|
|
71
|
+
type: string;
|
|
72
|
+
version: string;
|
|
73
|
+
dependencies: number;
|
|
74
|
+
}[];
|
|
75
|
+
edges: {
|
|
76
|
+
from: string;
|
|
77
|
+
to: string;
|
|
78
|
+
type: string;
|
|
79
|
+
}[];
|
|
75
80
|
};
|
|
76
81
|
/**
|
|
77
82
|
* Checks if a package has outdated dependencies
|
|
78
83
|
*/
|
|
79
|
-
declare function checkOutdatedDependencies(
|
|
84
|
+
declare function checkOutdatedDependencies(
|
|
85
|
+
packageInfo: PackageInfo
|
|
86
|
+
): DependencyInfo[];
|
|
80
87
|
/**
|
|
81
88
|
* Gets package size information
|
|
82
89
|
*/
|
|
83
90
|
declare function getPackageSize(packagePath: string): {
|
|
84
|
-
|
|
85
|
-
|
|
91
|
+
size: number;
|
|
92
|
+
files: number;
|
|
93
|
+
};
|
|
94
|
+
export {
|
|
95
|
+
scanMonorepo,
|
|
96
|
+
generateMonorepoStats,
|
|
97
|
+
findCircularDependencies,
|
|
98
|
+
generateDependencyGraph,
|
|
99
|
+
checkOutdatedDependencies,
|
|
100
|
+
getPackageSize,
|
|
101
|
+
calculatePackageHealth,
|
|
86
102
|
};
|
|
87
|
-
export { scanMonorepo, generateMonorepoStats, findCircularDependencies, generateDependencyGraph, checkOutdatedDependencies, getPackageSize, calculatePackageHealth, };
|
package/helpers.js
CHANGED
|
@@ -5,93 +5,90 @@ import path from 'path';
|
|
|
5
5
|
* Scans the monorepo and returns information about all packages
|
|
6
6
|
*/
|
|
7
7
|
function scanMonorepo(rootDir) {
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
}
|
|
8
|
+
const packages = [];
|
|
9
|
+
console.log('rootDir', rootDir);
|
|
10
|
+
// Scan packages directory
|
|
11
|
+
const packagesDir = path.join(rootDir, 'packages');
|
|
12
|
+
if (fs.existsSync(packagesDir)) {
|
|
13
|
+
const packageDirs = fs
|
|
14
|
+
.readdirSync(packagesDir, { withFileTypes: true })
|
|
15
|
+
.filter(dirent => dirent.isDirectory())
|
|
16
|
+
.map(dirent => dirent.name);
|
|
17
|
+
for (const packageName of packageDirs) {
|
|
18
|
+
const packagePath = path.join(packagesDir, packageName);
|
|
19
|
+
const packageInfo = parsePackageInfo(packagePath, packageName);
|
|
20
|
+
if (packageInfo) {
|
|
21
|
+
packages.push(packageInfo);
|
|
22
|
+
}
|
|
24
23
|
}
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
24
|
+
}
|
|
25
|
+
// Scan apps directory
|
|
26
|
+
const appsDir = path.join(rootDir, 'apps');
|
|
27
|
+
if (fs.existsSync(appsDir)) {
|
|
28
|
+
const appDirs = fs
|
|
29
|
+
.readdirSync(appsDir, { withFileTypes: true })
|
|
30
|
+
.filter(dirent => dirent.isDirectory())
|
|
31
|
+
.map(dirent => dirent.name);
|
|
32
|
+
for (const appName of appDirs) {
|
|
33
|
+
const appPath = path.join(appsDir, appName);
|
|
34
|
+
const appInfo = parsePackageInfo(appPath, appName, 'app');
|
|
35
|
+
if (appInfo) {
|
|
36
|
+
packages.push(appInfo);
|
|
37
|
+
}
|
|
39
38
|
}
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
39
|
+
}
|
|
40
|
+
// Scan libs directory
|
|
41
|
+
const libsDir = path.join(rootDir, 'libs');
|
|
42
|
+
if (fs.existsSync(libsDir)) {
|
|
43
|
+
const libDirs = fs
|
|
44
|
+
.readdirSync(libsDir, { withFileTypes: true })
|
|
45
|
+
.filter(dirent => dirent.isDirectory())
|
|
46
|
+
.map(dirent => dirent.name);
|
|
47
|
+
for (const libName of libDirs) {
|
|
48
|
+
const libPath = path.join(libsDir, libName);
|
|
49
|
+
const libInfo = parsePackageInfo(libPath, libName, 'lib');
|
|
50
|
+
if (libInfo) {
|
|
51
|
+
packages.push(libInfo);
|
|
52
|
+
}
|
|
54
53
|
}
|
|
55
|
-
|
|
54
|
+
}
|
|
55
|
+
return packages;
|
|
56
56
|
}
|
|
57
57
|
/*** Parses package.json and determines package type */
|
|
58
58
|
function parsePackageInfo(packagePath, packageName, forcedType) {
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
else if (packageJson.keywords && packageJson.keywords.includes('tool')) {
|
|
74
|
-
packageType = 'tool';
|
|
75
|
-
}
|
|
76
|
-
return {
|
|
77
|
-
name: packageJson.name || packageName,
|
|
78
|
-
version: packageJson.version || '0.0.0',
|
|
79
|
-
type: packageType,
|
|
80
|
-
path: packagePath,
|
|
81
|
-
dependencies: packageJson.dependencies || {},
|
|
82
|
-
devDependencies: packageJson.devDependencies || {},
|
|
83
|
-
peerDependencies: packageJson.peerDependencies || {},
|
|
84
|
-
scripts: packageJson.scripts || {},
|
|
85
|
-
maintainers: packageJson.maintainers || [],
|
|
86
|
-
description: packageJson.description,
|
|
87
|
-
license: packageJson.license,
|
|
88
|
-
repository: packageJson.repository || {},
|
|
89
|
-
};
|
|
90
|
-
}
|
|
91
|
-
catch (error) {
|
|
92
|
-
console.error(`Error parsing package.json for ${packageName}:`, error);
|
|
93
|
-
return null;
|
|
59
|
+
const packageJsonPath = path.join(packagePath, 'package.json');
|
|
60
|
+
if (!fs.existsSync(packageJsonPath)) {
|
|
61
|
+
return null;
|
|
62
|
+
}
|
|
63
|
+
try {
|
|
64
|
+
const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8'));
|
|
65
|
+
// Determine package type
|
|
66
|
+
let packageType = 'lib';
|
|
67
|
+
if (forcedType) {
|
|
68
|
+
packageType = forcedType;
|
|
69
|
+
} else if (packageJson.scripts && packageJson.scripts.start) {
|
|
70
|
+
packageType = 'app';
|
|
71
|
+
} else if (packageJson.keywords && packageJson.keywords.includes('tool')) {
|
|
72
|
+
packageType = 'tool';
|
|
94
73
|
}
|
|
74
|
+
return {
|
|
75
|
+
name: packageJson.name || packageName,
|
|
76
|
+
version: packageJson.version || '0.0.0',
|
|
77
|
+
type: packageType,
|
|
78
|
+
path: packagePath,
|
|
79
|
+
dependencies: packageJson.dependencies || {},
|
|
80
|
+
devDependencies: packageJson.devDependencies || {},
|
|
81
|
+
peerDependencies: packageJson.peerDependencies || {},
|
|
82
|
+
scripts: packageJson.scripts || {},
|
|
83
|
+
maintainers: packageJson.maintainers || [],
|
|
84
|
+
description: packageJson.description,
|
|
85
|
+
license: packageJson.license,
|
|
86
|
+
repository: packageJson.repository || {},
|
|
87
|
+
};
|
|
88
|
+
} catch (error) {
|
|
89
|
+
console.error(`Error parsing package.json for ${packageName}:`, error);
|
|
90
|
+
return null;
|
|
91
|
+
}
|
|
95
92
|
}
|
|
96
93
|
/**
|
|
97
94
|
* Analyzes dependencies and determines their status
|
|
@@ -110,161 +107,166 @@ function parsePackageInfo(packagePath, packageName, forcedType) {
|
|
|
110
107
|
/**
|
|
111
108
|
* Calculates package health score based on various metrics
|
|
112
109
|
*/
|
|
113
|
-
function calculatePackageHealth(
|
|
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
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
110
|
+
function calculatePackageHealth(
|
|
111
|
+
buildStatus,
|
|
112
|
+
testCoverage,
|
|
113
|
+
lintStatus,
|
|
114
|
+
securityAudit
|
|
115
|
+
) {
|
|
116
|
+
let score = 0;
|
|
117
|
+
// Build status (30 points)
|
|
118
|
+
switch (buildStatus) {
|
|
119
|
+
case 'success':
|
|
120
|
+
score += 30;
|
|
121
|
+
break;
|
|
122
|
+
case 'running':
|
|
123
|
+
score += 15;
|
|
124
|
+
break;
|
|
125
|
+
case 'failed':
|
|
126
|
+
score += 0;
|
|
127
|
+
break;
|
|
128
|
+
default:
|
|
129
|
+
score += 10;
|
|
130
|
+
}
|
|
131
|
+
// Test coverage (25 points)
|
|
132
|
+
score += Math.min(25, (testCoverage / 100) * 25);
|
|
133
|
+
// Lint status (25 points)
|
|
134
|
+
switch (lintStatus) {
|
|
135
|
+
case 'pass':
|
|
136
|
+
score += 25;
|
|
137
|
+
break;
|
|
138
|
+
case 'fail':
|
|
139
|
+
score += 0;
|
|
140
|
+
break;
|
|
141
|
+
default:
|
|
142
|
+
score += 10;
|
|
143
|
+
}
|
|
144
|
+
// Security audit (20 points)
|
|
145
|
+
switch (securityAudit) {
|
|
146
|
+
case 'pass':
|
|
147
|
+
score += 20;
|
|
148
|
+
break;
|
|
149
|
+
case 'fail':
|
|
150
|
+
score += 0;
|
|
151
|
+
break;
|
|
152
|
+
default:
|
|
153
|
+
score += 10;
|
|
154
|
+
}
|
|
155
|
+
return {
|
|
156
|
+
buildStatus,
|
|
157
|
+
testCoverage,
|
|
158
|
+
lintStatus,
|
|
159
|
+
securityAudit,
|
|
160
|
+
overallScore: Math.round(score),
|
|
161
|
+
};
|
|
160
162
|
}
|
|
161
163
|
/**
|
|
162
164
|
* Generates comprehensive monorepo statistics
|
|
163
165
|
*/
|
|
164
166
|
function generateMonorepoStats(packages) {
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
167
|
+
const stats = {
|
|
168
|
+
totalPackages: packages.length,
|
|
169
|
+
apps: packages.filter(p => p.type === 'app').length,
|
|
170
|
+
libraries: packages.filter(p => p.type === 'lib').length,
|
|
171
|
+
tools: packages.filter(p => p.type === 'tool').length,
|
|
172
|
+
healthyPackages: 0,
|
|
173
|
+
warningPackages: 0,
|
|
174
|
+
errorPackages: 0,
|
|
175
|
+
outdatedDependencies: 0,
|
|
176
|
+
totalDependencies: 0,
|
|
177
|
+
};
|
|
178
|
+
// Calculate dependency counts
|
|
179
|
+
packages.forEach(pkg => {
|
|
180
|
+
stats.totalDependencies += Object.keys(pkg.dependencies).length;
|
|
181
|
+
stats.totalDependencies += Object.keys(pkg.devDependencies).length;
|
|
182
|
+
stats.totalDependencies += Object.keys(pkg.peerDependencies ?? {}).length;
|
|
183
|
+
});
|
|
184
|
+
return stats;
|
|
183
185
|
}
|
|
184
186
|
/**
|
|
185
187
|
* Finds circular dependencies in the monorepo
|
|
186
188
|
*/
|
|
187
189
|
function findCircularDependencies(packages) {
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
}
|
|
202
|
-
if (visited.has(node)) {
|
|
203
|
-
return;
|
|
204
|
-
}
|
|
205
|
-
visited.add(node);
|
|
206
|
-
recursionStack.add(node);
|
|
207
|
-
path.push(node);
|
|
208
|
-
const dependencies = graph.get(node) || [];
|
|
209
|
-
for (const dep of dependencies) {
|
|
210
|
-
if (graph.has(dep)) {
|
|
211
|
-
dfs(dep, [...path]);
|
|
212
|
-
}
|
|
213
|
-
}
|
|
214
|
-
recursionStack.delete(node);
|
|
190
|
+
const graph = new Map();
|
|
191
|
+
const visited = new Set();
|
|
192
|
+
const recursionStack = new Set();
|
|
193
|
+
const circularDeps = [];
|
|
194
|
+
// Build dependency graph
|
|
195
|
+
packages.forEach(pkg => {
|
|
196
|
+
graph.set(pkg.name, Object.keys(pkg.dependencies));
|
|
197
|
+
});
|
|
198
|
+
function dfs(node, path) {
|
|
199
|
+
if (recursionStack.has(node)) {
|
|
200
|
+
const cycleStart = path.indexOf(node);
|
|
201
|
+
circularDeps.push(path.slice(cycleStart));
|
|
202
|
+
return;
|
|
215
203
|
}
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
204
|
+
if (visited.has(node)) {
|
|
205
|
+
return;
|
|
206
|
+
}
|
|
207
|
+
visited.add(node);
|
|
208
|
+
recursionStack.add(node);
|
|
209
|
+
path.push(node);
|
|
210
|
+
const dependencies = graph.get(node) || [];
|
|
211
|
+
for (const dep of dependencies) {
|
|
212
|
+
if (graph.has(dep)) {
|
|
213
|
+
dfs(dep, [...path]);
|
|
214
|
+
}
|
|
220
215
|
}
|
|
221
|
-
|
|
216
|
+
recursionStack.delete(node);
|
|
217
|
+
}
|
|
218
|
+
for (const node of graph.keys()) {
|
|
219
|
+
if (!visited.has(node)) {
|
|
220
|
+
dfs(node, []);
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
return circularDeps;
|
|
222
224
|
}
|
|
223
225
|
/**
|
|
224
226
|
* Generates a dependency graph for visualization
|
|
225
227
|
*/
|
|
226
228
|
function generateDependencyGraph(packages) {
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
});
|
|
244
|
-
}
|
|
229
|
+
const nodes = packages.map(pkg => ({
|
|
230
|
+
// id: pkg.name,
|
|
231
|
+
label: pkg.name,
|
|
232
|
+
type: pkg.type,
|
|
233
|
+
version: pkg.version,
|
|
234
|
+
dependencies: Object.keys(pkg.dependencies).length,
|
|
235
|
+
}));
|
|
236
|
+
const edges = [];
|
|
237
|
+
packages.forEach(pkg => {
|
|
238
|
+
Object.keys(pkg.dependencies).forEach(depName => {
|
|
239
|
+
// Only include internal dependencies
|
|
240
|
+
if (packages.some(p => p.name === depName)) {
|
|
241
|
+
edges.push({
|
|
242
|
+
from: pkg.name,
|
|
243
|
+
to: depName,
|
|
244
|
+
type: 'internal',
|
|
245
245
|
});
|
|
246
|
+
}
|
|
246
247
|
});
|
|
247
|
-
|
|
248
|
+
});
|
|
249
|
+
return { nodes, edges };
|
|
248
250
|
}
|
|
249
251
|
/**
|
|
250
252
|
* Checks if a package has outdated dependencies
|
|
251
253
|
*/
|
|
252
254
|
function checkOutdatedDependencies(packageInfo) {
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
255
|
+
const outdated = [];
|
|
256
|
+
// This would typically involve checking against npm registry
|
|
257
|
+
// For now, we'll simulate with some basic checks
|
|
258
|
+
Object.entries(packageInfo.dependencies).forEach(([name, version]) => {
|
|
259
|
+
if (version.startsWith('^') || version.startsWith('~')) {
|
|
260
|
+
// Could be outdated, would need registry check
|
|
261
|
+
outdated.push({
|
|
262
|
+
name,
|
|
263
|
+
version: version,
|
|
264
|
+
status: 'unknown',
|
|
265
|
+
type: 'dependency',
|
|
266
|
+
});
|
|
267
|
+
}
|
|
268
|
+
});
|
|
269
|
+
return outdated;
|
|
268
270
|
}
|
|
269
271
|
/**
|
|
270
272
|
* Formats version numbers for comparison
|
|
@@ -293,41 +295,45 @@ function checkOutdatedDependencies(packageInfo) {
|
|
|
293
295
|
* Gets package size information
|
|
294
296
|
*/
|
|
295
297
|
function getPackageSize(packagePath) {
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
catch (error) {
|
|
328
|
-
return { size: 0, files: 0 };
|
|
329
|
-
}
|
|
298
|
+
try {
|
|
299
|
+
let totalSize = 0;
|
|
300
|
+
let fileCount = 0;
|
|
301
|
+
const calculateSize = dirPath => {
|
|
302
|
+
const items = fs.readdirSync(dirPath, { withFileTypes: true });
|
|
303
|
+
for (const item of items) {
|
|
304
|
+
const fullPath = path.join(dirPath, item.name);
|
|
305
|
+
if (item.isDirectory()) {
|
|
306
|
+
// Skip node_modules and other build artifacts
|
|
307
|
+
if (!['node_modules', 'dist', 'build', '.git'].includes(item.name)) {
|
|
308
|
+
calculateSize(fullPath);
|
|
309
|
+
}
|
|
310
|
+
} else {
|
|
311
|
+
try {
|
|
312
|
+
const stats = fs.statSync(fullPath);
|
|
313
|
+
totalSize += stats.size;
|
|
314
|
+
fileCount++;
|
|
315
|
+
} catch (error) {
|
|
316
|
+
// Skip files we can't read
|
|
317
|
+
}
|
|
318
|
+
}
|
|
319
|
+
}
|
|
320
|
+
};
|
|
321
|
+
calculateSize(packagePath);
|
|
322
|
+
return {
|
|
323
|
+
size: totalSize,
|
|
324
|
+
files: fileCount,
|
|
325
|
+
};
|
|
326
|
+
} catch (error) {
|
|
327
|
+
return { size: 0, files: 0 };
|
|
328
|
+
}
|
|
330
329
|
}
|
|
331
|
-
export {
|
|
332
|
-
|
|
333
|
-
|
|
330
|
+
export {
|
|
331
|
+
scanMonorepo,
|
|
332
|
+
generateMonorepoStats,
|
|
333
|
+
findCircularDependencies,
|
|
334
|
+
generateDependencyGraph,
|
|
335
|
+
checkOutdatedDependencies,
|
|
336
|
+
getPackageSize,
|
|
337
|
+
// analyzeDependencies,
|
|
338
|
+
calculatePackageHealth,
|
|
339
|
+
};
|
package/helpers.ts
CHANGED
package/package.json
CHANGED