@kaitranntt/ccs 5.11.0 → 5.12.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 +2 -1
- package/VERSION +1 -1
- package/dist/cliproxy/model-catalog.d.ts.map +1 -1
- package/dist/cliproxy/model-catalog.js +11 -15
- package/dist/cliproxy/model-catalog.js.map +1 -1
- package/dist/commands/update-command.d.ts.map +1 -1
- package/dist/commands/update-command.js +7 -8
- package/dist/commands/update-command.js.map +1 -1
- package/dist/commands/version-command.d.ts.map +1 -1
- package/dist/commands/version-command.js +2 -3
- package/dist/commands/version-command.js.map +1 -1
- package/dist/ui/assets/{form-utils-Bjfa1DVH.js → form-utils-qlCULW73.js} +1 -1
- package/dist/ui/assets/icons-DOhZYYMX.js +1 -0
- package/dist/ui/assets/index-C5_mBoh5.js +72 -0
- package/dist/ui/assets/index-FXq1wR_S.css +1 -0
- package/dist/ui/assets/{radix-ui-Ba6LUgyw.js → radix-ui-CBdrr74O.js} +1 -1
- package/dist/ui/assets/react-vendor-B3ahKmcC.js +3 -0
- package/dist/ui/assets/{tanstack-MD0629v8.js → tanstack-D9KW3ciX.js} +1 -1
- package/dist/ui/index.html +15 -7
- package/dist/utils/version.d.ts +8 -0
- package/dist/utils/version.d.ts.map +1 -0
- package/dist/utils/version.js +51 -0
- package/dist/utils/version.js.map +1 -0
- package/dist/web-server/health-service.d.ts +14 -3
- package/dist/web-server/health-service.d.ts.map +1 -1
- package/dist/web-server/health-service.js +545 -66
- package/dist/web-server/health-service.js.map +1 -1
- package/dist/web-server/index.d.ts.map +1 -1
- package/dist/web-server/index.js +9 -0
- package/dist/web-server/index.js.map +1 -1
- package/dist/web-server/overview-routes.d.ts.map +1 -1
- package/dist/web-server/overview-routes.js +18 -4
- package/dist/web-server/overview-routes.js.map +1 -1
- package/dist/web-server/routes.js +11 -4
- package/dist/web-server/routes.js.map +1 -1
- package/dist/web-server/shared-routes.js +3 -2
- package/dist/web-server/shared-routes.js.map +1 -1
- package/dist/web-server/usage-routes.d.ts +27 -0
- package/dist/web-server/usage-routes.d.ts.map +1 -0
- package/dist/web-server/usage-routes.js +459 -0
- package/dist/web-server/usage-routes.js.map +1 -0
- package/package.json +2 -1
- package/dist/ui/assets/icons-D-Y22K9Z.js +0 -1
- package/dist/ui/assets/index-M5ru6OPu.js +0 -9
- package/dist/ui/assets/index-ZXu5MvSE.css +0 -1
- package/dist/ui/assets/react-vendor-CW-MU7-e.js +0 -3
|
@@ -0,0 +1,459 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Usage Analytics API Routes
|
|
4
|
+
*
|
|
5
|
+
* Provides REST endpoints for Claude Code usage analytics using better-ccusage library.
|
|
6
|
+
* Supports daily, monthly, and session-based usage data aggregation.
|
|
7
|
+
*
|
|
8
|
+
* Performance optimizations:
|
|
9
|
+
* - TTL-based caching to reduce better-ccusage library calls
|
|
10
|
+
* - Request coalescing to prevent duplicate concurrent requests
|
|
11
|
+
*/
|
|
12
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
13
|
+
exports.prewarmUsageCache = exports.clearUsageCache = exports.getLastFetchTimestamp = exports.usageRoutes = void 0;
|
|
14
|
+
const express_1 = require("express");
|
|
15
|
+
const data_loader_1 = require("better-ccusage/data-loader");
|
|
16
|
+
exports.usageRoutes = (0, express_1.Router)();
|
|
17
|
+
// Constants for validation
|
|
18
|
+
const MAX_LIMIT = 1000;
|
|
19
|
+
const DEFAULT_LIMIT = 50;
|
|
20
|
+
const DATE_REGEX = /^\d{8}$/; // YYYYMMDD format
|
|
21
|
+
// Cache TTLs (milliseconds)
|
|
22
|
+
const CACHE_TTL = {
|
|
23
|
+
daily: 60 * 1000, // 1 minute - changes frequently
|
|
24
|
+
monthly: 5 * 60 * 1000, // 5 minutes - aggregated data
|
|
25
|
+
session: 60 * 1000, // 1 minute - user may refresh
|
|
26
|
+
};
|
|
27
|
+
// Stale-while-revalidate: max age for stale data (1 hour)
|
|
28
|
+
const STALE_TTL = 60 * 60 * 1000;
|
|
29
|
+
// Track when data was last fetched (for UI indicator)
|
|
30
|
+
let lastFetchTimestamp = null;
|
|
31
|
+
/** Get timestamp of last successful data fetch */
|
|
32
|
+
function getLastFetchTimestamp() {
|
|
33
|
+
return lastFetchTimestamp;
|
|
34
|
+
}
|
|
35
|
+
exports.getLastFetchTimestamp = getLastFetchTimestamp;
|
|
36
|
+
// In-memory cache
|
|
37
|
+
const cache = new Map();
|
|
38
|
+
// Pending requests for coalescing (prevents duplicate concurrent calls)
|
|
39
|
+
const pendingRequests = new Map();
|
|
40
|
+
/**
|
|
41
|
+
* Get cached data or fetch from loader with TTL
|
|
42
|
+
* Also coalesces concurrent requests to prevent duplicate library calls
|
|
43
|
+
* Implements stale-while-revalidate pattern for instant responses
|
|
44
|
+
*/
|
|
45
|
+
async function getCachedData(key, ttl, loader) {
|
|
46
|
+
const cached = cache.get(key);
|
|
47
|
+
const now = Date.now();
|
|
48
|
+
// Fresh cache - return immediately
|
|
49
|
+
if (cached && now - cached.timestamp < ttl) {
|
|
50
|
+
return cached.data;
|
|
51
|
+
}
|
|
52
|
+
// Stale cache - return immediately, refresh in background (SWR pattern)
|
|
53
|
+
if (cached && now - cached.timestamp < STALE_TTL) {
|
|
54
|
+
// Fire and forget background refresh if not already pending
|
|
55
|
+
if (!pendingRequests.has(key)) {
|
|
56
|
+
const promise = loader()
|
|
57
|
+
.then((data) => {
|
|
58
|
+
cache.set(key, { data, timestamp: Date.now() });
|
|
59
|
+
lastFetchTimestamp = Date.now();
|
|
60
|
+
})
|
|
61
|
+
.catch((err) => {
|
|
62
|
+
console.error(`[!] Background refresh failed for ${key}:`, err);
|
|
63
|
+
})
|
|
64
|
+
.finally(() => {
|
|
65
|
+
pendingRequests.delete(key);
|
|
66
|
+
});
|
|
67
|
+
pendingRequests.set(key, promise);
|
|
68
|
+
}
|
|
69
|
+
return cached.data;
|
|
70
|
+
}
|
|
71
|
+
// No usable cache - check if request is already pending (coalesce)
|
|
72
|
+
const pending = pendingRequests.get(key);
|
|
73
|
+
if (pending) {
|
|
74
|
+
return pending;
|
|
75
|
+
}
|
|
76
|
+
// Create new request
|
|
77
|
+
const promise = loader()
|
|
78
|
+
.then((data) => {
|
|
79
|
+
cache.set(key, { data, timestamp: Date.now() });
|
|
80
|
+
lastFetchTimestamp = Date.now();
|
|
81
|
+
return data;
|
|
82
|
+
})
|
|
83
|
+
.finally(() => {
|
|
84
|
+
pendingRequests.delete(key);
|
|
85
|
+
});
|
|
86
|
+
pendingRequests.set(key, promise);
|
|
87
|
+
return promise;
|
|
88
|
+
}
|
|
89
|
+
/** Cached loader for daily usage data */
|
|
90
|
+
async function getCachedDailyData() {
|
|
91
|
+
return getCachedData('daily', CACHE_TTL.daily, async () => {
|
|
92
|
+
return (await (0, data_loader_1.loadDailyUsageData)());
|
|
93
|
+
});
|
|
94
|
+
}
|
|
95
|
+
/** Cached loader for monthly usage data */
|
|
96
|
+
async function getCachedMonthlyData() {
|
|
97
|
+
return getCachedData('monthly', CACHE_TTL.monthly, async () => {
|
|
98
|
+
return (await (0, data_loader_1.loadMonthlyUsageData)());
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
/** Cached loader for session data */
|
|
102
|
+
async function getCachedSessionData() {
|
|
103
|
+
return getCachedData('session', CACHE_TTL.session, async () => {
|
|
104
|
+
return (await (0, data_loader_1.loadSessionData)());
|
|
105
|
+
});
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* Clear all cached data (useful for manual refresh)
|
|
109
|
+
*/
|
|
110
|
+
function clearUsageCache() {
|
|
111
|
+
cache.clear();
|
|
112
|
+
}
|
|
113
|
+
exports.clearUsageCache = clearUsageCache;
|
|
114
|
+
/**
|
|
115
|
+
* Pre-warm usage caches on server startup
|
|
116
|
+
* Loads all usage data into cache so first user request is instant
|
|
117
|
+
* Returns timestamp when cache was populated
|
|
118
|
+
*/
|
|
119
|
+
async function prewarmUsageCache() {
|
|
120
|
+
const start = Date.now();
|
|
121
|
+
console.log('[i] Pre-warming usage cache...');
|
|
122
|
+
try {
|
|
123
|
+
await Promise.all([getCachedDailyData(), getCachedMonthlyData(), getCachedSessionData()]);
|
|
124
|
+
const elapsed = Date.now() - start;
|
|
125
|
+
lastFetchTimestamp = Date.now();
|
|
126
|
+
console.log(`[OK] Usage cache ready (${elapsed}ms)`);
|
|
127
|
+
return { timestamp: lastFetchTimestamp, elapsed };
|
|
128
|
+
}
|
|
129
|
+
catch (err) {
|
|
130
|
+
console.error('[!] Failed to prewarm usage cache:', err);
|
|
131
|
+
throw err;
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
exports.prewarmUsageCache = prewarmUsageCache;
|
|
135
|
+
// ============================================================================
|
|
136
|
+
// Validation Helpers
|
|
137
|
+
// ============================================================================
|
|
138
|
+
/**
|
|
139
|
+
* Validate date string in YYYYMMDD format
|
|
140
|
+
*/
|
|
141
|
+
function validateDate(dateString) {
|
|
142
|
+
if (!dateString)
|
|
143
|
+
return undefined;
|
|
144
|
+
if (!DATE_REGEX.test(dateString)) {
|
|
145
|
+
throw new Error('Invalid date format. Use YYYYMMDD');
|
|
146
|
+
}
|
|
147
|
+
// Basic range check
|
|
148
|
+
const year = parseInt(dateString.substring(0, 4), 10);
|
|
149
|
+
const month = parseInt(dateString.substring(4, 6), 10);
|
|
150
|
+
const day = parseInt(dateString.substring(6, 8), 10);
|
|
151
|
+
if (year < 2024 || year > 2100)
|
|
152
|
+
throw new Error('Year out of valid range');
|
|
153
|
+
if (month < 1 || month > 12)
|
|
154
|
+
throw new Error('Month out of valid range');
|
|
155
|
+
if (day < 1 || day > 31)
|
|
156
|
+
throw new Error('Day out of valid range');
|
|
157
|
+
return dateString;
|
|
158
|
+
}
|
|
159
|
+
/**
|
|
160
|
+
* Validate and parse limit parameter
|
|
161
|
+
*/
|
|
162
|
+
function validateLimit(limit) {
|
|
163
|
+
if (!limit)
|
|
164
|
+
return DEFAULT_LIMIT;
|
|
165
|
+
const num = parseInt(limit, 10);
|
|
166
|
+
if (isNaN(num) || num < 1 || num > MAX_LIMIT) {
|
|
167
|
+
throw new Error(`Limit must be between 1 and ${MAX_LIMIT}`);
|
|
168
|
+
}
|
|
169
|
+
return num;
|
|
170
|
+
}
|
|
171
|
+
/**
|
|
172
|
+
* Validate and parse offset parameter
|
|
173
|
+
*/
|
|
174
|
+
function validateOffset(offset) {
|
|
175
|
+
if (!offset)
|
|
176
|
+
return 0;
|
|
177
|
+
const num = parseInt(offset, 10);
|
|
178
|
+
if (isNaN(num) || num < 0) {
|
|
179
|
+
throw new Error('Offset must be a non-negative number');
|
|
180
|
+
}
|
|
181
|
+
return num;
|
|
182
|
+
}
|
|
183
|
+
/**
|
|
184
|
+
* Filter data by date range
|
|
185
|
+
*/
|
|
186
|
+
function filterByDateRange(data, since, until) {
|
|
187
|
+
if (!since && !until)
|
|
188
|
+
return data;
|
|
189
|
+
return data.filter((item) => {
|
|
190
|
+
// Get the date field (prioritize date, then month, then lastActivity)
|
|
191
|
+
const itemDate = item.date || item.month?.replace('-', '') || item.lastActivity?.replace(/-/g, '');
|
|
192
|
+
if (!itemDate)
|
|
193
|
+
return true;
|
|
194
|
+
// Normalize to YYYYMMDD for comparison
|
|
195
|
+
const normalizedDate = itemDate.replace(/-/g, '').substring(0, 8);
|
|
196
|
+
if (since && normalizedDate < since)
|
|
197
|
+
return false;
|
|
198
|
+
if (until && normalizedDate > until)
|
|
199
|
+
return false;
|
|
200
|
+
return true;
|
|
201
|
+
});
|
|
202
|
+
}
|
|
203
|
+
/**
|
|
204
|
+
* Create standard error response
|
|
205
|
+
*/
|
|
206
|
+
function errorResponse(res, error, defaultMessage) {
|
|
207
|
+
console.error(defaultMessage + ':', error);
|
|
208
|
+
const errorMessage = error instanceof Error ? error.message : 'Unknown error';
|
|
209
|
+
const isValidationError = errorMessage.includes('Invalid') ||
|
|
210
|
+
errorMessage.includes('format') ||
|
|
211
|
+
errorMessage.includes('range') ||
|
|
212
|
+
errorMessage.includes('must be');
|
|
213
|
+
const statusCode = isValidationError ? 400 : 500;
|
|
214
|
+
res.status(statusCode).json({
|
|
215
|
+
success: false,
|
|
216
|
+
error: isValidationError ? errorMessage : defaultMessage,
|
|
217
|
+
});
|
|
218
|
+
}
|
|
219
|
+
/**
|
|
220
|
+
* GET /api/usage/summary
|
|
221
|
+
*
|
|
222
|
+
* Returns usage summary data for quick dashboard display.
|
|
223
|
+
* Query: ?since=YYYYMMDD&until=YYYYMMDD
|
|
224
|
+
*/
|
|
225
|
+
exports.usageRoutes.get('/summary', async (req, res) => {
|
|
226
|
+
try {
|
|
227
|
+
const since = validateDate(req.query.since);
|
|
228
|
+
const until = validateDate(req.query.until);
|
|
229
|
+
const dailyData = await getCachedDailyData();
|
|
230
|
+
const filtered = filterByDateRange(dailyData, since, until);
|
|
231
|
+
// Calculate totals
|
|
232
|
+
let totalInputTokens = 0;
|
|
233
|
+
let totalOutputTokens = 0;
|
|
234
|
+
let totalCacheTokens = 0;
|
|
235
|
+
let totalCost = 0;
|
|
236
|
+
for (const day of filtered) {
|
|
237
|
+
totalInputTokens += day.inputTokens;
|
|
238
|
+
totalOutputTokens += day.outputTokens;
|
|
239
|
+
totalCacheTokens += day.cacheCreationTokens + day.cacheReadTokens;
|
|
240
|
+
totalCost += day.totalCost;
|
|
241
|
+
}
|
|
242
|
+
const totalTokens = totalInputTokens + totalOutputTokens;
|
|
243
|
+
res.json({
|
|
244
|
+
success: true,
|
|
245
|
+
data: {
|
|
246
|
+
totalTokens,
|
|
247
|
+
totalInputTokens,
|
|
248
|
+
totalOutputTokens,
|
|
249
|
+
totalCacheTokens,
|
|
250
|
+
totalCost: Math.round(totalCost * 100) / 100,
|
|
251
|
+
totalDays: filtered.length,
|
|
252
|
+
averageTokensPerDay: filtered.length > 0 ? Math.round(totalTokens / filtered.length) : 0,
|
|
253
|
+
averageCostPerDay: filtered.length > 0 ? Math.round((totalCost / filtered.length) * 100) / 100 : 0,
|
|
254
|
+
},
|
|
255
|
+
});
|
|
256
|
+
}
|
|
257
|
+
catch (error) {
|
|
258
|
+
errorResponse(res, error, 'Failed to fetch usage summary');
|
|
259
|
+
}
|
|
260
|
+
});
|
|
261
|
+
/**
|
|
262
|
+
* GET /api/usage/daily
|
|
263
|
+
*
|
|
264
|
+
* Returns daily usage trends for chart visualization.
|
|
265
|
+
* Query: ?since=YYYYMMDD&until=YYYYMMDD
|
|
266
|
+
*/
|
|
267
|
+
exports.usageRoutes.get('/daily', async (req, res) => {
|
|
268
|
+
try {
|
|
269
|
+
const since = validateDate(req.query.since);
|
|
270
|
+
const until = validateDate(req.query.until);
|
|
271
|
+
const dailyData = await getCachedDailyData();
|
|
272
|
+
const filtered = filterByDateRange(dailyData, since, until);
|
|
273
|
+
// Transform for chart consumption
|
|
274
|
+
const trends = filtered.map((day) => ({
|
|
275
|
+
date: day.date,
|
|
276
|
+
tokens: day.inputTokens + day.outputTokens,
|
|
277
|
+
inputTokens: day.inputTokens,
|
|
278
|
+
outputTokens: day.outputTokens,
|
|
279
|
+
cacheTokens: day.cacheCreationTokens + day.cacheReadTokens,
|
|
280
|
+
cost: Math.round(day.totalCost * 100) / 100,
|
|
281
|
+
modelsUsed: day.modelsUsed.length,
|
|
282
|
+
}));
|
|
283
|
+
res.json({
|
|
284
|
+
success: true,
|
|
285
|
+
data: trends,
|
|
286
|
+
});
|
|
287
|
+
}
|
|
288
|
+
catch (error) {
|
|
289
|
+
errorResponse(res, error, 'Failed to fetch daily usage');
|
|
290
|
+
}
|
|
291
|
+
});
|
|
292
|
+
/**
|
|
293
|
+
* GET /api/usage/models
|
|
294
|
+
*
|
|
295
|
+
* Returns usage breakdown by model for pie/bar charts.
|
|
296
|
+
* Query: ?since=YYYYMMDD&until=YYYYMMDD
|
|
297
|
+
*/
|
|
298
|
+
exports.usageRoutes.get('/models', async (req, res) => {
|
|
299
|
+
try {
|
|
300
|
+
const since = validateDate(req.query.since);
|
|
301
|
+
const until = validateDate(req.query.until);
|
|
302
|
+
const dailyData = await getCachedDailyData();
|
|
303
|
+
const filtered = filterByDateRange(dailyData, since, until);
|
|
304
|
+
// Aggregate model usage across all days
|
|
305
|
+
const modelMap = new Map();
|
|
306
|
+
for (const day of filtered) {
|
|
307
|
+
for (const breakdown of day.modelBreakdowns) {
|
|
308
|
+
const existing = modelMap.get(breakdown.modelName) || {
|
|
309
|
+
model: breakdown.modelName,
|
|
310
|
+
inputTokens: 0,
|
|
311
|
+
outputTokens: 0,
|
|
312
|
+
cacheTokens: 0,
|
|
313
|
+
cost: 0,
|
|
314
|
+
};
|
|
315
|
+
existing.inputTokens += breakdown.inputTokens;
|
|
316
|
+
existing.outputTokens += breakdown.outputTokens;
|
|
317
|
+
existing.cacheTokens += breakdown.cacheCreationTokens + breakdown.cacheReadTokens;
|
|
318
|
+
existing.cost += breakdown.cost;
|
|
319
|
+
modelMap.set(breakdown.modelName, existing);
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
// Calculate totals for percentage
|
|
323
|
+
const models = Array.from(modelMap.values());
|
|
324
|
+
const totalTokens = models.reduce((sum, m) => sum + m.inputTokens + m.outputTokens, 0);
|
|
325
|
+
// Add percentage and sort by tokens
|
|
326
|
+
const result = models
|
|
327
|
+
.map((m) => ({
|
|
328
|
+
...m,
|
|
329
|
+
tokens: m.inputTokens + m.outputTokens,
|
|
330
|
+
cost: Math.round(m.cost * 100) / 100,
|
|
331
|
+
percentage: totalTokens > 0
|
|
332
|
+
? Math.round(((m.inputTokens + m.outputTokens) / totalTokens) * 1000) / 10
|
|
333
|
+
: 0,
|
|
334
|
+
}))
|
|
335
|
+
.sort((a, b) => b.tokens - a.tokens);
|
|
336
|
+
res.json({
|
|
337
|
+
success: true,
|
|
338
|
+
data: result,
|
|
339
|
+
});
|
|
340
|
+
}
|
|
341
|
+
catch (error) {
|
|
342
|
+
errorResponse(res, error, 'Failed to fetch model usage');
|
|
343
|
+
}
|
|
344
|
+
});
|
|
345
|
+
/**
|
|
346
|
+
* GET /api/usage/sessions
|
|
347
|
+
*
|
|
348
|
+
* Returns paginated list of sessions.
|
|
349
|
+
* Query: ?since=YYYYMMDD&until=YYYYMMDD&limit=50&offset=0
|
|
350
|
+
*/
|
|
351
|
+
exports.usageRoutes.get('/sessions', async (req, res) => {
|
|
352
|
+
try {
|
|
353
|
+
const since = validateDate(req.query.since);
|
|
354
|
+
const until = validateDate(req.query.until);
|
|
355
|
+
const limit = validateLimit(req.query.limit);
|
|
356
|
+
const offset = validateOffset(req.query.offset);
|
|
357
|
+
const sessionData = await getCachedSessionData();
|
|
358
|
+
// Filter by date range using lastActivity
|
|
359
|
+
const filtered = filterByDateRange(sessionData, since, until);
|
|
360
|
+
// Sort by lastActivity descending
|
|
361
|
+
const sorted = [...filtered].sort((a, b) => new Date(b.lastActivity).getTime() - new Date(a.lastActivity).getTime());
|
|
362
|
+
// Paginate
|
|
363
|
+
const paginated = sorted.slice(offset, offset + limit);
|
|
364
|
+
// Transform for frontend
|
|
365
|
+
const sessions = paginated.map((s) => ({
|
|
366
|
+
sessionId: s.sessionId,
|
|
367
|
+
projectPath: s.projectPath,
|
|
368
|
+
tokens: s.inputTokens + s.outputTokens,
|
|
369
|
+
inputTokens: s.inputTokens,
|
|
370
|
+
outputTokens: s.outputTokens,
|
|
371
|
+
cost: Math.round(s.totalCost * 100) / 100,
|
|
372
|
+
lastActivity: s.lastActivity,
|
|
373
|
+
modelsUsed: s.modelsUsed,
|
|
374
|
+
}));
|
|
375
|
+
res.json({
|
|
376
|
+
success: true,
|
|
377
|
+
data: {
|
|
378
|
+
sessions,
|
|
379
|
+
total: filtered.length,
|
|
380
|
+
limit,
|
|
381
|
+
offset,
|
|
382
|
+
hasMore: offset + limit < filtered.length,
|
|
383
|
+
},
|
|
384
|
+
});
|
|
385
|
+
}
|
|
386
|
+
catch (error) {
|
|
387
|
+
errorResponse(res, error, 'Failed to fetch sessions');
|
|
388
|
+
}
|
|
389
|
+
});
|
|
390
|
+
/**
|
|
391
|
+
* GET /api/usage/monthly
|
|
392
|
+
*
|
|
393
|
+
* Returns monthly usage summary for charts.
|
|
394
|
+
* Query: ?since=YYYYMMDD&until=YYYYMMDD
|
|
395
|
+
*/
|
|
396
|
+
exports.usageRoutes.get('/monthly', async (req, res) => {
|
|
397
|
+
try {
|
|
398
|
+
const since = validateDate(req.query.since);
|
|
399
|
+
const until = validateDate(req.query.until);
|
|
400
|
+
const monthlyData = await getCachedMonthlyData();
|
|
401
|
+
// Filter by date range (convert month YYYY-MM to YYYYMM01 for comparison)
|
|
402
|
+
const filtered = since || until
|
|
403
|
+
? monthlyData.filter((m) => {
|
|
404
|
+
const monthDate = m.month.replace('-', '') + '01';
|
|
405
|
+
if (since && monthDate < since)
|
|
406
|
+
return false;
|
|
407
|
+
if (until && monthDate > until)
|
|
408
|
+
return false;
|
|
409
|
+
return true;
|
|
410
|
+
})
|
|
411
|
+
: monthlyData;
|
|
412
|
+
// Transform for charts
|
|
413
|
+
const result = filtered.map((m) => ({
|
|
414
|
+
month: m.month,
|
|
415
|
+
tokens: m.inputTokens + m.outputTokens,
|
|
416
|
+
inputTokens: m.inputTokens,
|
|
417
|
+
outputTokens: m.outputTokens,
|
|
418
|
+
cacheTokens: m.cacheCreationTokens + m.cacheReadTokens,
|
|
419
|
+
cost: Math.round(m.totalCost * 100) / 100,
|
|
420
|
+
modelsUsed: m.modelsUsed.length,
|
|
421
|
+
}));
|
|
422
|
+
res.json({
|
|
423
|
+
success: true,
|
|
424
|
+
data: result.sort((a, b) => a.month.localeCompare(b.month)),
|
|
425
|
+
});
|
|
426
|
+
}
|
|
427
|
+
catch (error) {
|
|
428
|
+
errorResponse(res, error, 'Failed to fetch monthly usage');
|
|
429
|
+
}
|
|
430
|
+
});
|
|
431
|
+
/**
|
|
432
|
+
* POST /api/usage/refresh
|
|
433
|
+
*
|
|
434
|
+
* Clears the usage cache to force fresh data fetch.
|
|
435
|
+
* Useful when user wants to see latest data immediately.
|
|
436
|
+
*/
|
|
437
|
+
exports.usageRoutes.post('/refresh', (_req, res) => {
|
|
438
|
+
clearUsageCache();
|
|
439
|
+
res.json({
|
|
440
|
+
success: true,
|
|
441
|
+
message: 'Usage cache cleared',
|
|
442
|
+
});
|
|
443
|
+
});
|
|
444
|
+
/**
|
|
445
|
+
* GET /api/usage/status
|
|
446
|
+
*
|
|
447
|
+
* Returns cache status including last fetch timestamp.
|
|
448
|
+
* Used by UI to show "Last updated: X ago" indicator.
|
|
449
|
+
*/
|
|
450
|
+
exports.usageRoutes.get('/status', (_req, res) => {
|
|
451
|
+
res.json({
|
|
452
|
+
success: true,
|
|
453
|
+
data: {
|
|
454
|
+
lastFetch: lastFetchTimestamp,
|
|
455
|
+
cacheSize: cache.size,
|
|
456
|
+
},
|
|
457
|
+
});
|
|
458
|
+
});
|
|
459
|
+
//# sourceMappingURL=usage-routes.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"usage-routes.js","sourceRoot":"","sources":["../../src/web-server/usage-routes.ts"],"names":[],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@kaitranntt/ccs",
|
|
3
|
-
"version": "5.
|
|
3
|
+
"version": "5.12.0",
|
|
4
4
|
"description": "Claude Code Switch - Instant profile switching between Claude Sonnet 4.5 and GLM 4.6",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"cli",
|
|
@@ -79,6 +79,7 @@
|
|
|
79
79
|
"postinstall": "node scripts/postinstall.js"
|
|
80
80
|
},
|
|
81
81
|
"dependencies": {
|
|
82
|
+
"better-ccusage": "^1.2.6",
|
|
82
83
|
"boxen": "^8.0.1",
|
|
83
84
|
"chalk": "^5.6.2",
|
|
84
85
|
"chokidar": "^5.0.0",
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
import{r as n}from"./react-vendor-CW-MU7-e.js";const M=t=>t.replace(/([a-z0-9])([A-Z])/g,"$1-$2").toLowerCase(),_=t=>t.replace(/^([A-Z])|[\s-_]+(\w)/g,(a,o,c)=>c?c.toUpperCase():o.toLowerCase()),r=t=>{const a=_(t);return a.charAt(0).toUpperCase()+a.slice(1)},y=(...t)=>t.filter((a,o,c)=>!!a&&a.trim()!==""&&c.indexOf(a)===o).join(" ").trim(),v=t=>{for(const a in t)if(a.startsWith("aria-")||a==="role"||a==="title")return!0};var f={xmlns:"http://www.w3.org/2000/svg",width:24,height:24,viewBox:"0 0 24 24",fill:"none",stroke:"currentColor",strokeWidth:2,strokeLinecap:"round",strokeLinejoin:"round"};const m=n.forwardRef(({color:t="currentColor",size:a=24,strokeWidth:o=2,absoluteStrokeWidth:c,className:h="",children:s,iconNode:k,...d},p)=>n.createElement("svg",{ref:p,...f,width:a,height:a,stroke:t,strokeWidth:c?Number(o)*24/Number(a):o,className:y("lucide",h),...!s&&!v(d)&&{"aria-hidden":"true"},...d},[...k.map(([i,l])=>n.createElement(i,l)),...Array.isArray(s)?s:[s]]));const e=(t,a)=>{const o=n.forwardRef(({className:c,...h},s)=>n.createElement(m,{ref:s,iconNode:a,className:y(`lucide-${M(r(t))}`,`lucide-${t}`,c),...h}));return o.displayName=r(t),o};const u=[["path",{d:"M22 12h-2.48a2 2 0 0 0-1.93 1.46l-2.35 8.36a.25.25 0 0 1-.48 0L9.24 2.18a.25.25 0 0 0-.48 0l-2.35 8.36A2 2 0 0 1 4.49 12H2",key:"169zse"}]],n1=e("activity",u);const w=[["path",{d:"M5 12h14",key:"1ays0h"}],["path",{d:"m12 5 7 7-7 7",key:"xquz4c"}]],h1=e("arrow-right",w);const $=[["path",{d:"M12 7v14",key:"1akyts"}],["path",{d:"M3 18a1 1 0 0 1-1-1V4a1 1 0 0 1 1-1h5a4 4 0 0 1 4 4 4 4 0 0 1 4-4h5a1 1 0 0 1 1 1v13a1 1 0 0 1-1 1h-6a3 3 0 0 0-3 3 3 3 0 0 0-3-3z",key:"ruj8y"}]],d1=e("book-open",$);const x=[["path",{d:"M12 8V4H8",key:"hb8ula"}],["rect",{width:"16",height:"12",x:"4",y:"8",rx:"2",key:"enze0r"}],["path",{d:"M2 14h2",key:"vft8re"}],["path",{d:"M20 14h2",key:"4cs60a"}],["path",{d:"M15 13v2",key:"1xurst"}],["path",{d:"M9 13v2",key:"rq6x2g"}]],r1=e("bot",x);const N=[["path",{d:"M20 6 9 17l-5-5",key:"1gmf2c"}]],y1=e("check",N);const g=[["path",{d:"m6 9 6 6 6-6",key:"qrunsl"}]],k1=e("chevron-down",g);const z=[["path",{d:"m9 18 6-6-6-6",key:"mthhwq"}]],p1=e("chevron-right",z);const b=[["path",{d:"M21.801 10A10 10 0 1 1 17 3.335",key:"yps3ct"}],["path",{d:"m9 11 3 3L22 4",key:"1pflzl"}]],i1=e("circle-check-big",b);const C=[["circle",{cx:"12",cy:"12",r:"10",key:"1mglay"}],["path",{d:"m15 9-6 6",key:"1uzhvr"}],["path",{d:"m9 9 6 6",key:"z0biqf"}]],l1=e("circle-x",C);const j=[["rect",{x:"2",y:"6",width:"20",height:"8",rx:"1",key:"1estib"}],["path",{d:"M17 14v7",key:"7m2elx"}],["path",{d:"M7 14v7",key:"1cm7wv"}],["path",{d:"M17 3v3",key:"1v4jwn"}],["path",{d:"M7 3v3",key:"7o6guu"}],["path",{d:"M10 14 2.3 6.3",key:"1023jk"}],["path",{d:"m14 6 7.7 7.7",key:"1s8pl2"}],["path",{d:"m8 6 8 8",key:"hl96qh"}]],M1=e("construction",j);const A=[["circle",{cx:"12",cy:"12",r:"1",key:"41hilf"}],["circle",{cx:"19",cy:"12",r:"1",key:"1wjl8i"}],["circle",{cx:"5",cy:"12",r:"1",key:"1pcz8c"}]],_1=e("ellipsis",A);const q=[["path",{d:"M10.733 5.076a10.744 10.744 0 0 1 11.205 6.575 1 1 0 0 1 0 .696 10.747 10.747 0 0 1-1.444 2.49",key:"ct8e1f"}],["path",{d:"M14.084 14.158a3 3 0 0 1-4.242-4.242",key:"151rxh"}],["path",{d:"M17.479 17.499a10.75 10.75 0 0 1-15.417-5.151 1 1 0 0 1 0-.696 10.75 10.75 0 0 1 4.446-5.143",key:"13bj9a"}],["path",{d:"m2 2 20 20",key:"1ooewy"}]],v1=e("eye-off",q);const H=[["path",{d:"M2.062 12.348a1 1 0 0 1 0-.696 10.75 10.75 0 0 1 19.876 0 1 1 0 0 1 0 .696 10.75 10.75 0 0 1-19.876 0",key:"1nclc0"}],["circle",{cx:"12",cy:"12",r:"3",key:"1v7zrd"}]],f1=e("eye",H);const L=[["path",{d:"M6 22a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h8a2.4 2.4 0 0 1 1.704.706l3.588 3.588A2.4 2.4 0 0 1 20 8v12a2 2 0 0 1-2 2z",key:"1oefj6"}],["path",{d:"M14 2v5a1 1 0 0 0 1 1h5",key:"wfsgrz"}],["path",{d:"M10 9H8",key:"b1mrlr"}],["path",{d:"M16 13H8",key:"t4e002"}],["path",{d:"M16 17H8",key:"z1uh3a"}]],m1=e("file-text",L);const V=[["path",{d:"m6 14 1.5-2.9A2 2 0 0 1 9.24 10H20a2 2 0 0 1 1.94 2.5l-1.54 6a2 2 0 0 1-1.95 1.5H4a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h3.9a2 2 0 0 1 1.69.9l.81 1.2a2 2 0 0 0 1.67.9H18a2 2 0 0 1 2 2v2",key:"usdka0"}]],u1=e("folder-open",V);const S=[["path",{d:"M15 21v-8a1 1 0 0 0-1-1h-4a1 1 0 0 0-1 1v8",key:"5wwlr5"}],["path",{d:"M3 10a2 2 0 0 1 .709-1.528l7-6a2 2 0 0 1 2.582 0l7 6A2 2 0 0 1 21 10v9a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2z",key:"r6nss1"}]],w1=e("house",S);const E=[["path",{d:"m15.5 7.5 2.3 2.3a1 1 0 0 0 1.4 0l2.1-2.1a1 1 0 0 0 0-1.4L19 4",key:"g0fldk"}],["path",{d:"m21 2-9.6 9.6",key:"1j0ho8"}],["circle",{cx:"7.5",cy:"15.5",r:"5.5",key:"yqb3hr"}]],$1=e("key",E);const B=[["path",{d:"M21 12a9 9 0 1 1-6.219-8.56",key:"13zald"}]],x1=e("loader-circle",B);const P=[["path",{d:"M20.985 12.486a9 9 0 1 1-9.473-9.472c.405-.022.617.46.402.803a6 6 0 0 0 8.268 8.268c.344-.215.825-.004.803.401",key:"kfwtm"}]],N1=e("moon",P);const R=[["rect",{width:"18",height:"18",x:"3",y:"3",rx:"2",key:"afitv7"}],["path",{d:"M9 3v18",key:"fh3hqa"}]],g1=e("panel-left",R);const W=[["path",{d:"M5 12h14",key:"1ays0h"}],["path",{d:"M12 5v14",key:"s699le"}]],z1=e("plus",W);const O=[["path",{d:"M3 12a9 9 0 0 1 9-9 9.75 9.75 0 0 1 6.74 2.74L21 8",key:"v9h5vc"}],["path",{d:"M21 3v5h-5",key:"1q7to0"}],["path",{d:"M21 12a9 9 0 0 1-9 9 9.75 9.75 0 0 1-6.74-2.74L3 16",key:"3uifl3"}],["path",{d:"M8 16H3v5",key:"1cv678"}]],b1=e("refresh-cw",O);const U=[["path",{d:"M15.2 3a2 2 0 0 1 1.4.6l3.8 3.8a2 2 0 0 1 .6 1.4V19a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2z",key:"1c8476"}],["path",{d:"M17 21v-7a1 1 0 0 0-1-1H8a1 1 0 0 0-1 1v7",key:"1ydtos"}],["path",{d:"M7 3v4a1 1 0 0 0 1 1h7",key:"t51u73"}]],C1=e("save",U);const T=[["path",{d:"M9.671 4.136a2.34 2.34 0 0 1 4.659 0 2.34 2.34 0 0 0 3.319 1.915 2.34 2.34 0 0 1 2.33 4.033 2.34 2.34 0 0 0 0 3.831 2.34 2.34 0 0 1-2.33 4.033 2.34 2.34 0 0 0-3.319 1.915 2.34 2.34 0 0 1-4.659 0 2.34 2.34 0 0 0-3.32-1.915 2.34 2.34 0 0 1-2.33-4.033 2.34 2.34 0 0 0 0-3.831A2.34 2.34 0 0 1 6.35 6.051a2.34 2.34 0 0 0 3.319-1.915",key:"1i5ecw"}],["circle",{cx:"12",cy:"12",r:"3",key:"1v7zrd"}]],j1=e("settings",T);const Z=[["path",{d:"M20 13c0 5-3.5 7.5-7.66 8.95a1 1 0 0 1-.67-.01C7.5 20.5 4 18 4 13V6a1 1 0 0 1 1-1c2 0 4.5-1.2 6.24-2.72a1.17 1.17 0 0 1 1.52 0C14.51 3.81 17 5 19 5a1 1 0 0 1 1 1z",key:"oel41y"}]],A1=e("shield",Z);const F=[["path",{d:"M11.017 2.814a1 1 0 0 1 1.966 0l1.051 5.558a2 2 0 0 0 1.594 1.594l5.558 1.051a1 1 0 0 1 0 1.966l-5.558 1.051a2 2 0 0 0-1.594 1.594l-1.051 5.558a1 1 0 0 1-1.966 0l-1.051-5.558a2 2 0 0 0-1.594-1.594l-5.558-1.051a1 1 0 0 1 0-1.966l5.558-1.051a2 2 0 0 0 1.594-1.594z",key:"1s2grr"}],["path",{d:"M20 2v4",key:"1rf3ol"}],["path",{d:"M22 4h-4",key:"gwowj6"}],["circle",{cx:"4",cy:"20",r:"2",key:"6kqj1y"}]],q1=e("sparkles",F);const K=[["path",{d:"M12 3H5a2 2 0 0 0-2 2v14a2 2 0 0 0 2 2h14a2 2 0 0 0 2-2v-7",key:"1m0v6g"}],["path",{d:"M18.375 2.625a1 1 0 0 1 3 3l-9.013 9.014a2 2 0 0 1-.853.505l-2.873.84a.5.5 0 0 1-.62-.62l.84-2.873a2 2 0 0 1 .506-.852z",key:"ohrbg2"}]],H1=e("square-pen",K);const I=[["path",{d:"M11.525 2.295a.53.53 0 0 1 .95 0l2.31 4.679a2.123 2.123 0 0 0 1.595 1.16l5.166.756a.53.53 0 0 1 .294.904l-3.736 3.638a2.123 2.123 0 0 0-.611 1.878l.882 5.14a.53.53 0 0 1-.771.56l-4.618-2.428a2.122 2.122 0 0 0-1.973 0L6.396 21.01a.53.53 0 0 1-.77-.56l.881-5.139a2.122 2.122 0 0 0-.611-1.879L2.16 9.795a.53.53 0 0 1 .294-.906l5.165-.755a2.122 2.122 0 0 0 1.597-1.16z",key:"r04s7s"}]],L1=e("star",I);const X=[["path",{d:"M11 2v2",key:"1539x4"}],["path",{d:"M5 2v2",key:"1yf1q8"}],["path",{d:"M5 3H4a2 2 0 0 0-2 2v4a6 6 0 0 0 12 0V5a2 2 0 0 0-2-2h-1",key:"rb5t3r"}],["path",{d:"M8 15a6 6 0 0 0 12 0v-3",key:"x18d4x"}],["circle",{cx:"20",cy:"10",r:"2",key:"ts1r5v"}]],V1=e("stethoscope",X);const D=[["circle",{cx:"12",cy:"12",r:"4",key:"4exip2"}],["path",{d:"M12 2v2",key:"tus03m"}],["path",{d:"M12 20v2",key:"1lh1kg"}],["path",{d:"m4.93 4.93 1.41 1.41",key:"149t6j"}],["path",{d:"m17.66 17.66 1.41 1.41",key:"ptbguv"}],["path",{d:"M2 12h2",key:"1t8f8n"}],["path",{d:"M20 12h2",key:"1q8mjw"}],["path",{d:"m6.34 17.66-1.41 1.41",key:"1m8zz5"}],["path",{d:"m19.07 4.93-1.41 1.41",key:"1shlcs"}]],S1=e("sun",D);const G=[["path",{d:"M10 11v6",key:"nco0om"}],["path",{d:"M14 11v6",key:"outv1u"}],["path",{d:"M19 6v14a2 2 0 0 1-2 2H7a2 2 0 0 1-2-2V6",key:"miytrc"}],["path",{d:"M3 6h18",key:"d0wm0j"}],["path",{d:"M8 6V4a2 2 0 0 1 2-2h4a2 2 0 0 1 2 2v2",key:"e791ji"}]],E1=e("trash-2",G);const J=[["path",{d:"m21.73 18-8-14a2 2 0 0 0-3.48 0l-8 14A2 2 0 0 0 4 21h16a2 2 0 0 0 1.73-3",key:"wmoenq"}],["path",{d:"M12 9v4",key:"juzpu7"}],["path",{d:"M12 17h.01",key:"p32p05"}]],B1=e("triangle-alert",J);const Q=[["path",{d:"M19 21v-2a4 4 0 0 0-4-4H9a4 4 0 0 0-4 4v2",key:"975kel"}],["circle",{cx:"12",cy:"7",r:"4",key:"17ys0d"}]],P1=e("user",Q);const Y=[["path",{d:"M16 21v-2a4 4 0 0 0-4-4H6a4 4 0 0 0-4 4v2",key:"1yyitq"}],["path",{d:"M16 3.128a4 4 0 0 1 0 7.744",key:"16gr8j"}],["path",{d:"M22 21v-2a4 4 0 0 0-3-3.87",key:"kshegd"}],["circle",{cx:"9",cy:"7",r:"4",key:"nufk8"}]],R1=e("users",Y);const e1=[["path",{d:"M12 20h.01",key:"zekei9"}],["path",{d:"M8.5 16.429a5 5 0 0 1 7 0",key:"1bycff"}],["path",{d:"M5 12.859a10 10 0 0 1 5.17-2.69",key:"1dl1wf"}],["path",{d:"M19 12.859a10 10 0 0 0-2.007-1.523",key:"4k23kn"}],["path",{d:"M2 8.82a15 15 0 0 1 4.177-2.643",key:"1grhjp"}],["path",{d:"M22 8.82a15 15 0 0 0-11.288-3.764",key:"z3jwby"}],["path",{d:"m2 2 20 20",key:"1ooewy"}]],W1=e("wifi-off",e1);const a1=[["path",{d:"M12 20h.01",key:"zekei9"}],["path",{d:"M2 8.82a15 15 0 0 1 20 0",key:"dnpr2z"}],["path",{d:"M5 12.859a10 10 0 0 1 14 0",key:"1x1e6c"}],["path",{d:"M8.5 16.429a5 5 0 0 1 7 0",key:"1bycff"}]],O1=e("wifi",a1);const t1=[["path",{d:"M14.7 6.3a1 1 0 0 0 0 1.4l1.6 1.6a1 1 0 0 0 1.4 0l3.106-3.105c.32-.322.863-.22.983.218a6 6 0 0 1-8.259 7.057l-7.91 7.91a1 1 0 0 1-2.999-3l7.91-7.91a6 6 0 0 1 7.057-8.259c.438.12.54.662.219.984z",key:"1ngwbx"}]],U1=e("wrench",t1);const o1=[["path",{d:"M18 6 6 18",key:"1bl5f8"}],["path",{d:"m6 6 12 12",key:"d8bk6v"}]],T1=e("x",o1);const c1=[["path",{d:"M4 14a1 1 0 0 1-.78-1.63l9.9-10.2a.5.5 0 0 1 .86.46l-1.92 6.02A1 1 0 0 0 13 10h7a1 1 0 0 1 .78 1.63l-9.9 10.2a.5.5 0 0 1-.86-.46l1.92-6.02A1 1 0 0 0 11 14z",key:"1xq2db"}]],Z1=e("zap",c1);export{n1 as A,d1 as B,p1 as C,_1 as E,u1 as F,w1 as H,$1 as K,x1 as L,N1 as M,g1 as P,b1 as R,j1 as S,B1 as T,R1 as U,W1 as W,T1 as X,Z1 as Z,S1 as a,O1 as b,A1 as c,z1 as d,V1 as e,H1 as f,E1 as g,v1 as h,f1 as i,C1 as j,P1 as k,y1 as l,L1 as m,k1 as n,M1 as o,h1 as p,l1 as q,i1 as r,U1 as s,m1 as t,q1 as u,r1 as v};
|