@mr-aftab-ahmad-khan/drift-check 0.1.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.cjs ADDED
@@ -0,0 +1,360 @@
1
+ #!/usr/bin/env node
2
+ "use strict";
3
+ var __create = Object.create;
4
+ var __defProp = Object.defineProperty;
5
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
6
+ var __getOwnPropNames = Object.getOwnPropertyNames;
7
+ var __getProtoOf = Object.getPrototypeOf;
8
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
9
+ var __copyProps = (to, from, except, desc) => {
10
+ if (from && typeof from === "object" || typeof from === "function") {
11
+ for (let key of __getOwnPropNames(from))
12
+ if (!__hasOwnProp.call(to, key) && key !== except)
13
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
14
+ }
15
+ return to;
16
+ };
17
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
18
+ // If the importer is in node compatibility mode or this is not an ESM
19
+ // file that has been converted to a CommonJS file using a Babel-
20
+ // compatible transform (i.e. "__esModule" has not been set), then set
21
+ // "default" to the CommonJS "module.exports" for node compatibility.
22
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
23
+ mod
24
+ ));
25
+
26
+ // src/cli.ts
27
+ var import_commander = require("commander");
28
+ var import_node_fs4 = require("fs");
29
+ var import_node_path3 = require("path");
30
+
31
+ // src/workspace.ts
32
+ var import_node_fs = require("fs");
33
+ var import_node_path = require("path");
34
+ var import_yaml = __toESM(require("yaml"), 1);
35
+ function readJson(file) {
36
+ try {
37
+ return JSON.parse((0, import_node_fs.readFileSync)(file, "utf8"));
38
+ } catch {
39
+ return void 0;
40
+ }
41
+ }
42
+ function detectWorkspaces(root, extraGlobs) {
43
+ root = (0, import_node_path.resolve)(root);
44
+ const candidates = /* @__PURE__ */ new Set();
45
+ const pnpm = (0, import_node_path.join)(root, "pnpm-workspace.yaml");
46
+ if ((0, import_node_fs.existsSync)(pnpm)) {
47
+ try {
48
+ const doc = import_yaml.default.parse((0, import_node_fs.readFileSync)(pnpm, "utf8"));
49
+ for (const g of doc.packages ?? []) expandGlob(root, g, candidates);
50
+ } catch {
51
+ }
52
+ }
53
+ const rootPkg = readJson((0, import_node_path.join)(root, "package.json"));
54
+ if (rootPkg) {
55
+ const ws = rootPkg.workspaces;
56
+ if (Array.isArray(ws)) {
57
+ for (const g of ws) if (typeof g === "string") expandGlob(root, g, candidates);
58
+ } else if (ws && typeof ws === "object" && Array.isArray(ws.packages)) {
59
+ for (const g of ws.packages) expandGlob(root, g, candidates);
60
+ }
61
+ }
62
+ for (const g of extraGlobs ?? ["packages/*", "apps/*"]) expandGlob(root, g, candidates);
63
+ if (candidates.size === 0) {
64
+ candidates.add((0, import_node_path.join)(root, "package.json"));
65
+ }
66
+ return [...candidates];
67
+ }
68
+ function expandGlob(root, glob, out) {
69
+ if (!glob.includes("*")) {
70
+ const pj = (0, import_node_path.join)(root, glob, "package.json");
71
+ if ((0, import_node_fs.existsSync)(pj)) out.add(pj);
72
+ return;
73
+ }
74
+ const segments = glob.split("/");
75
+ expand(root, segments, 0, out);
76
+ }
77
+ function expand(dir, segments, idx, out) {
78
+ if (idx === segments.length) {
79
+ const pj = (0, import_node_path.join)(dir, "package.json");
80
+ if ((0, import_node_fs.existsSync)(pj)) out.add(pj);
81
+ return;
82
+ }
83
+ const seg = segments[idx];
84
+ if (seg === "*") {
85
+ if (!(0, import_node_fs.existsSync)(dir)) return;
86
+ for (const entry of (0, import_node_fs.readdirSync)(dir)) {
87
+ const sub = (0, import_node_path.join)(dir, entry);
88
+ try {
89
+ if (!(0, import_node_fs.statSync)(sub).isDirectory()) continue;
90
+ } catch {
91
+ continue;
92
+ }
93
+ expand(sub, segments, idx + 1, out);
94
+ }
95
+ } else if (seg === "**") {
96
+ walk(dir, (d) => expand(d, segments, idx + 1, out));
97
+ } else {
98
+ expand((0, import_node_path.join)(dir, seg), segments, idx + 1, out);
99
+ }
100
+ }
101
+ function walk(dir, visit) {
102
+ if (!(0, import_node_fs.existsSync)(dir)) return;
103
+ visit(dir);
104
+ for (const entry of (0, import_node_fs.readdirSync)(dir)) {
105
+ if (entry === "node_modules" || entry.startsWith(".")) continue;
106
+ const sub = (0, import_node_path.join)(dir, entry);
107
+ try {
108
+ if ((0, import_node_fs.statSync)(sub).isDirectory()) walk(sub, visit);
109
+ } catch {
110
+ }
111
+ }
112
+ }
113
+ function loadWorkspaces(root, extraGlobs) {
114
+ return detectWorkspaces(root, extraGlobs).map((file) => {
115
+ const pj = readJson(file);
116
+ if (!pj) return void 0;
117
+ return {
118
+ name: pj.name ?? (0, import_node_path.dirname)(file),
119
+ version: pj.version ?? "0.0.0",
120
+ packageJsonPath: file,
121
+ dependencies: pj.dependencies ?? {},
122
+ devDependencies: pj.devDependencies ?? {},
123
+ peerDependencies: pj.peerDependencies ?? {},
124
+ optionalDependencies: pj.optionalDependencies ?? {}
125
+ };
126
+ }).filter((p) => Boolean(p));
127
+ }
128
+ function readDriftIgnore(root) {
129
+ const file = (0, import_node_path.join)(root, ".driftignore");
130
+ if (!(0, import_node_fs.existsSync)(file)) return [];
131
+ return (0, import_node_fs.readFileSync)(file, "utf8").split(/\r?\n/).map((s) => s.trim()).filter((s) => s && !s.startsWith("#"));
132
+ }
133
+
134
+ // src/semver.ts
135
+ var SEMVER_RE = /(\d+)\.(\d+)\.(\d+)(?:-([0-9A-Za-z-.]+))?/;
136
+ function extractVersion(range) {
137
+ const m = SEMVER_RE.exec(range);
138
+ if (!m) return void 0;
139
+ return { major: Number(m[1]), minor: Number(m[2]), patch: Number(m[3]) };
140
+ }
141
+ function severity(a, b) {
142
+ const va = extractVersion(a);
143
+ const vb = extractVersion(b);
144
+ if (!va || !vb) return a === b ? "none" : "patch";
145
+ if (va.major !== vb.major) return "major";
146
+ if (va.minor !== vb.minor) return "minor";
147
+ if (va.patch !== vb.patch) return "patch";
148
+ if (a !== b) return "patch";
149
+ return "none";
150
+ }
151
+ function highestRange(ranges) {
152
+ let best = ranges[0];
153
+ let bestVer = extractVersion(best);
154
+ for (const r of ranges) {
155
+ const v = extractVersion(r);
156
+ if (!v) continue;
157
+ if (!bestVer || compare(v, bestVer) > 0) {
158
+ best = r;
159
+ bestVer = v;
160
+ }
161
+ }
162
+ return best;
163
+ }
164
+ function compare(a, b) {
165
+ if (a.major !== b.major) return a.major - b.major;
166
+ if (a.minor !== b.minor) return a.minor - b.minor;
167
+ return a.patch - b.patch;
168
+ }
169
+ var SEVERITY_RANK = { patch: 0, minor: 1, major: 2 };
170
+ function maxSeverity(values) {
171
+ let max = "patch";
172
+ for (const v of values) if (SEVERITY_RANK[v] > SEVERITY_RANK[max]) max = v;
173
+ return max;
174
+ }
175
+
176
+ // src/scan.ts
177
+ var import_node_fs2 = require("fs");
178
+ var FIELDS = ["dependencies", "devDependencies", "peerDependencies", "optionalDependencies"];
179
+ function scan(options = {}) {
180
+ const cwd = options.cwd ?? process.cwd();
181
+ const workspaces = loadWorkspaces(cwd, options.workspaceGlobs);
182
+ const ignore = /* @__PURE__ */ new Set([...options.ignore ?? [], ...readDriftIgnore(cwd)]);
183
+ const byDep = /* @__PURE__ */ new Map();
184
+ for (const ws of workspaces) {
185
+ for (const field of FIELDS) {
186
+ for (const [name, version] of Object.entries(ws[field])) {
187
+ if (ignore.has(name)) continue;
188
+ const list = byDep.get(name) ?? [];
189
+ list.push({ workspace: ws.name, version, field, path: ws.packageJsonPath });
190
+ byDep.set(name, list);
191
+ }
192
+ }
193
+ }
194
+ const drifted = [];
195
+ for (const [name, occurrences] of byDep) {
196
+ const unique = [...new Set(occurrences.map((o) => o.version))];
197
+ if (unique.length <= 1) continue;
198
+ const severities = [];
199
+ for (let i = 0; i < unique.length; i++) {
200
+ for (let j = i + 1; j < unique.length; j++) {
201
+ const s = severity(unique[i], unique[j]);
202
+ if (s !== "none") severities.push(s);
203
+ }
204
+ }
205
+ drifted.push({
206
+ name,
207
+ severity: maxSeverity(severities),
208
+ versions: occurrences
209
+ });
210
+ }
211
+ drifted.sort((a, b) => a.name.localeCompare(b.name));
212
+ return { workspaces, drifted, generatedAt: (/* @__PURE__ */ new Date()).toISOString() };
213
+ }
214
+ function planFix(report, options = {}) {
215
+ const changes = [];
216
+ for (const d of report.drifted) {
217
+ if (options.only && d.name !== options.only) continue;
218
+ const target = options.target && options.target !== "latest" ? options.target : highestRange(d.versions.map((v) => v.version));
219
+ for (const v of d.versions) {
220
+ if (v.version !== target) {
221
+ changes.push({ path: v.path, field: v.field, name: d.name, from: v.version, to: target });
222
+ }
223
+ }
224
+ }
225
+ return { changes };
226
+ }
227
+ function applyFix(plan) {
228
+ const grouped = /* @__PURE__ */ new Map();
229
+ for (const c of plan.changes) {
230
+ const arr = grouped.get(c.path) ?? [];
231
+ arr.push(c);
232
+ grouped.set(c.path, arr);
233
+ }
234
+ for (const [file, changes] of grouped) {
235
+ const src = (0, import_node_fs2.readFileSync)(file, "utf8");
236
+ let next = src;
237
+ for (const c of changes) {
238
+ const re = new RegExp(`("${escape(c.name)}"\\s*:\\s*")${escape(c.from)}(")`);
239
+ next = next.replace(re, (_m, p1, p2) => `${p1}${c.to}${p2}`);
240
+ }
241
+ (0, import_node_fs2.writeFileSync)(file, next, "utf8");
242
+ }
243
+ return { filesChanged: grouped.size };
244
+ }
245
+ function escape(s) {
246
+ return s.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
247
+ }
248
+
249
+ // src/format.ts
250
+ var import_picocolors = __toESM(require("picocolors"), 1);
251
+ var sevColor = {
252
+ patch: import_picocolors.default.gray,
253
+ minor: import_picocolors.default.yellow,
254
+ major: import_picocolors.default.red
255
+ };
256
+ function formatPretty(report) {
257
+ if (report.drifted.length === 0) {
258
+ return import_picocolors.default.green(`drift-check: no drift across ${report.workspaces.length} workspaces`);
259
+ }
260
+ const lines = [];
261
+ lines.push(import_picocolors.default.bold(`drift-check \u2014 ${report.drifted.length} drifted package(s) across ${report.workspaces.length} workspaces`));
262
+ for (const d of report.drifted) {
263
+ lines.push("");
264
+ lines.push(` ${sevColor[d.severity](d.severity.toUpperCase())} ${import_picocolors.default.bold(d.name)}`);
265
+ for (const v of d.versions) {
266
+ lines.push(` - ${v.version} ${import_picocolors.default.gray(`(${v.workspace} / ${v.field})`)}`);
267
+ }
268
+ }
269
+ return lines.join("\n");
270
+ }
271
+ function formatJson(report) {
272
+ return JSON.stringify(report, null, 2);
273
+ }
274
+
275
+ // src/config.ts
276
+ var import_node_fs3 = require("fs");
277
+ var import_node_path2 = require("path");
278
+ function loadConfig(cwd) {
279
+ const f = (0, import_node_path2.join)(cwd, ".driftrc.json");
280
+ if (!(0, import_node_fs3.existsSync)(f)) return {};
281
+ try {
282
+ return JSON.parse((0, import_node_fs3.readFileSync)(f, "utf8"));
283
+ } catch {
284
+ return {};
285
+ }
286
+ }
287
+
288
+ // src/cli.ts
289
+ var program = new import_commander.Command();
290
+ program.name("drift-check").description("Detect and fix dependency drift across workspaces").option("--cwd <path>", "Working directory", process.cwd());
291
+ program.command("scan", { isDefault: true }).description("Scan workspaces and print a drift report").option("--format <fmt>", "pretty | json", "pretty").option("--fail-on <severity>", "patch | minor | major").action((opts) => {
292
+ const cwd = program.opts().cwd;
293
+ const cfg = loadConfig(cwd);
294
+ const report = scan({
295
+ cwd,
296
+ ignore: cfg.ignore ?? [],
297
+ ...cfg.workspaceGlobs ? { workspaceGlobs: cfg.workspaceGlobs } : {}
298
+ });
299
+ process.stdout.write(opts.format === "json" ? formatJson(report) : formatPretty(report));
300
+ process.stdout.write("\n");
301
+ const failOn = opts.failOn ?? cfg.failOn;
302
+ if (failOn) {
303
+ const hit = report.drifted.some((d) => SEVERITY_RANK[d.severity] >= SEVERITY_RANK[failOn]);
304
+ if (hit) process.exit(1);
305
+ }
306
+ });
307
+ program.command("fix").description("Bump drifted packages to a single version").option("--target <target>", "Target version (e.g. 18.2.0) or 'latest' to pick the highest currently used", "latest").option("--pkg <name>", "Limit to a single package name").option("--dry-run", "Show planned changes without writing", false).action((opts) => {
308
+ const cwd = program.opts().cwd;
309
+ const cfg = loadConfig(cwd);
310
+ const report = scan({
311
+ cwd,
312
+ ignore: cfg.ignore ?? [],
313
+ ...cfg.workspaceGlobs ? { workspaceGlobs: cfg.workspaceGlobs } : {}
314
+ });
315
+ const plan = planFix(report, {
316
+ target: opts.target,
317
+ ...opts.pkg !== void 0 ? { only: opts.pkg } : {}
318
+ });
319
+ if (plan.changes.length === 0) {
320
+ process.stdout.write("drift-check: nothing to fix\n");
321
+ return;
322
+ }
323
+ for (const c of plan.changes) {
324
+ process.stdout.write(` ${c.name}: ${c.from} \u2192 ${c.to} ${c.path}
325
+ `);
326
+ }
327
+ if (opts.dryRun) {
328
+ process.stdout.write("drift-check: dry-run, no files modified\n");
329
+ return;
330
+ }
331
+ const result = applyFix(plan);
332
+ process.stdout.write(`drift-check: updated ${result.filesChanged} file(s). Run your package manager's install.
333
+ `);
334
+ });
335
+ program.command("ignore <name>").description("Append a package name to .driftignore").action((name) => {
336
+ const cwd = program.opts().cwd;
337
+ const file = (0, import_node_path3.join)(cwd, ".driftignore");
338
+ const prefix = (0, import_node_fs4.existsSync)(file) ? "" : "# drift-check ignore list\n";
339
+ (0, import_node_fs4.appendFileSync)(file, `${prefix}${name}
340
+ `, "utf8");
341
+ process.stdout.write(`drift-check: added ${name} to .driftignore
342
+ `);
343
+ });
344
+ program.command("report").description("Alias for scan --format json (machine readable)").option("--format <fmt>", "pretty | json", "json").action((opts) => {
345
+ const cwd = program.opts().cwd;
346
+ const cfg = loadConfig(cwd);
347
+ const report = scan({
348
+ cwd,
349
+ ignore: cfg.ignore ?? [],
350
+ ...cfg.workspaceGlobs ? { workspaceGlobs: cfg.workspaceGlobs } : {}
351
+ });
352
+ process.stdout.write(opts.format === "pretty" ? formatPretty(report) : formatJson(report));
353
+ process.stdout.write("\n");
354
+ });
355
+ program.parseAsync(process.argv).catch((err) => {
356
+ process.stderr.write(`drift-check: ${err instanceof Error ? err.message : String(err)}
357
+ `);
358
+ process.exit(1);
359
+ });
360
+ //# sourceMappingURL=cli.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/cli.ts","../src/workspace.ts","../src/semver.ts","../src/scan.ts","../src/format.ts","../src/config.ts"],"sourcesContent":["#!/usr/bin/env node\nimport { Command } from \"commander\";\nimport { existsSync, appendFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport { scan, planFix, applyFix } from \"./scan.js\";\nimport { formatJson, formatPretty } from \"./format.js\";\nimport { loadConfig } from \"./config.js\";\nimport { SEVERITY_RANK } from \"./semver.js\";\nimport type { Severity } from \"./types.js\";\n\nconst program = new Command();\nprogram\n .name(\"drift-check\")\n .description(\"Detect and fix dependency drift across workspaces\")\n .option(\"--cwd <path>\", \"Working directory\", process.cwd());\n\nprogram\n .command(\"scan\", { isDefault: true })\n .description(\"Scan workspaces and print a drift report\")\n .option(\"--format <fmt>\", \"pretty | json\", \"pretty\")\n .option(\"--fail-on <severity>\", \"patch | minor | major\")\n .action((opts: { format: string; failOn?: string }) => {\n const cwd = (program.opts() as { cwd: string }).cwd;\n const cfg = loadConfig(cwd);\n const report = scan({\n cwd,\n ignore: cfg.ignore ?? [],\n ...(cfg.workspaceGlobs ? { workspaceGlobs: cfg.workspaceGlobs } : {}),\n });\n process.stdout.write(opts.format === \"json\" ? formatJson(report) : formatPretty(report));\n process.stdout.write(\"\\n\");\n const failOn = (opts.failOn ?? cfg.failOn) as Severity | undefined;\n if (failOn) {\n const hit = report.drifted.some((d) => SEVERITY_RANK[d.severity] >= SEVERITY_RANK[failOn]);\n if (hit) process.exit(1);\n }\n });\n\nprogram\n .command(\"fix\")\n .description(\"Bump drifted packages to a single version\")\n .option(\"--target <target>\", \"Target version (e.g. 18.2.0) or 'latest' to pick the highest currently used\", \"latest\")\n .option(\"--pkg <name>\", \"Limit to a single package name\")\n .option(\"--dry-run\", \"Show planned changes without writing\", false)\n .action((opts: { target: string; pkg?: string; dryRun: boolean }) => {\n const cwd = (program.opts() as { cwd: string }).cwd;\n const cfg = loadConfig(cwd);\n const report = scan({\n cwd,\n ignore: cfg.ignore ?? [],\n ...(cfg.workspaceGlobs ? { workspaceGlobs: cfg.workspaceGlobs } : {}),\n });\n const plan = planFix(report, {\n target: opts.target,\n ...(opts.pkg !== undefined ? { only: opts.pkg } : {}),\n });\n if (plan.changes.length === 0) {\n process.stdout.write(\"drift-check: nothing to fix\\n\");\n return;\n }\n for (const c of plan.changes) {\n process.stdout.write(` ${c.name}: ${c.from} → ${c.to} ${c.path}\\n`);\n }\n if (opts.dryRun) {\n process.stdout.write(\"drift-check: dry-run, no files modified\\n\");\n return;\n }\n const result = applyFix(plan);\n process.stdout.write(`drift-check: updated ${result.filesChanged} file(s). Run your package manager's install.\\n`);\n });\n\nprogram\n .command(\"ignore <name>\")\n .description(\"Append a package name to .driftignore\")\n .action((name: string) => {\n const cwd = (program.opts() as { cwd: string }).cwd;\n const file = join(cwd, \".driftignore\");\n const prefix = existsSync(file) ? \"\" : \"# drift-check ignore list\\n\";\n appendFileSync(file, `${prefix}${name}\\n`, \"utf8\");\n process.stdout.write(`drift-check: added ${name} to .driftignore\\n`);\n });\n\nprogram\n .command(\"report\")\n .description(\"Alias for scan --format json (machine readable)\")\n .option(\"--format <fmt>\", \"pretty | json\", \"json\")\n .action((opts: { format: string }) => {\n const cwd = (program.opts() as { cwd: string }).cwd;\n const cfg = loadConfig(cwd);\n const report = scan({\n cwd,\n ignore: cfg.ignore ?? [],\n ...(cfg.workspaceGlobs ? { workspaceGlobs: cfg.workspaceGlobs } : {}),\n });\n process.stdout.write(opts.format === \"pretty\" ? formatPretty(report) : formatJson(report));\n process.stdout.write(\"\\n\");\n });\n\nprogram.parseAsync(process.argv).catch((err) => {\n process.stderr.write(`drift-check: ${err instanceof Error ? err.message : String(err)}\\n`);\n process.exit(1);\n});\n","import { existsSync, readFileSync, readdirSync, statSync } from \"node:fs\";\nimport { dirname, join, resolve } from \"node:path\";\nimport YAML from \"yaml\";\nimport type { WorkspacePackage } from \"./types.js\";\n\nfunction readJson(file: string): Record<string, unknown> | undefined {\n try {\n return JSON.parse(readFileSync(file, \"utf8\")) as Record<string, unknown>;\n } catch {\n return undefined;\n }\n}\n\n/** Detect workspace format and return absolute paths to every workspace `package.json`. */\nexport function detectWorkspaces(root: string, extraGlobs?: string[]): string[] {\n root = resolve(root);\n const candidates = new Set<string>();\n const pnpm = join(root, \"pnpm-workspace.yaml\");\n if (existsSync(pnpm)) {\n try {\n const doc = YAML.parse(readFileSync(pnpm, \"utf8\")) as { packages?: string[] };\n for (const g of doc.packages ?? []) expandGlob(root, g, candidates);\n } catch {\n // ignore\n }\n }\n const rootPkg = readJson(join(root, \"package.json\"));\n if (rootPkg) {\n const ws = rootPkg.workspaces;\n if (Array.isArray(ws)) {\n for (const g of ws) if (typeof g === \"string\") expandGlob(root, g, candidates);\n } else if (ws && typeof ws === \"object\" && Array.isArray((ws as { packages?: string[] }).packages)) {\n for (const g of (ws as { packages: string[] }).packages) expandGlob(root, g, candidates);\n }\n }\n for (const g of extraGlobs ?? [\"packages/*\", \"apps/*\"]) expandGlob(root, g, candidates);\n\n if (candidates.size === 0) {\n candidates.add(join(root, \"package.json\"));\n }\n return [...candidates];\n}\n\nfunction expandGlob(root: string, glob: string, out: Set<string>): void {\n if (!glob.includes(\"*\")) {\n const pj = join(root, glob, \"package.json\");\n if (existsSync(pj)) out.add(pj);\n return;\n }\n const segments = glob.split(\"/\");\n expand(root, segments, 0, out);\n}\n\nfunction expand(dir: string, segments: string[], idx: number, out: Set<string>): void {\n if (idx === segments.length) {\n const pj = join(dir, \"package.json\");\n if (existsSync(pj)) out.add(pj);\n return;\n }\n const seg = segments[idx]!;\n if (seg === \"*\") {\n if (!existsSync(dir)) return;\n for (const entry of readdirSync(dir)) {\n const sub = join(dir, entry);\n try { if (!statSync(sub).isDirectory()) continue; } catch { continue; }\n expand(sub, segments, idx + 1, out);\n }\n } else if (seg === \"**\") {\n walk(dir, (d) => expand(d, segments, idx + 1, out));\n } else {\n expand(join(dir, seg), segments, idx + 1, out);\n }\n}\n\nfunction walk(dir: string, visit: (d: string) => void): void {\n if (!existsSync(dir)) return;\n visit(dir);\n for (const entry of readdirSync(dir)) {\n if (entry === \"node_modules\" || entry.startsWith(\".\")) continue;\n const sub = join(dir, entry);\n try { if (statSync(sub).isDirectory()) walk(sub, visit); } catch { /* skip */ }\n }\n}\n\nexport function loadWorkspaces(root: string, extraGlobs?: string[]): WorkspacePackage[] {\n return detectWorkspaces(root, extraGlobs)\n .map((file) => {\n const pj = readJson(file);\n if (!pj) return undefined;\n return {\n name: (pj.name as string) ?? dirname(file),\n version: (pj.version as string) ?? \"0.0.0\",\n packageJsonPath: file,\n dependencies: (pj.dependencies as Record<string, string>) ?? {},\n devDependencies: (pj.devDependencies as Record<string, string>) ?? {},\n peerDependencies: (pj.peerDependencies as Record<string, string>) ?? {},\n optionalDependencies: (pj.optionalDependencies as Record<string, string>) ?? {},\n };\n })\n .filter((p): p is WorkspacePackage => Boolean(p));\n}\n\nexport function readDriftIgnore(root: string): string[] {\n const file = join(root, \".driftignore\");\n if (!existsSync(file)) return [];\n return readFileSync(file, \"utf8\")\n .split(/\\r?\\n/)\n .map((s) => s.trim())\n .filter((s) => s && !s.startsWith(\"#\"));\n}\n","import type { Severity } from \"./types.js\";\n\nconst SEMVER_RE = /(\\d+)\\.(\\d+)\\.(\\d+)(?:-([0-9A-Za-z-.]+))?/;\n\nexport function extractVersion(range: string): { major: number; minor: number; patch: number } | undefined {\n const m = SEMVER_RE.exec(range);\n if (!m) return undefined;\n return { major: Number(m[1]), minor: Number(m[2]), patch: Number(m[3]) };\n}\n\nexport function severity(a: string, b: string): Severity | \"none\" {\n const va = extractVersion(a);\n const vb = extractVersion(b);\n if (!va || !vb) return a === b ? \"none\" : \"patch\";\n if (va.major !== vb.major) return \"major\";\n if (va.minor !== vb.minor) return \"minor\";\n if (va.patch !== vb.patch) return \"patch\";\n if (a !== b) return \"patch\";\n return \"none\";\n}\n\nexport function highestRange(ranges: string[]): string {\n let best = ranges[0]!;\n let bestVer = extractVersion(best);\n for (const r of ranges) {\n const v = extractVersion(r);\n if (!v) continue;\n if (!bestVer || compare(v, bestVer) > 0) {\n best = r;\n bestVer = v;\n }\n }\n return best;\n}\n\nfunction compare(a: { major: number; minor: number; patch: number }, b: { major: number; minor: number; patch: number }): number {\n if (a.major !== b.major) return a.major - b.major;\n if (a.minor !== b.minor) return a.minor - b.minor;\n return a.patch - b.patch;\n}\n\nexport const SEVERITY_RANK: Record<Severity, number> = { patch: 0, minor: 1, major: 2 };\n\nexport function maxSeverity(values: Severity[]): Severity {\n let max: Severity = \"patch\";\n for (const v of values) if (SEVERITY_RANK[v] > SEVERITY_RANK[max]) max = v;\n return max;\n}\n","import { loadWorkspaces, readDriftIgnore } from \"./workspace.js\";\nimport { highestRange, maxSeverity, severity } from \"./semver.js\";\nimport type { DepField, DriftReport, DriftedDependency, ScanOptions, Severity } from \"./types.js\";\n\nconst FIELDS: DepField[] = [\"dependencies\", \"devDependencies\", \"peerDependencies\", \"optionalDependencies\"];\n\nexport function scan(options: ScanOptions = {}): DriftReport {\n const cwd = options.cwd ?? process.cwd();\n const workspaces = loadWorkspaces(cwd, options.workspaceGlobs);\n const ignore = new Set([...(options.ignore ?? []), ...readDriftIgnore(cwd)]);\n\n const byDep = new Map<string, DriftedDependency[\"versions\"]>();\n for (const ws of workspaces) {\n for (const field of FIELDS) {\n for (const [name, version] of Object.entries(ws[field])) {\n if (ignore.has(name)) continue;\n const list = byDep.get(name) ?? [];\n list.push({ workspace: ws.name, version, field, path: ws.packageJsonPath });\n byDep.set(name, list);\n }\n }\n }\n\n const drifted: DriftedDependency[] = [];\n for (const [name, occurrences] of byDep) {\n const unique = [...new Set(occurrences.map((o) => o.version))];\n if (unique.length <= 1) continue;\n const severities: Severity[] = [];\n for (let i = 0; i < unique.length; i++) {\n for (let j = i + 1; j < unique.length; j++) {\n const s = severity(unique[i]!, unique[j]!);\n if (s !== \"none\") severities.push(s);\n }\n }\n drifted.push({\n name,\n severity: maxSeverity(severities),\n versions: occurrences,\n });\n }\n\n drifted.sort((a, b) => a.name.localeCompare(b.name));\n return { workspaces, drifted, generatedAt: new Date().toISOString() };\n}\n\nexport interface FixPlan {\n changes: {\n path: string;\n field: DepField;\n name: string;\n from: string;\n to: string;\n }[];\n}\n\nexport function planFix(report: DriftReport, options: { only?: string; target?: \"latest\" | string } = {}): FixPlan {\n const changes: FixPlan[\"changes\"] = [];\n for (const d of report.drifted) {\n if (options.only && d.name !== options.only) continue;\n const target = options.target && options.target !== \"latest\" ? options.target : highestRange(d.versions.map((v) => v.version));\n for (const v of d.versions) {\n if (v.version !== target) {\n changes.push({ path: v.path, field: v.field, name: d.name, from: v.version, to: target });\n }\n }\n }\n return { changes };\n}\n\nimport { readFileSync, writeFileSync } from \"node:fs\";\n\nexport function applyFix(plan: FixPlan): { filesChanged: number } {\n const grouped = new Map<string, FixPlan[\"changes\"]>();\n for (const c of plan.changes) {\n const arr = grouped.get(c.path) ?? [];\n arr.push(c);\n grouped.set(c.path, arr);\n }\n for (const [file, changes] of grouped) {\n const src = readFileSync(file, \"utf8\");\n let next = src;\n for (const c of changes) {\n const re = new RegExp(`(\"${escape(c.name)}\"\\\\s*:\\\\s*\")${escape(c.from)}(\")`);\n next = next.replace(re, (_m, p1: string, p2: string) => `${p1}${c.to}${p2}`);\n }\n writeFileSync(file, next, \"utf8\");\n }\n return { filesChanged: grouped.size };\n}\n\nfunction escape(s: string): string {\n return s.replace(/[.*+?^${}()|[\\]\\\\]/g, \"\\\\$&\");\n}\n","import pc from \"picocolors\";\nimport type { DriftReport, Severity } from \"./types.js\";\n\nconst sevColor: Record<Severity, (s: string) => string> = {\n patch: pc.gray,\n minor: pc.yellow,\n major: pc.red,\n};\n\nexport function formatPretty(report: DriftReport): string {\n if (report.drifted.length === 0) {\n return pc.green(`drift-check: no drift across ${report.workspaces.length} workspaces`);\n }\n const lines: string[] = [];\n lines.push(pc.bold(`drift-check — ${report.drifted.length} drifted package(s) across ${report.workspaces.length} workspaces`));\n for (const d of report.drifted) {\n lines.push(\"\");\n lines.push(` ${sevColor[d.severity](d.severity.toUpperCase())} ${pc.bold(d.name)}`);\n for (const v of d.versions) {\n lines.push(` - ${v.version} ${pc.gray(`(${v.workspace} / ${v.field})`)}`);\n }\n }\n return lines.join(\"\\n\");\n}\n\nexport function formatJson(report: DriftReport): string {\n return JSON.stringify(report, null, 2);\n}\n","import { existsSync, readFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport type { Severity } from \"./types.js\";\n\nexport interface DriftConfig {\n failOn?: Severity;\n ignore?: string[];\n workspaceGlobs?: string[];\n}\n\nexport function loadConfig(cwd: string): DriftConfig {\n const f = join(cwd, \".driftrc.json\");\n if (!existsSync(f)) return {};\n try {\n return JSON.parse(readFileSync(f, \"utf8\")) as DriftConfig;\n } catch {\n return {};\n }\n}\n"],"mappings":"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","names":["import_node_fs","import_node_path","YAML","import_node_fs","pc","import_node_fs","import_node_path"]}
package/dist/cli.d.cts ADDED
@@ -0,0 +1 @@
1
+ #!/usr/bin/env node
package/dist/cli.d.ts ADDED
@@ -0,0 +1 @@
1
+ #!/usr/bin/env node
package/dist/cli.js ADDED
@@ -0,0 +1,87 @@
1
+ #!/usr/bin/env node
2
+ import {
3
+ SEVERITY_RANK,
4
+ applyFix,
5
+ formatJson,
6
+ formatPretty,
7
+ loadConfig,
8
+ planFix,
9
+ scan
10
+ } from "./chunk-Q6ZBG4TH.js";
11
+
12
+ // src/cli.ts
13
+ import { Command } from "commander";
14
+ import { existsSync, appendFileSync } from "fs";
15
+ import { join } from "path";
16
+ var program = new Command();
17
+ program.name("drift-check").description("Detect and fix dependency drift across workspaces").option("--cwd <path>", "Working directory", process.cwd());
18
+ program.command("scan", { isDefault: true }).description("Scan workspaces and print a drift report").option("--format <fmt>", "pretty | json", "pretty").option("--fail-on <severity>", "patch | minor | major").action((opts) => {
19
+ const cwd = program.opts().cwd;
20
+ const cfg = loadConfig(cwd);
21
+ const report = scan({
22
+ cwd,
23
+ ignore: cfg.ignore ?? [],
24
+ ...cfg.workspaceGlobs ? { workspaceGlobs: cfg.workspaceGlobs } : {}
25
+ });
26
+ process.stdout.write(opts.format === "json" ? formatJson(report) : formatPretty(report));
27
+ process.stdout.write("\n");
28
+ const failOn = opts.failOn ?? cfg.failOn;
29
+ if (failOn) {
30
+ const hit = report.drifted.some((d) => SEVERITY_RANK[d.severity] >= SEVERITY_RANK[failOn]);
31
+ if (hit) process.exit(1);
32
+ }
33
+ });
34
+ program.command("fix").description("Bump drifted packages to a single version").option("--target <target>", "Target version (e.g. 18.2.0) or 'latest' to pick the highest currently used", "latest").option("--pkg <name>", "Limit to a single package name").option("--dry-run", "Show planned changes without writing", false).action((opts) => {
35
+ const cwd = program.opts().cwd;
36
+ const cfg = loadConfig(cwd);
37
+ const report = scan({
38
+ cwd,
39
+ ignore: cfg.ignore ?? [],
40
+ ...cfg.workspaceGlobs ? { workspaceGlobs: cfg.workspaceGlobs } : {}
41
+ });
42
+ const plan = planFix(report, {
43
+ target: opts.target,
44
+ ...opts.pkg !== void 0 ? { only: opts.pkg } : {}
45
+ });
46
+ if (plan.changes.length === 0) {
47
+ process.stdout.write("drift-check: nothing to fix\n");
48
+ return;
49
+ }
50
+ for (const c of plan.changes) {
51
+ process.stdout.write(` ${c.name}: ${c.from} \u2192 ${c.to} ${c.path}
52
+ `);
53
+ }
54
+ if (opts.dryRun) {
55
+ process.stdout.write("drift-check: dry-run, no files modified\n");
56
+ return;
57
+ }
58
+ const result = applyFix(plan);
59
+ process.stdout.write(`drift-check: updated ${result.filesChanged} file(s). Run your package manager's install.
60
+ `);
61
+ });
62
+ program.command("ignore <name>").description("Append a package name to .driftignore").action((name) => {
63
+ const cwd = program.opts().cwd;
64
+ const file = join(cwd, ".driftignore");
65
+ const prefix = existsSync(file) ? "" : "# drift-check ignore list\n";
66
+ appendFileSync(file, `${prefix}${name}
67
+ `, "utf8");
68
+ process.stdout.write(`drift-check: added ${name} to .driftignore
69
+ `);
70
+ });
71
+ program.command("report").description("Alias for scan --format json (machine readable)").option("--format <fmt>", "pretty | json", "json").action((opts) => {
72
+ const cwd = program.opts().cwd;
73
+ const cfg = loadConfig(cwd);
74
+ const report = scan({
75
+ cwd,
76
+ ignore: cfg.ignore ?? [],
77
+ ...cfg.workspaceGlobs ? { workspaceGlobs: cfg.workspaceGlobs } : {}
78
+ });
79
+ process.stdout.write(opts.format === "pretty" ? formatPretty(report) : formatJson(report));
80
+ process.stdout.write("\n");
81
+ });
82
+ program.parseAsync(process.argv).catch((err) => {
83
+ process.stderr.write(`drift-check: ${err instanceof Error ? err.message : String(err)}
84
+ `);
85
+ process.exit(1);
86
+ });
87
+ //# sourceMappingURL=cli.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/cli.ts"],"sourcesContent":["#!/usr/bin/env node\nimport { Command } from \"commander\";\nimport { existsSync, appendFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport { scan, planFix, applyFix } from \"./scan.js\";\nimport { formatJson, formatPretty } from \"./format.js\";\nimport { loadConfig } from \"./config.js\";\nimport { SEVERITY_RANK } from \"./semver.js\";\nimport type { Severity } from \"./types.js\";\n\nconst program = new Command();\nprogram\n .name(\"drift-check\")\n .description(\"Detect and fix dependency drift across workspaces\")\n .option(\"--cwd <path>\", \"Working directory\", process.cwd());\n\nprogram\n .command(\"scan\", { isDefault: true })\n .description(\"Scan workspaces and print a drift report\")\n .option(\"--format <fmt>\", \"pretty | json\", \"pretty\")\n .option(\"--fail-on <severity>\", \"patch | minor | major\")\n .action((opts: { format: string; failOn?: string }) => {\n const cwd = (program.opts() as { cwd: string }).cwd;\n const cfg = loadConfig(cwd);\n const report = scan({\n cwd,\n ignore: cfg.ignore ?? [],\n ...(cfg.workspaceGlobs ? { workspaceGlobs: cfg.workspaceGlobs } : {}),\n });\n process.stdout.write(opts.format === \"json\" ? formatJson(report) : formatPretty(report));\n process.stdout.write(\"\\n\");\n const failOn = (opts.failOn ?? cfg.failOn) as Severity | undefined;\n if (failOn) {\n const hit = report.drifted.some((d) => SEVERITY_RANK[d.severity] >= SEVERITY_RANK[failOn]);\n if (hit) process.exit(1);\n }\n });\n\nprogram\n .command(\"fix\")\n .description(\"Bump drifted packages to a single version\")\n .option(\"--target <target>\", \"Target version (e.g. 18.2.0) or 'latest' to pick the highest currently used\", \"latest\")\n .option(\"--pkg <name>\", \"Limit to a single package name\")\n .option(\"--dry-run\", \"Show planned changes without writing\", false)\n .action((opts: { target: string; pkg?: string; dryRun: boolean }) => {\n const cwd = (program.opts() as { cwd: string }).cwd;\n const cfg = loadConfig(cwd);\n const report = scan({\n cwd,\n ignore: cfg.ignore ?? [],\n ...(cfg.workspaceGlobs ? { workspaceGlobs: cfg.workspaceGlobs } : {}),\n });\n const plan = planFix(report, {\n target: opts.target,\n ...(opts.pkg !== undefined ? { only: opts.pkg } : {}),\n });\n if (plan.changes.length === 0) {\n process.stdout.write(\"drift-check: nothing to fix\\n\");\n return;\n }\n for (const c of plan.changes) {\n process.stdout.write(` ${c.name}: ${c.from} → ${c.to} ${c.path}\\n`);\n }\n if (opts.dryRun) {\n process.stdout.write(\"drift-check: dry-run, no files modified\\n\");\n return;\n }\n const result = applyFix(plan);\n process.stdout.write(`drift-check: updated ${result.filesChanged} file(s). Run your package manager's install.\\n`);\n });\n\nprogram\n .command(\"ignore <name>\")\n .description(\"Append a package name to .driftignore\")\n .action((name: string) => {\n const cwd = (program.opts() as { cwd: string }).cwd;\n const file = join(cwd, \".driftignore\");\n const prefix = existsSync(file) ? \"\" : \"# drift-check ignore list\\n\";\n appendFileSync(file, `${prefix}${name}\\n`, \"utf8\");\n process.stdout.write(`drift-check: added ${name} to .driftignore\\n`);\n });\n\nprogram\n .command(\"report\")\n .description(\"Alias for scan --format json (machine readable)\")\n .option(\"--format <fmt>\", \"pretty | json\", \"json\")\n .action((opts: { format: string }) => {\n const cwd = (program.opts() as { cwd: string }).cwd;\n const cfg = loadConfig(cwd);\n const report = scan({\n cwd,\n ignore: cfg.ignore ?? [],\n ...(cfg.workspaceGlobs ? { workspaceGlobs: cfg.workspaceGlobs } : {}),\n });\n process.stdout.write(opts.format === \"pretty\" ? formatPretty(report) : formatJson(report));\n process.stdout.write(\"\\n\");\n });\n\nprogram.parseAsync(process.argv).catch((err) => {\n process.stderr.write(`drift-check: ${err instanceof Error ? err.message : String(err)}\\n`);\n process.exit(1);\n});\n"],"mappings":"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","names":[]}