meter-ai 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/README.md +124 -0
- package/dist/auth/credentials.d.ts +9 -0
- package/dist/auth/credentials.d.ts.map +1 -0
- package/dist/auth/credentials.js +32 -0
- package/dist/auth/credentials.js.map +1 -0
- package/dist/auth/detect.d.ts +7 -0
- package/dist/auth/detect.d.ts.map +1 -0
- package/dist/auth/detect.js +20 -0
- package/dist/auth/detect.js.map +1 -0
- package/dist/commands/config.d.ts +2 -0
- package/dist/commands/config.d.ts.map +1 -0
- package/dist/commands/config.js +26 -0
- package/dist/commands/config.js.map +1 -0
- package/dist/commands/history.d.ts +2 -0
- package/dist/commands/history.d.ts.map +1 -0
- package/dist/commands/history.js +19 -0
- package/dist/commands/history.js.map +1 -0
- package/dist/commands/init.d.ts +10 -0
- package/dist/commands/init.d.ts.map +1 -0
- package/dist/commands/init.js +58 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/report.d.ts +2 -0
- package/dist/commands/report.d.ts.map +1 -0
- package/dist/commands/report.js +37 -0
- package/dist/commands/report.js.map +1 -0
- package/dist/commands/status.d.ts +2 -0
- package/dist/commands/status.d.ts.map +1 -0
- package/dist/commands/status.js +20 -0
- package/dist/commands/status.js.map +1 -0
- package/dist/commands/uninstall.d.ts +2 -0
- package/dist/commands/uninstall.d.ts.map +1 -0
- package/dist/commands/uninstall.js +20 -0
- package/dist/commands/uninstall.js.map +1 -0
- package/dist/commands/wrap.d.ts +2 -0
- package/dist/commands/wrap.d.ts.map +1 -0
- package/dist/commands/wrap.js +189 -0
- package/dist/commands/wrap.js.map +1 -0
- package/dist/constants.d.ts +30 -0
- package/dist/constants.d.ts.map +1 -0
- package/dist/constants.js +48 -0
- package/dist/constants.js.map +1 -0
- package/dist/estimation/heuristics.d.ts +8 -0
- package/dist/estimation/heuristics.d.ts.map +1 -0
- package/dist/estimation/heuristics.js +28 -0
- package/dist/estimation/heuristics.js.map +1 -0
- package/dist/estimation/history-matcher.d.ts +4 -0
- package/dist/estimation/history-matcher.d.ts.map +1 -0
- package/dist/estimation/history-matcher.js +27 -0
- package/dist/estimation/history-matcher.js.map +1 -0
- package/dist/estimation/llm-precheck.d.ts +3 -0
- package/dist/estimation/llm-precheck.d.ts.map +1 -0
- package/dist/estimation/llm-precheck.js +21 -0
- package/dist/estimation/llm-precheck.js.map +1 -0
- package/dist/estimation/pipeline.d.ts +16 -0
- package/dist/estimation/pipeline.d.ts.map +1 -0
- package/dist/estimation/pipeline.js +48 -0
- package/dist/estimation/pipeline.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +45 -0
- package/dist/index.js.map +1 -0
- package/dist/pty/resize.d.ts +7 -0
- package/dist/pty/resize.d.ts.map +1 -0
- package/dist/pty/resize.js +10 -0
- package/dist/pty/resize.js.map +1 -0
- package/dist/pty/screen.d.ts +6 -0
- package/dist/pty/screen.d.ts.map +1 -0
- package/dist/pty/screen.js +15 -0
- package/dist/pty/screen.js.map +1 -0
- package/dist/pty/wrapper.d.ts +15 -0
- package/dist/pty/wrapper.d.ts.map +1 -0
- package/dist/pty/wrapper.js +49 -0
- package/dist/pty/wrapper.js.map +1 -0
- package/dist/shell/binary-resolver.d.ts +2 -0
- package/dist/shell/binary-resolver.d.ts.map +1 -0
- package/dist/shell/binary-resolver.js +18 -0
- package/dist/shell/binary-resolver.js.map +1 -0
- package/dist/shell/detect.d.ts +5 -0
- package/dist/shell/detect.d.ts.map +1 -0
- package/dist/shell/detect.js +32 -0
- package/dist/shell/detect.js.map +1 -0
- package/dist/shell/path-inject.d.ts +5 -0
- package/dist/shell/path-inject.d.ts.map +1 -0
- package/dist/shell/path-inject.js +30 -0
- package/dist/shell/path-inject.js.map +1 -0
- package/dist/shell/shim-writer.d.ts +3 -0
- package/dist/shell/shim-writer.d.ts.map +1 -0
- package/dist/shell/shim-writer.js +22 -0
- package/dist/shell/shim-writer.js.map +1 -0
- package/dist/storage/config-store.d.ts +10 -0
- package/dist/storage/config-store.d.ts.map +1 -0
- package/dist/storage/config-store.js +42 -0
- package/dist/storage/config-store.js.map +1 -0
- package/dist/storage/db.d.ts +7 -0
- package/dist/storage/db.d.ts.map +1 -0
- package/dist/storage/db.js +53 -0
- package/dist/storage/db.js.map +1 -0
- package/dist/tracking/cost.d.ts +6 -0
- package/dist/tracking/cost.d.ts.map +1 -0
- package/dist/tracking/cost.js +6 -0
- package/dist/tracking/cost.js.map +1 -0
- package/dist/tracking/plan-usage.d.ts +6 -0
- package/dist/tracking/plan-usage.d.ts.map +1 -0
- package/dist/tracking/plan-usage.js +53 -0
- package/dist/tracking/plan-usage.js.map +1 -0
- package/dist/tracking/tokens.d.ts +7 -0
- package/dist/tracking/tokens.d.ts.map +1 -0
- package/dist/tracking/tokens.js +14 -0
- package/dist/tracking/tokens.js.map +1 -0
- package/dist/types.d.ts +70 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/dist/ui/keypress.d.ts +3 -0
- package/dist/ui/keypress.d.ts.map +1 -0
- package/dist/ui/keypress.js +20 -0
- package/dist/ui/keypress.js.map +1 -0
- package/dist/ui/notification.d.ts +11 -0
- package/dist/ui/notification.d.ts.map +1 -0
- package/dist/ui/notification.js +18 -0
- package/dist/ui/notification.js.map +1 -0
- package/dist/ui/statusbar.d.ts +15 -0
- package/dist/ui/statusbar.d.ts.map +1 -0
- package/dist/ui/statusbar.js +53 -0
- package/dist/ui/statusbar.js.map +1 -0
- package/package.json +34 -0
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
const YELLOW = '\x1b[33m';
|
|
2
|
+
const BOLD = '\x1b[1m';
|
|
3
|
+
const RESET = '\x1b[0m';
|
|
4
|
+
const DIM = '\x1b[2m';
|
|
5
|
+
export function renderNotification(state) {
|
|
6
|
+
let message;
|
|
7
|
+
if (state.mode === 'api' && state.elapsedCost !== null && state.budgetUsd !== null) {
|
|
8
|
+
message = `${YELLOW}⚠ ${state.thresholdPct}% budget hit${RESET} ($${state.elapsedCost.toFixed(2)} / $${state.budgetUsd.toFixed(2)})`;
|
|
9
|
+
}
|
|
10
|
+
else {
|
|
11
|
+
message = `${YELLOW}⚠ ${state.windowPct}% window used${RESET}`;
|
|
12
|
+
}
|
|
13
|
+
const actions = state.nextModel
|
|
14
|
+
? ` ${BOLD}[s]${RESET}${DIM}witch to ${state.nextModel}${RESET} ${BOLD}[d]${RESET}${DIM}ismiss${RESET} ${BOLD}[c]${RESET}${DIM}ancel${RESET}`
|
|
15
|
+
: ` ${BOLD}[d]${RESET}${DIM}ismiss${RESET} ${BOLD}[c]${RESET}${DIM}ancel${RESET}`;
|
|
16
|
+
return message + actions;
|
|
17
|
+
}
|
|
18
|
+
//# sourceMappingURL=notification.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"notification.js","sourceRoot":"","sources":["../../src/ui/notification.ts"],"names":[],"mappings":"AAWA,MAAM,MAAM,GAAG,UAAU,CAAA;AACzB,MAAM,IAAI,GAAG,SAAS,CAAA;AACtB,MAAM,KAAK,GAAG,SAAS,CAAA;AACvB,MAAM,GAAG,GAAG,SAAS,CAAA;AAErB,MAAM,UAAU,kBAAkB,CAAC,KAAwB;IACzD,IAAI,OAAe,CAAA;IAEnB,IAAI,KAAK,CAAC,IAAI,KAAK,KAAK,IAAI,KAAK,CAAC,WAAW,KAAK,IAAI,IAAI,KAAK,CAAC,SAAS,KAAK,IAAI,EAAE,CAAC;QACnF,OAAO,GAAG,GAAG,MAAM,MAAM,KAAK,CAAC,YAAY,eAAe,KAAK,OAAO,KAAK,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,KAAK,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,CAAA;IACxI,CAAC;SAAM,CAAC;QACN,OAAO,GAAG,GAAG,MAAM,MAAM,KAAK,CAAC,SAAS,gBAAgB,KAAK,EAAE,CAAA;IACjE,CAAC;IAED,MAAM,OAAO,GAAG,KAAK,CAAC,SAAS;QAC7B,CAAC,CAAC,KAAK,IAAI,MAAM,KAAK,GAAG,GAAG,YAAY,KAAK,CAAC,SAAS,GAAG,KAAK,KAAK,IAAI,MAAM,KAAK,GAAG,GAAG,SAAS,KAAK,KAAK,IAAI,MAAM,KAAK,GAAG,GAAG,QAAQ,KAAK,EAAE;QAChJ,CAAC,CAAC,KAAK,IAAI,MAAM,KAAK,GAAG,GAAG,SAAS,KAAK,KAAK,IAAI,MAAM,KAAK,GAAG,GAAG,QAAQ,KAAK,EAAE,CAAA;IAErF,OAAO,OAAO,GAAG,OAAO,CAAA;AAC1B,CAAC"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import type { Complexity, UserMode } from '../types.js';
|
|
2
|
+
export interface StatusBarState {
|
|
3
|
+
model: string;
|
|
4
|
+
estimatedCost: number | null;
|
|
5
|
+
complexity: Complexity | null;
|
|
6
|
+
mode: UserMode;
|
|
7
|
+
elapsedCost: number | null;
|
|
8
|
+
budgetUsd: number | null;
|
|
9
|
+
windowPct: number | null;
|
|
10
|
+
windowResetIn: string | null;
|
|
11
|
+
}
|
|
12
|
+
export declare function renderStatusBar(state: StatusBarState): string;
|
|
13
|
+
export declare function injectStatusBar(content: string, linesAbove: number): void;
|
|
14
|
+
export declare function clearStatusBar(linesAbove: number): void;
|
|
15
|
+
//# sourceMappingURL=statusbar.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"statusbar.d.ts","sourceRoot":"","sources":["../../src/ui/statusbar.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAA;AAEvD,MAAM,WAAW,cAAc;IAC7B,KAAK,EAAE,MAAM,CAAA;IACb,aAAa,EAAE,MAAM,GAAG,IAAI,CAAA;IAC5B,UAAU,EAAE,UAAU,GAAG,IAAI,CAAA;IAC7B,IAAI,EAAE,QAAQ,CAAA;IACd,WAAW,EAAE,MAAM,GAAG,IAAI,CAAA;IAC1B,SAAS,EAAE,MAAM,GAAG,IAAI,CAAA;IACxB,SAAS,EAAE,MAAM,GAAG,IAAI,CAAA;IACxB,aAAa,EAAE,MAAM,GAAG,IAAI,CAAA;CAC7B;AAiBD,wBAAgB,eAAe,CAAC,KAAK,EAAE,cAAc,GAAG,MAAM,CAyB7D;AAED,wBAAgB,eAAe,CAAC,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,GAAG,IAAI,CAQzE;AAED,wBAAgB,cAAc,CAAC,UAAU,EAAE,MAAM,GAAG,IAAI,CAOvD"}
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import ansiEscapes from 'ansi-escapes';
|
|
2
|
+
const RESET = '\x1b[0m';
|
|
3
|
+
const DIM = '\x1b[2m';
|
|
4
|
+
const BOLD = '\x1b[1m';
|
|
5
|
+
const YELLOW = '\x1b[33m';
|
|
6
|
+
const RED = '\x1b[31m';
|
|
7
|
+
const GREEN = '\x1b[32m';
|
|
8
|
+
const CYAN = '\x1b[36m';
|
|
9
|
+
function progressBar(pct, width = 12) {
|
|
10
|
+
const filled = Math.round((pct / 100) * width);
|
|
11
|
+
const empty = width - filled;
|
|
12
|
+
const color = pct >= 80 ? RED : pct >= 60 ? YELLOW : GREEN;
|
|
13
|
+
return `${color}${'█'.repeat(filled)}${DIM}${'░'.repeat(empty)}${RESET}`;
|
|
14
|
+
}
|
|
15
|
+
export function renderStatusBar(state) {
|
|
16
|
+
const parts = [`${CYAN}◆ meter${RESET}`, `${BOLD}${state.model}${RESET}`];
|
|
17
|
+
if (state.estimatedCost !== null) {
|
|
18
|
+
parts.push(`${DIM}~$${state.estimatedCost.toFixed(2)}${RESET}`);
|
|
19
|
+
}
|
|
20
|
+
if (state.complexity) {
|
|
21
|
+
parts.push(`${DIM}${state.complexity}${RESET}`);
|
|
22
|
+
}
|
|
23
|
+
parts.push('│');
|
|
24
|
+
if (state.mode === 'api' && state.elapsedCost !== null && state.budgetUsd !== null) {
|
|
25
|
+
const pct = (state.elapsedCost / state.budgetUsd) * 100;
|
|
26
|
+
parts.push(progressBar(pct));
|
|
27
|
+
parts.push(`$${state.elapsedCost.toFixed(2)} / $${state.budgetUsd.toFixed(2)}`);
|
|
28
|
+
}
|
|
29
|
+
else if (state.mode === 'plan' && state.windowPct !== null) {
|
|
30
|
+
parts.push(progressBar(state.windowPct));
|
|
31
|
+
parts.push(`${state.windowPct}% 5hr window`);
|
|
32
|
+
if (state.windowResetIn)
|
|
33
|
+
parts.push(`${DIM}reset in ${state.windowResetIn}${RESET}`);
|
|
34
|
+
}
|
|
35
|
+
else {
|
|
36
|
+
parts.push(`${DIM}usage unavailable${RESET}`);
|
|
37
|
+
}
|
|
38
|
+
return parts.join(' ');
|
|
39
|
+
}
|
|
40
|
+
export function injectStatusBar(content, linesAbove) {
|
|
41
|
+
process.stdout.write(ansiEscapes.cursorSavePosition +
|
|
42
|
+
ansiEscapes.cursorUp(linesAbove) +
|
|
43
|
+
ansiEscapes.eraseLine +
|
|
44
|
+
content +
|
|
45
|
+
ansiEscapes.cursorRestorePosition);
|
|
46
|
+
}
|
|
47
|
+
export function clearStatusBar(linesAbove) {
|
|
48
|
+
process.stdout.write(ansiEscapes.cursorSavePosition +
|
|
49
|
+
ansiEscapes.cursorUp(linesAbove) +
|
|
50
|
+
ansiEscapes.eraseLine +
|
|
51
|
+
ansiEscapes.cursorRestorePosition);
|
|
52
|
+
}
|
|
53
|
+
//# sourceMappingURL=statusbar.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"statusbar.js","sourceRoot":"","sources":["../../src/ui/statusbar.ts"],"names":[],"mappings":"AAAA,OAAO,WAAW,MAAM,cAAc,CAAA;AActC,MAAM,KAAK,GAAG,SAAS,CAAA;AACvB,MAAM,GAAG,GAAG,SAAS,CAAA;AACrB,MAAM,IAAI,GAAG,SAAS,CAAA;AACtB,MAAM,MAAM,GAAG,UAAU,CAAA;AACzB,MAAM,GAAG,GAAG,UAAU,CAAA;AACtB,MAAM,KAAK,GAAG,UAAU,CAAA;AACxB,MAAM,IAAI,GAAG,UAAU,CAAA;AAEvB,SAAS,WAAW,CAAC,GAAW,EAAE,KAAK,GAAG,EAAE;IAC1C,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,GAAG,GAAG,CAAC,GAAG,KAAK,CAAC,CAAA;IAC9C,MAAM,KAAK,GAAG,KAAK,GAAG,MAAM,CAAA;IAC5B,MAAM,KAAK,GAAG,GAAG,IAAI,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,IAAI,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAA;IAC1D,OAAO,GAAG,KAAK,GAAG,GAAG,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,GAAG,GAAG,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,KAAK,EAAE,CAAA;AAC1E,CAAC;AAED,MAAM,UAAU,eAAe,CAAC,KAAqB;IACnD,MAAM,KAAK,GAAa,CAAC,GAAG,IAAI,UAAU,KAAK,EAAE,EAAE,GAAG,IAAI,GAAG,KAAK,CAAC,KAAK,GAAG,KAAK,EAAE,CAAC,CAAA;IAEnF,IAAI,KAAK,CAAC,aAAa,KAAK,IAAI,EAAE,CAAC;QACjC,KAAK,CAAC,IAAI,CAAC,GAAG,GAAG,KAAK,KAAK,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,KAAK,EAAE,CAAC,CAAA;IACjE,CAAC;IACD,IAAI,KAAK,CAAC,UAAU,EAAE,CAAC;QACrB,KAAK,CAAC,IAAI,CAAC,GAAG,GAAG,GAAG,KAAK,CAAC,UAAU,GAAG,KAAK,EAAE,CAAC,CAAA;IACjD,CAAC;IAED,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;IAEf,IAAI,KAAK,CAAC,IAAI,KAAK,KAAK,IAAI,KAAK,CAAC,WAAW,KAAK,IAAI,IAAI,KAAK,CAAC,SAAS,KAAK,IAAI,EAAE,CAAC;QACnF,MAAM,GAAG,GAAG,CAAC,KAAK,CAAC,WAAW,GAAG,KAAK,CAAC,SAAS,CAAC,GAAG,GAAG,CAAA;QACvD,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,CAAA;QAC5B,KAAK,CAAC,IAAI,CAAC,IAAI,KAAK,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,KAAK,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAA;IACjF,CAAC;SAAM,IAAI,KAAK,CAAC,IAAI,KAAK,MAAM,IAAI,KAAK,CAAC,SAAS,KAAK,IAAI,EAAE,CAAC;QAC7D,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAA;QACxC,KAAK,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,SAAS,cAAc,CAAC,CAAA;QAC5C,IAAI,KAAK,CAAC,aAAa;YAAE,KAAK,CAAC,IAAI,CAAC,GAAG,GAAG,YAAY,KAAK,CAAC,aAAa,GAAG,KAAK,EAAE,CAAC,CAAA;IACtF,CAAC;SAAM,CAAC;QACN,KAAK,CAAC,IAAI,CAAC,GAAG,GAAG,oBAAoB,KAAK,EAAE,CAAC,CAAA;IAC/C,CAAC;IAED,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;AACzB,CAAC;AAED,MAAM,UAAU,eAAe,CAAC,OAAe,EAAE,UAAkB;IACjE,OAAO,CAAC,MAAM,CAAC,KAAK,CAClB,WAAW,CAAC,kBAAkB;QAC9B,WAAW,CAAC,QAAQ,CAAC,UAAU,CAAC;QAChC,WAAW,CAAC,SAAS;QACrB,OAAO;QACP,WAAW,CAAC,qBAAqB,CAClC,CAAA;AACH,CAAC;AAED,MAAM,UAAU,cAAc,CAAC,UAAkB;IAC/C,OAAO,CAAC,MAAM,CAAC,KAAK,CAClB,WAAW,CAAC,kBAAkB;QAC9B,WAAW,CAAC,QAAQ,CAAC,UAAU,CAAC;QAChC,WAAW,CAAC,SAAS;QACrB,WAAW,CAAC,qBAAqB,CAClC,CAAA;AACH,CAAC"}
|
package/package.json
ADDED
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "meter-ai",
|
|
3
|
+
"version": "0.1.0",
|
|
4
|
+
"description": "Intelligent CLI wrapper for Claude Code — pre-task cost estimation, live status bar, budget protection",
|
|
5
|
+
"bin": {
|
|
6
|
+
"meter": "./dist/index.js"
|
|
7
|
+
},
|
|
8
|
+
"main": "./dist/index.js",
|
|
9
|
+
"type": "module",
|
|
10
|
+
"engines": { "node": ">=20.0.0" },
|
|
11
|
+
"scripts": {
|
|
12
|
+
"build": "tsc -p tsconfig.build.json && node -e \"const fs=require('fs');const f='dist/index.js';const c=fs.readFileSync(f,'utf8');if(!c.startsWith('#!/usr/bin/env node')){fs.writeFileSync(f,'#!/usr/bin/env node\\n'+c)}\"",
|
|
13
|
+
"dev": "tsx src/index.ts",
|
|
14
|
+
"test": "vitest run",
|
|
15
|
+
"test:watch": "vitest",
|
|
16
|
+
"typecheck": "tsc --noEmit",
|
|
17
|
+
"prepublishOnly": "npm run build && npm test"
|
|
18
|
+
},
|
|
19
|
+
"keywords": ["claude", "ai", "cost", "cli", "agent"],
|
|
20
|
+
"license": "MIT",
|
|
21
|
+
"dependencies": {
|
|
22
|
+
"@anthropic-ai/sdk": "^0.39.0",
|
|
23
|
+
"ansi-escapes": "^7.0.0",
|
|
24
|
+
"better-sqlite3": "^11.0.0",
|
|
25
|
+
"node-pty": "^1.0.0"
|
|
26
|
+
},
|
|
27
|
+
"devDependencies": {
|
|
28
|
+
"@types/better-sqlite3": "^7.6.0",
|
|
29
|
+
"@types/node": "^20.0.0",
|
|
30
|
+
"tsx": "^4.0.0",
|
|
31
|
+
"typescript": "^5.4.0",
|
|
32
|
+
"vitest": "^2.0.0"
|
|
33
|
+
}
|
|
34
|
+
}
|