whyinstall 0.1.1 → 0.3.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/dist/cli.js +13 -4
- package/dist/cli.js.map +1 -1
- package/dist/fileFinder.d.ts +1 -0
- package/dist/fileFinder.d.ts.map +1 -1
- package/dist/fileFinder.js +1 -0
- package/dist/fileFinder.js.map +1 -1
- package/dist/formatter.d.ts +2 -1
- package/dist/formatter.d.ts.map +1 -1
- package/dist/formatter.js +18 -0
- package/dist/formatter.js.map +1 -1
- package/dist/impactAnalyzer.d.ts +3 -0
- package/dist/impactAnalyzer.d.ts.map +1 -0
- package/dist/impactAnalyzer.js +117 -0
- package/dist/impactAnalyzer.js.map +1 -0
- package/dist/sizeMapAnalyzer.d.ts +3 -0
- package/dist/sizeMapAnalyzer.d.ts.map +1 -0
- package/dist/sizeMapAnalyzer.js +134 -0
- package/dist/sizeMapAnalyzer.js.map +1 -0
- package/dist/types.d.ts +11 -0
- package/dist/types.d.ts.map +1 -1
- package/package.json +1 -1
- package/src/cli.ts +14 -6
- package/src/fileFinder.ts +1 -1
- package/src/formatter.ts +23 -1
- package/src/sizeMapAnalyzer.ts +145 -0
- package/src/types.ts +13 -0
package/dist/cli.js
CHANGED
|
@@ -9,14 +9,16 @@ const chalk_1 = __importDefault(require("chalk"));
|
|
|
9
9
|
const analyzer_1 = require("./analyzer");
|
|
10
10
|
const formatter_1 = require("./formatter");
|
|
11
11
|
const packageManager_1 = require("./packageManager");
|
|
12
|
+
const sizeMapAnalyzer_1 = require("./sizeMapAnalyzer");
|
|
12
13
|
const program = new commander_1.Command();
|
|
13
14
|
program
|
|
14
15
|
.name('whyinstall')
|
|
15
16
|
.description('Find why a dependency exists in your JS/TS project')
|
|
16
|
-
.version('0.
|
|
17
|
+
.version('0.3.0')
|
|
17
18
|
.argument('<package-name>', 'Package name to analyze')
|
|
18
19
|
.option('-j, --json', 'Output as JSON')
|
|
19
20
|
.option('-c, --cwd <path>', 'Working directory', process.cwd())
|
|
21
|
+
.option('-s, --size-map', 'Show bundle size impact breakdown')
|
|
20
22
|
.action((packageName, options) => {
|
|
21
23
|
try {
|
|
22
24
|
const cwd = options.cwd || process.cwd();
|
|
@@ -24,9 +26,16 @@ program
|
|
|
24
26
|
if (!options.json) {
|
|
25
27
|
console.log(`\n${chalk_1.default.gray(`Detected package manager: ${pm}`)}\n`);
|
|
26
28
|
}
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
29
|
+
if (options.sizeMap) {
|
|
30
|
+
const result = (0, sizeMapAnalyzer_1.analyzeSizeMap)(packageName, cwd);
|
|
31
|
+
const output = (0, formatter_1.formatSizeMap)(result, options.json);
|
|
32
|
+
console.log(output);
|
|
33
|
+
}
|
|
34
|
+
else {
|
|
35
|
+
const result = (0, analyzer_1.analyzePackage)(packageName, cwd);
|
|
36
|
+
const output = (0, formatter_1.formatOutput)(result, options.json);
|
|
37
|
+
console.log(output);
|
|
38
|
+
}
|
|
30
39
|
process.exit(0);
|
|
31
40
|
}
|
|
32
41
|
catch (error) {
|
package/dist/cli.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":";;;;;;AAEA,yCAAoC;AACpC,kDAA0B;AAC1B,yCAA4C;AAC5C,
|
|
1
|
+
{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":";;;;;;AAEA,yCAAoC;AACpC,kDAA0B;AAC1B,yCAA4C;AAC5C,2CAA0D;AAC1D,qDAAwD;AACxD,uDAAmD;AAEnD,MAAM,OAAO,GAAG,IAAI,mBAAO,EAAE,CAAC;AAE9B,OAAO;KACJ,IAAI,CAAC,YAAY,CAAC;KAClB,WAAW,CAAC,oDAAoD,CAAC;KACjE,OAAO,CAAC,OAAO,CAAC;KAChB,QAAQ,CAAC,gBAAgB,EAAE,yBAAyB,CAAC;KACrD,MAAM,CAAC,YAAY,EAAE,gBAAgB,CAAC;KACtC,MAAM,CAAC,kBAAkB,EAAE,mBAAmB,EAAE,OAAO,CAAC,GAAG,EAAE,CAAC;KAC9D,MAAM,CAAC,gBAAgB,EAAE,mCAAmC,CAAC;KAC7D,MAAM,CAAC,CAAC,WAAmB,EAAE,OAA4D,EAAE,EAAE;IAC5F,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,IAAI,OAAO,CAAC,GAAG,EAAE,CAAC;QACzC,MAAM,EAAE,GAAG,IAAA,qCAAoB,EAAC,GAAG,CAAC,CAAC;QAErC,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;YAClB,OAAO,CAAC,GAAG,CAAC,KAAK,eAAK,CAAC,IAAI,CAAC,6BAA6B,EAAE,EAAE,CAAC,IAAI,CAAC,CAAC;QACtE,CAAC;QAED,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpB,MAAM,MAAM,GAAG,IAAA,gCAAc,EAAC,WAAW,EAAE,GAAG,CAAC,CAAC;YAChD,MAAM,MAAM,GAAG,IAAA,yBAAa,EAAC,MAAM,EAAE,OAAO,CAAC,IAAI,CAAC,CAAC;YACnD,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QACtB,CAAC;aAAM,CAAC;YACN,MAAM,MAAM,GAAG,IAAA,yBAAc,EAAC,WAAW,EAAE,GAAG,CAAC,CAAC;YAChD,MAAM,MAAM,GAAG,IAAA,wBAAY,EAAC,MAAM,EAAE,OAAO,CAAC,IAAI,CAAC,CAAC;YAClD,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QACtB,CAAC;QAED,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,MAAM,OAAO,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QACvE,OAAO,CAAC,KAAK,CAAC,eAAK,CAAC,GAAG,CAAC,UAAU,OAAO,EAAE,CAAC,CAAC,CAAC;QAC9C,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC,CAAC,CAAC;AAEL,OAAO,CAAC,KAAK,EAAE,CAAC"}
|
package/dist/fileFinder.d.ts
CHANGED
package/dist/fileFinder.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"fileFinder.d.ts","sourceRoot":"","sources":["../src/fileFinder.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"fileFinder.d.ts","sourceRoot":"","sources":["../src/fileFinder.ts"],"names":[],"mappings":"AAcA,wBAAgB,eAAe,CAAC,GAAG,EAAE,MAAM,EAAE,QAAQ,GAAE,MAAU,EAAE,YAAY,GAAE,MAAU,GAAG,MAAM,EAAE,CAgCrG;AAmBD,wBAAgB,qBAAqB,CAAC,WAAW,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,GAAG,MAAM,EAAE,CAYhF"}
|
package/dist/fileFinder.js
CHANGED
package/dist/fileFinder.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"fileFinder.js","sourceRoot":"","sources":["../src/fileFinder.ts"],"names":[],"mappings":";;
|
|
1
|
+
{"version":3,"file":"fileFinder.js","sourceRoot":"","sources":["../src/fileFinder.ts"],"names":[],"mappings":"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"}
|
package/dist/formatter.d.ts
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
1
|
-
import { AnalyzeResult } from './types';
|
|
1
|
+
import { AnalyzeResult, SizeMapResult } from './types';
|
|
2
2
|
export declare function formatOutput(result: AnalyzeResult, json?: boolean): string;
|
|
3
|
+
export declare function formatSizeMap(result: SizeMapResult, json?: boolean): string;
|
|
3
4
|
//# sourceMappingURL=formatter.d.ts.map
|
package/dist/formatter.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"formatter.d.ts","sourceRoot":"","sources":["../src/formatter.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,aAAa,EAAkB,MAAM,SAAS,CAAC;
|
|
1
|
+
{"version":3,"file":"formatter.d.ts","sourceRoot":"","sources":["../src/formatter.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,aAAa,EAAkB,aAAa,EAAE,MAAM,SAAS,CAAC;AAwDvE,wBAAgB,YAAY,CAAC,MAAM,EAAE,aAAa,EAAE,IAAI,GAAE,OAAe,GAAG,MAAM,CA8CjF;AAED,wBAAgB,aAAa,CAAC,MAAM,EAAE,aAAa,EAAE,IAAI,GAAE,OAAe,GAAG,MAAM,CAoBlF"}
|
package/dist/formatter.js
CHANGED
|
@@ -4,6 +4,7 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
|
4
4
|
};
|
|
5
5
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
6
|
exports.formatOutput = formatOutput;
|
|
7
|
+
exports.formatSizeMap = formatSizeMap;
|
|
7
8
|
const chalk_1 = __importDefault(require("chalk"));
|
|
8
9
|
function formatSize(bytes) {
|
|
9
10
|
if (!bytes)
|
|
@@ -98,4 +99,21 @@ function formatOutput(result, json = false) {
|
|
|
98
99
|
}
|
|
99
100
|
return output;
|
|
100
101
|
}
|
|
102
|
+
function formatSizeMap(result, json = false) {
|
|
103
|
+
if (json) {
|
|
104
|
+
return JSON.stringify(result, null, 2);
|
|
105
|
+
}
|
|
106
|
+
let output = '';
|
|
107
|
+
output += chalk_1.default.bold.cyan(`Size map for: ${result.packageName}\n\n`);
|
|
108
|
+
output += chalk_1.default.bold(`${result.packageName} total impact: `) + chalk_1.default.green(formatSize(result.totalSize)) + '\n\n';
|
|
109
|
+
output += chalk_1.default.bold('Breakdown:\n');
|
|
110
|
+
for (const item of result.breakdown) {
|
|
111
|
+
const sizeStr = formatSize(item.size);
|
|
112
|
+
output += chalk_1.default.gray('- ') + chalk_1.default.white(item.name) + chalk_1.default.gray(': ') + chalk_1.default.yellow(sizeStr) + '\n';
|
|
113
|
+
}
|
|
114
|
+
if (result.percentOfNodeModules !== undefined && result.percentOfNodeModules > 0) {
|
|
115
|
+
output += '\n' + chalk_1.default.dim(`This package contributes ${result.percentOfNodeModules.toFixed(1)}% of your vendor bundle.\n`);
|
|
116
|
+
}
|
|
117
|
+
return output;
|
|
118
|
+
}
|
|
101
119
|
//# sourceMappingURL=formatter.js.map
|
package/dist/formatter.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"formatter.js","sourceRoot":"","sources":["../src/formatter.ts"],"names":[],"mappings":";;;;;AAyDA,oCA8CC;
|
|
1
|
+
{"version":3,"file":"formatter.js","sourceRoot":"","sources":["../src/formatter.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"impactAnalyzer.d.ts","sourceRoot":"","sources":["../src/impactAnalyzer.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,SAAS,CAAC;AAqGzC,wBAAgB,aAAa,CAAC,WAAW,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,GAAG,cAAc,CAqC9E"}
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.analyzeImpact = analyzeImpact;
|
|
4
|
+
const fileFinder_1 = require("./fileFinder");
|
|
5
|
+
function analyzeUsageContext(usage, content) {
|
|
6
|
+
// Analyze actual usage patterns in THIS codebase, not generic assumptions
|
|
7
|
+
const lines = content.split('\n');
|
|
8
|
+
const contextLines = [];
|
|
9
|
+
// Get context around usage lines
|
|
10
|
+
usage.lines.forEach(lineNum => {
|
|
11
|
+
const start = Math.max(0, lineNum - 2);
|
|
12
|
+
const end = Math.min(lines.length, lineNum + 2);
|
|
13
|
+
contextLines.push(...lines.slice(start, end));
|
|
14
|
+
});
|
|
15
|
+
const context = contextLines.join('\n').toLowerCase();
|
|
16
|
+
// Simple heuristics based on actual code patterns, not package names
|
|
17
|
+
if (context.includes('console.') || context.includes('process.stdout') || context.includes('process.stderr')) {
|
|
18
|
+
return 'Console/output';
|
|
19
|
+
}
|
|
20
|
+
if (context.includes('response') || context.includes('res.') || context.includes('req.')) {
|
|
21
|
+
return 'HTTP/API';
|
|
22
|
+
}
|
|
23
|
+
if (context.includes('database') || context.includes('db.') || context.includes('query')) {
|
|
24
|
+
return 'Database';
|
|
25
|
+
}
|
|
26
|
+
if (context.includes('test') || context.includes('spec') || context.includes('describe')) {
|
|
27
|
+
return 'Testing';
|
|
28
|
+
}
|
|
29
|
+
return 'General usage';
|
|
30
|
+
}
|
|
31
|
+
function computeRiskLevel(files, contexts) {
|
|
32
|
+
// Risk assessment based on WHERE it's used, not WHAT package it is
|
|
33
|
+
const hasDatabase = contexts.some(c => c === 'Database');
|
|
34
|
+
const hasHTTP = contexts.some(c => c === 'HTTP/API');
|
|
35
|
+
const hasTesting = contexts.every(c => c === 'Testing');
|
|
36
|
+
const hasConsole = contexts.every(c => c === 'Console/output');
|
|
37
|
+
// High risk: Database, HTTP/API (core functionality)
|
|
38
|
+
if (hasDatabase || hasHTTP) {
|
|
39
|
+
return 'High';
|
|
40
|
+
}
|
|
41
|
+
// Low risk: Only in tests, or only console output
|
|
42
|
+
if (hasTesting || (hasConsole && files.length === 1)) {
|
|
43
|
+
return 'Low';
|
|
44
|
+
}
|
|
45
|
+
// Medium: Everything else (could be important, needs review)
|
|
46
|
+
return 'Medium';
|
|
47
|
+
}
|
|
48
|
+
function generateImpacts(files, contexts, packageName) {
|
|
49
|
+
const impacts = [];
|
|
50
|
+
const uniqueContexts = [...new Set(contexts)];
|
|
51
|
+
// Base impacts on actual usage in THIS codebase
|
|
52
|
+
if (files.length === 0) {
|
|
53
|
+
impacts.push(`No direct usage found in source files`);
|
|
54
|
+
impacts.push(`May be a transitive dependency or unused`);
|
|
55
|
+
}
|
|
56
|
+
else {
|
|
57
|
+
const totalMethods = new Set(files.flatMap(f => f.methods));
|
|
58
|
+
if (totalMethods.size > 0) {
|
|
59
|
+
impacts.push(`Methods used: ${Array.from(totalMethods).slice(0, 8).join(', ')}${totalMethods.size > 8 ? ` (+${totalMethods.size - 8} more)` : ''}`);
|
|
60
|
+
}
|
|
61
|
+
else {
|
|
62
|
+
impacts.push(`Package imported but no method calls detected`);
|
|
63
|
+
impacts.push(`May be used as default export or namespace`);
|
|
64
|
+
}
|
|
65
|
+
impacts.push(`Used in ${files.length} file${files.length !== 1 ? 's' : ''}: ${uniqueContexts.join(', ')}`);
|
|
66
|
+
// Context-specific guidance (based on actual usage, not package name)
|
|
67
|
+
if (uniqueContexts.includes('Database')) {
|
|
68
|
+
impacts.push(`⚠️ Used in database operations - removal may break data access`);
|
|
69
|
+
}
|
|
70
|
+
if (uniqueContexts.includes('HTTP/API')) {
|
|
71
|
+
impacts.push(`⚠️ Used in API/HTTP layer - removal may break endpoints`);
|
|
72
|
+
}
|
|
73
|
+
if (uniqueContexts.includes('Testing')) {
|
|
74
|
+
impacts.push(`✓ Only used in tests - safe to remove from production dependencies`);
|
|
75
|
+
}
|
|
76
|
+
if (uniqueContexts.includes('Console/output') && files.length === 1) {
|
|
77
|
+
impacts.push(`✓ Only used for console output - low impact, can be replaced`);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
impacts.push(`\n💡 Review the files above to assess actual impact in your codebase`);
|
|
81
|
+
return impacts;
|
|
82
|
+
}
|
|
83
|
+
function analyzeImpact(packageName, cwd) {
|
|
84
|
+
const sourceFiles = (0, fileFinder_1.findSourceFiles)(cwd);
|
|
85
|
+
const fileUsages = [];
|
|
86
|
+
for (const file of sourceFiles) {
|
|
87
|
+
const usage = (0, fileFinder_1.analyzeFileUsage)(file, packageName, cwd);
|
|
88
|
+
if (usage) {
|
|
89
|
+
fileUsages.push(usage);
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
// Analyze actual usage context in THIS codebase
|
|
93
|
+
const contexts = fileUsages.map(usage => {
|
|
94
|
+
try {
|
|
95
|
+
const fs = require('fs');
|
|
96
|
+
const path = require('path');
|
|
97
|
+
const fullPath = usage.file.startsWith(cwd) ? usage.file : path.join(cwd, usage.file);
|
|
98
|
+
const content = fs.readFileSync(fullPath, 'utf-8');
|
|
99
|
+
return analyzeUsageContext(usage, content);
|
|
100
|
+
}
|
|
101
|
+
catch {
|
|
102
|
+
return 'Unknown';
|
|
103
|
+
}
|
|
104
|
+
});
|
|
105
|
+
const riskLevel = computeRiskLevel(fileUsages, contexts);
|
|
106
|
+
const impacts = generateImpacts(fileUsages, contexts, packageName);
|
|
107
|
+
// Add contexts to file usages
|
|
108
|
+
fileUsages.forEach((usage, index) => {
|
|
109
|
+
usage.purpose = contexts[index];
|
|
110
|
+
});
|
|
111
|
+
return {
|
|
112
|
+
files: fileUsages,
|
|
113
|
+
riskLevel,
|
|
114
|
+
impacts
|
|
115
|
+
};
|
|
116
|
+
}
|
|
117
|
+
//# sourceMappingURL=impactAnalyzer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"impactAnalyzer.js","sourceRoot":"","sources":["../src/impactAnalyzer.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sizeMapAnalyzer.d.ts","sourceRoot":"","sources":["../src/sizeMapAnalyzer.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,aAAa,EAAiB,MAAM,SAAS,CAAC;AAmHvD,wBAAgB,cAAc,CAAC,WAAW,EAAE,MAAM,EAAE,GAAG,GAAE,MAAsB,GAAG,aAAa,CA0B9F"}
|
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.analyzeSizeMap = analyzeSizeMap;
|
|
4
|
+
const fs_1 = require("fs");
|
|
5
|
+
const path_1 = require("path");
|
|
6
|
+
const JS_EXTENSIONS = new Set(['.js', '.mjs', '.cjs']);
|
|
7
|
+
const EXCLUDE_DIRS = new Set(['test', 'tests', '__tests__', 'docs', 'doc', 'types', '@types', 'typings', '.d.ts']);
|
|
8
|
+
function readPackageJson(path) {
|
|
9
|
+
try {
|
|
10
|
+
if ((0, fs_1.existsSync)(path)) {
|
|
11
|
+
return JSON.parse((0, fs_1.readFileSync)(path, 'utf-8'));
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
catch { }
|
|
15
|
+
return null;
|
|
16
|
+
}
|
|
17
|
+
function shouldExclude(name) {
|
|
18
|
+
const lower = name.toLowerCase();
|
|
19
|
+
return EXCLUDE_DIRS.has(lower) || lower.endsWith('.d.ts') || lower.startsWith('.');
|
|
20
|
+
}
|
|
21
|
+
function calculateJsSize(dir) {
|
|
22
|
+
let total = 0;
|
|
23
|
+
try {
|
|
24
|
+
const entries = (0, fs_1.readdirSync)(dir);
|
|
25
|
+
for (const entry of entries) {
|
|
26
|
+
if (shouldExclude(entry))
|
|
27
|
+
continue;
|
|
28
|
+
const fullPath = (0, path_1.join)(dir, entry);
|
|
29
|
+
try {
|
|
30
|
+
const stats = (0, fs_1.statSync)(fullPath);
|
|
31
|
+
if (stats.isDirectory()) {
|
|
32
|
+
if (entry !== 'node_modules') {
|
|
33
|
+
total += calculateJsSize(fullPath);
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
else if (JS_EXTENSIONS.has((0, path_1.extname)(entry).toLowerCase())) {
|
|
37
|
+
total += stats.size;
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
catch { }
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
catch { }
|
|
44
|
+
return total;
|
|
45
|
+
}
|
|
46
|
+
function getNodeModulesSize(cwd) {
|
|
47
|
+
const nmPath = (0, path_1.join)(cwd, 'node_modules');
|
|
48
|
+
let total = 0;
|
|
49
|
+
function walkDir(dir) {
|
|
50
|
+
try {
|
|
51
|
+
const entries = (0, fs_1.readdirSync)(dir);
|
|
52
|
+
for (const entry of entries) {
|
|
53
|
+
const fullPath = (0, path_1.join)(dir, entry);
|
|
54
|
+
try {
|
|
55
|
+
const stats = (0, fs_1.statSync)(fullPath);
|
|
56
|
+
if (stats.isDirectory()) {
|
|
57
|
+
walkDir(fullPath);
|
|
58
|
+
}
|
|
59
|
+
else if (JS_EXTENSIONS.has((0, path_1.extname)(entry).toLowerCase())) {
|
|
60
|
+
total += stats.size;
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
catch { }
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
catch { }
|
|
67
|
+
}
|
|
68
|
+
walkDir(nmPath);
|
|
69
|
+
return total;
|
|
70
|
+
}
|
|
71
|
+
function findPackagePath(packageName, cwd) {
|
|
72
|
+
const directPath = (0, path_1.join)(cwd, 'node_modules', packageName);
|
|
73
|
+
if ((0, fs_1.existsSync)(directPath))
|
|
74
|
+
return directPath;
|
|
75
|
+
// Handle scoped packages
|
|
76
|
+
if (packageName.startsWith('@')) {
|
|
77
|
+
const parts = packageName.split('/');
|
|
78
|
+
const scopedPath = (0, path_1.join)(cwd, 'node_modules', parts[0], parts[1]);
|
|
79
|
+
if ((0, fs_1.existsSync)(scopedPath))
|
|
80
|
+
return scopedPath;
|
|
81
|
+
}
|
|
82
|
+
return null;
|
|
83
|
+
}
|
|
84
|
+
function getDependencyTree(packagePath, cwd, visited = new Set()) {
|
|
85
|
+
const breakdown = [];
|
|
86
|
+
const pkgJsonPath = (0, path_1.join)(packagePath, 'package.json');
|
|
87
|
+
const pkg = readPackageJson(pkgJsonPath);
|
|
88
|
+
if (!pkg?.dependencies)
|
|
89
|
+
return breakdown;
|
|
90
|
+
for (const depName of Object.keys(pkg.dependencies)) {
|
|
91
|
+
if (visited.has(depName))
|
|
92
|
+
continue;
|
|
93
|
+
visited.add(depName);
|
|
94
|
+
// Check nested node_modules first
|
|
95
|
+
let depPath = (0, path_1.join)(packagePath, 'node_modules', depName);
|
|
96
|
+
if (!(0, fs_1.existsSync)(depPath)) {
|
|
97
|
+
depPath = (0, path_1.join)(cwd, 'node_modules', depName);
|
|
98
|
+
}
|
|
99
|
+
if ((0, fs_1.existsSync)(depPath)) {
|
|
100
|
+
const size = calculateJsSize(depPath);
|
|
101
|
+
if (size > 0) {
|
|
102
|
+
breakdown.push({ name: depName, size });
|
|
103
|
+
}
|
|
104
|
+
// Recursively get sub-dependencies
|
|
105
|
+
const subDeps = getDependencyTree(depPath, cwd, visited);
|
|
106
|
+
breakdown.push(...subDeps);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
return breakdown;
|
|
110
|
+
}
|
|
111
|
+
function analyzeSizeMap(packageName, cwd = process.cwd()) {
|
|
112
|
+
const packagePath = findPackagePath(packageName, cwd);
|
|
113
|
+
if (!packagePath) {
|
|
114
|
+
throw new Error(`Package "${packageName}" not found in node_modules`);
|
|
115
|
+
}
|
|
116
|
+
const ownSize = calculateJsSize(packagePath);
|
|
117
|
+
const visited = new Set([packageName]);
|
|
118
|
+
const depBreakdown = getDependencyTree(packagePath, cwd, visited);
|
|
119
|
+
const breakdown = [
|
|
120
|
+
{ name: packageName, size: ownSize },
|
|
121
|
+
...depBreakdown.sort((a, b) => b.size - a.size)
|
|
122
|
+
];
|
|
123
|
+
const totalSize = breakdown.reduce((sum, item) => sum + item.size, 0);
|
|
124
|
+
const nodeModulesSize = getNodeModulesSize(cwd);
|
|
125
|
+
const percentOfNodeModules = nodeModulesSize > 0 ? (totalSize / nodeModulesSize) * 100 : 0;
|
|
126
|
+
return {
|
|
127
|
+
packageName,
|
|
128
|
+
totalSize,
|
|
129
|
+
breakdown,
|
|
130
|
+
nodeModulesSize,
|
|
131
|
+
percentOfNodeModules
|
|
132
|
+
};
|
|
133
|
+
}
|
|
134
|
+
//# sourceMappingURL=sizeMapAnalyzer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sizeMapAnalyzer.js","sourceRoot":"","sources":["../src/sizeMapAnalyzer.ts"],"names":[],"mappings":"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"}
|
package/dist/types.d.ts
CHANGED
|
@@ -16,4 +16,15 @@ export interface AnalyzeResult {
|
|
|
16
16
|
suggestions: string[];
|
|
17
17
|
}
|
|
18
18
|
export type PackageManager = 'npm' | 'yarn' | 'pnpm';
|
|
19
|
+
export interface SizeBreakdown {
|
|
20
|
+
name: string;
|
|
21
|
+
size: number;
|
|
22
|
+
}
|
|
23
|
+
export interface SizeMapResult {
|
|
24
|
+
packageName: string;
|
|
25
|
+
totalSize: number;
|
|
26
|
+
breakdown: SizeBreakdown[];
|
|
27
|
+
nodeModulesSize?: number;
|
|
28
|
+
percentOfNodeModules?: number;
|
|
29
|
+
}
|
|
19
30
|
//# sourceMappingURL=types.d.ts.map
|
package/dist/types.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,cAAc;IAC7B,KAAK,EAAE,MAAM,EAAE,CAAC;IAChB,IAAI,EAAE,MAAM,GAAG,KAAK,GAAG,MAAM,GAAG,UAAU,CAAC;IAC3C,eAAe,CAAC,EAAE,MAAM,CAAC;CAC1B;AAED,MAAM,WAAW,WAAW;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;IAChB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,cAAc,EAAE,CAAC;IACxB,WAAW,CAAC,EAAE,MAAM,EAAE,CAAC;CACxB;AAED,MAAM,WAAW,aAAa;IAC5B,OAAO,EAAE,WAAW,CAAC;IACrB,WAAW,EAAE,MAAM,EAAE,CAAC;CACvB;AAED,MAAM,MAAM,cAAc,GAAG,KAAK,GAAG,MAAM,GAAG,MAAM,CAAC"}
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,cAAc;IAC7B,KAAK,EAAE,MAAM,EAAE,CAAC;IAChB,IAAI,EAAE,MAAM,GAAG,KAAK,GAAG,MAAM,GAAG,UAAU,CAAC;IAC3C,eAAe,CAAC,EAAE,MAAM,CAAC;CAC1B;AAED,MAAM,WAAW,WAAW;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;IAChB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,cAAc,EAAE,CAAC;IACxB,WAAW,CAAC,EAAE,MAAM,EAAE,CAAC;CACxB;AAED,MAAM,WAAW,aAAa;IAC5B,OAAO,EAAE,WAAW,CAAC;IACrB,WAAW,EAAE,MAAM,EAAE,CAAC;CACvB;AAED,MAAM,MAAM,cAAc,GAAG,KAAK,GAAG,MAAM,GAAG,MAAM,CAAC;AAErD,MAAM,WAAW,aAAa;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;CACd;AAED,MAAM,WAAW,aAAa;IAC5B,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,aAAa,EAAE,CAAC;IAC3B,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,oBAAoB,CAAC,EAAE,MAAM,CAAC;CAC/B"}
|
package/package.json
CHANGED
package/src/cli.ts
CHANGED
|
@@ -3,19 +3,21 @@
|
|
|
3
3
|
import { Command } from 'commander';
|
|
4
4
|
import chalk from 'chalk';
|
|
5
5
|
import { analyzePackage } from './analyzer';
|
|
6
|
-
import { formatOutput } from './formatter';
|
|
6
|
+
import { formatOutput, formatSizeMap } from './formatter';
|
|
7
7
|
import { detectPackageManager } from './packageManager';
|
|
8
|
+
import { analyzeSizeMap } from './sizeMapAnalyzer';
|
|
8
9
|
|
|
9
10
|
const program = new Command();
|
|
10
11
|
|
|
11
12
|
program
|
|
12
13
|
.name('whyinstall')
|
|
13
14
|
.description('Find why a dependency exists in your JS/TS project')
|
|
14
|
-
.version('0.
|
|
15
|
+
.version('0.3.0')
|
|
15
16
|
.argument('<package-name>', 'Package name to analyze')
|
|
16
17
|
.option('-j, --json', 'Output as JSON')
|
|
17
18
|
.option('-c, --cwd <path>', 'Working directory', process.cwd())
|
|
18
|
-
.
|
|
19
|
+
.option('-s, --size-map', 'Show bundle size impact breakdown')
|
|
20
|
+
.action((packageName: string, options: { json?: boolean; cwd?: string; sizeMap?: boolean }) => {
|
|
19
21
|
try {
|
|
20
22
|
const cwd = options.cwd || process.cwd();
|
|
21
23
|
const pm = detectPackageManager(cwd);
|
|
@@ -24,9 +26,15 @@ program
|
|
|
24
26
|
console.log(`\n${chalk.gray(`Detected package manager: ${pm}`)}\n`);
|
|
25
27
|
}
|
|
26
28
|
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
29
|
+
if (options.sizeMap) {
|
|
30
|
+
const result = analyzeSizeMap(packageName, cwd);
|
|
31
|
+
const output = formatSizeMap(result, options.json);
|
|
32
|
+
console.log(output);
|
|
33
|
+
} else {
|
|
34
|
+
const result = analyzePackage(packageName, cwd);
|
|
35
|
+
const output = formatOutput(result, options.json);
|
|
36
|
+
console.log(output);
|
|
37
|
+
}
|
|
30
38
|
|
|
31
39
|
process.exit(0);
|
|
32
40
|
} catch (error) {
|
package/src/fileFinder.ts
CHANGED
|
@@ -12,7 +12,7 @@ function isSourceFile(filePath: string): boolean {
|
|
|
12
12
|
return SOURCE_EXTENSIONS.includes(extname(filePath));
|
|
13
13
|
}
|
|
14
14
|
|
|
15
|
-
function findSourceFiles(dir: string, maxDepth: number = 5, currentDepth: number = 0): string[] {
|
|
15
|
+
export function findSourceFiles(dir: string, maxDepth: number = 5, currentDepth: number = 0): string[] {
|
|
16
16
|
if (currentDepth > maxDepth) {
|
|
17
17
|
return [];
|
|
18
18
|
}
|
package/src/formatter.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import chalk from 'chalk';
|
|
2
|
-
import { AnalyzeResult, DependencyPath } from './types';
|
|
2
|
+
import { AnalyzeResult, DependencyPath, SizeMapResult } from './types';
|
|
3
3
|
|
|
4
4
|
function formatSize(bytes: number | undefined): string {
|
|
5
5
|
if (!bytes) return '';
|
|
@@ -102,3 +102,25 @@ export function formatOutput(result: AnalyzeResult, json: boolean = false): stri
|
|
|
102
102
|
|
|
103
103
|
return output;
|
|
104
104
|
}
|
|
105
|
+
|
|
106
|
+
export function formatSizeMap(result: SizeMapResult, json: boolean = false): string {
|
|
107
|
+
if (json) {
|
|
108
|
+
return JSON.stringify(result, null, 2);
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
let output = '';
|
|
112
|
+
output += chalk.bold.cyan(`Size map for: ${result.packageName}\n\n`);
|
|
113
|
+
output += chalk.bold(`${result.packageName} total impact: `) + chalk.green(formatSize(result.totalSize)) + '\n\n';
|
|
114
|
+
output += chalk.bold('Breakdown:\n');
|
|
115
|
+
|
|
116
|
+
for (const item of result.breakdown) {
|
|
117
|
+
const sizeStr = formatSize(item.size);
|
|
118
|
+
output += chalk.gray('- ') + chalk.white(item.name) + chalk.gray(': ') + chalk.yellow(sizeStr) + '\n';
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
if (result.percentOfNodeModules !== undefined && result.percentOfNodeModules > 0) {
|
|
122
|
+
output += '\n' + chalk.dim(`This package contributes ${result.percentOfNodeModules.toFixed(1)}% of your vendor bundle.\n`);
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
return output;
|
|
126
|
+
}
|
|
@@ -0,0 +1,145 @@
|
|
|
1
|
+
import { readFileSync, existsSync, readdirSync, statSync } from 'fs';
|
|
2
|
+
import { join, dirname, extname } from 'path';
|
|
3
|
+
import { SizeMapResult, SizeBreakdown } from './types';
|
|
4
|
+
|
|
5
|
+
interface PackageJson {
|
|
6
|
+
name?: string;
|
|
7
|
+
dependencies?: Record<string, string>;
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
const JS_EXTENSIONS = new Set(['.js', '.mjs', '.cjs']);
|
|
11
|
+
const EXCLUDE_DIRS = new Set(['test', 'tests', '__tests__', 'docs', 'doc', 'types', '@types', 'typings', '.d.ts']);
|
|
12
|
+
|
|
13
|
+
function readPackageJson(path: string): PackageJson | null {
|
|
14
|
+
try {
|
|
15
|
+
if (existsSync(path)) {
|
|
16
|
+
return JSON.parse(readFileSync(path, 'utf-8'));
|
|
17
|
+
}
|
|
18
|
+
} catch {}
|
|
19
|
+
return null;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
function shouldExclude(name: string): boolean {
|
|
23
|
+
const lower = name.toLowerCase();
|
|
24
|
+
return EXCLUDE_DIRS.has(lower) || lower.endsWith('.d.ts') || lower.startsWith('.');
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
function calculateJsSize(dir: string): number {
|
|
28
|
+
let total = 0;
|
|
29
|
+
try {
|
|
30
|
+
const entries = readdirSync(dir);
|
|
31
|
+
for (const entry of entries) {
|
|
32
|
+
if (shouldExclude(entry)) continue;
|
|
33
|
+
const fullPath = join(dir, entry);
|
|
34
|
+
try {
|
|
35
|
+
const stats = statSync(fullPath);
|
|
36
|
+
if (stats.isDirectory()) {
|
|
37
|
+
if (entry !== 'node_modules') {
|
|
38
|
+
total += calculateJsSize(fullPath);
|
|
39
|
+
}
|
|
40
|
+
} else if (JS_EXTENSIONS.has(extname(entry).toLowerCase())) {
|
|
41
|
+
total += stats.size;
|
|
42
|
+
}
|
|
43
|
+
} catch {}
|
|
44
|
+
}
|
|
45
|
+
} catch {}
|
|
46
|
+
return total;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
function getNodeModulesSize(cwd: string): number {
|
|
50
|
+
const nmPath = join(cwd, 'node_modules');
|
|
51
|
+
let total = 0;
|
|
52
|
+
|
|
53
|
+
function walkDir(dir: string): void {
|
|
54
|
+
try {
|
|
55
|
+
const entries = readdirSync(dir);
|
|
56
|
+
for (const entry of entries) {
|
|
57
|
+
const fullPath = join(dir, entry);
|
|
58
|
+
try {
|
|
59
|
+
const stats = statSync(fullPath);
|
|
60
|
+
if (stats.isDirectory()) {
|
|
61
|
+
walkDir(fullPath);
|
|
62
|
+
} else if (JS_EXTENSIONS.has(extname(entry).toLowerCase())) {
|
|
63
|
+
total += stats.size;
|
|
64
|
+
}
|
|
65
|
+
} catch {}
|
|
66
|
+
}
|
|
67
|
+
} catch {}
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
walkDir(nmPath);
|
|
71
|
+
return total;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
function findPackagePath(packageName: string, cwd: string): string | null {
|
|
75
|
+
const directPath = join(cwd, 'node_modules', packageName);
|
|
76
|
+
if (existsSync(directPath)) return directPath;
|
|
77
|
+
|
|
78
|
+
// Handle scoped packages
|
|
79
|
+
if (packageName.startsWith('@')) {
|
|
80
|
+
const parts = packageName.split('/');
|
|
81
|
+
const scopedPath = join(cwd, 'node_modules', parts[0], parts[1]);
|
|
82
|
+
if (existsSync(scopedPath)) return scopedPath;
|
|
83
|
+
}
|
|
84
|
+
return null;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
function getDependencyTree(packagePath: string, cwd: string, visited: Set<string> = new Set()): SizeBreakdown[] {
|
|
88
|
+
const breakdown: SizeBreakdown[] = [];
|
|
89
|
+
const pkgJsonPath = join(packagePath, 'package.json');
|
|
90
|
+
const pkg = readPackageJson(pkgJsonPath);
|
|
91
|
+
|
|
92
|
+
if (!pkg?.dependencies) return breakdown;
|
|
93
|
+
|
|
94
|
+
for (const depName of Object.keys(pkg.dependencies)) {
|
|
95
|
+
if (visited.has(depName)) continue;
|
|
96
|
+
visited.add(depName);
|
|
97
|
+
|
|
98
|
+
// Check nested node_modules first
|
|
99
|
+
let depPath = join(packagePath, 'node_modules', depName);
|
|
100
|
+
if (!existsSync(depPath)) {
|
|
101
|
+
depPath = join(cwd, 'node_modules', depName);
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
if (existsSync(depPath)) {
|
|
105
|
+
const size = calculateJsSize(depPath);
|
|
106
|
+
if (size > 0) {
|
|
107
|
+
breakdown.push({ name: depName, size });
|
|
108
|
+
}
|
|
109
|
+
// Recursively get sub-dependencies
|
|
110
|
+
const subDeps = getDependencyTree(depPath, cwd, visited);
|
|
111
|
+
breakdown.push(...subDeps);
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
return breakdown;
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
export function analyzeSizeMap(packageName: string, cwd: string = process.cwd()): SizeMapResult {
|
|
119
|
+
const packagePath = findPackagePath(packageName, cwd);
|
|
120
|
+
if (!packagePath) {
|
|
121
|
+
throw new Error(`Package "${packageName}" not found in node_modules`);
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
const ownSize = calculateJsSize(packagePath);
|
|
125
|
+
const visited = new Set<string>([packageName]);
|
|
126
|
+
const depBreakdown = getDependencyTree(packagePath, cwd, visited);
|
|
127
|
+
|
|
128
|
+
const breakdown: SizeBreakdown[] = [
|
|
129
|
+
{ name: packageName, size: ownSize },
|
|
130
|
+
...depBreakdown.sort((a, b) => b.size - a.size)
|
|
131
|
+
];
|
|
132
|
+
|
|
133
|
+
const totalSize = breakdown.reduce((sum, item) => sum + item.size, 0);
|
|
134
|
+
const nodeModulesSize = getNodeModulesSize(cwd);
|
|
135
|
+
const percentOfNodeModules = nodeModulesSize > 0 ? (totalSize / nodeModulesSize) * 100 : 0;
|
|
136
|
+
|
|
137
|
+
return {
|
|
138
|
+
packageName,
|
|
139
|
+
totalSize,
|
|
140
|
+
breakdown,
|
|
141
|
+
nodeModulesSize,
|
|
142
|
+
percentOfNodeModules
|
|
143
|
+
};
|
|
144
|
+
}
|
|
145
|
+
|
package/src/types.ts
CHANGED
|
@@ -19,3 +19,16 @@ export interface AnalyzeResult {
|
|
|
19
19
|
}
|
|
20
20
|
|
|
21
21
|
export type PackageManager = 'npm' | 'yarn' | 'pnpm';
|
|
22
|
+
|
|
23
|
+
export interface SizeBreakdown {
|
|
24
|
+
name: string;
|
|
25
|
+
size: number;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export interface SizeMapResult {
|
|
29
|
+
packageName: string;
|
|
30
|
+
totalSize: number;
|
|
31
|
+
breakdown: SizeBreakdown[];
|
|
32
|
+
nodeModulesSize?: number;
|
|
33
|
+
percentOfNodeModules?: number;
|
|
34
|
+
}
|