@code-insights/cli 1.2.0 → 2.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/CHANGELOG.md +26 -0
- package/README.md +101 -9
- package/dist/commands/config.d.ts +3 -0
- package/dist/commands/config.d.ts.map +1 -0
- package/dist/commands/config.js +130 -0
- package/dist/commands/config.js.map +1 -0
- package/dist/commands/connect.d.ts.map +1 -1
- package/dist/commands/connect.js +7 -1
- package/dist/commands/connect.js.map +1 -1
- package/dist/commands/init.d.ts.map +1 -1
- package/dist/commands/init.js +52 -0
- package/dist/commands/init.js.map +1 -1
- package/dist/commands/install-hook.d.ts.map +1 -1
- package/dist/commands/install-hook.js +16 -3
- package/dist/commands/install-hook.js.map +1 -1
- package/dist/commands/open.d.ts +9 -0
- package/dist/commands/open.d.ts.map +1 -0
- package/dist/commands/open.js +59 -0
- package/dist/commands/open.js.map +1 -0
- package/dist/commands/reset.d.ts.map +1 -1
- package/dist/commands/reset.js +14 -1
- package/dist/commands/reset.js.map +1 -1
- package/dist/commands/stats/actions/cost.d.ts +3 -0
- package/dist/commands/stats/actions/cost.d.ts.map +1 -0
- package/dist/commands/stats/actions/cost.js +139 -0
- package/dist/commands/stats/actions/cost.js.map +1 -0
- package/dist/commands/stats/actions/error-handler.d.ts +6 -0
- package/dist/commands/stats/actions/error-handler.d.ts.map +1 -0
- package/dist/commands/stats/actions/error-handler.js +42 -0
- package/dist/commands/stats/actions/error-handler.js.map +1 -0
- package/dist/commands/stats/actions/models.d.ts +3 -0
- package/dist/commands/stats/actions/models.d.ts.map +1 -0
- package/dist/commands/stats/actions/models.js +101 -0
- package/dist/commands/stats/actions/models.js.map +1 -0
- package/dist/commands/stats/actions/overview.d.ts +3 -0
- package/dist/commands/stats/actions/overview.d.ts.map +1 -0
- package/dist/commands/stats/actions/overview.js +147 -0
- package/dist/commands/stats/actions/overview.js.map +1 -0
- package/dist/commands/stats/actions/projects.d.ts +3 -0
- package/dist/commands/stats/actions/projects.d.ts.map +1 -0
- package/dist/commands/stats/actions/projects.js +90 -0
- package/dist/commands/stats/actions/projects.js.map +1 -0
- package/dist/commands/stats/actions/today.d.ts +3 -0
- package/dist/commands/stats/actions/today.d.ts.map +1 -0
- package/dist/commands/stats/actions/today.js +118 -0
- package/dist/commands/stats/actions/today.js.map +1 -0
- package/dist/commands/stats/data/aggregation.d.ts +60 -0
- package/dist/commands/stats/data/aggregation.d.ts.map +1 -0
- package/dist/commands/stats/data/aggregation.js +614 -0
- package/dist/commands/stats/data/aggregation.js.map +1 -0
- package/dist/commands/stats/data/cache.d.ts +29 -0
- package/dist/commands/stats/data/cache.d.ts.map +1 -0
- package/dist/commands/stats/data/cache.js +197 -0
- package/dist/commands/stats/data/cache.js.map +1 -0
- package/dist/commands/stats/data/firestore.d.ts +13 -0
- package/dist/commands/stats/data/firestore.d.ts.map +1 -0
- package/dist/commands/stats/data/firestore.js +155 -0
- package/dist/commands/stats/data/firestore.js.map +1 -0
- package/dist/commands/stats/data/fuzzy-match.d.ts +5 -0
- package/dist/commands/stats/data/fuzzy-match.d.ts.map +1 -0
- package/dist/commands/stats/data/fuzzy-match.js +36 -0
- package/dist/commands/stats/data/fuzzy-match.js.map +1 -0
- package/dist/commands/stats/data/local.d.ts +12 -0
- package/dist/commands/stats/data/local.d.ts.map +1 -0
- package/dist/commands/stats/data/local.js +77 -0
- package/dist/commands/stats/data/local.js.map +1 -0
- package/dist/commands/stats/data/source.d.ts +13 -0
- package/dist/commands/stats/data/source.d.ts.map +1 -0
- package/dist/commands/stats/data/source.js +41 -0
- package/dist/commands/stats/data/source.js.map +1 -0
- package/dist/commands/stats/data/types.d.ts +229 -0
- package/dist/commands/stats/data/types.d.ts.map +1 -0
- package/dist/commands/stats/data/types.js +50 -0
- package/dist/commands/stats/data/types.js.map +1 -0
- package/dist/commands/stats/index.d.ts +3 -0
- package/dist/commands/stats/index.d.ts.map +1 -0
- package/dist/commands/stats/index.js +41 -0
- package/dist/commands/stats/index.js.map +1 -0
- package/dist/commands/stats/render/charts.d.ts +9 -0
- package/dist/commands/stats/render/charts.d.ts.map +1 -0
- package/dist/commands/stats/render/charts.js +45 -0
- package/dist/commands/stats/render/charts.js.map +1 -0
- package/dist/commands/stats/render/colors.d.ts +21 -0
- package/dist/commands/stats/render/colors.d.ts.map +1 -0
- package/dist/commands/stats/render/colors.js +47 -0
- package/dist/commands/stats/render/colors.js.map +1 -0
- package/dist/commands/stats/render/format.d.ts +10 -0
- package/dist/commands/stats/render/format.d.ts.map +1 -0
- package/dist/commands/stats/render/format.js +57 -0
- package/dist/commands/stats/render/format.js.map +1 -0
- package/dist/commands/stats/render/layout.d.ts +10 -0
- package/dist/commands/stats/render/layout.d.ts.map +1 -0
- package/dist/commands/stats/render/layout.js +48 -0
- package/dist/commands/stats/render/layout.js.map +1 -0
- package/dist/commands/stats/shared.d.ts +6 -0
- package/dist/commands/stats/shared.d.ts.map +1 -0
- package/dist/commands/stats/shared.js +26 -0
- package/dist/commands/stats/shared.js.map +1 -0
- package/dist/commands/status.d.ts.map +1 -1
- package/dist/commands/status.js +50 -37
- package/dist/commands/status.js.map +1 -1
- package/dist/commands/sync.d.ts +16 -1
- package/dist/commands/sync.d.ts.map +1 -1
- package/dist/commands/sync.js +173 -80
- package/dist/commands/sync.js.map +1 -1
- package/dist/commands/telemetry.d.ts +3 -0
- package/dist/commands/telemetry.d.ts.map +1 -0
- package/dist/commands/telemetry.js +106 -0
- package/dist/commands/telemetry.js.map +1 -0
- package/dist/firebase/client.d.ts +5 -0
- package/dist/firebase/client.d.ts.map +1 -1
- package/dist/firebase/client.js +5 -2
- package/dist/firebase/client.js.map +1 -1
- package/dist/index.js +21 -4
- package/dist/index.js.map +1 -1
- package/dist/parser/jsonl.js +3 -2
- package/dist/parser/jsonl.js.map +1 -1
- package/dist/providers/claude-code.d.ts.map +1 -1
- package/dist/providers/claude-code.js +5 -1
- package/dist/providers/claude-code.js.map +1 -1
- package/dist/providers/codex.d.ts +14 -0
- package/dist/providers/codex.d.ts.map +1 -0
- package/dist/providers/codex.js +364 -0
- package/dist/providers/codex.js.map +1 -0
- package/dist/providers/copilot-cli.d.ts +14 -0
- package/dist/providers/copilot-cli.d.ts.map +1 -0
- package/dist/providers/copilot-cli.js +363 -0
- package/dist/providers/copilot-cli.js.map +1 -0
- package/dist/providers/cursor.d.ts +27 -0
- package/dist/providers/cursor.d.ts.map +1 -0
- package/dist/providers/cursor.js +356 -0
- package/dist/providers/cursor.js.map +1 -0
- package/dist/providers/registry.d.ts +0 -4
- package/dist/providers/registry.d.ts.map +1 -1
- package/dist/providers/registry.js +9 -6
- package/dist/providers/registry.js.map +1 -1
- package/dist/types.d.ts +5 -1
- package/dist/types.d.ts.map +1 -1
- package/dist/utils/config.d.ts +10 -1
- package/dist/utils/config.d.ts.map +1 -1
- package/dist/utils/config.js +21 -0
- package/dist/utils/config.js.map +1 -1
- package/dist/utils/paths.d.ts +10 -0
- package/dist/utils/paths.d.ts.map +1 -0
- package/dist/utils/paths.js +16 -0
- package/dist/utils/paths.js.map +1 -0
- package/dist/utils/telemetry.d.ts +52 -0
- package/dist/utils/telemetry.d.ts.map +1 -0
- package/dist/utils/telemetry.js +304 -0
- package/dist/utils/telemetry.js.map +1 -0
- package/dist/utils/tips.d.ts +11 -0
- package/dist/utils/tips.d.ts.map +1 -0
- package/dist/utils/tips.js +106 -0
- package/dist/utils/tips.js.map +1 -0
- package/dist/utils/welcome.d.ts +11 -0
- package/dist/utils/welcome.d.ts.map +1 -0
- package/dist/utils/welcome.js +101 -0
- package/dist/utils/welcome.js.map +1 -0
- package/package.json +9 -2
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import { execFile } from 'child_process';
|
|
2
|
+
import path from 'path';
|
|
3
|
+
import chalk from 'chalk';
|
|
4
|
+
import { loadConfig } from '../utils/config.js';
|
|
5
|
+
import { trackEvent } from '../utils/telemetry.js';
|
|
6
|
+
const DEFAULT_DASHBOARD_URL = 'https://code-insights.app';
|
|
7
|
+
/**
|
|
8
|
+
* Open the web dashboard in the default browser.
|
|
9
|
+
*/
|
|
10
|
+
export async function openCommand(options) {
|
|
11
|
+
const config = loadConfig();
|
|
12
|
+
const baseUrl = config?.dashboardUrl || DEFAULT_DASHBOARD_URL;
|
|
13
|
+
let url = baseUrl;
|
|
14
|
+
// If --project flag, try to detect current project name from cwd
|
|
15
|
+
if (options.project) {
|
|
16
|
+
const projectName = getCurrentProjectName();
|
|
17
|
+
if (projectName) {
|
|
18
|
+
url = `${baseUrl}/sessions?project=${encodeURIComponent(projectName)}`;
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
console.log(chalk.cyan(`\n Opening ${url}\n`));
|
|
22
|
+
try {
|
|
23
|
+
openInBrowser(url);
|
|
24
|
+
trackEvent('open', true);
|
|
25
|
+
}
|
|
26
|
+
catch {
|
|
27
|
+
console.log(chalk.yellow(' Could not open browser automatically.'));
|
|
28
|
+
console.log(chalk.white(` Visit: ${chalk.bold.underline(url)}\n`));
|
|
29
|
+
trackEvent('open', false);
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Open a URL in the default browser using platform-specific commands.
|
|
34
|
+
* Uses execFile (not exec) to prevent shell injection.
|
|
35
|
+
*/
|
|
36
|
+
function openInBrowser(url) {
|
|
37
|
+
const platform = process.platform;
|
|
38
|
+
if (platform === 'darwin') {
|
|
39
|
+
execFile('open', [url]);
|
|
40
|
+
}
|
|
41
|
+
else if (platform === 'win32') {
|
|
42
|
+
execFile('cmd', ['/c', 'start', '', url]);
|
|
43
|
+
}
|
|
44
|
+
else {
|
|
45
|
+
execFile('xdg-open', [url]);
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* Get the current directory name as a project name guess.
|
|
50
|
+
*/
|
|
51
|
+
function getCurrentProjectName() {
|
|
52
|
+
try {
|
|
53
|
+
return path.basename(process.cwd()) || null;
|
|
54
|
+
}
|
|
55
|
+
catch {
|
|
56
|
+
return null;
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
//# sourceMappingURL=open.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"open.js","sourceRoot":"","sources":["../../src/commands/open.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AACzC,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAChD,OAAO,EAAE,UAAU,EAAE,MAAM,uBAAuB,CAAC;AAEnD,MAAM,qBAAqB,GAAG,2BAA2B,CAAC;AAM1D;;GAEG;AACH,MAAM,CAAC,KAAK,UAAU,WAAW,CAAC,OAAoB;IACpD,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,OAAO,GAAG,MAAM,EAAE,YAAY,IAAI,qBAAqB,CAAC;IAE9D,IAAI,GAAG,GAAG,OAAO,CAAC;IAElB,iEAAiE;IACjE,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;QACpB,MAAM,WAAW,GAAG,qBAAqB,EAAE,CAAC;QAC5C,IAAI,WAAW,EAAE,CAAC;YAChB,GAAG,GAAG,GAAG,OAAO,qBAAqB,kBAAkB,CAAC,WAAW,CAAC,EAAE,CAAC;QACzE,CAAC;IACH,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,CAAC,CAAC;IAEhD,IAAI,CAAC;QACH,aAAa,CAAC,GAAG,CAAC,CAAC;QACnB,UAAU,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;IAC3B,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,yCAAyC,CAAC,CAAC,CAAC;QACrE,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,YAAY,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC;QACpE,UAAU,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;IAC5B,CAAC;AACH,CAAC;AAED;;;GAGG;AACH,SAAS,aAAa,CAAC,GAAW;IAChC,MAAM,QAAQ,GAAG,OAAO,CAAC,QAAQ,CAAC;IAClC,IAAI,QAAQ,KAAK,QAAQ,EAAE,CAAC;QAC1B,QAAQ,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IAC1B,CAAC;SAAM,IAAI,QAAQ,KAAK,OAAO,EAAE,CAAC;QAChC,QAAQ,CAAC,KAAK,EAAE,CAAC,IAAI,EAAE,OAAO,EAAE,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC;IAC5C,CAAC;SAAM,CAAC;QACN,QAAQ,CAAC,UAAU,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IAC9B,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,qBAAqB;IAC5B,IAAI,CAAC;QACH,OAAO,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,IAAI,IAAI,CAAC;IAC9C,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"reset.d.ts","sourceRoot":"","sources":["../../src/commands/reset.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;
|
|
1
|
+
{"version":3,"file":"reset.d.ts","sourceRoot":"","sources":["../../src/commands/reset.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAYpC,eAAO,MAAM,YAAY,SAyFrB,CAAC"}
|
package/dist/commands/reset.js
CHANGED
|
@@ -5,12 +5,20 @@ import admin from 'firebase-admin';
|
|
|
5
5
|
import { existsSync, unlinkSync } from 'fs';
|
|
6
6
|
import { join } from 'path';
|
|
7
7
|
import { homedir } from 'os';
|
|
8
|
-
import { loadConfig } from '../utils/config.js';
|
|
8
|
+
import { loadConfig, resolveDataSourcePreference } from '../utils/config.js';
|
|
9
|
+
import { trackEvent } from '../utils/telemetry.js';
|
|
9
10
|
const SYNC_STATE_FILE = join(homedir(), '.code-insights', 'sync-state.json');
|
|
10
11
|
export const resetCommand = new Command('reset')
|
|
11
12
|
.description('Delete all data from Firestore and reset local sync state')
|
|
12
13
|
.option('--confirm', 'Skip confirmation prompt')
|
|
13
14
|
.action(async (options) => {
|
|
15
|
+
const preference = resolveDataSourcePreference();
|
|
16
|
+
if (preference === 'local') {
|
|
17
|
+
console.log(chalk.yellow('\n ⚠ Data source is local. Nothing to reset in Firestore.\n'));
|
|
18
|
+
console.log(chalk.gray(' To clear the local stats cache:'));
|
|
19
|
+
console.log(chalk.gray(' rm ~/.code-insights/stats-cache.json\n'));
|
|
20
|
+
process.exit(0);
|
|
21
|
+
}
|
|
14
22
|
console.log(chalk.red.bold('\n⚠️ WARNING: This will permanently delete ALL data from your Firestore database!'));
|
|
15
23
|
console.log(chalk.yellow('Collections to be deleted: projects, sessions, insights, messages\n'));
|
|
16
24
|
if (!options.confirm) {
|
|
@@ -35,6 +43,10 @@ export const resetCommand = new Command('reset')
|
|
|
35
43
|
console.error(chalk.red('Error: Not configured. Run `code-insights init` first.'));
|
|
36
44
|
process.exit(1);
|
|
37
45
|
}
|
|
46
|
+
if (!config.firebase) {
|
|
47
|
+
console.error(chalk.red('Firebase not configured. Nothing to reset.'));
|
|
48
|
+
process.exit(1);
|
|
49
|
+
}
|
|
38
50
|
// Initialize Firebase
|
|
39
51
|
if (admin.apps.length === 0) {
|
|
40
52
|
admin.initializeApp({
|
|
@@ -73,6 +85,7 @@ export const resetCommand = new Command('reset')
|
|
|
73
85
|
syncSpinner.fail(`Failed to remove sync state: ${error}`);
|
|
74
86
|
}
|
|
75
87
|
console.log(chalk.green('\n✓ Reset complete. Run `code-insights sync` to re-sync all sessions.\n'));
|
|
88
|
+
trackEvent('reset', true);
|
|
76
89
|
process.exit(0);
|
|
77
90
|
});
|
|
78
91
|
async function deleteCollection(db, collectionName) {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"reset.js","sourceRoot":"","sources":["../../src/commands/reset.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,GAAG,MAAM,KAAK,CAAC;AACtB,OAAO,KAAK,MAAM,gBAAgB,CAAC;AACnC,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,IAAI,CAAC;AAC5C,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,EAAE,OAAO,EAAE,MAAM,IAAI,CAAC;AAC7B,OAAO,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;
|
|
1
|
+
{"version":3,"file":"reset.js","sourceRoot":"","sources":["../../src/commands/reset.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cost.d.ts","sourceRoot":"","sources":["../../../../src/commands/stats/actions/cost.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAE,UAAU,EAAuB,MAAM,kBAAkB,CAAC;AAcxE,wBAAsB,UAAU,CAAC,KAAK,EAAE,UAAU,GAAG,OAAO,CAAC,IAAI,CAAC,CA6IjE"}
|
|
@@ -0,0 +1,139 @@
|
|
|
1
|
+
// ──────────────────────────────────────────────────────
|
|
2
|
+
// stats cost — Cost breakdown by project, model, tokens
|
|
3
|
+
// ──────────────────────────────────────────────────────
|
|
4
|
+
import ora from 'ora';
|
|
5
|
+
import { trackEvent } from '../../../utils/telemetry.js';
|
|
6
|
+
import { resolveDataSource } from '../data/source.js';
|
|
7
|
+
import { periodStartDate, computeCostBreakdown } from '../data/aggregation.js';
|
|
8
|
+
import { colors } from '../render/colors.js';
|
|
9
|
+
import { handleStatsError } from './error-handler.js';
|
|
10
|
+
import { formatMoney, formatTokens, formatPercent, formatPeriodLabel, } from '../render/format.js';
|
|
11
|
+
import { sparkline, sparklineLabels } from '../render/charts.js';
|
|
12
|
+
import { barChart } from '../render/charts.js';
|
|
13
|
+
import { sectionHeader, metricGrid, getBarWidth } from '../render/layout.js';
|
|
14
|
+
import { showTip } from '../../../utils/tips.js';
|
|
15
|
+
export async function costAction(flags) {
|
|
16
|
+
try {
|
|
17
|
+
const source = await resolveDataSource(flags);
|
|
18
|
+
const spinner = ora({ text: 'Syncing...', indent: 2 }).start();
|
|
19
|
+
try {
|
|
20
|
+
const prepResult = await source.prepare(flags);
|
|
21
|
+
spinner.succeed(prepResult.message);
|
|
22
|
+
}
|
|
23
|
+
catch {
|
|
24
|
+
spinner.warn('Sync failed (showing cached data)');
|
|
25
|
+
}
|
|
26
|
+
// Resolve project filter
|
|
27
|
+
const opts = {
|
|
28
|
+
periodStart: periodStartDate(flags.period),
|
|
29
|
+
sourceTool: flags.source,
|
|
30
|
+
};
|
|
31
|
+
if (flags.project) {
|
|
32
|
+
const resolved = await source.resolveProjectId(flags.project);
|
|
33
|
+
opts.projectId = resolved.projectId;
|
|
34
|
+
}
|
|
35
|
+
const sessions = await source.getSessions(opts);
|
|
36
|
+
// Empty state
|
|
37
|
+
if (sessions.length === 0) {
|
|
38
|
+
const periodLabel = formatPeriodLabel(flags.period);
|
|
39
|
+
console.log(sectionHeader('COST BREAKDOWN', periodLabel));
|
|
40
|
+
console.log(`\n No sessions found in the ${periodLabel.toLowerCase()}.\n`);
|
|
41
|
+
console.log(colors.hint('Run stats --period 30d to expand the time range'));
|
|
42
|
+
console.log();
|
|
43
|
+
return;
|
|
44
|
+
}
|
|
45
|
+
const cost = computeCostBreakdown(sessions, flags.period);
|
|
46
|
+
const periodLabel = formatPeriodLabel(flags.period);
|
|
47
|
+
// No cost data at all
|
|
48
|
+
if (cost.sessionsWithCostCount === 0) {
|
|
49
|
+
console.log(sectionHeader('COST BREAKDOWN', periodLabel));
|
|
50
|
+
console.log(`\n No cost data available.\n`);
|
|
51
|
+
console.log(` Cost tracking is supported by:`);
|
|
52
|
+
console.log(` ${colors.success('\u25CF')} Claude Code (automatic)`);
|
|
53
|
+
console.log(` ${colors.success('\u25CF')} Cursor (via usage export)`);
|
|
54
|
+
console.log();
|
|
55
|
+
console.log(colors.hint('Run code-insights sync to refresh data'));
|
|
56
|
+
console.log();
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
// Header
|
|
60
|
+
console.log(sectionHeader('COST BREAKDOWN', periodLabel));
|
|
61
|
+
// Cost data coverage warning
|
|
62
|
+
if (cost.sessionsWithCostCount < cost.sessionCount * 0.5) {
|
|
63
|
+
console.log(`\n ${colors.warning(`\u26A0 ${cost.sessionCount - cost.sessionsWithCostCount} sessions have no cost data`)}`);
|
|
64
|
+
}
|
|
65
|
+
// Metric grid
|
|
66
|
+
console.log();
|
|
67
|
+
console.log(metricGrid([
|
|
68
|
+
{ label: 'Total', value: formatMoney(cost.totalCost) },
|
|
69
|
+
{ label: 'Avg/day', value: formatMoney(cost.avgPerDay) },
|
|
70
|
+
{ label: 'Avg/session', value: formatMoney(cost.avgPerSession) },
|
|
71
|
+
{ label: 'Sessions', value: `${cost.sessionCount} (${cost.sessionsWithCostCount} with cost data)` },
|
|
72
|
+
]));
|
|
73
|
+
// Daily trend sparkline
|
|
74
|
+
const spark = sparkline(cost.dailyTrend.map(d => d.value));
|
|
75
|
+
const labels = sparklineLabels(flags.period);
|
|
76
|
+
console.log(sectionHeader('DAILY TREND', spark));
|
|
77
|
+
if (labels) {
|
|
78
|
+
console.log(` ${colors.label(labels)}`);
|
|
79
|
+
}
|
|
80
|
+
if (cost.peakDay) {
|
|
81
|
+
console.log(` ${colors.label('Peak:')} ${colors.value(cost.peakDay.date)} ${colors.money(cost.peakDay.cost)} ${colors.label(`(${cost.peakDay.sessions} sessions)`)}`);
|
|
82
|
+
}
|
|
83
|
+
// By project
|
|
84
|
+
if (cost.byProject.length > 0) {
|
|
85
|
+
console.log(sectionHeader('BY PROJECT'));
|
|
86
|
+
const barWidth = getBarWidth();
|
|
87
|
+
const lines = barChart(cost.byProject.map(p => ({
|
|
88
|
+
label: p.name,
|
|
89
|
+
value: p.cost,
|
|
90
|
+
suffix: `${colors.money(p.cost)} ${formatPercent(p.percent)} ${colors.label(`${p.count} sessions`)}`,
|
|
91
|
+
})), barWidth);
|
|
92
|
+
for (const line of lines) {
|
|
93
|
+
console.log(line);
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
// By model
|
|
97
|
+
if (cost.byModel.length > 0) {
|
|
98
|
+
console.log(sectionHeader('BY MODEL'));
|
|
99
|
+
const barWidth = getBarWidth();
|
|
100
|
+
const lines = barChart(cost.byModel.map(m => ({
|
|
101
|
+
label: m.name,
|
|
102
|
+
value: m.cost,
|
|
103
|
+
suffix: `${colors.money(m.cost)} ${formatPercent(m.percent)} ${colors.label(`${m.count} sessions`)}`,
|
|
104
|
+
})), barWidth);
|
|
105
|
+
for (const line of lines) {
|
|
106
|
+
console.log(line);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
// Token breakdown
|
|
110
|
+
const tb = cost.tokenBreakdown;
|
|
111
|
+
if (tb.inputTokens + tb.outputTokens > 0) {
|
|
112
|
+
console.log(sectionHeader('TOKEN BREAKDOWN'));
|
|
113
|
+
const barWidth = getBarWidth();
|
|
114
|
+
const tokenItems = [
|
|
115
|
+
{ label: 'Input tokens', value: tb.inputCost, suffix: `${formatTokens(tb.inputTokens)} ${colors.money(tb.inputCost)}` },
|
|
116
|
+
{ label: 'Output tokens', value: tb.outputCost, suffix: `${formatTokens(tb.outputTokens)} ${colors.money(tb.outputCost)}` },
|
|
117
|
+
{ label: 'Cache creation', value: tb.cacheCreationCost, suffix: `${formatTokens(tb.cacheCreation)} ${colors.money(tb.cacheCreationCost)}` },
|
|
118
|
+
{ label: 'Cache reads', value: tb.cacheReadCost, suffix: `${formatTokens(tb.cacheReads)} ${colors.money(tb.cacheReadCost)}` },
|
|
119
|
+
];
|
|
120
|
+
const lines = barChart(tokenItems, barWidth);
|
|
121
|
+
for (const line of lines) {
|
|
122
|
+
console.log(line);
|
|
123
|
+
}
|
|
124
|
+
console.log(` ${colors.label('Cache hit rate')} ${formatPercent(tb.cacheHitRate * 100)}`);
|
|
125
|
+
}
|
|
126
|
+
// Hints
|
|
127
|
+
console.log();
|
|
128
|
+
console.log(colors.hint(`Run stats cost --period 30d for monthly trends`));
|
|
129
|
+
console.log(colors.hint('Run stats models for detailed model analysis'));
|
|
130
|
+
console.log();
|
|
131
|
+
trackEvent('stats', true, 'cost');
|
|
132
|
+
showTip('stats cost');
|
|
133
|
+
}
|
|
134
|
+
catch (err) {
|
|
135
|
+
trackEvent('stats', false, 'cost');
|
|
136
|
+
handleStatsError(err);
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
//# sourceMappingURL=cost.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cost.js","sourceRoot":"","sources":["../../../../src/commands/stats/actions/cost.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"error-handler.d.ts","sourceRoot":"","sources":["../../../../src/commands/stats/actions/error-handler.ts"],"names":[],"mappings":"AAYA;;;GAGG;AACH,wBAAgB,gBAAgB,CAAC,GAAG,EAAE,OAAO,GAAG,KAAK,CA+BpD"}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
// ──────────────────────────────────────────────────────
|
|
2
|
+
// Shared error handler for all stats action handlers
|
|
3
|
+
// ──────────────────────────────────────────────────────
|
|
4
|
+
import { ProjectNotFoundError, FirestoreIndexError, ConfigNotFoundError, InvalidPeriodError, } from '../data/types.js';
|
|
5
|
+
import { colors } from '../render/colors.js';
|
|
6
|
+
/**
|
|
7
|
+
* Handle known stats errors with user-friendly output.
|
|
8
|
+
* Rethrows unknown errors.
|
|
9
|
+
*/
|
|
10
|
+
export function handleStatsError(err) {
|
|
11
|
+
if (err instanceof InvalidPeriodError) {
|
|
12
|
+
console.error(`\n ${colors.error(err.message)}`);
|
|
13
|
+
console.log(colors.hint('Expected: 7d, 30d, 90d, or all'));
|
|
14
|
+
process.exit(1);
|
|
15
|
+
}
|
|
16
|
+
if (err instanceof ConfigNotFoundError) {
|
|
17
|
+
console.error(`\n ${colors.error('\u2717')} ${err.message}`);
|
|
18
|
+
process.exit(1);
|
|
19
|
+
}
|
|
20
|
+
if (err instanceof ProjectNotFoundError) {
|
|
21
|
+
console.error(`\n ${colors.error(`Project "${err.projectName}" not found.`)}`);
|
|
22
|
+
if (err.suggestions.length > 0) {
|
|
23
|
+
console.log(`\n Did you mean?`);
|
|
24
|
+
for (const s of err.suggestions) {
|
|
25
|
+
console.log(` ${colors.success('\u25CF')} ${s}`);
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
console.log(`\n Available projects:`);
|
|
29
|
+
for (const p of err.availableProjects) {
|
|
30
|
+
console.log(` ${colors.success('\u25CF')} ${p.name}`);
|
|
31
|
+
}
|
|
32
|
+
process.exit(1);
|
|
33
|
+
}
|
|
34
|
+
if (err instanceof FirestoreIndexError) {
|
|
35
|
+
console.error(`\n ${colors.error('\u2717')} Failed to load stats`);
|
|
36
|
+
console.log(`\n ${colors.error('Error:')} Missing Firestore index for sessions query.`);
|
|
37
|
+
console.log(` Create it here: ${err.indexUrl}`);
|
|
38
|
+
process.exit(1);
|
|
39
|
+
}
|
|
40
|
+
throw err;
|
|
41
|
+
}
|
|
42
|
+
//# sourceMappingURL=error-handler.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"error-handler.js","sourceRoot":"","sources":["../../../../src/commands/stats/actions/error-handler.ts"],"names":[],"mappings":"AAAA,yDAAyD;AACzD,qDAAqD;AACrD,yDAAyD;AAEzD,OAAO,EACL,oBAAoB,EACpB,mBAAmB,EACnB,mBAAmB,EACnB,kBAAkB,GACnB,MAAM,kBAAkB,CAAC;AAC1B,OAAO,EAAE,MAAM,EAAE,MAAM,qBAAqB,CAAC;AAE7C;;;GAGG;AACH,MAAM,UAAU,gBAAgB,CAAC,GAAY;IAC3C,IAAI,GAAG,YAAY,kBAAkB,EAAE,CAAC;QACtC,OAAO,CAAC,KAAK,CAAC,OAAO,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;QAClD,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,gCAAgC,CAAC,CAAC,CAAC;QAC3D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IACD,IAAI,GAAG,YAAY,mBAAmB,EAAE,CAAC;QACvC,OAAO,CAAC,KAAK,CAAC,OAAO,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC;QAC9D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IACD,IAAI,GAAG,YAAY,oBAAoB,EAAE,CAAC;QACxC,OAAO,CAAC,KAAK,CAAC,OAAO,MAAM,CAAC,KAAK,CAAC,YAAY,GAAG,CAAC,WAAW,cAAc,CAAC,EAAE,CAAC,CAAC;QAChF,IAAI,GAAG,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC/B,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAC,CAAC;YACjC,KAAK,MAAM,CAAC,IAAI,GAAG,CAAC,WAAW,EAAE,CAAC;gBAChC,OAAO,CAAC,GAAG,CAAC,OAAO,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACtD,CAAC;QACH,CAAC;QACD,OAAO,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAC;QACvC,KAAK,MAAM,CAAC,IAAI,GAAG,CAAC,iBAAiB,EAAE,CAAC;YACtC,OAAO,CAAC,GAAG,CAAC,OAAO,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;QAC3D,CAAC;QACD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IACD,IAAI,GAAG,YAAY,mBAAmB,EAAE,CAAC;QACvC,OAAO,CAAC,KAAK,CAAC,OAAO,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,uBAAuB,CAAC,CAAC;QACpE,OAAO,CAAC,GAAG,CAAC,OAAO,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,8CAA8C,CAAC,CAAC;QACzF,OAAO,CAAC,GAAG,CAAC,qBAAqB,GAAG,CAAC,QAAQ,EAAE,CAAC,CAAC;QACjD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IACD,MAAM,GAAG,CAAC;AACZ,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"models.d.ts","sourceRoot":"","sources":["../../../../src/commands/stats/actions/models.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAE,UAAU,EAAuB,MAAM,kBAAkB,CAAC;AAexE,wBAAsB,YAAY,CAAC,KAAK,EAAE,UAAU,GAAG,OAAO,CAAC,IAAI,CAAC,CAoGnE"}
|
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
// ──────────────────────────────────────────────────────
|
|
2
|
+
// stats models — Model usage distribution + cost
|
|
3
|
+
// ──────────────────────────────────────────────────────
|
|
4
|
+
import ora from 'ora';
|
|
5
|
+
import { trackEvent } from '../../../utils/telemetry.js';
|
|
6
|
+
import { resolveDataSource } from '../data/source.js';
|
|
7
|
+
import { periodStartDate, computeModelStats } from '../data/aggregation.js';
|
|
8
|
+
import { colors } from '../render/colors.js';
|
|
9
|
+
import { handleStatsError } from './error-handler.js';
|
|
10
|
+
import { formatMoney, formatTokens, formatPercent, formatCount, formatPeriodLabel, } from '../render/format.js';
|
|
11
|
+
import { sparkline } from '../render/charts.js';
|
|
12
|
+
import { barChart } from '../render/charts.js';
|
|
13
|
+
import { sectionHeader, metricGrid, getBarWidth, projectCardHeader } from '../render/layout.js';
|
|
14
|
+
import { showTip } from '../../../utils/tips.js';
|
|
15
|
+
export async function modelsAction(flags) {
|
|
16
|
+
try {
|
|
17
|
+
const source = await resolveDataSource(flags);
|
|
18
|
+
const spinner = ora({ text: 'Syncing...', indent: 2 }).start();
|
|
19
|
+
try {
|
|
20
|
+
const prepResult = await source.prepare(flags);
|
|
21
|
+
spinner.succeed(prepResult.message);
|
|
22
|
+
}
|
|
23
|
+
catch {
|
|
24
|
+
spinner.warn('Sync failed (showing cached data)');
|
|
25
|
+
}
|
|
26
|
+
// Resolve project filter
|
|
27
|
+
const opts = {
|
|
28
|
+
periodStart: periodStartDate(flags.period),
|
|
29
|
+
sourceTool: flags.source,
|
|
30
|
+
};
|
|
31
|
+
if (flags.project) {
|
|
32
|
+
const resolved = await source.resolveProjectId(flags.project);
|
|
33
|
+
opts.projectId = resolved.projectId;
|
|
34
|
+
}
|
|
35
|
+
const sessions = await source.getSessions(opts);
|
|
36
|
+
const periodLabel = formatPeriodLabel(flags.period);
|
|
37
|
+
// Empty state
|
|
38
|
+
if (sessions.length === 0) {
|
|
39
|
+
console.log(sectionHeader('MODEL USAGE', periodLabel));
|
|
40
|
+
console.log(`\n No sessions found in the ${periodLabel.toLowerCase()}.\n`);
|
|
41
|
+
console.log(colors.hint('Run stats --period 30d to expand the time range'));
|
|
42
|
+
console.log();
|
|
43
|
+
return;
|
|
44
|
+
}
|
|
45
|
+
const models = computeModelStats(sessions, flags.period);
|
|
46
|
+
// No model data
|
|
47
|
+
if (models.length === 0) {
|
|
48
|
+
console.log(sectionHeader('MODEL USAGE', periodLabel));
|
|
49
|
+
console.log(`\n No model data available.\n`);
|
|
50
|
+
console.log(colors.hint('Model data is captured from Claude Code usage logs'));
|
|
51
|
+
console.log();
|
|
52
|
+
return;
|
|
53
|
+
}
|
|
54
|
+
// Header
|
|
55
|
+
console.log(sectionHeader('MODEL USAGE', periodLabel));
|
|
56
|
+
// Summary
|
|
57
|
+
const totalSessions = models.reduce((sum, m) => sum + m.sessionCount, 0);
|
|
58
|
+
console.log(`\n ${colors.value(String(models.length))} models across ${colors.value(formatCount(totalSessions))} sessions`);
|
|
59
|
+
// Model cards
|
|
60
|
+
for (const model of models) {
|
|
61
|
+
console.log(projectCardHeader(model.displayName));
|
|
62
|
+
console.log(metricGrid([
|
|
63
|
+
{ label: 'Sessions', value: `${formatCount(model.sessionCount)} (${formatPercent(model.sessionPercent)})` },
|
|
64
|
+
{ label: 'Cost', value: `${formatMoney(model.totalCost)} (${formatPercent(model.costPercent)})` },
|
|
65
|
+
{ label: 'Tokens', value: formatTokens(model.totalTokens) },
|
|
66
|
+
{ label: 'Avg/session', value: formatMoney(model.avgCostPerSession) },
|
|
67
|
+
]));
|
|
68
|
+
// Input / Output / Cache cost breakdown
|
|
69
|
+
console.log(` ${colors.label('Input')} ${colors.money(model.inputCost)}${''.padEnd(10)}${colors.label('Output')} ${colors.money(model.outputCost)}${''.padEnd(6)}${colors.label('Cache')} ${colors.money(model.cacheCost)}`);
|
|
70
|
+
// Trend sparkline
|
|
71
|
+
const spark = sparkline(model.trend.map(d => d.value));
|
|
72
|
+
if (spark) {
|
|
73
|
+
console.log(` ${colors.label('Trend')} ${spark}`);
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
// Cost distribution bar chart
|
|
77
|
+
if (models.length > 1) {
|
|
78
|
+
console.log(sectionHeader('COST DISTRIBUTION'));
|
|
79
|
+
const barWidth = getBarWidth();
|
|
80
|
+
const lines = barChart(models.map(m => ({
|
|
81
|
+
label: m.displayName,
|
|
82
|
+
value: m.totalCost,
|
|
83
|
+
suffix: `${formatPercent(m.costPercent)} ${colors.money(m.totalCost)}`,
|
|
84
|
+
})), barWidth);
|
|
85
|
+
for (const line of lines) {
|
|
86
|
+
console.log(line);
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
// Hints
|
|
90
|
+
console.log();
|
|
91
|
+
console.log(colors.hint('Run stats cost for time-based cost analysis'));
|
|
92
|
+
console.log();
|
|
93
|
+
trackEvent('stats', true, 'models');
|
|
94
|
+
showTip('stats models');
|
|
95
|
+
}
|
|
96
|
+
catch (err) {
|
|
97
|
+
trackEvent('stats', false, 'models');
|
|
98
|
+
handleStatsError(err);
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
//# sourceMappingURL=models.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"models.js","sourceRoot":"","sources":["../../../../src/commands/stats/actions/models.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"overview.d.ts","sourceRoot":"","sources":["../../../../src/commands/stats/actions/overview.ts"],"names":[],"mappings":"AAaA,OAAO,KAAK,EAAE,UAAU,EAAuB,MAAM,kBAAkB,CAAC;AAkBxE,wBAAsB,cAAc,CAAC,KAAK,EAAE,UAAU,GAAG,OAAO,CAAC,IAAI,CAAC,CAiJrE"}
|
|
@@ -0,0 +1,147 @@
|
|
|
1
|
+
// ──────────────────────────────────────────────────────
|
|
2
|
+
// stats (no args) — Dashboard overview
|
|
3
|
+
// ──────────────────────────────────────────────────────
|
|
4
|
+
import ora from 'ora';
|
|
5
|
+
import { trackEvent } from '../../../utils/telemetry.js';
|
|
6
|
+
import { resolveDataSource } from '../data/source.js';
|
|
7
|
+
import { periodStartDate, computeOverview, resolveTitle, shortenModelName, } from '../data/aggregation.js';
|
|
8
|
+
import { colors } from '../render/colors.js';
|
|
9
|
+
import { handleStatsError } from './error-handler.js';
|
|
10
|
+
import { formatMoney, formatTokens, formatDuration, formatRelativeDate, formatCount, formatPeriodLabel, } from '../render/format.js';
|
|
11
|
+
import { sparkline, sparklineLabels } from '../render/charts.js';
|
|
12
|
+
import { barChart } from '../render/charts.js';
|
|
13
|
+
import { sectionHeader, metricGrid, getBarWidth } from '../render/layout.js';
|
|
14
|
+
import { showWelcomeIfFirstRun } from '../../../utils/welcome.js';
|
|
15
|
+
import { showTip } from '../../../utils/tips.js';
|
|
16
|
+
import { isConfigured } from '../../../utils/config.js';
|
|
17
|
+
export async function overviewAction(flags) {
|
|
18
|
+
try {
|
|
19
|
+
const source = await resolveDataSource(flags);
|
|
20
|
+
const spinner = ora({ text: 'Syncing...', indent: 2 }).start();
|
|
21
|
+
try {
|
|
22
|
+
const prepResult = await source.prepare(flags);
|
|
23
|
+
spinner.succeed(prepResult.message);
|
|
24
|
+
}
|
|
25
|
+
catch {
|
|
26
|
+
spinner.warn('Sync failed (showing cached data)');
|
|
27
|
+
}
|
|
28
|
+
// Show welcome message on first ever run (no config file)
|
|
29
|
+
if (!isConfigured()) {
|
|
30
|
+
showWelcomeIfFirstRun();
|
|
31
|
+
}
|
|
32
|
+
// Resolve project filter
|
|
33
|
+
const opts = {
|
|
34
|
+
periodStart: periodStartDate(flags.period),
|
|
35
|
+
sourceTool: flags.source,
|
|
36
|
+
};
|
|
37
|
+
if (flags.project) {
|
|
38
|
+
const resolved = await source.resolveProjectId(flags.project);
|
|
39
|
+
opts.projectId = resolved.projectId;
|
|
40
|
+
}
|
|
41
|
+
const sessions = await source.getSessions(opts);
|
|
42
|
+
// Empty state: no sessions at all
|
|
43
|
+
if (sessions.length === 0 && !opts.periodStart) {
|
|
44
|
+
console.log(`\n No sessions found.\n`);
|
|
45
|
+
console.log(` Get started:`);
|
|
46
|
+
console.log(` 1. Use Claude Code, Cursor, or Codex on a project`);
|
|
47
|
+
console.log(` 2. Run ${colors.value('code-insights sync')} to upload your sessions`);
|
|
48
|
+
console.log(` 3. Run ${colors.value('code-insights stats')} to see your analytics\n`);
|
|
49
|
+
showTip('stats');
|
|
50
|
+
return;
|
|
51
|
+
}
|
|
52
|
+
// Empty state: no sessions in period
|
|
53
|
+
if (sessions.length === 0) {
|
|
54
|
+
const periodLabel = formatPeriodLabel(flags.period);
|
|
55
|
+
console.log(sectionHeader('CODE INSIGHTS', periodLabel));
|
|
56
|
+
console.log(`\n No sessions in the ${periodLabel.toLowerCase()}.\n`);
|
|
57
|
+
const lastSession = await source.getLastSession();
|
|
58
|
+
if (lastSession) {
|
|
59
|
+
const title = resolveTitle(lastSession);
|
|
60
|
+
console.log(` Last session: ${formatRelativeDate(lastSession.endedAt)} in ${colors.project(lastSession.projectName)} — ${colors.value(title)}`);
|
|
61
|
+
console.log();
|
|
62
|
+
}
|
|
63
|
+
console.log(colors.hint(`Run stats --period 30d to expand the time range`));
|
|
64
|
+
console.log();
|
|
65
|
+
return;
|
|
66
|
+
}
|
|
67
|
+
const stats = computeOverview(sessions, flags.period);
|
|
68
|
+
const periodLabel = formatPeriodLabel(flags.period);
|
|
69
|
+
// Header
|
|
70
|
+
let headerTitle = 'CODE INSIGHTS';
|
|
71
|
+
if (flags.project)
|
|
72
|
+
headerTitle += ` \u2014 ${flags.project}`;
|
|
73
|
+
if (flags.source)
|
|
74
|
+
headerTitle += ` \u2014 ${flags.source} only`;
|
|
75
|
+
console.log(sectionHeader(headerTitle, periodLabel));
|
|
76
|
+
// Metric grid
|
|
77
|
+
const isProjectScoped = !!flags.project;
|
|
78
|
+
const modelCount = new Set(sessions.map(s => s.primaryModel).filter(Boolean)).size;
|
|
79
|
+
console.log();
|
|
80
|
+
console.log(metricGrid([
|
|
81
|
+
{ label: 'Sessions', value: formatCount(stats.sessionCount) },
|
|
82
|
+
{ label: 'Cost', value: formatMoney(stats.totalCost) },
|
|
83
|
+
{ label: 'Time', value: formatDuration(stats.totalTimeMinutes) },
|
|
84
|
+
{ label: 'Messages', value: formatCount(stats.messageCount) },
|
|
85
|
+
{ label: 'Tokens', value: formatTokens(stats.totalTokens) },
|
|
86
|
+
{ label: isProjectScoped ? 'Models' : 'Projects', value: formatCount(isProjectScoped ? modelCount : stats.projectCount) },
|
|
87
|
+
]));
|
|
88
|
+
// Activity sparkline
|
|
89
|
+
const spark = sparkline(stats.activityByDay.map(d => d.value));
|
|
90
|
+
const labels = sparklineLabels(flags.period);
|
|
91
|
+
console.log(sectionHeader('ACTIVITY', spark));
|
|
92
|
+
if (labels) {
|
|
93
|
+
// Right-align labels under the sparkline in the header
|
|
94
|
+
console.log(` ${colors.label(labels)}`);
|
|
95
|
+
}
|
|
96
|
+
// Quick stats: today / yesterday / this week
|
|
97
|
+
console.log();
|
|
98
|
+
console.log(` ${colors.value('Today'.padEnd(14))} ${colors.label(`${stats.todayStats.sessionCount} sessions`)} ${colors.money(stats.todayStats.totalCost)} ${colors.label(formatDuration(stats.todayStats.totalMinutes))}`);
|
|
99
|
+
console.log(` ${colors.value('Yesterday'.padEnd(14))} ${colors.label(`${stats.yesterdayStats.sessionCount} sessions`)} ${colors.money(stats.yesterdayStats.totalCost)} ${colors.label(formatDuration(stats.yesterdayStats.totalMinutes))}`);
|
|
100
|
+
console.log(` ${colors.value('This week'.padEnd(14))} ${colors.label(`${stats.weekStats.sessionCount} sessions`)} ${colors.money(stats.weekStats.totalCost)} ${colors.label(formatDuration(stats.weekStats.totalMinutes))}`);
|
|
101
|
+
// Top projects (or models if project-scoped)
|
|
102
|
+
if (isProjectScoped) {
|
|
103
|
+
const models = new Set(sessions.map(s => s.primaryModel).filter((m) => m != null));
|
|
104
|
+
if (models.size > 0) {
|
|
105
|
+
console.log(`\n ${colors.label('Models used:')} ${[...models].map(m => colors.model(shortenModelName(m))).join(', ')}`);
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
else if (stats.topProjects.length > 0) {
|
|
109
|
+
console.log(sectionHeader('TOP PROJECTS'));
|
|
110
|
+
const barWidth = getBarWidth();
|
|
111
|
+
const lines = barChart(stats.topProjects.map(p => ({
|
|
112
|
+
label: p.name,
|
|
113
|
+
value: p.count,
|
|
114
|
+
suffix: `${colors.label(`${p.count} sessions`)} ${colors.money(p.cost)}`,
|
|
115
|
+
})), barWidth);
|
|
116
|
+
for (const line of lines) {
|
|
117
|
+
console.log(line);
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
// Sources section (only if 2+ source tools)
|
|
121
|
+
if (stats.sourceTools.length >= 2) {
|
|
122
|
+
console.log(sectionHeader('SOURCES'));
|
|
123
|
+
const barWidth = getBarWidth();
|
|
124
|
+
const lines = barChart(stats.sourceTools.map(s => ({
|
|
125
|
+
label: s.name,
|
|
126
|
+
value: s.count,
|
|
127
|
+
suffix: `${colors.label(`${s.count} sessions`)} ${colors.money(s.cost)}`,
|
|
128
|
+
})), barWidth);
|
|
129
|
+
for (const line of lines) {
|
|
130
|
+
console.log(line);
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
// Hints
|
|
134
|
+
console.log();
|
|
135
|
+
console.log(colors.hint('Run stats cost for cost breakdown'));
|
|
136
|
+
console.log(colors.hint("Run stats today for today's sessions"));
|
|
137
|
+
console.log(colors.hint('Run stats projects for project details'));
|
|
138
|
+
console.log();
|
|
139
|
+
trackEvent('stats', true);
|
|
140
|
+
showTip('stats');
|
|
141
|
+
}
|
|
142
|
+
catch (err) {
|
|
143
|
+
trackEvent('stats', false);
|
|
144
|
+
handleStatsError(err);
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
//# sourceMappingURL=overview.js.map
|