@sporesec/arcana 2.3.1 → 3.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli.d.ts +0 -1
- package/dist/cli.js +140 -10
- package/dist/command-registry.d.ts +10 -0
- package/dist/command-registry.js +65 -0
- package/dist/commands/audit.d.ts +0 -1
- package/dist/commands/audit.js +16 -6
- package/dist/commands/benchmark.d.ts +4 -0
- package/dist/commands/benchmark.js +178 -0
- package/dist/commands/clean.d.ts +2 -1
- package/dist/commands/clean.js +198 -47
- package/dist/commands/compact.d.ts +6 -0
- package/dist/commands/compact.js +239 -0
- package/dist/commands/completions.d.ts +3 -0
- package/dist/commands/completions.js +104 -0
- package/dist/commands/config.d.ts +0 -1
- package/dist/commands/config.js +15 -6
- package/dist/commands/create.d.ts +0 -1
- package/dist/commands/create.js +1 -1
- package/dist/commands/diff.d.ts +4 -0
- package/dist/commands/diff.js +166 -0
- package/dist/commands/doctor.d.ts +0 -1
- package/dist/commands/doctor.js +153 -24
- package/dist/commands/export-cmd.d.ts +4 -0
- package/dist/commands/export-cmd.js +66 -0
- package/dist/commands/import-cmd.d.ts +4 -0
- package/dist/commands/import-cmd.js +131 -0
- package/dist/commands/info.d.ts +0 -1
- package/dist/commands/info.js +29 -4
- package/dist/commands/init.d.ts +0 -1
- package/dist/commands/init.js +156 -117
- package/dist/commands/install.d.ts +1 -1
- package/dist/commands/install.js +118 -205
- package/dist/commands/list.d.ts +0 -1
- package/dist/commands/list.js +12 -4
- package/dist/commands/lock.d.ts +4 -0
- package/dist/commands/lock.js +171 -0
- package/dist/commands/optimize.d.ts +3 -0
- package/dist/commands/optimize.js +356 -0
- package/dist/commands/outdated.d.ts +4 -0
- package/dist/commands/outdated.js +159 -0
- package/dist/commands/profile.d.ts +3 -0
- package/dist/commands/profile.js +274 -0
- package/dist/commands/providers.d.ts +0 -1
- package/dist/commands/providers.js +1 -4
- package/dist/commands/recommend.d.ts +5 -0
- package/dist/commands/recommend.js +96 -0
- package/dist/commands/scan.d.ts +0 -1
- package/dist/commands/scan.js +13 -7
- package/dist/commands/search.d.ts +2 -1
- package/dist/commands/search.js +32 -9
- package/dist/commands/stats.d.ts +0 -1
- package/dist/commands/stats.js +83 -16
- package/dist/commands/team.d.ts +3 -0
- package/dist/commands/team.js +291 -0
- package/dist/commands/uninstall.d.ts +0 -1
- package/dist/commands/uninstall.js +18 -4
- package/dist/commands/update.d.ts +0 -1
- package/dist/commands/update.js +155 -155
- package/dist/commands/validate.d.ts +0 -1
- package/dist/commands/validate.js +14 -6
- package/dist/commands/verify.d.ts +4 -0
- package/dist/commands/verify.js +116 -0
- package/dist/constants.d.ts +10 -0
- package/dist/constants.js +13 -0
- package/dist/index.d.ts +0 -1
- package/dist/index.js +0 -1
- package/dist/interactive/browse.d.ts +4 -0
- package/dist/interactive/browse.js +103 -0
- package/dist/interactive/categories.d.ts +4 -0
- package/dist/interactive/categories.js +87 -0
- package/dist/interactive/health.d.ts +1 -0
- package/dist/interactive/health.js +57 -0
- package/dist/interactive/helpers.d.ts +11 -0
- package/dist/interactive/helpers.js +66 -0
- package/dist/interactive/index.d.ts +1 -0
- package/dist/interactive/index.js +1 -0
- package/dist/interactive/manage.d.ts +2 -0
- package/dist/interactive/manage.js +187 -0
- package/dist/interactive/menu.d.ts +1 -0
- package/dist/interactive/menu.js +107 -0
- package/dist/interactive/search.d.ts +2 -0
- package/dist/interactive/search.js +66 -0
- package/dist/interactive/setup.d.ts +2 -0
- package/dist/interactive/setup.js +48 -0
- package/dist/interactive/skill-detail.d.ts +5 -0
- package/dist/interactive/skill-detail.js +126 -0
- package/dist/interactive.d.ts +0 -1
- package/dist/interactive.js +89 -66
- package/dist/providers/arcana.d.ts +0 -1
- package/dist/providers/arcana.js +0 -1
- package/dist/providers/base.d.ts +0 -1
- package/dist/providers/base.js +0 -1
- package/dist/providers/github.d.ts +0 -1
- package/dist/providers/github.js +8 -3
- package/dist/registry.d.ts +0 -1
- package/dist/registry.js +1 -4
- package/dist/types.d.ts +10 -1
- package/dist/types.js +0 -1
- package/dist/utils/atomic.d.ts +0 -1
- package/dist/utils/atomic.js +3 -2
- package/dist/utils/cache.d.ts +0 -1
- package/dist/utils/cache.js +3 -2
- package/dist/utils/config.d.ts +2 -1
- package/dist/utils/config.js +30 -5
- package/dist/utils/conflict-check.d.ts +8 -0
- package/dist/utils/conflict-check.js +72 -0
- package/dist/utils/errors.d.ts +0 -1
- package/dist/utils/errors.js +0 -1
- package/dist/utils/frontmatter.d.ts +0 -1
- package/dist/utils/frontmatter.js +37 -10
- package/dist/utils/fs.d.ts +19 -1
- package/dist/utils/fs.js +105 -8
- package/dist/utils/help.d.ts +0 -1
- package/dist/utils/help.js +15 -28
- package/dist/utils/history.d.ts +0 -1
- package/dist/utils/history.js +0 -1
- package/dist/utils/http.d.ts +0 -1
- package/dist/utils/http.js +14 -5
- package/dist/utils/install-core.d.ts +48 -0
- package/dist/utils/install-core.js +108 -0
- package/dist/utils/integrity.d.ts +17 -0
- package/dist/utils/integrity.js +84 -0
- package/dist/utils/parallel.d.ts +0 -1
- package/dist/utils/parallel.js +0 -1
- package/dist/utils/project-context.d.ts +19 -0
- package/dist/utils/project-context.js +283 -0
- package/dist/utils/scanner.d.ts +0 -1
- package/dist/utils/scanner.js +138 -10
- package/dist/utils/scoring.d.ts +10 -0
- package/dist/utils/scoring.js +84 -0
- package/dist/utils/ui.d.ts +0 -1
- package/dist/utils/ui.js +11 -4
- package/dist/utils/validate.d.ts +0 -1
- package/dist/utils/validate.js +4 -1
- package/package.json +19 -7
- package/dist/cli.d.ts.map +0 -1
- package/dist/cli.js.map +0 -1
- package/dist/commands/audit.d.ts.map +0 -1
- package/dist/commands/audit.js.map +0 -1
- package/dist/commands/audit.test.d.ts +0 -2
- package/dist/commands/audit.test.d.ts.map +0 -1
- package/dist/commands/audit.test.js +0 -217
- package/dist/commands/audit.test.js.map +0 -1
- package/dist/commands/clean.d.ts.map +0 -1
- package/dist/commands/clean.js.map +0 -1
- package/dist/commands/config.d.ts.map +0 -1
- package/dist/commands/config.js.map +0 -1
- package/dist/commands/create.d.ts.map +0 -1
- package/dist/commands/create.js.map +0 -1
- package/dist/commands/doctor.d.ts.map +0 -1
- package/dist/commands/doctor.js.map +0 -1
- package/dist/commands/info.d.ts.map +0 -1
- package/dist/commands/info.js.map +0 -1
- package/dist/commands/init.d.ts.map +0 -1
- package/dist/commands/init.js.map +0 -1
- package/dist/commands/install.d.ts.map +0 -1
- package/dist/commands/install.js.map +0 -1
- package/dist/commands/list.d.ts.map +0 -1
- package/dist/commands/list.js.map +0 -1
- package/dist/commands/providers.d.ts.map +0 -1
- package/dist/commands/providers.js.map +0 -1
- package/dist/commands/scan.d.ts.map +0 -1
- package/dist/commands/scan.js.map +0 -1
- package/dist/commands/search.d.ts.map +0 -1
- package/dist/commands/search.js.map +0 -1
- package/dist/commands/stats.d.ts.map +0 -1
- package/dist/commands/stats.js.map +0 -1
- package/dist/commands/uninstall.d.ts.map +0 -1
- package/dist/commands/uninstall.js.map +0 -1
- package/dist/commands/update.d.ts.map +0 -1
- package/dist/commands/update.js.map +0 -1
- package/dist/commands/validate.d.ts.map +0 -1
- package/dist/commands/validate.js.map +0 -1
- package/dist/index.d.ts.map +0 -1
- package/dist/index.js.map +0 -1
- package/dist/interactive.d.ts.map +0 -1
- package/dist/interactive.js.map +0 -1
- package/dist/providers/arcana.d.ts.map +0 -1
- package/dist/providers/arcana.js.map +0 -1
- package/dist/providers/base.d.ts.map +0 -1
- package/dist/providers/base.js.map +0 -1
- package/dist/providers/github.d.ts.map +0 -1
- package/dist/providers/github.js.map +0 -1
- package/dist/registry.d.ts.map +0 -1
- package/dist/registry.js.map +0 -1
- package/dist/types.d.ts.map +0 -1
- package/dist/types.js.map +0 -1
- package/dist/utils/atomic.d.ts.map +0 -1
- package/dist/utils/atomic.js.map +0 -1
- package/dist/utils/atomic.test.d.ts +0 -2
- package/dist/utils/atomic.test.d.ts.map +0 -1
- package/dist/utils/atomic.test.js +0 -31
- package/dist/utils/atomic.test.js.map +0 -1
- package/dist/utils/cache.d.ts.map +0 -1
- package/dist/utils/cache.js.map +0 -1
- package/dist/utils/config.d.ts.map +0 -1
- package/dist/utils/config.js.map +0 -1
- package/dist/utils/config.test.d.ts +0 -2
- package/dist/utils/config.test.d.ts.map +0 -1
- package/dist/utils/config.test.js +0 -38
- package/dist/utils/config.test.js.map +0 -1
- package/dist/utils/errors.d.ts.map +0 -1
- package/dist/utils/errors.js.map +0 -1
- package/dist/utils/frontmatter.d.ts.map +0 -1
- package/dist/utils/frontmatter.js.map +0 -1
- package/dist/utils/frontmatter.test.d.ts +0 -2
- package/dist/utils/frontmatter.test.d.ts.map +0 -1
- package/dist/utils/frontmatter.test.js +0 -152
- package/dist/utils/frontmatter.test.js.map +0 -1
- package/dist/utils/fs.d.ts.map +0 -1
- package/dist/utils/fs.js.map +0 -1
- package/dist/utils/fs.test.d.ts +0 -2
- package/dist/utils/fs.test.d.ts.map +0 -1
- package/dist/utils/fs.test.js +0 -145
- package/dist/utils/fs.test.js.map +0 -1
- package/dist/utils/help.d.ts.map +0 -1
- package/dist/utils/help.js.map +0 -1
- package/dist/utils/help.test.d.ts +0 -2
- package/dist/utils/help.test.d.ts.map +0 -1
- package/dist/utils/help.test.js +0 -66
- package/dist/utils/help.test.js.map +0 -1
- package/dist/utils/history.d.ts.map +0 -1
- package/dist/utils/history.js.map +0 -1
- package/dist/utils/http.d.ts.map +0 -1
- package/dist/utils/http.js.map +0 -1
- package/dist/utils/http.test.d.ts +0 -2
- package/dist/utils/http.test.d.ts.map +0 -1
- package/dist/utils/http.test.js +0 -55
- package/dist/utils/http.test.js.map +0 -1
- package/dist/utils/parallel.d.ts.map +0 -1
- package/dist/utils/parallel.js.map +0 -1
- package/dist/utils/scanner.d.ts.map +0 -1
- package/dist/utils/scanner.js.map +0 -1
- package/dist/utils/ui.d.ts.map +0 -1
- package/dist/utils/ui.js.map +0 -1
- package/dist/utils/ui.test.d.ts +0 -2
- package/dist/utils/ui.test.d.ts.map +0 -1
- package/dist/utils/ui.test.js +0 -31
- package/dist/utils/ui.test.js.map +0 -1
- package/dist/utils/validate.d.ts.map +0 -1
- package/dist/utils/validate.js.map +0 -1
|
@@ -0,0 +1,171 @@
|
|
|
1
|
+
import { readdirSync, readFileSync, existsSync, lstatSync } from "node:fs";
|
|
2
|
+
import { join } from "node:path";
|
|
3
|
+
import * as p from "@clack/prompts";
|
|
4
|
+
import chalk from "chalk";
|
|
5
|
+
import { getInstallDir, readSkillMeta } from "../utils/fs.js";
|
|
6
|
+
import { readLockfile, writeLockfile, computeHash } from "../utils/integrity.js";
|
|
7
|
+
import { renderBanner } from "../utils/help.js";
|
|
8
|
+
import { printErrorWithHint } from "../utils/ui.js";
|
|
9
|
+
function readSkillFiles(skillDir) {
|
|
10
|
+
const files = [];
|
|
11
|
+
const queue = [skillDir];
|
|
12
|
+
while (queue.length > 0) {
|
|
13
|
+
const dir = queue.pop();
|
|
14
|
+
for (const entry of readdirSync(dir)) {
|
|
15
|
+
const full = join(dir, entry);
|
|
16
|
+
const stat = lstatSync(full);
|
|
17
|
+
if (stat.isDirectory())
|
|
18
|
+
queue.push(full);
|
|
19
|
+
else if (stat.isFile()) {
|
|
20
|
+
const relPath = full.slice(skillDir.length + 1).replace(/\\/g, "/");
|
|
21
|
+
files.push({ path: relPath, content: readFileSync(full, "utf-8") });
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
return files;
|
|
26
|
+
}
|
|
27
|
+
export async function lockCommand(opts) {
|
|
28
|
+
if (opts.ci) {
|
|
29
|
+
return ciMode(opts.json);
|
|
30
|
+
}
|
|
31
|
+
return generateMode(opts.json);
|
|
32
|
+
}
|
|
33
|
+
async function generateMode(json) {
|
|
34
|
+
if (!json) {
|
|
35
|
+
console.log(renderBanner());
|
|
36
|
+
console.log();
|
|
37
|
+
p.intro(chalk.bold("Generate lockfile"));
|
|
38
|
+
}
|
|
39
|
+
const installDir = getInstallDir();
|
|
40
|
+
if (!existsSync(installDir)) {
|
|
41
|
+
if (json) {
|
|
42
|
+
console.log(JSON.stringify({ action: "generate", entries: 0, path: "~/.arcana/arcana-lock.json" }));
|
|
43
|
+
}
|
|
44
|
+
else {
|
|
45
|
+
p.log.info("No skills installed. Lockfile written with 0 entries.");
|
|
46
|
+
}
|
|
47
|
+
writeLockfile([]);
|
|
48
|
+
return;
|
|
49
|
+
}
|
|
50
|
+
const dirs = readdirSync(installDir).filter((d) => {
|
|
51
|
+
try {
|
|
52
|
+
return lstatSync(join(installDir, d)).isDirectory();
|
|
53
|
+
}
|
|
54
|
+
catch {
|
|
55
|
+
return false;
|
|
56
|
+
}
|
|
57
|
+
});
|
|
58
|
+
const entries = [];
|
|
59
|
+
for (const name of dirs) {
|
|
60
|
+
const skillDir = join(installDir, name);
|
|
61
|
+
const files = readSkillFiles(skillDir);
|
|
62
|
+
const sorted = [...files].sort((a, b) => a.path.localeCompare(b.path));
|
|
63
|
+
const concatenated = sorted.map((f) => f.content).join("");
|
|
64
|
+
const hash = computeHash(concatenated);
|
|
65
|
+
const meta = readSkillMeta(name);
|
|
66
|
+
entries.push({
|
|
67
|
+
skill: name,
|
|
68
|
+
version: meta?.version ?? "0.0.0",
|
|
69
|
+
hash,
|
|
70
|
+
source: meta?.source ?? "unknown",
|
|
71
|
+
installedAt: meta?.installedAt ?? new Date().toISOString(),
|
|
72
|
+
});
|
|
73
|
+
}
|
|
74
|
+
writeLockfile(entries);
|
|
75
|
+
if (json) {
|
|
76
|
+
console.log(JSON.stringify({ action: "generate", entries: entries.length, path: "~/.arcana/arcana-lock.json" }));
|
|
77
|
+
}
|
|
78
|
+
else {
|
|
79
|
+
p.log.success(`Lockfile written with ${entries.length} entries.`);
|
|
80
|
+
p.outro(chalk.dim("~/.arcana/arcana-lock.json"));
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
async function ciMode(json) {
|
|
84
|
+
if (!json) {
|
|
85
|
+
console.log(renderBanner());
|
|
86
|
+
console.log();
|
|
87
|
+
p.intro(chalk.bold("Validate lockfile"));
|
|
88
|
+
}
|
|
89
|
+
const existing = readLockfile();
|
|
90
|
+
if (existing.length === 0) {
|
|
91
|
+
const lockPath = join(getInstallDir(), "..", "arcana-lock.json");
|
|
92
|
+
if (!existsSync(lockPath)) {
|
|
93
|
+
if (json) {
|
|
94
|
+
console.log(JSON.stringify({
|
|
95
|
+
action: "ci",
|
|
96
|
+
valid: false,
|
|
97
|
+
mismatches: [],
|
|
98
|
+
missing: [],
|
|
99
|
+
extra: [],
|
|
100
|
+
error: "No lockfile found",
|
|
101
|
+
}));
|
|
102
|
+
}
|
|
103
|
+
else {
|
|
104
|
+
printErrorWithHint(new Error("No lockfile found. Run `arcana lock` first to generate one."), true);
|
|
105
|
+
}
|
|
106
|
+
process.exit(1);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
const installDir = getInstallDir();
|
|
110
|
+
const installedDirs = existsSync(installDir)
|
|
111
|
+
? readdirSync(installDir).filter((d) => {
|
|
112
|
+
try {
|
|
113
|
+
return lstatSync(join(installDir, d)).isDirectory();
|
|
114
|
+
}
|
|
115
|
+
catch {
|
|
116
|
+
return false;
|
|
117
|
+
}
|
|
118
|
+
})
|
|
119
|
+
: [];
|
|
120
|
+
const lockedNames = new Set(existing.map((e) => e.skill));
|
|
121
|
+
const installedNames = new Set(installedDirs);
|
|
122
|
+
const mismatches = [];
|
|
123
|
+
const missing = [];
|
|
124
|
+
const extra = [];
|
|
125
|
+
// Check each lockfile entry against installed state
|
|
126
|
+
for (const entry of existing) {
|
|
127
|
+
if (!installedNames.has(entry.skill)) {
|
|
128
|
+
missing.push(entry.skill);
|
|
129
|
+
continue;
|
|
130
|
+
}
|
|
131
|
+
const skillDir = join(installDir, entry.skill);
|
|
132
|
+
const files = readSkillFiles(skillDir);
|
|
133
|
+
const sorted = [...files].sort((a, b) => a.path.localeCompare(b.path));
|
|
134
|
+
const concatenated = sorted.map((f) => f.content).join("");
|
|
135
|
+
const hash = computeHash(concatenated);
|
|
136
|
+
if (hash !== entry.hash) {
|
|
137
|
+
mismatches.push(entry.skill);
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
// Check for extra skills not in lockfile
|
|
141
|
+
for (const name of installedDirs) {
|
|
142
|
+
if (!lockedNames.has(name)) {
|
|
143
|
+
extra.push(name);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
const valid = mismatches.length === 0 && missing.length === 0 && extra.length === 0;
|
|
147
|
+
if (json) {
|
|
148
|
+
console.log(JSON.stringify({ action: "ci", valid, mismatches, missing, extra }));
|
|
149
|
+
}
|
|
150
|
+
else {
|
|
151
|
+
if (valid) {
|
|
152
|
+
p.log.success("Lockfile matches installed state.");
|
|
153
|
+
p.outro(chalk.dim("All entries verified."));
|
|
154
|
+
}
|
|
155
|
+
else {
|
|
156
|
+
if (mismatches.length > 0) {
|
|
157
|
+
p.log.error(`Hash mismatch: ${mismatches.join(", ")}`);
|
|
158
|
+
}
|
|
159
|
+
if (missing.length > 0) {
|
|
160
|
+
p.log.error(`Missing from disk: ${missing.join(", ")}`);
|
|
161
|
+
}
|
|
162
|
+
if (extra.length > 0) {
|
|
163
|
+
p.log.warn(`Extra (not in lockfile): ${extra.join(", ")}`);
|
|
164
|
+
}
|
|
165
|
+
p.outro(chalk.dim("Lockfile validation failed."));
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
if (!valid) {
|
|
169
|
+
process.exit(1);
|
|
170
|
+
}
|
|
171
|
+
}
|
|
@@ -0,0 +1,356 @@
|
|
|
1
|
+
import { existsSync, readFileSync, readdirSync, statSync } from "node:fs";
|
|
2
|
+
import { join } from "node:path";
|
|
3
|
+
import { homedir } from "node:os";
|
|
4
|
+
import { ui, banner } from "../utils/ui.js";
|
|
5
|
+
import { getInstallDir, getDirSize } from "../utils/fs.js";
|
|
6
|
+
function readSettings() {
|
|
7
|
+
const settingsPath = join(homedir(), ".claude", "settings.json");
|
|
8
|
+
if (!existsSync(settingsPath))
|
|
9
|
+
return null;
|
|
10
|
+
try {
|
|
11
|
+
return JSON.parse(readFileSync(settingsPath, "utf-8"));
|
|
12
|
+
}
|
|
13
|
+
catch {
|
|
14
|
+
return null;
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
function checkAutocompact() {
|
|
18
|
+
const settings = readSettings();
|
|
19
|
+
if (!settings) {
|
|
20
|
+
return {
|
|
21
|
+
area: "Autocompact",
|
|
22
|
+
status: "suggest",
|
|
23
|
+
message: "No settings.json found",
|
|
24
|
+
action: "Set CLAUDE_AUTOCOMPACT_PCT_OVERRIDE=80 in ~/.claude/settings.json env block",
|
|
25
|
+
};
|
|
26
|
+
}
|
|
27
|
+
const env = settings.env;
|
|
28
|
+
const val = env?.CLAUDE_AUTOCOMPACT_PCT_OVERRIDE;
|
|
29
|
+
if (!val) {
|
|
30
|
+
return {
|
|
31
|
+
area: "Autocompact",
|
|
32
|
+
status: "suggest",
|
|
33
|
+
message: "Not configured (defaults to high threshold)",
|
|
34
|
+
action: "Set to 80 to compact earlier and save tokens",
|
|
35
|
+
};
|
|
36
|
+
}
|
|
37
|
+
const pct = parseInt(val);
|
|
38
|
+
if (pct <= 70) {
|
|
39
|
+
return {
|
|
40
|
+
area: "Autocompact",
|
|
41
|
+
status: "warn",
|
|
42
|
+
message: `Set to ${pct}%. Too aggressive, may lose context.`,
|
|
43
|
+
action: "Raise to 75-80% for better balance",
|
|
44
|
+
};
|
|
45
|
+
}
|
|
46
|
+
if (pct > 85) {
|
|
47
|
+
return {
|
|
48
|
+
area: "Autocompact",
|
|
49
|
+
status: "suggest",
|
|
50
|
+
message: `Set to ${pct}%. Compaction happens late, less room for reasoning.`,
|
|
51
|
+
action: "Lower to 80% for better quality (research-backed)",
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
return { area: "Autocompact", status: "good", message: `Set to ${pct}% (optimal range)` };
|
|
55
|
+
}
|
|
56
|
+
function checkEffortLevel() {
|
|
57
|
+
const settings = readSettings();
|
|
58
|
+
const env = (settings?.env ?? {});
|
|
59
|
+
const val = env.CLAUDE_CODE_EFFORT_LEVEL;
|
|
60
|
+
if (!val) {
|
|
61
|
+
return { area: "Effort level", status: "good", message: "Using default (high)" };
|
|
62
|
+
}
|
|
63
|
+
if (val === "low") {
|
|
64
|
+
return {
|
|
65
|
+
area: "Effort level",
|
|
66
|
+
status: "suggest",
|
|
67
|
+
message: "Set to 'low'. Faster but may miss details.",
|
|
68
|
+
action: "Use 'medium' for daily work, 'high' for complex tasks",
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
if (val === "medium") {
|
|
72
|
+
return { area: "Effort level", status: "good", message: "Set to 'medium'. Good balance of speed and quality." };
|
|
73
|
+
}
|
|
74
|
+
return { area: "Effort level", status: "good", message: `Set to '${val}'` };
|
|
75
|
+
}
|
|
76
|
+
function checkNonEssentialCalls() {
|
|
77
|
+
const settings = readSettings();
|
|
78
|
+
const env = (settings?.env ?? {});
|
|
79
|
+
const val = env.DISABLE_NON_ESSENTIAL_MODEL_CALLS;
|
|
80
|
+
if (val === "1" || val === "true") {
|
|
81
|
+
return { area: "Non-essential calls", status: "good", message: "Disabled (saves tokens)" };
|
|
82
|
+
}
|
|
83
|
+
return {
|
|
84
|
+
area: "Non-essential calls",
|
|
85
|
+
status: "suggest",
|
|
86
|
+
message: "Not disabled",
|
|
87
|
+
action: "Set DISABLE_NON_ESSENTIAL_MODEL_CALLS=1 in settings.json env to save tokens",
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
function checkSkillTokenBudget() {
|
|
91
|
+
const dir = getInstallDir();
|
|
92
|
+
if (!existsSync(dir)) {
|
|
93
|
+
return { area: "Skill token budget", status: "good", message: "No skills installed" };
|
|
94
|
+
}
|
|
95
|
+
let totalKB = 0;
|
|
96
|
+
let skillCount = 0;
|
|
97
|
+
const large = [];
|
|
98
|
+
for (const entry of readdirSync(dir)) {
|
|
99
|
+
const skillDir = join(dir, entry);
|
|
100
|
+
if (!statSync(skillDir).isDirectory())
|
|
101
|
+
continue;
|
|
102
|
+
skillCount++;
|
|
103
|
+
const kb = getDirSize(skillDir) / 1024;
|
|
104
|
+
totalKB += kb;
|
|
105
|
+
if (kb > 50)
|
|
106
|
+
large.push({ name: entry, kb });
|
|
107
|
+
}
|
|
108
|
+
const estTokens = Math.round(totalKB * 256);
|
|
109
|
+
if (totalKB > 500) {
|
|
110
|
+
large.sort((a, b) => b.kb - a.kb);
|
|
111
|
+
const topNames = large
|
|
112
|
+
.slice(0, 3)
|
|
113
|
+
.map((s) => s.name)
|
|
114
|
+
.join(", ");
|
|
115
|
+
return {
|
|
116
|
+
area: "Skill token budget",
|
|
117
|
+
status: "warn",
|
|
118
|
+
message: `${skillCount} skills, ${totalKB.toFixed(0)} KB (~${(estTokens / 1000).toFixed(0)}K tokens). Heavy context load.`,
|
|
119
|
+
action: `Consider uninstalling rarely used skills. Largest: ${topNames}`,
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
if (totalKB > 200) {
|
|
123
|
+
return {
|
|
124
|
+
area: "Skill token budget",
|
|
125
|
+
status: "suggest",
|
|
126
|
+
message: `${skillCount} skills, ${totalKB.toFixed(0)} KB (~${(estTokens / 1000).toFixed(0)}K tokens)`,
|
|
127
|
+
action: "Review installed skills with 'arcana list --installed' and remove unused ones",
|
|
128
|
+
};
|
|
129
|
+
}
|
|
130
|
+
return {
|
|
131
|
+
area: "Skill token budget",
|
|
132
|
+
status: "good",
|
|
133
|
+
message: `${skillCount} skills, ${totalKB.toFixed(0)} KB (~${(estTokens / 1000).toFixed(0)}K tokens)`,
|
|
134
|
+
};
|
|
135
|
+
}
|
|
136
|
+
function checkDiskHealth() {
|
|
137
|
+
const claudeDir = join(homedir(), ".claude");
|
|
138
|
+
if (!existsSync(claudeDir)) {
|
|
139
|
+
return { area: "Disk health", status: "good", message: "No Claude data directory" };
|
|
140
|
+
}
|
|
141
|
+
const totalMB = getDirSize(claudeDir) / (1024 * 1024);
|
|
142
|
+
if (totalMB > 1000) {
|
|
143
|
+
return {
|
|
144
|
+
area: "Disk health",
|
|
145
|
+
status: "warn",
|
|
146
|
+
message: `${totalMB.toFixed(0)} MB total Claude data`,
|
|
147
|
+
action: "Run: arcana compact (removes agent logs, keeps sessions)",
|
|
148
|
+
};
|
|
149
|
+
}
|
|
150
|
+
if (totalMB > 500) {
|
|
151
|
+
return {
|
|
152
|
+
area: "Disk health",
|
|
153
|
+
status: "suggest",
|
|
154
|
+
message: `${totalMB.toFixed(0)} MB total Claude data`,
|
|
155
|
+
action: "Run: arcana compact",
|
|
156
|
+
};
|
|
157
|
+
}
|
|
158
|
+
return { area: "Disk health", status: "good", message: `${totalMB.toFixed(0)} MB total Claude data` };
|
|
159
|
+
}
|
|
160
|
+
function checkPreCompactHook() {
|
|
161
|
+
// Check both global and local settings for PreCompact hooks
|
|
162
|
+
const paths = [join(homedir(), ".claude", "settings.json"), join(homedir(), ".claude", "settings.local.json")];
|
|
163
|
+
for (const settingsPath of paths) {
|
|
164
|
+
if (!existsSync(settingsPath))
|
|
165
|
+
continue;
|
|
166
|
+
try {
|
|
167
|
+
const settings = JSON.parse(readFileSync(settingsPath, "utf-8"));
|
|
168
|
+
const hooks = settings?.hooks?.PreCompact;
|
|
169
|
+
if (hooks && Array.isArray(hooks) && hooks.length > 0) {
|
|
170
|
+
return { area: "PreCompact hook", status: "good", message: "Installed. Context preserved before compaction." };
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
catch {
|
|
174
|
+
continue;
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
return {
|
|
178
|
+
area: "PreCompact hook",
|
|
179
|
+
status: "suggest",
|
|
180
|
+
message: "Not installed. Context is lost during auto-compaction.",
|
|
181
|
+
action: "Run: arcana init --tool claude (adds PreCompact hook to preserve context)",
|
|
182
|
+
};
|
|
183
|
+
}
|
|
184
|
+
function checkMemorySize() {
|
|
185
|
+
const projectsDir = join(homedir(), ".claude", "projects");
|
|
186
|
+
if (!existsSync(projectsDir)) {
|
|
187
|
+
return { area: "MEMORY.md sizes", status: "good", message: "No project memory files" };
|
|
188
|
+
}
|
|
189
|
+
const oversized = [];
|
|
190
|
+
for (const entry of readdirSync(projectsDir)) {
|
|
191
|
+
const memDir = join(projectsDir, entry, "memory");
|
|
192
|
+
const memFile = join(memDir, "MEMORY.md");
|
|
193
|
+
if (!existsSync(memFile))
|
|
194
|
+
continue;
|
|
195
|
+
try {
|
|
196
|
+
const content = readFileSync(memFile, "utf-8");
|
|
197
|
+
const lineCount = content.split("\n").length;
|
|
198
|
+
if (lineCount > 200) {
|
|
199
|
+
oversized.push({ project: entry, lines: lineCount });
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
catch {
|
|
203
|
+
continue;
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
if (oversized.length === 0) {
|
|
207
|
+
return { area: "MEMORY.md sizes", status: "good", message: "All under 200 lines (auto-load limit)" };
|
|
208
|
+
}
|
|
209
|
+
oversized.sort((a, b) => b.lines - a.lines);
|
|
210
|
+
const top = oversized[0];
|
|
211
|
+
return {
|
|
212
|
+
area: "MEMORY.md sizes",
|
|
213
|
+
status: "warn",
|
|
214
|
+
message: `${oversized.length} MEMORY.md file${oversized.length > 1 ? "s" : ""} exceed 200 lines. Only first 200 auto-load. Worst: ${top.project} (${top.lines} lines)`,
|
|
215
|
+
action: "Trim to 200 lines. Move detailed notes to separate topic files in memory/",
|
|
216
|
+
};
|
|
217
|
+
}
|
|
218
|
+
function checkAgentBloat() {
|
|
219
|
+
const projectsDir = join(homedir(), ".claude", "projects");
|
|
220
|
+
if (!existsSync(projectsDir)) {
|
|
221
|
+
return { area: "Agent log bloat", status: "good", message: "No session data" };
|
|
222
|
+
}
|
|
223
|
+
let agentBytes = 0;
|
|
224
|
+
let mainBytes = 0;
|
|
225
|
+
let agentCount = 0;
|
|
226
|
+
let mainCount = 0;
|
|
227
|
+
for (const project of readdirSync(projectsDir)) {
|
|
228
|
+
const projDir = join(projectsDir, project);
|
|
229
|
+
if (!statSync(projDir).isDirectory())
|
|
230
|
+
continue;
|
|
231
|
+
for (const file of readdirSync(projDir)) {
|
|
232
|
+
if (!file.endsWith(".jsonl"))
|
|
233
|
+
continue;
|
|
234
|
+
try {
|
|
235
|
+
const size = statSync(join(projDir, file)).size;
|
|
236
|
+
if (file.startsWith("agent-")) {
|
|
237
|
+
agentBytes += size;
|
|
238
|
+
agentCount++;
|
|
239
|
+
}
|
|
240
|
+
else {
|
|
241
|
+
mainBytes += size;
|
|
242
|
+
mainCount++;
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
catch {
|
|
246
|
+
continue;
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
const totalMB = (agentBytes + mainBytes) / (1024 * 1024);
|
|
251
|
+
const agentMB = agentBytes / (1024 * 1024);
|
|
252
|
+
const agentPct = totalMB > 0 ? Math.round((agentMB / totalMB) * 100) : 0;
|
|
253
|
+
if (agentPct > 70 && agentMB > 50) {
|
|
254
|
+
return {
|
|
255
|
+
area: "Agent log bloat",
|
|
256
|
+
status: "warn",
|
|
257
|
+
message: `${agentCount} agent logs (${agentMB.toFixed(0)} MB, ${agentPct}% of all logs). ${mainCount} main sessions (${(mainBytes / (1024 * 1024)).toFixed(0)} MB).`,
|
|
258
|
+
action: "Run: arcana compact (removes agent logs, keeps main sessions)",
|
|
259
|
+
};
|
|
260
|
+
}
|
|
261
|
+
if (agentMB > 20) {
|
|
262
|
+
return {
|
|
263
|
+
area: "Agent log bloat",
|
|
264
|
+
status: "suggest",
|
|
265
|
+
message: `${agentCount} agent logs (${agentMB.toFixed(0)} MB, ${agentPct}%). ${mainCount} main sessions.`,
|
|
266
|
+
action: "Run: arcana compact",
|
|
267
|
+
};
|
|
268
|
+
}
|
|
269
|
+
return {
|
|
270
|
+
area: "Agent log bloat",
|
|
271
|
+
status: "good",
|
|
272
|
+
message: `${agentCount} agent logs (${agentMB.toFixed(0)} MB), ${mainCount} main sessions (${(mainBytes / (1024 * 1024)).toFixed(0)} MB)`,
|
|
273
|
+
};
|
|
274
|
+
}
|
|
275
|
+
function checkLargestSkills() {
|
|
276
|
+
const dir = getInstallDir();
|
|
277
|
+
if (!existsSync(dir)) {
|
|
278
|
+
return { area: "Top skills by size", status: "good", message: "No skills installed" };
|
|
279
|
+
}
|
|
280
|
+
const skills = [];
|
|
281
|
+
for (const entry of readdirSync(dir)) {
|
|
282
|
+
const skillDir = join(dir, entry);
|
|
283
|
+
if (!statSync(skillDir).isDirectory())
|
|
284
|
+
continue;
|
|
285
|
+
const kb = getDirSize(skillDir) / 1024;
|
|
286
|
+
skills.push({ name: entry, kb });
|
|
287
|
+
}
|
|
288
|
+
if (skills.length === 0) {
|
|
289
|
+
return { area: "Top skills by size", status: "good", message: "No skills installed" };
|
|
290
|
+
}
|
|
291
|
+
skills.sort((a, b) => b.kb - a.kb);
|
|
292
|
+
const totalKB = skills.reduce((s, sk) => s + sk.kb, 0);
|
|
293
|
+
const totalMB = totalKB / 1024;
|
|
294
|
+
const top5 = skills
|
|
295
|
+
.slice(0, 5)
|
|
296
|
+
.map((s) => `${s.name} (${s.kb.toFixed(0)} KB)`)
|
|
297
|
+
.join(", ");
|
|
298
|
+
if (totalMB > 3) {
|
|
299
|
+
return {
|
|
300
|
+
area: "Top skills by size",
|
|
301
|
+
status: "warn",
|
|
302
|
+
message: `${skills.length} skills, ${totalMB.toFixed(1)} MB total. Top 5: ${top5}`,
|
|
303
|
+
action: "Review large skills with 'arcana list --installed'. Uninstall unused ones.",
|
|
304
|
+
};
|
|
305
|
+
}
|
|
306
|
+
if (totalMB > 1.5) {
|
|
307
|
+
return {
|
|
308
|
+
area: "Top skills by size",
|
|
309
|
+
status: "suggest",
|
|
310
|
+
message: `${skills.length} skills, ${totalMB.toFixed(1)} MB total. Top 5: ${top5}`,
|
|
311
|
+
action: "Consider removing rarely used skills to save context tokens",
|
|
312
|
+
};
|
|
313
|
+
}
|
|
314
|
+
return {
|
|
315
|
+
area: "Top skills by size",
|
|
316
|
+
status: "good",
|
|
317
|
+
message: `${skills.length} skills, ${totalMB.toFixed(1)} MB total. Top 5: ${top5}`,
|
|
318
|
+
};
|
|
319
|
+
}
|
|
320
|
+
export async function optimizeCommand(opts) {
|
|
321
|
+
if (!opts.json) {
|
|
322
|
+
banner();
|
|
323
|
+
console.log(ui.bold(" Claude Code Optimization Report\n"));
|
|
324
|
+
}
|
|
325
|
+
const recommendations = [
|
|
326
|
+
checkAutocompact(),
|
|
327
|
+
checkEffortLevel(),
|
|
328
|
+
checkNonEssentialCalls(),
|
|
329
|
+
checkPreCompactHook(),
|
|
330
|
+
checkSkillTokenBudget(),
|
|
331
|
+
checkMemorySize(),
|
|
332
|
+
checkAgentBloat(),
|
|
333
|
+
checkDiskHealth(),
|
|
334
|
+
checkLargestSkills(),
|
|
335
|
+
];
|
|
336
|
+
if (opts.json) {
|
|
337
|
+
console.log(JSON.stringify({ recommendations }, null, 2));
|
|
338
|
+
return;
|
|
339
|
+
}
|
|
340
|
+
for (const rec of recommendations) {
|
|
341
|
+
const icon = rec.status === "good" ? ui.success("[OK]") : rec.status === "suggest" ? ui.cyan("[>>]") : ui.warn("[!!]");
|
|
342
|
+
console.log(` ${icon} ${ui.bold(rec.area)}: ${rec.message}`);
|
|
343
|
+
if (rec.action) {
|
|
344
|
+
console.log(ui.dim(` ${rec.action}`));
|
|
345
|
+
}
|
|
346
|
+
}
|
|
347
|
+
const actionable = recommendations.filter((r) => r.status !== "good");
|
|
348
|
+
console.log();
|
|
349
|
+
if (actionable.length === 0) {
|
|
350
|
+
console.log(ui.success(" Your setup is well optimized."));
|
|
351
|
+
}
|
|
352
|
+
else {
|
|
353
|
+
console.log(ui.dim(` ${actionable.length} suggestion${actionable.length > 1 ? "s" : ""} to improve token usage and performance.`));
|
|
354
|
+
}
|
|
355
|
+
console.log();
|
|
356
|
+
}
|
|
@@ -0,0 +1,159 @@
|
|
|
1
|
+
import { existsSync, readdirSync, statSync } from "node:fs";
|
|
2
|
+
import { join } from "node:path";
|
|
3
|
+
import semver from "semver";
|
|
4
|
+
import { getInstallDir, readSkillMeta } from "../utils/fs.js";
|
|
5
|
+
import { getProvider, getProviders } from "../registry.js";
|
|
6
|
+
import { loadConfig } from "../utils/config.js";
|
|
7
|
+
function listInstalledSkills(installDir) {
|
|
8
|
+
if (!existsSync(installDir))
|
|
9
|
+
return [];
|
|
10
|
+
let entries;
|
|
11
|
+
try {
|
|
12
|
+
entries = readdirSync(installDir);
|
|
13
|
+
}
|
|
14
|
+
catch {
|
|
15
|
+
return [];
|
|
16
|
+
}
|
|
17
|
+
const results = [];
|
|
18
|
+
for (const entry of entries) {
|
|
19
|
+
const fullPath = join(installDir, entry);
|
|
20
|
+
try {
|
|
21
|
+
const stat = statSync(fullPath);
|
|
22
|
+
if (stat.isDirectory() && existsSync(join(fullPath, "SKILL.md"))) {
|
|
23
|
+
results.push(entry);
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
catch {
|
|
27
|
+
// skip unreadable
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
return results;
|
|
31
|
+
}
|
|
32
|
+
export async function outdatedCommand(opts) {
|
|
33
|
+
const installDir = getInstallDir();
|
|
34
|
+
const skills = listInstalledSkills(installDir);
|
|
35
|
+
if (skills.length === 0) {
|
|
36
|
+
if (opts.json) {
|
|
37
|
+
console.log(JSON.stringify({ outdated: [], upToDate: 0, total: 0 }));
|
|
38
|
+
}
|
|
39
|
+
else {
|
|
40
|
+
console.log("No skills installed.");
|
|
41
|
+
}
|
|
42
|
+
process.exit(0);
|
|
43
|
+
}
|
|
44
|
+
const providerName = opts.provider ?? loadConfig().defaultProvider;
|
|
45
|
+
const providers = opts.provider ? [getProvider(providerName)] : getProviders();
|
|
46
|
+
if (providers.length === 0) {
|
|
47
|
+
if (opts.json) {
|
|
48
|
+
console.log(JSON.stringify({
|
|
49
|
+
error: "No providers configured",
|
|
50
|
+
outdated: [],
|
|
51
|
+
upToDate: 0,
|
|
52
|
+
total: 0,
|
|
53
|
+
}));
|
|
54
|
+
}
|
|
55
|
+
else {
|
|
56
|
+
console.error("No providers configured. Run: arcana providers --add owner/repo");
|
|
57
|
+
}
|
|
58
|
+
process.exit(0);
|
|
59
|
+
}
|
|
60
|
+
const outdated = [];
|
|
61
|
+
let upToDate = 0;
|
|
62
|
+
let checked = 0;
|
|
63
|
+
for (const skillName of skills) {
|
|
64
|
+
const meta = readSkillMeta(skillName);
|
|
65
|
+
const localVersion = meta?.version ?? "0.0.0";
|
|
66
|
+
const preferredSource = meta?.source;
|
|
67
|
+
// Try the provider that installed this skill first, then fall back to others
|
|
68
|
+
const orderedProviders = [...providers];
|
|
69
|
+
if (preferredSource) {
|
|
70
|
+
const idx = orderedProviders.findIndex((p) => p.name === preferredSource);
|
|
71
|
+
if (idx > 0) {
|
|
72
|
+
const [pref] = orderedProviders.splice(idx, 1);
|
|
73
|
+
orderedProviders.unshift(pref);
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
let found = false;
|
|
77
|
+
for (const provider of orderedProviders) {
|
|
78
|
+
try {
|
|
79
|
+
const remoteInfo = await provider.info(skillName);
|
|
80
|
+
if (!remoteInfo)
|
|
81
|
+
continue;
|
|
82
|
+
const remoteVersion = remoteInfo.version;
|
|
83
|
+
const coercedRemote = semver.valid(semver.coerce(remoteVersion)) ?? "0.0.0";
|
|
84
|
+
const coercedLocal = semver.valid(semver.coerce(localVersion)) ?? "0.0.0";
|
|
85
|
+
if (semver.gt(coercedRemote, coercedLocal)) {
|
|
86
|
+
outdated.push({
|
|
87
|
+
name: skillName,
|
|
88
|
+
current: localVersion,
|
|
89
|
+
available: remoteVersion,
|
|
90
|
+
source: provider.name,
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
else {
|
|
94
|
+
upToDate++;
|
|
95
|
+
}
|
|
96
|
+
found = true;
|
|
97
|
+
checked++;
|
|
98
|
+
break;
|
|
99
|
+
}
|
|
100
|
+
catch {
|
|
101
|
+
// try next provider
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
if (!found) {
|
|
105
|
+
// Could not check this skill, count it as up-to-date for totals
|
|
106
|
+
upToDate++;
|
|
107
|
+
checked++;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
const result = {
|
|
111
|
+
outdated,
|
|
112
|
+
upToDate,
|
|
113
|
+
total: checked,
|
|
114
|
+
};
|
|
115
|
+
if (opts.json) {
|
|
116
|
+
console.log(JSON.stringify(result, null, 2));
|
|
117
|
+
process.exit(0);
|
|
118
|
+
}
|
|
119
|
+
// Console output: aligned table
|
|
120
|
+
console.log(`Checked ${result.total} installed skills.`);
|
|
121
|
+
console.log();
|
|
122
|
+
if (outdated.length === 0) {
|
|
123
|
+
console.log("All skills are up to date.");
|
|
124
|
+
process.exit(0);
|
|
125
|
+
}
|
|
126
|
+
// Calculate column widths
|
|
127
|
+
const nameWidth = Math.max(4, ...outdated.map((e) => e.name.length));
|
|
128
|
+
const currentWidth = Math.max(7, ...outdated.map((e) => e.current.length));
|
|
129
|
+
const availableWidth = Math.max(9, ...outdated.map((e) => e.available.length));
|
|
130
|
+
const sourceWidth = Math.max(6, ...outdated.map((e) => e.source.length));
|
|
131
|
+
const header = "Name".padEnd(nameWidth) +
|
|
132
|
+
" " +
|
|
133
|
+
"Current".padEnd(currentWidth) +
|
|
134
|
+
" " +
|
|
135
|
+
"Available".padEnd(availableWidth) +
|
|
136
|
+
" " +
|
|
137
|
+
"Source".padEnd(sourceWidth);
|
|
138
|
+
const separator = "-".repeat(nameWidth) +
|
|
139
|
+
" " +
|
|
140
|
+
"-".repeat(currentWidth) +
|
|
141
|
+
" " +
|
|
142
|
+
"-".repeat(availableWidth) +
|
|
143
|
+
" " +
|
|
144
|
+
"-".repeat(sourceWidth);
|
|
145
|
+
console.log(header);
|
|
146
|
+
console.log(separator);
|
|
147
|
+
for (const entry of outdated) {
|
|
148
|
+
console.log(entry.name.padEnd(nameWidth) +
|
|
149
|
+
" " +
|
|
150
|
+
entry.current.padEnd(currentWidth) +
|
|
151
|
+
" " +
|
|
152
|
+
entry.available.padEnd(availableWidth) +
|
|
153
|
+
" " +
|
|
154
|
+
entry.source.padEnd(sourceWidth));
|
|
155
|
+
}
|
|
156
|
+
console.log();
|
|
157
|
+
console.log(`${outdated.length} outdated, ${upToDate} up to date, ${result.total} total`);
|
|
158
|
+
process.exit(0);
|
|
159
|
+
}
|