@nocoo/pew 0.2.0 → 0.6.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.map +1 -1
- package/dist/cli.js +117 -8
- package/dist/cli.js.map +1 -1
- package/dist/commands/session-sync.d.ts +62 -0
- package/dist/commands/session-sync.d.ts.map +1 -0
- package/dist/commands/session-sync.js +443 -0
- package/dist/commands/session-sync.js.map +1 -0
- package/dist/commands/session-upload.d.ts +55 -0
- package/dist/commands/session-upload.d.ts.map +1 -0
- package/dist/commands/session-upload.js +177 -0
- package/dist/commands/session-upload.js.map +1 -0
- package/dist/commands/status.d.ts +9 -0
- package/dist/commands/status.d.ts.map +1 -1
- package/dist/commands/status.js +22 -11
- package/dist/commands/status.js.map +1 -1
- package/dist/commands/sync.d.ts +13 -1
- package/dist/commands/sync.d.ts.map +1 -1
- package/dist/commands/sync.js +207 -9
- package/dist/commands/sync.js.map +1 -1
- package/dist/discovery/sources.d.ts +5 -0
- package/dist/discovery/sources.d.ts.map +1 -1
- package/dist/discovery/sources.js +7 -0
- package/dist/discovery/sources.js.map +1 -1
- package/dist/parsers/claude-session.d.ts +19 -0
- package/dist/parsers/claude-session.d.ts.map +1 -0
- package/dist/parsers/claude-session.js +131 -0
- package/dist/parsers/claude-session.js.map +1 -0
- package/dist/parsers/codex-session.d.ts +24 -0
- package/dist/parsers/codex-session.d.ts.map +1 -0
- package/dist/parsers/codex-session.js +140 -0
- package/dist/parsers/codex-session.js.map +1 -0
- package/dist/parsers/codex.d.ts +37 -0
- package/dist/parsers/codex.d.ts.map +1 -0
- package/dist/parsers/codex.js +136 -0
- package/dist/parsers/codex.js.map +1 -0
- package/dist/parsers/gemini-session.d.ts +19 -0
- package/dist/parsers/gemini-session.d.ts.map +1 -0
- package/dist/parsers/gemini-session.js +103 -0
- package/dist/parsers/gemini-session.js.map +1 -0
- package/dist/parsers/openclaw-session.d.ts +20 -0
- package/dist/parsers/openclaw-session.d.ts.map +1 -0
- package/dist/parsers/openclaw-session.js +122 -0
- package/dist/parsers/openclaw-session.js.map +1 -0
- package/dist/parsers/opencode-session.d.ts +15 -0
- package/dist/parsers/opencode-session.d.ts.map +1 -0
- package/dist/parsers/opencode-session.js +131 -0
- package/dist/parsers/opencode-session.js.map +1 -0
- package/dist/parsers/opencode-sqlite-db.d.ts +29 -0
- package/dist/parsers/opencode-sqlite-db.d.ts.map +1 -0
- package/dist/parsers/opencode-sqlite-db.js +71 -0
- package/dist/parsers/opencode-sqlite-db.js.map +1 -0
- package/dist/parsers/opencode-sqlite-session.d.ts +32 -0
- package/dist/parsers/opencode-sqlite-session.d.ts.map +1 -0
- package/dist/parsers/opencode-sqlite-session.js +121 -0
- package/dist/parsers/opencode-sqlite-session.js.map +1 -0
- package/dist/parsers/opencode-sqlite.d.ts +53 -0
- package/dist/parsers/opencode-sqlite.d.ts.map +1 -0
- package/dist/parsers/opencode-sqlite.js +104 -0
- package/dist/parsers/opencode-sqlite.js.map +1 -0
- package/dist/storage/session-cursor-store.d.ts +14 -0
- package/dist/storage/session-cursor-store.d.ts.map +1 -0
- package/dist/storage/session-cursor-store.js +34 -0
- package/dist/storage/session-cursor-store.js.map +1 -0
- package/dist/storage/session-queue.d.ts +28 -0
- package/dist/storage/session-queue.d.ts.map +1 -0
- package/dist/storage/session-queue.js +65 -0
- package/dist/storage/session-queue.js.map +1 -0
- package/dist/utils/paths.d.ts +4 -0
- package/dist/utils/paths.d.ts.map +1 -1
- package/dist/utils/paths.js +5 -0
- package/dist/utils/paths.js.map +1 -1
- package/package.json +1 -1
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* OpenCode SQLite session collector.
|
|
3
|
+
*
|
|
4
|
+
* Queries the `session` and `message` tables from opencode.db
|
|
5
|
+
* to produce SessionSnapshot records. Uses dependency injection
|
|
6
|
+
* (pre-fetched rows) for testability without bun:sqlite.
|
|
7
|
+
*/
|
|
8
|
+
/**
|
|
9
|
+
* Coerce an epoch value to milliseconds.
|
|
10
|
+
* Values < 1e12 are treated as seconds and multiplied by 1000.
|
|
11
|
+
*/
|
|
12
|
+
function coerceEpochMs(v) {
|
|
13
|
+
const n = Number(v);
|
|
14
|
+
if (!Number.isFinite(n) || n <= 0)
|
|
15
|
+
return 0;
|
|
16
|
+
if (n < 1e12)
|
|
17
|
+
return Math.floor(n * 1000);
|
|
18
|
+
return Math.floor(n);
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Collect session snapshots from pre-fetched SQLite rows.
|
|
22
|
+
*
|
|
23
|
+
* Groups messages by session, counts roles, tracks timestamps and model.
|
|
24
|
+
* Uses session table time_created/time_updated as fallback timestamps
|
|
25
|
+
* when message-level time is unavailable.
|
|
26
|
+
*/
|
|
27
|
+
export function collectOpenCodeSqliteSessions(sessions, messages) {
|
|
28
|
+
if (sessions.length === 0)
|
|
29
|
+
return [];
|
|
30
|
+
// Group messages by session_id
|
|
31
|
+
const msgMap = new Map();
|
|
32
|
+
for (const msg of messages) {
|
|
33
|
+
const list = msgMap.get(msg.session_id);
|
|
34
|
+
if (list) {
|
|
35
|
+
list.push(msg);
|
|
36
|
+
}
|
|
37
|
+
else {
|
|
38
|
+
msgMap.set(msg.session_id, [msg]);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
const snapshots = [];
|
|
42
|
+
for (const session of sessions) {
|
|
43
|
+
const sessionMessages = msgMap.get(session.id) ?? [];
|
|
44
|
+
let userMessages = 0;
|
|
45
|
+
let assistantMessages = 0;
|
|
46
|
+
const totalMessages = sessionMessages.length;
|
|
47
|
+
let minEpochMs = null;
|
|
48
|
+
let maxEpochMs = null;
|
|
49
|
+
let lastModel = null;
|
|
50
|
+
for (const msg of sessionMessages) {
|
|
51
|
+
// Count by role from the role column (faster than parsing JSON)
|
|
52
|
+
if (msg.role === "user") {
|
|
53
|
+
userMessages++;
|
|
54
|
+
}
|
|
55
|
+
else if (msg.role === "assistant") {
|
|
56
|
+
assistantMessages++;
|
|
57
|
+
}
|
|
58
|
+
// Parse data JSON for time and model extraction
|
|
59
|
+
let data = null;
|
|
60
|
+
try {
|
|
61
|
+
data = JSON.parse(msg.data);
|
|
62
|
+
}
|
|
63
|
+
catch {
|
|
64
|
+
// If data is corrupted, we still count the message but skip time/model
|
|
65
|
+
}
|
|
66
|
+
if (data) {
|
|
67
|
+
const time = data.time;
|
|
68
|
+
if (time) {
|
|
69
|
+
const completedMs = coerceEpochMs(time.completed);
|
|
70
|
+
const createdMs = coerceEpochMs(time.created);
|
|
71
|
+
const msgStart = createdMs || completedMs;
|
|
72
|
+
const msgEnd = completedMs || createdMs;
|
|
73
|
+
if (msgStart) {
|
|
74
|
+
if (!minEpochMs || msgStart < minEpochMs)
|
|
75
|
+
minEpochMs = msgStart;
|
|
76
|
+
}
|
|
77
|
+
if (msgEnd) {
|
|
78
|
+
if (!maxEpochMs || msgEnd > maxEpochMs)
|
|
79
|
+
maxEpochMs = msgEnd;
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
// Track model from assistant messages
|
|
83
|
+
const model = typeof data.modelID === "string"
|
|
84
|
+
? data.modelID.trim()
|
|
85
|
+
: typeof data.model === "string"
|
|
86
|
+
? data.model.trim()
|
|
87
|
+
: null;
|
|
88
|
+
if (model) {
|
|
89
|
+
lastModel = model;
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
// Fallback to session table timestamps if messages lack time
|
|
94
|
+
if (!minEpochMs) {
|
|
95
|
+
minEpochMs = coerceEpochMs(session.time_created) || null;
|
|
96
|
+
}
|
|
97
|
+
if (!maxEpochMs) {
|
|
98
|
+
maxEpochMs = coerceEpochMs(session.time_updated) || minEpochMs;
|
|
99
|
+
}
|
|
100
|
+
// Must have at least a start time
|
|
101
|
+
if (!minEpochMs)
|
|
102
|
+
continue;
|
|
103
|
+
const endMs = maxEpochMs ?? minEpochMs;
|
|
104
|
+
snapshots.push({
|
|
105
|
+
sessionKey: `opencode:${session.id}`,
|
|
106
|
+
source: "opencode",
|
|
107
|
+
kind: "human",
|
|
108
|
+
startedAt: new Date(minEpochMs).toISOString(),
|
|
109
|
+
lastMessageAt: new Date(endMs).toISOString(),
|
|
110
|
+
durationSeconds: Math.max(0, Math.floor((endMs - minEpochMs) / 1000)),
|
|
111
|
+
userMessages,
|
|
112
|
+
assistantMessages,
|
|
113
|
+
totalMessages,
|
|
114
|
+
projectRef: session.project_id ?? null,
|
|
115
|
+
model: lastModel,
|
|
116
|
+
snapshotAt: new Date().toISOString(),
|
|
117
|
+
});
|
|
118
|
+
}
|
|
119
|
+
return snapshots;
|
|
120
|
+
}
|
|
121
|
+
//# sourceMappingURL=opencode-sqlite-session.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"opencode-sqlite-session.js","sourceRoot":"","sources":["../../src/parsers/opencode-sqlite-session.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAqBH;;;GAGG;AACH,SAAS,aAAa,CAAC,CAAU;IAC/B,MAAM,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;IACpB,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC;QAAE,OAAO,CAAC,CAAC;IAC5C,IAAI,CAAC,GAAG,IAAI;QAAE,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC;IAC1C,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;AACvB,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,6BAA6B,CAC3C,QAAsB,EACtB,QAA6B;IAE7B,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,EAAE,CAAC;IAErC,+BAA+B;IAC/B,MAAM,MAAM,GAAG,IAAI,GAAG,EAA+B,CAAC;IACtD,KAAK,MAAM,GAAG,IAAI,QAAQ,EAAE,CAAC;QAC3B,MAAM,IAAI,GAAG,MAAM,CAAC,GAAG,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QACxC,IAAI,IAAI,EAAE,CAAC;YACT,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACjB,CAAC;aAAM,CAAC;YACN,MAAM,CAAC,GAAG,CAAC,GAAG,CAAC,UAAU,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;QACpC,CAAC;IACH,CAAC;IAED,MAAM,SAAS,GAAsB,EAAE,CAAC;IAExC,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;QAC/B,MAAM,eAAe,GAAG,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;QAErD,IAAI,YAAY,GAAG,CAAC,CAAC;QACrB,IAAI,iBAAiB,GAAG,CAAC,CAAC;QAC1B,MAAM,aAAa,GAAG,eAAe,CAAC,MAAM,CAAC;QAE7C,IAAI,UAAU,GAAkB,IAAI,CAAC;QACrC,IAAI,UAAU,GAAkB,IAAI,CAAC;QACrC,IAAI,SAAS,GAAkB,IAAI,CAAC;QAEpC,KAAK,MAAM,GAAG,IAAI,eAAe,EAAE,CAAC;YAClC,gEAAgE;YAChE,IAAI,GAAG,CAAC,IAAI,KAAK,MAAM,EAAE,CAAC;gBACxB,YAAY,EAAE,CAAC;YACjB,CAAC;iBAAM,IAAI,GAAG,CAAC,IAAI,KAAK,WAAW,EAAE,CAAC;gBACpC,iBAAiB,EAAE,CAAC;YACtB,CAAC;YAED,gDAAgD;YAChD,IAAI,IAAI,GAAmC,IAAI,CAAC;YAChD,IAAI,CAAC;gBACH,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YAC9B,CAAC;YAAC,MAAM,CAAC;gBACP,uEAAuE;YACzE,CAAC;YAED,IAAI,IAAI,EAAE,CAAC;gBACT,MAAM,IAAI,GAAG,IAAI,CAAC,IAA2C,CAAC;gBAC9D,IAAI,IAAI,EAAE,CAAC;oBACT,MAAM,WAAW,GAAG,aAAa,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;oBAClD,MAAM,SAAS,GAAG,aAAa,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;oBAE9C,MAAM,QAAQ,GAAG,SAAS,IAAI,WAAW,CAAC;oBAC1C,MAAM,MAAM,GAAG,WAAW,IAAI,SAAS,CAAC;oBAExC,IAAI,QAAQ,EAAE,CAAC;wBACb,IAAI,CAAC,UAAU,IAAI,QAAQ,GAAG,UAAU;4BAAE,UAAU,GAAG,QAAQ,CAAC;oBAClE,CAAC;oBACD,IAAI,MAAM,EAAE,CAAC;wBACX,IAAI,CAAC,UAAU,IAAI,MAAM,GAAG,UAAU;4BAAE,UAAU,GAAG,MAAM,CAAC;oBAC9D,CAAC;gBACH,CAAC;gBAED,sCAAsC;gBACtC,MAAM,KAAK,GACT,OAAO,IAAI,CAAC,OAAO,KAAK,QAAQ;oBAC9B,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE;oBACrB,CAAC,CAAC,OAAO,IAAI,CAAC,KAAK,KAAK,QAAQ;wBAC9B,CAAC,CAAE,IAAI,CAAC,KAAgB,CAAC,IAAI,EAAE;wBAC/B,CAAC,CAAC,IAAI,CAAC;gBACb,IAAI,KAAK,EAAE,CAAC;oBACV,SAAS,GAAG,KAAK,CAAC;gBACpB,CAAC;YACH,CAAC;QACH,CAAC;QAED,6DAA6D;QAC7D,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,UAAU,GAAG,aAAa,CAAC,OAAO,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC;QAC3D,CAAC;QACD,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,UAAU,GAAG,aAAa,CAAC,OAAO,CAAC,YAAY,CAAC,IAAI,UAAU,CAAC;QACjE,CAAC;QAED,kCAAkC;QAClC,IAAI,CAAC,UAAU;YAAE,SAAS;QAE1B,MAAM,KAAK,GAAG,UAAU,IAAI,UAAU,CAAC;QAEvC,SAAS,CAAC,IAAI,CAAC;YACb,UAAU,EAAE,YAAY,OAAO,CAAC,EAAE,EAAE;YACpC,MAAM,EAAE,UAAoB;YAC5B,IAAI,EAAE,OAAO;YACb,SAAS,EAAE,IAAI,IAAI,CAAC,UAAU,CAAC,CAAC,WAAW,EAAE;YAC7C,aAAa,EAAE,IAAI,IAAI,CAAC,KAAK,CAAC,CAAC,WAAW,EAAE;YAC5C,eAAe,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC,KAAK,GAAG,UAAU,CAAC,GAAG,IAAI,CAAC,CAAC;YACrE,YAAY;YACZ,iBAAiB;YACjB,aAAa;YACb,UAAU,EAAE,OAAO,CAAC,UAAU,IAAI,IAAI;YACtC,KAAK,EAAE,SAAS;YAChB,UAAU,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;SACrC,CAAC,CAAC;IACL,CAAC;IAED,OAAO,SAAS,CAAC;AACnB,CAAC"}
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import type { ParsedDelta } from "./claude.js";
|
|
2
|
+
/** Result of parsing OpenCode SQLite database */
|
|
3
|
+
export interface OpenCodeSqliteResult {
|
|
4
|
+
/** Parsed token deltas (one per assistant message) */
|
|
5
|
+
deltas: ParsedDelta[];
|
|
6
|
+
/** Message keys for dedup: "sessionId|messageId" */
|
|
7
|
+
messageKeys: Set<string>;
|
|
8
|
+
/** Highest time_created seen (for cursor advancement) */
|
|
9
|
+
maxTimeCreated: number;
|
|
10
|
+
/** DB file inode (for detecting replacement/recreation) */
|
|
11
|
+
inode: number;
|
|
12
|
+
}
|
|
13
|
+
/** Row shape from the message table */
|
|
14
|
+
export interface MessageRow {
|
|
15
|
+
id: string;
|
|
16
|
+
session_id: string;
|
|
17
|
+
time_created: number;
|
|
18
|
+
/** Extracted via json_extract(data, '$.role') at the SQL level */
|
|
19
|
+
role: string | null;
|
|
20
|
+
data: string;
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* Function that queries the message table.
|
|
24
|
+
* Accepts lastTimeCreated and returns rows where time_created >= lastTimeCreated.
|
|
25
|
+
* Callers must filter out previously-processed IDs from the prior batch
|
|
26
|
+
* to handle same-millisecond boundary dedup.
|
|
27
|
+
*/
|
|
28
|
+
export type QueryMessagesFn = (lastTimeCreated: number) => MessageRow[];
|
|
29
|
+
/**
|
|
30
|
+
* Parse message rows from OpenCode's SQLite database for token usage records.
|
|
31
|
+
*
|
|
32
|
+
* Processes rows from the `message` table where `time_created > lastTimeCreated`.
|
|
33
|
+
*
|
|
34
|
+
* Unlike the JSON file parser, no diffTotals is needed — each SQLite row
|
|
35
|
+
* is an independent message with absolute token values.
|
|
36
|
+
*
|
|
37
|
+
* The `queryMessages` function is injected to decouple from `bun:sqlite`
|
|
38
|
+
* for testability. Use `openMessageDb()` from `opencode-sqlite-db.ts` to
|
|
39
|
+
* create the real adapter at runtime.
|
|
40
|
+
*/
|
|
41
|
+
export declare function processOpenCodeMessages(rows: MessageRow[]): Omit<OpenCodeSqliteResult, "inode">;
|
|
42
|
+
/**
|
|
43
|
+
* High-level entry: parse OpenCode SQLite database for token usage.
|
|
44
|
+
*
|
|
45
|
+
* Opens the database, queries new messages, and returns parsed deltas.
|
|
46
|
+
* The `queryMessages` function provides the database access layer.
|
|
47
|
+
*/
|
|
48
|
+
export declare function parseOpenCodeSqlite(opts: {
|
|
49
|
+
dbPath: string;
|
|
50
|
+
lastTimeCreated: number;
|
|
51
|
+
queryMessages?: QueryMessagesFn;
|
|
52
|
+
}): Promise<OpenCodeSqliteResult>;
|
|
53
|
+
//# sourceMappingURL=opencode-sqlite.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"opencode-sqlite.d.ts","sourceRoot":"","sources":["../../src/parsers/opencode-sqlite.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAG/C,iDAAiD;AACjD,MAAM,WAAW,oBAAoB;IACnC,sDAAsD;IACtD,MAAM,EAAE,WAAW,EAAE,CAAC;IACtB,oDAAoD;IACpD,WAAW,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC;IACzB,yDAAyD;IACzD,cAAc,EAAE,MAAM,CAAC;IACvB,2DAA2D;IAC3D,KAAK,EAAE,MAAM,CAAC;CACf;AAED,uCAAuC;AACvC,MAAM,WAAW,UAAU;IACzB,EAAE,EAAE,MAAM,CAAC;IACX,UAAU,EAAE,MAAM,CAAC;IACnB,YAAY,EAAE,MAAM,CAAC;IACrB,kEAAkE;IAClE,IAAI,EAAE,MAAM,GAAG,IAAI,CAAC;IACpB,IAAI,EAAE,MAAM,CAAC;CACd;AAED;;;;;GAKG;AACH,MAAM,MAAM,eAAe,GAAG,CAAC,eAAe,EAAE,MAAM,KAAK,UAAU,EAAE,CAAC;AAiBxE;;;;;;;;;;;GAWG;AACH,wBAAgB,uBAAuB,CACrC,IAAI,EAAE,UAAU,EAAE,GACjB,IAAI,CAAC,oBAAoB,EAAE,OAAO,CAAC,CAsDrC;AAED;;;;;GAKG;AACH,wBAAsB,mBAAmB,CAAC,IAAI,EAAE;IAC9C,MAAM,EAAE,MAAM,CAAC;IACf,eAAe,EAAE,MAAM,CAAC;IACxB,aAAa,CAAC,EAAE,eAAe,CAAC;CACjC,GAAG,OAAO,CAAC,oBAAoB,CAAC,CAgChC"}
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
import { stat } from "node:fs/promises";
|
|
2
|
+
import { normalizeOpenCodeTokens, coerceEpochMs } from "./opencode.js";
|
|
3
|
+
/** Check if a TokenDelta is all zeros */
|
|
4
|
+
function isAllZero(d) {
|
|
5
|
+
return (d.inputTokens === 0 &&
|
|
6
|
+
d.cachedInputTokens === 0 &&
|
|
7
|
+
d.outputTokens === 0 &&
|
|
8
|
+
d.reasoningOutputTokens === 0);
|
|
9
|
+
}
|
|
10
|
+
/**
|
|
11
|
+
* Parse message rows from OpenCode's SQLite database for token usage records.
|
|
12
|
+
*
|
|
13
|
+
* Processes rows from the `message` table where `time_created > lastTimeCreated`.
|
|
14
|
+
*
|
|
15
|
+
* Unlike the JSON file parser, no diffTotals is needed — each SQLite row
|
|
16
|
+
* is an independent message with absolute token values.
|
|
17
|
+
*
|
|
18
|
+
* The `queryMessages` function is injected to decouple from `bun:sqlite`
|
|
19
|
+
* for testability. Use `openMessageDb()` from `opencode-sqlite-db.ts` to
|
|
20
|
+
* create the real adapter at runtime.
|
|
21
|
+
*/
|
|
22
|
+
export function processOpenCodeMessages(rows) {
|
|
23
|
+
const deltas = [];
|
|
24
|
+
const messageKeys = new Set();
|
|
25
|
+
let maxTimeCreated = 0;
|
|
26
|
+
for (const row of rows) {
|
|
27
|
+
// Track max time_created for cursor
|
|
28
|
+
if (row.time_created > maxTimeCreated) {
|
|
29
|
+
maxTimeCreated = row.time_created;
|
|
30
|
+
}
|
|
31
|
+
// Only process assistant messages (role is pre-extracted at SQL level)
|
|
32
|
+
if (row.role !== "assistant")
|
|
33
|
+
continue;
|
|
34
|
+
// Parse data JSON
|
|
35
|
+
let msg;
|
|
36
|
+
try {
|
|
37
|
+
msg = JSON.parse(row.data);
|
|
38
|
+
}
|
|
39
|
+
catch {
|
|
40
|
+
continue;
|
|
41
|
+
}
|
|
42
|
+
// Build message key for dedup
|
|
43
|
+
messageKeys.add(`${row.session_id}|${row.id}`);
|
|
44
|
+
// Normalize tokens
|
|
45
|
+
const tokens = normalizeOpenCodeTokens(msg.tokens);
|
|
46
|
+
if (!tokens || isAllZero(tokens))
|
|
47
|
+
continue;
|
|
48
|
+
// Extract timestamp from time.completed or time.created
|
|
49
|
+
const time = msg.time;
|
|
50
|
+
const timestampMs = coerceEpochMs(time?.completed) || coerceEpochMs(time?.created);
|
|
51
|
+
if (!timestampMs)
|
|
52
|
+
continue;
|
|
53
|
+
// Extract model
|
|
54
|
+
const model = typeof msg.modelID === "string"
|
|
55
|
+
? msg.modelID.trim()
|
|
56
|
+
: typeof msg.model === "string"
|
|
57
|
+
? msg.model.trim()
|
|
58
|
+
: "unknown";
|
|
59
|
+
deltas.push({
|
|
60
|
+
source: "opencode",
|
|
61
|
+
model,
|
|
62
|
+
timestamp: new Date(timestampMs).toISOString(),
|
|
63
|
+
tokens,
|
|
64
|
+
});
|
|
65
|
+
}
|
|
66
|
+
return { deltas, messageKeys, maxTimeCreated };
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* High-level entry: parse OpenCode SQLite database for token usage.
|
|
70
|
+
*
|
|
71
|
+
* Opens the database, queries new messages, and returns parsed deltas.
|
|
72
|
+
* The `queryMessages` function provides the database access layer.
|
|
73
|
+
*/
|
|
74
|
+
export async function parseOpenCodeSqlite(opts) {
|
|
75
|
+
const { dbPath, lastTimeCreated, queryMessages } = opts;
|
|
76
|
+
const empty = {
|
|
77
|
+
deltas: [],
|
|
78
|
+
messageKeys: new Set(),
|
|
79
|
+
maxTimeCreated: 0,
|
|
80
|
+
inode: 0,
|
|
81
|
+
};
|
|
82
|
+
// Check if database file exists and get inode
|
|
83
|
+
let fileInode;
|
|
84
|
+
try {
|
|
85
|
+
const st = await stat(dbPath);
|
|
86
|
+
fileInode = st.ino;
|
|
87
|
+
}
|
|
88
|
+
catch {
|
|
89
|
+
return empty;
|
|
90
|
+
}
|
|
91
|
+
if (!queryMessages) {
|
|
92
|
+
return empty;
|
|
93
|
+
}
|
|
94
|
+
let rows;
|
|
95
|
+
try {
|
|
96
|
+
rows = queryMessages(lastTimeCreated);
|
|
97
|
+
}
|
|
98
|
+
catch {
|
|
99
|
+
return empty;
|
|
100
|
+
}
|
|
101
|
+
const result = processOpenCodeMessages(rows);
|
|
102
|
+
return { ...result, inode: fileInode };
|
|
103
|
+
}
|
|
104
|
+
//# sourceMappingURL=opencode-sqlite.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"opencode-sqlite.js","sourceRoot":"","sources":["../../src/parsers/opencode-sqlite.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
import type { SessionCursorState } from "@pew/core";
|
|
2
|
+
/**
|
|
3
|
+
* Persists session file cursors (mtime + size dual-check) to disk.
|
|
4
|
+
* Stored at ~/.config/pew/session-cursors.json
|
|
5
|
+
*/
|
|
6
|
+
export declare class SessionCursorStore {
|
|
7
|
+
readonly filePath: string;
|
|
8
|
+
constructor(storeDir: string);
|
|
9
|
+
/** Load cursor state from disk. Returns empty state if file doesn't exist or is corrupted. */
|
|
10
|
+
load(): Promise<SessionCursorState>;
|
|
11
|
+
/** Save cursor state to disk, creating the directory if needed. */
|
|
12
|
+
save(state: SessionCursorState): Promise<void>;
|
|
13
|
+
}
|
|
14
|
+
//# sourceMappingURL=session-cursor-store.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"session-cursor-store.d.ts","sourceRoot":"","sources":["../../src/storage/session-cursor-store.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,WAAW,CAAC;AASpD;;;GAGG;AACH,qBAAa,kBAAkB;IAC7B,QAAQ,CAAC,QAAQ,EAAE,MAAM,CAAC;gBAEd,QAAQ,EAAE,MAAM;IAI5B,8FAA8F;IACxF,IAAI,IAAI,OAAO,CAAC,kBAAkB,CAAC;IASzC,mEAAmE;IAC7D,IAAI,CAAC,KAAK,EAAE,kBAAkB,GAAG,OAAO,CAAC,IAAI,CAAC;CAKrD"}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import { readFile, writeFile, mkdir } from "node:fs/promises";
|
|
2
|
+
import { join, dirname } from "node:path";
|
|
3
|
+
const CURSORS_FILE = "session-cursors.json";
|
|
4
|
+
/** Creates a fresh empty session cursor state */
|
|
5
|
+
function emptyCursorState() {
|
|
6
|
+
return { version: 1, files: {}, updatedAt: null };
|
|
7
|
+
}
|
|
8
|
+
/**
|
|
9
|
+
* Persists session file cursors (mtime + size dual-check) to disk.
|
|
10
|
+
* Stored at ~/.config/pew/session-cursors.json
|
|
11
|
+
*/
|
|
12
|
+
export class SessionCursorStore {
|
|
13
|
+
filePath;
|
|
14
|
+
constructor(storeDir) {
|
|
15
|
+
this.filePath = join(storeDir, CURSORS_FILE);
|
|
16
|
+
}
|
|
17
|
+
/** Load cursor state from disk. Returns empty state if file doesn't exist or is corrupted. */
|
|
18
|
+
async load() {
|
|
19
|
+
try {
|
|
20
|
+
const raw = await readFile(this.filePath, "utf-8");
|
|
21
|
+
return JSON.parse(raw);
|
|
22
|
+
}
|
|
23
|
+
catch {
|
|
24
|
+
return emptyCursorState();
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
/** Save cursor state to disk, creating the directory if needed. */
|
|
28
|
+
async save(state) {
|
|
29
|
+
const dir = dirname(this.filePath);
|
|
30
|
+
await mkdir(dir, { recursive: true });
|
|
31
|
+
await writeFile(this.filePath, JSON.stringify(state, null, 2) + "\n");
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
//# sourceMappingURL=session-cursor-store.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"session-cursor-store.js","sourceRoot":"","sources":["../../src/storage/session-cursor-store.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,kBAAkB,CAAC;AAC9D,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAG1C,MAAM,YAAY,GAAG,sBAAsB,CAAC;AAE5C,iDAAiD;AACjD,SAAS,gBAAgB;IACvB,OAAO,EAAE,OAAO,EAAE,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC;AACpD,CAAC;AAED;;;GAGG;AACH,MAAM,OAAO,kBAAkB;IACpB,QAAQ,CAAS;IAE1B,YAAY,QAAgB;QAC1B,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,EAAE,YAAY,CAAC,CAAC;IAC/C,CAAC;IAED,8FAA8F;IAC9F,KAAK,CAAC,IAAI;QACR,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,MAAM,QAAQ,CAAC,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;YACnD,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAuB,CAAC;QAC/C,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,gBAAgB,EAAE,CAAC;QAC5B,CAAC;IACH,CAAC;IAED,mEAAmE;IACnE,KAAK,CAAC,IAAI,CAAC,KAAyB;QAClC,MAAM,GAAG,GAAG,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QACnC,MAAM,KAAK,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QACtC,MAAM,SAAS,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC;IACxE,CAAC;CACF"}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import type { SessionQueueRecord } from "@pew/core";
|
|
2
|
+
/**
|
|
3
|
+
* Append-only local queue for session records.
|
|
4
|
+
* Mirrors LocalQueue but uses separate files and SessionQueueRecord type.
|
|
5
|
+
*/
|
|
6
|
+
export declare class SessionQueue {
|
|
7
|
+
readonly queuePath: string;
|
|
8
|
+
private readonly statePath;
|
|
9
|
+
private readonly dir;
|
|
10
|
+
constructor(storeDir: string);
|
|
11
|
+
/** Ensure the directory exists */
|
|
12
|
+
private ensureDir;
|
|
13
|
+
/** Append multiple records to the queue in a single write */
|
|
14
|
+
appendBatch(records: SessionQueueRecord[]): Promise<void>;
|
|
15
|
+
/**
|
|
16
|
+
* Read records from the queue starting at a byte offset.
|
|
17
|
+
* Returns parsed records and the new offset (end of file).
|
|
18
|
+
*/
|
|
19
|
+
readFromOffset(offset: number): Promise<{
|
|
20
|
+
records: SessionQueueRecord[];
|
|
21
|
+
newOffset: number;
|
|
22
|
+
}>;
|
|
23
|
+
/** Save the upload byte offset to the state file */
|
|
24
|
+
saveOffset(offset: number): Promise<void>;
|
|
25
|
+
/** Load the upload byte offset. Returns 0 if not found or corrupted. */
|
|
26
|
+
loadOffset(): Promise<number>;
|
|
27
|
+
}
|
|
28
|
+
//# sourceMappingURL=session-queue.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"session-queue.d.ts","sourceRoot":"","sources":["../../src/storage/session-queue.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,WAAW,CAAC;AAKpD;;;GAGG;AACH,qBAAa,YAAY;IACvB,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC;IAC3B,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAS;IACnC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAS;gBAEjB,QAAQ,EAAE,MAAM;IAM5B,kCAAkC;YACpB,SAAS;IAIvB,6DAA6D;IACvD,WAAW,CAAC,OAAO,EAAE,kBAAkB,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;IAO/D;;;OAGG;IACG,cAAc,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC;QAC5C,OAAO,EAAE,kBAAkB,EAAE,CAAC;QAC9B,SAAS,EAAE,MAAM,CAAC;KACnB,CAAC;IAkBF,oDAAoD;IAC9C,UAAU,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAK/C,wEAAwE;IAClE,UAAU,IAAI,OAAO,CAAC,MAAM,CAAC;CASpC"}
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import { readFile, writeFile, appendFile, mkdir } from "node:fs/promises";
|
|
2
|
+
import { join } from "node:path";
|
|
3
|
+
const QUEUE_FILE = "session-queue.jsonl";
|
|
4
|
+
const STATE_FILE = "session-queue.state.json";
|
|
5
|
+
/**
|
|
6
|
+
* Append-only local queue for session records.
|
|
7
|
+
* Mirrors LocalQueue but uses separate files and SessionQueueRecord type.
|
|
8
|
+
*/
|
|
9
|
+
export class SessionQueue {
|
|
10
|
+
queuePath;
|
|
11
|
+
statePath;
|
|
12
|
+
dir;
|
|
13
|
+
constructor(storeDir) {
|
|
14
|
+
this.dir = storeDir;
|
|
15
|
+
this.queuePath = join(storeDir, QUEUE_FILE);
|
|
16
|
+
this.statePath = join(storeDir, STATE_FILE);
|
|
17
|
+
}
|
|
18
|
+
/** Ensure the directory exists */
|
|
19
|
+
async ensureDir() {
|
|
20
|
+
await mkdir(this.dir, { recursive: true });
|
|
21
|
+
}
|
|
22
|
+
/** Append multiple records to the queue in a single write */
|
|
23
|
+
async appendBatch(records) {
|
|
24
|
+
if (records.length === 0)
|
|
25
|
+
return;
|
|
26
|
+
await this.ensureDir();
|
|
27
|
+
const data = records.map((r) => JSON.stringify(r)).join("\n") + "\n";
|
|
28
|
+
await appendFile(this.queuePath, data);
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Read records from the queue starting at a byte offset.
|
|
32
|
+
* Returns parsed records and the new offset (end of file).
|
|
33
|
+
*/
|
|
34
|
+
async readFromOffset(offset) {
|
|
35
|
+
let raw;
|
|
36
|
+
try {
|
|
37
|
+
raw = await readFile(this.queuePath, "utf-8");
|
|
38
|
+
}
|
|
39
|
+
catch {
|
|
40
|
+
return { records: [], newOffset: 0 };
|
|
41
|
+
}
|
|
42
|
+
const slice = raw.slice(offset);
|
|
43
|
+
const lines = slice.split("\n").filter((line) => line.trim().length > 0);
|
|
44
|
+
const records = lines.map((line) => JSON.parse(line));
|
|
45
|
+
const newOffset = Buffer.byteLength(raw, "utf-8");
|
|
46
|
+
return { records, newOffset };
|
|
47
|
+
}
|
|
48
|
+
/** Save the upload byte offset to the state file */
|
|
49
|
+
async saveOffset(offset) {
|
|
50
|
+
await this.ensureDir();
|
|
51
|
+
await writeFile(this.statePath, JSON.stringify({ offset }) + "\n");
|
|
52
|
+
}
|
|
53
|
+
/** Load the upload byte offset. Returns 0 if not found or corrupted. */
|
|
54
|
+
async loadOffset() {
|
|
55
|
+
try {
|
|
56
|
+
const raw = await readFile(this.statePath, "utf-8");
|
|
57
|
+
const state = JSON.parse(raw);
|
|
58
|
+
return state.offset ?? 0;
|
|
59
|
+
}
|
|
60
|
+
catch {
|
|
61
|
+
return 0;
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
//# sourceMappingURL=session-queue.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"session-queue.js","sourceRoot":"","sources":["../../src/storage/session-queue.ts"],"names":[],"mappings":"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"}
|
package/dist/utils/paths.d.ts
CHANGED
|
@@ -7,10 +7,14 @@ export declare function resolveDefaultPaths(home?: string): {
|
|
|
7
7
|
stateDir: string;
|
|
8
8
|
/** Claude Code data: ~/.claude */
|
|
9
9
|
claudeDir: string;
|
|
10
|
+
/** Codex CLI sessions: ~/.codex/sessions (or $CODEX_HOME/sessions) */
|
|
11
|
+
codexSessionsDir: string;
|
|
10
12
|
/** Gemini CLI data: ~/.gemini */
|
|
11
13
|
geminiDir: string;
|
|
12
14
|
/** OpenCode message storage: ~/.local/share/opencode/storage/message */
|
|
13
15
|
openCodeMessageDir: string;
|
|
16
|
+
/** OpenCode SQLite database: ~/.local/share/opencode/opencode.db */
|
|
17
|
+
openCodeDbPath: string;
|
|
14
18
|
/** OpenClaw data: ~/.openclaw */
|
|
15
19
|
openclawDir: string;
|
|
16
20
|
};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"paths.d.ts","sourceRoot":"","sources":["../../src/utils/paths.ts"],"names":[],"mappings":"AAGA;;;GAGG;AACH,wBAAgB,mBAAmB,CAAC,IAAI,SAAY;
|
|
1
|
+
{"version":3,"file":"paths.d.ts","sourceRoot":"","sources":["../../src/utils/paths.ts"],"names":[],"mappings":"AAGA;;;GAGG;AACH,wBAAgB,mBAAmB,CAAC,IAAI,SAAY;IAGhD,0CAA0C;;IAE1C,kCAAkC;;IAElC,sEAAsE;;IAEtE,iCAAiC;;IAEjC,wEAAwE;;IASxE,oEAAoE;;IAEpE,iCAAiC;;EAGpC"}
|
package/dist/utils/paths.js
CHANGED
|
@@ -5,15 +5,20 @@ import { join } from "node:path";
|
|
|
5
5
|
* All paths can be overridden for testing.
|
|
6
6
|
*/
|
|
7
7
|
export function resolveDefaultPaths(home = homedir()) {
|
|
8
|
+
const codexHome = process.env.CODEX_HOME || join(home, ".codex");
|
|
8
9
|
return {
|
|
9
10
|
/** Pew state directory: ~/.config/pew/ */
|
|
10
11
|
stateDir: join(home, ".config", "pew"),
|
|
11
12
|
/** Claude Code data: ~/.claude */
|
|
12
13
|
claudeDir: join(home, ".claude"),
|
|
14
|
+
/** Codex CLI sessions: ~/.codex/sessions (or $CODEX_HOME/sessions) */
|
|
15
|
+
codexSessionsDir: join(codexHome, "sessions"),
|
|
13
16
|
/** Gemini CLI data: ~/.gemini */
|
|
14
17
|
geminiDir: join(home, ".gemini"),
|
|
15
18
|
/** OpenCode message storage: ~/.local/share/opencode/storage/message */
|
|
16
19
|
openCodeMessageDir: join(home, ".local", "share", "opencode", "storage", "message"),
|
|
20
|
+
/** OpenCode SQLite database: ~/.local/share/opencode/opencode.db */
|
|
21
|
+
openCodeDbPath: join(home, ".local", "share", "opencode", "opencode.db"),
|
|
17
22
|
/** OpenClaw data: ~/.openclaw */
|
|
18
23
|
openclawDir: join(home, ".openclaw"),
|
|
19
24
|
};
|
package/dist/utils/paths.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"paths.js","sourceRoot":"","sources":["../../src/utils/paths.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAEjC;;;GAGG;AACH,MAAM,UAAU,mBAAmB,CAAC,IAAI,GAAG,OAAO,EAAE;IAClD,OAAO;QACL,0CAA0C;QAC1C,QAAQ,EAAE,IAAI,CAAC,IAAI,EAAE,SAAS,EAAE,KAAK,CAAC;QACtC,kCAAkC;QAClC,SAAS,EAAE,IAAI,CAAC,IAAI,EAAE,SAAS,CAAC;QAChC,iCAAiC;QACjC,SAAS,EAAE,IAAI,CAAC,IAAI,EAAE,SAAS,CAAC;QAChC,wEAAwE;QACxE,kBAAkB,EAAE,IAAI,CACtB,IAAI,EACJ,QAAQ,EACR,OAAO,EACP,UAAU,EACV,SAAS,EACT,SAAS,CACV;QACD,iCAAiC;QACjC,WAAW,EAAE,IAAI,CAAC,IAAI,EAAE,WAAW,CAAC;KACrC,CAAC;AACJ,CAAC"}
|
|
1
|
+
{"version":3,"file":"paths.js","sourceRoot":"","sources":["../../src/utils/paths.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAClC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAEjC;;;GAGG;AACH,MAAM,UAAU,mBAAmB,CAAC,IAAI,GAAG,OAAO,EAAE;IAClD,MAAM,SAAS,GAAG,OAAO,CAAC,GAAG,CAAC,UAAU,IAAI,IAAI,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;IACjE,OAAO;QACL,0CAA0C;QAC1C,QAAQ,EAAE,IAAI,CAAC,IAAI,EAAE,SAAS,EAAE,KAAK,CAAC;QACtC,kCAAkC;QAClC,SAAS,EAAE,IAAI,CAAC,IAAI,EAAE,SAAS,CAAC;QAChC,sEAAsE;QACtE,gBAAgB,EAAE,IAAI,CAAC,SAAS,EAAE,UAAU,CAAC;QAC7C,iCAAiC;QACjC,SAAS,EAAE,IAAI,CAAC,IAAI,EAAE,SAAS,CAAC;QAChC,wEAAwE;QACxE,kBAAkB,EAAE,IAAI,CACtB,IAAI,EACJ,QAAQ,EACR,OAAO,EACP,UAAU,EACV,SAAS,EACT,SAAS,CACV;QACD,oEAAoE;QACpE,cAAc,EAAE,IAAI,CAAC,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,UAAU,EAAE,aAAa,CAAC;QACxE,iCAAiC;QACjC,WAAW,EAAE,IAAI,CAAC,IAAI,EAAE,WAAW,CAAC;KACrC,CAAC;AACJ,CAAC"}
|