@fml-inc/panopticon 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (124) hide show
  1. package/.claude-plugin/plugin.json +10 -0
  2. package/LICENSE +5 -0
  3. package/README.md +363 -0
  4. package/bin/hook-handler +3 -0
  5. package/bin/mcp-server +3 -0
  6. package/bin/panopticon +3 -0
  7. package/bin/proxy +3 -0
  8. package/bin/server +3 -0
  9. package/dist/api/client.d.ts +67 -0
  10. package/dist/api/client.js +48 -0
  11. package/dist/api/client.js.map +1 -0
  12. package/dist/chunk-3BUJ7URA.js +387 -0
  13. package/dist/chunk-3BUJ7URA.js.map +1 -0
  14. package/dist/chunk-3TZAKV3M.js +158 -0
  15. package/dist/chunk-3TZAKV3M.js.map +1 -0
  16. package/dist/chunk-4SM2H22C.js +169 -0
  17. package/dist/chunk-4SM2H22C.js.map +1 -0
  18. package/dist/chunk-7Q3BJMLG.js +62 -0
  19. package/dist/chunk-7Q3BJMLG.js.map +1 -0
  20. package/dist/chunk-BVOE7A2Z.js +412 -0
  21. package/dist/chunk-BVOE7A2Z.js.map +1 -0
  22. package/dist/chunk-CF4GPWLI.js +170 -0
  23. package/dist/chunk-CF4GPWLI.js.map +1 -0
  24. package/dist/chunk-DZ5HJFB4.js +467 -0
  25. package/dist/chunk-DZ5HJFB4.js.map +1 -0
  26. package/dist/chunk-HQCY722C.js +428 -0
  27. package/dist/chunk-HQCY722C.js.map +1 -0
  28. package/dist/chunk-HRCEIYKU.js +134 -0
  29. package/dist/chunk-HRCEIYKU.js.map +1 -0
  30. package/dist/chunk-K7YUPLES.js +76 -0
  31. package/dist/chunk-K7YUPLES.js.map +1 -0
  32. package/dist/chunk-L7G27XWF.js +130 -0
  33. package/dist/chunk-L7G27XWF.js.map +1 -0
  34. package/dist/chunk-LWXF7YRG.js +626 -0
  35. package/dist/chunk-LWXF7YRG.js.map +1 -0
  36. package/dist/chunk-NXH7AONS.js +1120 -0
  37. package/dist/chunk-NXH7AONS.js.map +1 -0
  38. package/dist/chunk-QK5442ZP.js +55 -0
  39. package/dist/chunk-QK5442ZP.js.map +1 -0
  40. package/dist/chunk-QVK6VGCV.js +1703 -0
  41. package/dist/chunk-QVK6VGCV.js.map +1 -0
  42. package/dist/chunk-RX2RXHBH.js +1699 -0
  43. package/dist/chunk-RX2RXHBH.js.map +1 -0
  44. package/dist/chunk-SEXU2WYG.js +788 -0
  45. package/dist/chunk-SEXU2WYG.js.map +1 -0
  46. package/dist/chunk-SUGSQ4YI.js +264 -0
  47. package/dist/chunk-SUGSQ4YI.js.map +1 -0
  48. package/dist/chunk-TGXFVAID.js +138 -0
  49. package/dist/chunk-TGXFVAID.js.map +1 -0
  50. package/dist/chunk-WLBNFVIG.js +447 -0
  51. package/dist/chunk-WLBNFVIG.js.map +1 -0
  52. package/dist/chunk-XLTCUH5A.js +1072 -0
  53. package/dist/chunk-XLTCUH5A.js.map +1 -0
  54. package/dist/chunk-YVRWVDIA.js +146 -0
  55. package/dist/chunk-YVRWVDIA.js.map +1 -0
  56. package/dist/chunk-ZEC4LRKS.js +176 -0
  57. package/dist/chunk-ZEC4LRKS.js.map +1 -0
  58. package/dist/cli.d.ts +1 -0
  59. package/dist/cli.js +1084 -0
  60. package/dist/cli.js.map +1 -0
  61. package/dist/config-NwoZC-GM.d.ts +20 -0
  62. package/dist/db.d.ts +46 -0
  63. package/dist/db.js +15 -0
  64. package/dist/db.js.map +1 -0
  65. package/dist/doctor.d.ts +37 -0
  66. package/dist/doctor.js +14 -0
  67. package/dist/doctor.js.map +1 -0
  68. package/dist/hooks/handler.d.ts +23 -0
  69. package/dist/hooks/handler.js +295 -0
  70. package/dist/hooks/handler.js.map +1 -0
  71. package/dist/index.d.ts +57 -0
  72. package/dist/index.js +101 -0
  73. package/dist/index.js.map +1 -0
  74. package/dist/mcp/server.d.ts +1 -0
  75. package/dist/mcp/server.js +243 -0
  76. package/dist/mcp/server.js.map +1 -0
  77. package/dist/otlp/server.d.ts +7 -0
  78. package/dist/otlp/server.js +17 -0
  79. package/dist/otlp/server.js.map +1 -0
  80. package/dist/permissions.d.ts +33 -0
  81. package/dist/permissions.js +14 -0
  82. package/dist/permissions.js.map +1 -0
  83. package/dist/pricing.d.ts +29 -0
  84. package/dist/pricing.js +13 -0
  85. package/dist/pricing.js.map +1 -0
  86. package/dist/proxy/server.d.ts +10 -0
  87. package/dist/proxy/server.js +20 -0
  88. package/dist/proxy/server.js.map +1 -0
  89. package/dist/prune.d.ts +18 -0
  90. package/dist/prune.js +13 -0
  91. package/dist/prune.js.map +1 -0
  92. package/dist/query.d.ts +56 -0
  93. package/dist/query.js +27 -0
  94. package/dist/query.js.map +1 -0
  95. package/dist/reparse-636YZCE3.js +14 -0
  96. package/dist/reparse-636YZCE3.js.map +1 -0
  97. package/dist/repo.d.ts +17 -0
  98. package/dist/repo.js +9 -0
  99. package/dist/repo.js.map +1 -0
  100. package/dist/scanner.d.ts +73 -0
  101. package/dist/scanner.js +15 -0
  102. package/dist/scanner.js.map +1 -0
  103. package/dist/sdk.d.ts +82 -0
  104. package/dist/sdk.js +208 -0
  105. package/dist/sdk.js.map +1 -0
  106. package/dist/server.d.ts +5 -0
  107. package/dist/server.js +25 -0
  108. package/dist/server.js.map +1 -0
  109. package/dist/setup.d.ts +35 -0
  110. package/dist/setup.js +19 -0
  111. package/dist/setup.js.map +1 -0
  112. package/dist/sync/index.d.ts +29 -0
  113. package/dist/sync/index.js +32 -0
  114. package/dist/sync/index.js.map +1 -0
  115. package/dist/targets.d.ts +279 -0
  116. package/dist/targets.js +20 -0
  117. package/dist/targets.js.map +1 -0
  118. package/dist/types-D-MYCBol.d.ts +128 -0
  119. package/dist/types.d.ts +164 -0
  120. package/dist/types.js +1 -0
  121. package/dist/types.js.map +1 -0
  122. package/hooks/hooks.json +274 -0
  123. package/package.json +124 -0
  124. package/skills/panopticon-optimize/SKILL.md +222 -0
@@ -0,0 +1,467 @@
1
+ import {
2
+ config
3
+ } from "./chunk-K7YUPLES.js";
4
+
5
+ // src/db/schema.ts
6
+ import fs from "fs";
7
+ import { gunzipSync } from "zlib";
8
+ import Database from "better-sqlite3";
9
+ var SCHEMA_SQL = `
10
+
11
+ -- \u2500\u2500 OTLP tables \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
12
+
13
+ CREATE TABLE IF NOT EXISTS otel_logs (
14
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
15
+ timestamp_ns INTEGER NOT NULL,
16
+ observed_timestamp_ns INTEGER,
17
+ severity_number INTEGER,
18
+ severity_text TEXT,
19
+ body TEXT,
20
+ attributes JSON,
21
+ resource_attributes JSON,
22
+ session_id TEXT,
23
+ prompt_id TEXT,
24
+ trace_id TEXT,
25
+ span_id TEXT,
26
+ sync_id TEXT DEFAULT (hex(randomblob(8)))
27
+ );
28
+
29
+ CREATE TABLE IF NOT EXISTS otel_metrics (
30
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
31
+ timestamp_ns INTEGER NOT NULL,
32
+ name TEXT NOT NULL,
33
+ value REAL NOT NULL,
34
+ metric_type TEXT,
35
+ unit TEXT,
36
+ attributes JSON,
37
+ resource_attributes JSON,
38
+ session_id TEXT,
39
+ sync_id TEXT DEFAULT (hex(randomblob(8)))
40
+ );
41
+
42
+ CREATE TABLE IF NOT EXISTS otel_spans (
43
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
44
+ trace_id TEXT NOT NULL,
45
+ span_id TEXT NOT NULL,
46
+ parent_span_id TEXT,
47
+ name TEXT NOT NULL,
48
+ kind INTEGER,
49
+ start_time_ns INTEGER NOT NULL,
50
+ end_time_ns INTEGER NOT NULL,
51
+ status_code INTEGER,
52
+ status_message TEXT,
53
+ attributes JSON,
54
+ resource_attributes JSON,
55
+ session_id TEXT,
56
+ UNIQUE(trace_id, span_id)
57
+ );
58
+
59
+ -- \u2500\u2500 Hook events \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
60
+
61
+ CREATE TABLE IF NOT EXISTS hook_events (
62
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
63
+ session_id TEXT NOT NULL,
64
+ event_type TEXT NOT NULL,
65
+ timestamp_ms INTEGER NOT NULL,
66
+ cwd TEXT,
67
+ repository TEXT,
68
+ tool_name TEXT,
69
+ payload BLOB NOT NULL,
70
+ user_prompt TEXT,
71
+ file_path TEXT,
72
+ command TEXT,
73
+ plan TEXT,
74
+ allowed_prompts TEXT,
75
+ tool_result TEXT,
76
+ target TEXT,
77
+ sync_id TEXT DEFAULT (hex(randomblob(8)))
78
+ );
79
+
80
+ CREATE VIRTUAL TABLE IF NOT EXISTS hook_events_fts USING fts5(
81
+ payload,
82
+ content='',
83
+ contentless_delete=1,
84
+ tokenize='trigram'
85
+ );
86
+
87
+ -- \u2500\u2500 Sessions \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
88
+
89
+ CREATE TABLE IF NOT EXISTS sessions (
90
+ session_id TEXT PRIMARY KEY,
91
+ target TEXT,
92
+ started_at_ms INTEGER,
93
+ ended_at_ms INTEGER,
94
+ cwd TEXT,
95
+ first_prompt TEXT,
96
+ permission_mode TEXT,
97
+ agent_version TEXT,
98
+ model TEXT,
99
+ cli_version TEXT,
100
+ scanner_file_path TEXT,
101
+ total_input_tokens INTEGER DEFAULT 0,
102
+ total_output_tokens INTEGER DEFAULT 0,
103
+ total_cache_read_tokens INTEGER DEFAULT 0,
104
+ total_cache_creation_tokens INTEGER DEFAULT 0,
105
+ total_reasoning_tokens INTEGER DEFAULT 0,
106
+ turn_count INTEGER DEFAULT 0,
107
+ otel_input_tokens INTEGER DEFAULT 0,
108
+ otel_output_tokens INTEGER DEFAULT 0,
109
+ otel_cache_read_tokens INTEGER DEFAULT 0,
110
+ otel_cache_creation_tokens INTEGER DEFAULT 0,
111
+ models TEXT,
112
+ has_hooks INTEGER DEFAULT 0,
113
+ has_otel INTEGER DEFAULT 0,
114
+ has_scanner INTEGER DEFAULT 0,
115
+ summary TEXT,
116
+ summary_version INTEGER DEFAULT 0,
117
+ sync_dirty INTEGER DEFAULT 0,
118
+ sync_seq INTEGER DEFAULT 0,
119
+ tool_counts JSON DEFAULT '{}',
120
+ hook_tool_counts JSON DEFAULT '{}',
121
+ event_type_counts JSON DEFAULT '{}',
122
+ hook_event_type_counts JSON DEFAULT '{}',
123
+ project TEXT,
124
+ machine TEXT NOT NULL DEFAULT 'local',
125
+ message_count INTEGER DEFAULT 0,
126
+ user_message_count INTEGER DEFAULT 0,
127
+ parent_session_id TEXT,
128
+ relationship_type TEXT DEFAULT '',
129
+ is_automated INTEGER DEFAULT 0,
130
+ created_at INTEGER
131
+ );
132
+
133
+ CREATE TABLE IF NOT EXISTS session_repositories (
134
+ session_id TEXT NOT NULL,
135
+ repository TEXT NOT NULL,
136
+ first_seen_ms INTEGER NOT NULL,
137
+ git_user_name TEXT,
138
+ git_user_email TEXT,
139
+ branch TEXT,
140
+ UNIQUE(session_id, repository)
141
+ );
142
+
143
+ CREATE TABLE IF NOT EXISTS session_cwds (
144
+ session_id TEXT NOT NULL,
145
+ cwd TEXT NOT NULL,
146
+ first_seen_ms INTEGER NOT NULL,
147
+ UNIQUE(session_id, cwd)
148
+ );
149
+
150
+ -- \u2500\u2500 Messages \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
151
+
152
+ CREATE TABLE IF NOT EXISTS messages (
153
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
154
+ session_id TEXT NOT NULL,
155
+ ordinal INTEGER NOT NULL,
156
+ role TEXT NOT NULL,
157
+ content TEXT NOT NULL,
158
+ timestamp_ms INTEGER,
159
+ has_thinking INTEGER NOT NULL DEFAULT 0,
160
+ has_tool_use INTEGER NOT NULL DEFAULT 0,
161
+ content_length INTEGER NOT NULL DEFAULT 0,
162
+ is_system INTEGER NOT NULL DEFAULT 0,
163
+ model TEXT NOT NULL DEFAULT '',
164
+ token_usage TEXT NOT NULL DEFAULT '',
165
+ context_tokens INTEGER NOT NULL DEFAULT 0,
166
+ output_tokens INTEGER NOT NULL DEFAULT 0,
167
+ has_context_tokens INTEGER NOT NULL DEFAULT 0,
168
+ has_output_tokens INTEGER NOT NULL DEFAULT 0,
169
+ uuid TEXT,
170
+ parent_uuid TEXT,
171
+ UNIQUE(session_id, ordinal)
172
+ );
173
+
174
+ CREATE VIRTUAL TABLE IF NOT EXISTS messages_fts USING fts5(
175
+ content,
176
+ content='',
177
+ contentless_delete=1,
178
+ tokenize='trigram'
179
+ );
180
+
181
+ -- \u2500\u2500 Tool calls \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
182
+
183
+ CREATE TABLE IF NOT EXISTS tool_calls (
184
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
185
+ message_id INTEGER NOT NULL,
186
+ session_id TEXT NOT NULL,
187
+ tool_name TEXT NOT NULL,
188
+ category TEXT NOT NULL,
189
+ tool_use_id TEXT,
190
+ input_json TEXT,
191
+ skill_name TEXT,
192
+ result_content_length INTEGER,
193
+ result_content TEXT,
194
+ duration_ms INTEGER,
195
+ subagent_session_id TEXT,
196
+ sync_id TEXT DEFAULT (hex(randomblob(8)))
197
+ );
198
+
199
+ -- \u2500\u2500 Scanner tables \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
200
+
201
+ CREATE TABLE IF NOT EXISTS scanner_turns (
202
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
203
+ session_id TEXT NOT NULL,
204
+ source TEXT NOT NULL,
205
+ turn_index INTEGER NOT NULL,
206
+ timestamp_ms INTEGER NOT NULL,
207
+ model TEXT,
208
+ role TEXT,
209
+ content_preview TEXT,
210
+ input_tokens INTEGER DEFAULT 0,
211
+ output_tokens INTEGER DEFAULT 0,
212
+ cache_read_tokens INTEGER DEFAULT 0,
213
+ cache_creation_tokens INTEGER DEFAULT 0,
214
+ reasoning_tokens INTEGER DEFAULT 0,
215
+ sync_id TEXT DEFAULT (hex(randomblob(8))),
216
+ UNIQUE(session_id, source, turn_index)
217
+ );
218
+
219
+ CREATE TABLE IF NOT EXISTS scanner_events (
220
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
221
+ session_id TEXT NOT NULL,
222
+ source TEXT NOT NULL,
223
+ event_type TEXT NOT NULL,
224
+ timestamp_ms INTEGER NOT NULL,
225
+ tool_name TEXT,
226
+ tool_input TEXT,
227
+ tool_output TEXT,
228
+ content TEXT,
229
+ metadata JSON,
230
+ sync_id TEXT DEFAULT (hex(randomblob(8))),
231
+ UNIQUE(session_id, source, event_type, timestamp_ms, tool_name)
232
+ );
233
+
234
+ CREATE TABLE IF NOT EXISTS scanner_file_watermarks (
235
+ file_path TEXT PRIMARY KEY,
236
+ byte_offset INTEGER NOT NULL DEFAULT 0,
237
+ last_scanned_ms INTEGER NOT NULL,
238
+ archived_size INTEGER DEFAULT 0
239
+ );
240
+
241
+ -- \u2500\u2500 Session summaries \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
242
+
243
+ CREATE TABLE IF NOT EXISTS session_summary_deltas (
244
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
245
+ session_id TEXT NOT NULL,
246
+ delta_index INTEGER NOT NULL,
247
+ created_at_ms INTEGER NOT NULL,
248
+ from_turn INTEGER NOT NULL,
249
+ to_turn INTEGER NOT NULL,
250
+ content TEXT NOT NULL,
251
+ method TEXT NOT NULL DEFAULT 'deterministic',
252
+ UNIQUE(session_id, delta_index)
253
+ );
254
+
255
+ -- \u2500\u2500 Model pricing \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
256
+
257
+ CREATE TABLE IF NOT EXISTS model_pricing (
258
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
259
+ model_id TEXT NOT NULL,
260
+ input_per_m REAL NOT NULL,
261
+ output_per_m REAL NOT NULL,
262
+ cache_read_per_m REAL NOT NULL DEFAULT 0,
263
+ cache_write_per_m REAL NOT NULL DEFAULT 0,
264
+ updated_ms INTEGER NOT NULL
265
+ );
266
+
267
+ -- \u2500\u2500 Config snapshots \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
268
+
269
+ CREATE TABLE IF NOT EXISTS user_config_snapshots (
270
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
271
+ device_name TEXT NOT NULL,
272
+ snapshot_at_ms INTEGER NOT NULL,
273
+ content_hash TEXT NOT NULL,
274
+ permissions JSON NOT NULL DEFAULT '{}',
275
+ enabled_plugins JSON NOT NULL DEFAULT '[]',
276
+ hooks JSON NOT NULL DEFAULT '[]',
277
+ commands JSON NOT NULL DEFAULT '[]',
278
+ rules JSON NOT NULL DEFAULT '[]',
279
+ skills JSON NOT NULL DEFAULT '[]'
280
+ );
281
+
282
+ CREATE TABLE IF NOT EXISTS repo_config_snapshots (
283
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
284
+ repository TEXT NOT NULL,
285
+ cwd TEXT NOT NULL,
286
+ session_id TEXT,
287
+ snapshot_at_ms INTEGER NOT NULL,
288
+ content_hash TEXT NOT NULL,
289
+ hooks JSON NOT NULL DEFAULT '[]',
290
+ mcp_servers JSON NOT NULL DEFAULT '[]',
291
+ commands JSON NOT NULL DEFAULT '[]',
292
+ agents JSON NOT NULL DEFAULT '[]',
293
+ rules JSON NOT NULL DEFAULT '[]',
294
+ local_hooks JSON NOT NULL DEFAULT '[]',
295
+ local_mcp_servers JSON NOT NULL DEFAULT '[]',
296
+ local_permissions JSON NOT NULL DEFAULT '{}',
297
+ local_is_gitignored INTEGER NOT NULL DEFAULT 1,
298
+ instructions JSON NOT NULL DEFAULT '[]'
299
+ );
300
+
301
+ -- \u2500\u2500 Sync watermarks \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
302
+
303
+ CREATE TABLE IF NOT EXISTS watermarks (
304
+ key TEXT PRIMARY KEY,
305
+ value INTEGER NOT NULL
306
+ );
307
+
308
+ -- \u2500\u2500 Per-session sync state \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
309
+
310
+ DROP TABLE IF EXISTS pending_session_sync;
311
+
312
+ CREATE TABLE IF NOT EXISTS target_session_sync (
313
+ session_id TEXT NOT NULL,
314
+ target TEXT NOT NULL,
315
+ confirmed INTEGER DEFAULT 0,
316
+ sync_seq INTEGER DEFAULT 0,
317
+ synced_seq INTEGER DEFAULT 0,
318
+ wm_messages INTEGER DEFAULT 0,
319
+ wm_tool_calls INTEGER DEFAULT 0,
320
+ wm_scanner_turns INTEGER DEFAULT 0,
321
+ wm_scanner_events INTEGER DEFAULT 0,
322
+ wm_hook_events INTEGER DEFAULT 0,
323
+ wm_otel_logs INTEGER DEFAULT 0,
324
+ wm_otel_metrics INTEGER DEFAULT 0,
325
+ wm_otel_spans INTEGER DEFAULT 0,
326
+ PRIMARY KEY (session_id, target)
327
+ );
328
+
329
+ -- \u2500\u2500 Indexes \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500
330
+
331
+ -- otel_logs
332
+ CREATE INDEX IF NOT EXISTS idx_logs_session ON otel_logs(session_id);
333
+ CREATE INDEX IF NOT EXISTS idx_logs_body ON otel_logs(body);
334
+ CREATE INDEX IF NOT EXISTS idx_logs_ts ON otel_logs(timestamp_ns);
335
+ CREATE INDEX IF NOT EXISTS idx_logs_prompt ON otel_logs(prompt_id);
336
+
337
+ -- otel_metrics
338
+ CREATE INDEX IF NOT EXISTS idx_metrics_session ON otel_metrics(session_id);
339
+ CREATE INDEX IF NOT EXISTS idx_metrics_name ON otel_metrics(name);
340
+ CREATE INDEX IF NOT EXISTS idx_metrics_ts ON otel_metrics(timestamp_ns);
341
+
342
+ -- otel_spans
343
+ CREATE INDEX IF NOT EXISTS idx_spans_session ON otel_spans(session_id);
344
+ CREATE INDEX IF NOT EXISTS idx_spans_trace ON otel_spans(trace_id);
345
+ CREATE INDEX IF NOT EXISTS idx_spans_start ON otel_spans(start_time_ns);
346
+
347
+ -- hook_events
348
+ CREATE INDEX IF NOT EXISTS idx_hooks_session ON hook_events(session_id);
349
+ CREATE INDEX IF NOT EXISTS idx_hooks_type ON hook_events(event_type);
350
+ CREATE INDEX IF NOT EXISTS idx_hooks_tool ON hook_events(tool_name);
351
+ CREATE INDEX IF NOT EXISTS idx_hooks_ts ON hook_events(timestamp_ms);
352
+ CREATE INDEX IF NOT EXISTS idx_hooks_file_path ON hook_events(file_path);
353
+ CREATE INDEX IF NOT EXISTS idx_hooks_target ON hook_events(target);
354
+
355
+ -- sessions
356
+ CREATE INDEX IF NOT EXISTS idx_sessions_target ON sessions(target);
357
+ CREATE INDEX IF NOT EXISTS idx_sessions_started ON sessions(started_at_ms);
358
+ CREATE INDEX IF NOT EXISTS idx_sessions_sync_seq ON sessions(sync_seq);
359
+ CREATE INDEX IF NOT EXISTS idx_sessions_project ON sessions(project);
360
+ CREATE INDEX IF NOT EXISTS idx_sessions_machine ON sessions(machine);
361
+ CREATE INDEX IF NOT EXISTS idx_sessions_parent ON sessions(parent_session_id)
362
+ WHERE parent_session_id IS NOT NULL;
363
+
364
+ -- session_repositories
365
+ CREATE INDEX IF NOT EXISTS idx_session_repos_session ON session_repositories(session_id);
366
+ CREATE INDEX IF NOT EXISTS idx_session_repos_repo ON session_repositories(repository);
367
+
368
+ -- session_cwds
369
+ CREATE INDEX IF NOT EXISTS idx_session_cwds_session ON session_cwds(session_id);
370
+
371
+ -- messages
372
+ CREATE INDEX IF NOT EXISTS idx_messages_session_ordinal ON messages(session_id, ordinal);
373
+ CREATE INDEX IF NOT EXISTS idx_messages_session_role ON messages(session_id, role);
374
+
375
+ -- tool_calls
376
+ CREATE INDEX IF NOT EXISTS idx_tool_calls_session ON tool_calls(session_id);
377
+ CREATE INDEX IF NOT EXISTS idx_tool_calls_category ON tool_calls(category);
378
+ CREATE INDEX IF NOT EXISTS idx_tool_calls_skill ON tool_calls(skill_name)
379
+ WHERE skill_name IS NOT NULL;
380
+ CREATE INDEX IF NOT EXISTS idx_tool_calls_subagent ON tool_calls(subagent_session_id)
381
+ WHERE subagent_session_id IS NOT NULL;
382
+
383
+ -- scanner_turns
384
+ CREATE INDEX IF NOT EXISTS idx_scanner_turns_session ON scanner_turns(session_id);
385
+ CREATE INDEX IF NOT EXISTS idx_scanner_turns_ts ON scanner_turns(timestamp_ms);
386
+
387
+ -- scanner_events
388
+ CREATE INDEX IF NOT EXISTS idx_scanner_events_session ON scanner_events(session_id);
389
+ CREATE INDEX IF NOT EXISTS idx_scanner_events_type ON scanner_events(event_type);
390
+
391
+ -- session_summary_deltas
392
+ CREATE INDEX IF NOT EXISTS idx_summary_deltas_session ON session_summary_deltas(session_id);
393
+
394
+ -- model_pricing
395
+ CREATE INDEX IF NOT EXISTS idx_model_pricing_model ON model_pricing(model_id, updated_ms);
396
+
397
+ -- user_config_snapshots
398
+ CREATE INDEX IF NOT EXISTS idx_user_config_ts ON user_config_snapshots(snapshot_at_ms);
399
+ CREATE INDEX IF NOT EXISTS idx_user_config_device_hash ON user_config_snapshots(device_name, content_hash);
400
+
401
+ -- repo_config_snapshots
402
+ CREATE INDEX IF NOT EXISTS idx_repo_config_repo_ts ON repo_config_snapshots(repository, snapshot_at_ms);
403
+ CREATE INDEX IF NOT EXISTS idx_repo_config_session ON repo_config_snapshots(session_id);
404
+ CREATE INDEX IF NOT EXISTS idx_repo_config_repo_hash ON repo_config_snapshots(repository, content_hash);
405
+
406
+ `;
407
+ var SCANNER_DATA_VERSION = 1;
408
+ var _db = null;
409
+ var _needsResync = false;
410
+ function registerCompressionFunctions(db) {
411
+ db.function(
412
+ "decompress",
413
+ (blob) => blob ? gunzipSync(blob).toString() : null
414
+ );
415
+ }
416
+ function getDb() {
417
+ if (_db && !fs.existsSync(config.dbPath)) {
418
+ try {
419
+ _db.close();
420
+ } catch {
421
+ }
422
+ _db = null;
423
+ }
424
+ if (_db) return _db;
425
+ if (!fs.existsSync(config.dataDir)) {
426
+ throw new Error(
427
+ `Panopticon data directory not found: ${config.dataDir}. Run "panopticon install" to set up.`
428
+ );
429
+ }
430
+ _db = new Database(config.dbPath);
431
+ _db.pragma("auto_vacuum = INCREMENTAL");
432
+ _db.pragma("journal_mode = WAL");
433
+ _db.pragma("busy_timeout = 5000");
434
+ registerCompressionFunctions(_db);
435
+ _db.exec(SCHEMA_SQL);
436
+ const currentVersion = _db.pragma("user_version", { simple: true }) ?? 0;
437
+ if (currentVersion < SCANNER_DATA_VERSION) {
438
+ _needsResync = true;
439
+ } else {
440
+ _needsResync = false;
441
+ }
442
+ return _db;
443
+ }
444
+ function needsResync() {
445
+ return _needsResync;
446
+ }
447
+ function markResyncComplete() {
448
+ _needsResync = false;
449
+ const db = getDb();
450
+ db.pragma(`user_version = ${SCANNER_DATA_VERSION}`);
451
+ }
452
+ function closeDb() {
453
+ if (_db) {
454
+ _db.close();
455
+ _db = null;
456
+ }
457
+ }
458
+
459
+ export {
460
+ SCHEMA_SQL,
461
+ SCANNER_DATA_VERSION,
462
+ getDb,
463
+ needsResync,
464
+ markResyncComplete,
465
+ closeDb
466
+ };
467
+ //# sourceMappingURL=chunk-DZ5HJFB4.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/db/schema.ts"],"sourcesContent":["import fs from \"node:fs\";\nimport { gunzipSync } from \"node:zlib\";\nimport Database from \"better-sqlite3\";\nimport { config } from \"../config.js\";\n\nexport const SCHEMA_SQL = `\n\n-- ── OTLP tables ─────────────────────────────────────────────────────────────\n\nCREATE TABLE IF NOT EXISTS otel_logs (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n timestamp_ns INTEGER NOT NULL,\n observed_timestamp_ns INTEGER,\n severity_number INTEGER,\n severity_text TEXT,\n body TEXT,\n attributes JSON,\n resource_attributes JSON,\n session_id TEXT,\n prompt_id TEXT,\n trace_id TEXT,\n span_id TEXT,\n sync_id TEXT DEFAULT (hex(randomblob(8)))\n);\n\nCREATE TABLE IF NOT EXISTS otel_metrics (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n timestamp_ns INTEGER NOT NULL,\n name TEXT NOT NULL,\n value REAL NOT NULL,\n metric_type TEXT,\n unit TEXT,\n attributes JSON,\n resource_attributes JSON,\n session_id TEXT,\n sync_id TEXT DEFAULT (hex(randomblob(8)))\n);\n\nCREATE TABLE IF NOT EXISTS otel_spans (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n trace_id TEXT NOT NULL,\n span_id TEXT NOT NULL,\n parent_span_id TEXT,\n name TEXT NOT NULL,\n kind INTEGER,\n start_time_ns INTEGER NOT NULL,\n end_time_ns INTEGER NOT NULL,\n status_code INTEGER,\n status_message TEXT,\n attributes JSON,\n resource_attributes JSON,\n session_id TEXT,\n UNIQUE(trace_id, span_id)\n);\n\n-- ── Hook events ─────────────────────────────────────────────────────────────\n\nCREATE TABLE IF NOT EXISTS hook_events (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n session_id TEXT NOT NULL,\n event_type TEXT NOT NULL,\n timestamp_ms INTEGER NOT NULL,\n cwd TEXT,\n repository TEXT,\n tool_name TEXT,\n payload BLOB NOT NULL,\n user_prompt TEXT,\n file_path TEXT,\n command TEXT,\n plan TEXT,\n allowed_prompts TEXT,\n tool_result TEXT,\n target TEXT,\n sync_id TEXT DEFAULT (hex(randomblob(8)))\n);\n\nCREATE VIRTUAL TABLE IF NOT EXISTS hook_events_fts USING fts5(\n payload,\n content='',\n contentless_delete=1,\n tokenize='trigram'\n);\n\n-- ── Sessions ────────────────────────────────────────────────────────────────\n\nCREATE TABLE IF NOT EXISTS sessions (\n session_id TEXT PRIMARY KEY,\n target TEXT,\n started_at_ms INTEGER,\n ended_at_ms INTEGER,\n cwd TEXT,\n first_prompt TEXT,\n permission_mode TEXT,\n agent_version TEXT,\n model TEXT,\n cli_version TEXT,\n scanner_file_path TEXT,\n total_input_tokens INTEGER DEFAULT 0,\n total_output_tokens INTEGER DEFAULT 0,\n total_cache_read_tokens INTEGER DEFAULT 0,\n total_cache_creation_tokens INTEGER DEFAULT 0,\n total_reasoning_tokens INTEGER DEFAULT 0,\n turn_count INTEGER DEFAULT 0,\n otel_input_tokens INTEGER DEFAULT 0,\n otel_output_tokens INTEGER DEFAULT 0,\n otel_cache_read_tokens INTEGER DEFAULT 0,\n otel_cache_creation_tokens INTEGER DEFAULT 0,\n models TEXT,\n has_hooks INTEGER DEFAULT 0,\n has_otel INTEGER DEFAULT 0,\n has_scanner INTEGER DEFAULT 0,\n summary TEXT,\n summary_version INTEGER DEFAULT 0,\n sync_dirty INTEGER DEFAULT 0,\n sync_seq INTEGER DEFAULT 0,\n tool_counts JSON DEFAULT '{}',\n hook_tool_counts JSON DEFAULT '{}',\n event_type_counts JSON DEFAULT '{}',\n hook_event_type_counts JSON DEFAULT '{}',\n project TEXT,\n machine TEXT NOT NULL DEFAULT 'local',\n message_count INTEGER DEFAULT 0,\n user_message_count INTEGER DEFAULT 0,\n parent_session_id TEXT,\n relationship_type TEXT DEFAULT '',\n is_automated INTEGER DEFAULT 0,\n created_at INTEGER\n);\n\nCREATE TABLE IF NOT EXISTS session_repositories (\n session_id TEXT NOT NULL,\n repository TEXT NOT NULL,\n first_seen_ms INTEGER NOT NULL,\n git_user_name TEXT,\n git_user_email TEXT,\n branch TEXT,\n UNIQUE(session_id, repository)\n);\n\nCREATE TABLE IF NOT EXISTS session_cwds (\n session_id TEXT NOT NULL,\n cwd TEXT NOT NULL,\n first_seen_ms INTEGER NOT NULL,\n UNIQUE(session_id, cwd)\n);\n\n-- ── Messages ────────────────────────────────────────────────────────────────\n\nCREATE TABLE IF NOT EXISTS messages (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n session_id TEXT NOT NULL,\n ordinal INTEGER NOT NULL,\n role TEXT NOT NULL,\n content TEXT NOT NULL,\n timestamp_ms INTEGER,\n has_thinking INTEGER NOT NULL DEFAULT 0,\n has_tool_use INTEGER NOT NULL DEFAULT 0,\n content_length INTEGER NOT NULL DEFAULT 0,\n is_system INTEGER NOT NULL DEFAULT 0,\n model TEXT NOT NULL DEFAULT '',\n token_usage TEXT NOT NULL DEFAULT '',\n context_tokens INTEGER NOT NULL DEFAULT 0,\n output_tokens INTEGER NOT NULL DEFAULT 0,\n has_context_tokens INTEGER NOT NULL DEFAULT 0,\n has_output_tokens INTEGER NOT NULL DEFAULT 0,\n uuid TEXT,\n parent_uuid TEXT,\n UNIQUE(session_id, ordinal)\n);\n\nCREATE VIRTUAL TABLE IF NOT EXISTS messages_fts USING fts5(\n content,\n content='',\n contentless_delete=1,\n tokenize='trigram'\n);\n\n-- ── Tool calls ──────────────────────────────────────────────────────────────\n\nCREATE TABLE IF NOT EXISTS tool_calls (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n message_id INTEGER NOT NULL,\n session_id TEXT NOT NULL,\n tool_name TEXT NOT NULL,\n category TEXT NOT NULL,\n tool_use_id TEXT,\n input_json TEXT,\n skill_name TEXT,\n result_content_length INTEGER,\n result_content TEXT,\n duration_ms INTEGER,\n subagent_session_id TEXT,\n sync_id TEXT DEFAULT (hex(randomblob(8)))\n);\n\n-- ── Scanner tables ──────────────────────────────────────────────────────────\n\nCREATE TABLE IF NOT EXISTS scanner_turns (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n session_id TEXT NOT NULL,\n source TEXT NOT NULL,\n turn_index INTEGER NOT NULL,\n timestamp_ms INTEGER NOT NULL,\n model TEXT,\n role TEXT,\n content_preview TEXT,\n input_tokens INTEGER DEFAULT 0,\n output_tokens INTEGER DEFAULT 0,\n cache_read_tokens INTEGER DEFAULT 0,\n cache_creation_tokens INTEGER DEFAULT 0,\n reasoning_tokens INTEGER DEFAULT 0,\n sync_id TEXT DEFAULT (hex(randomblob(8))),\n UNIQUE(session_id, source, turn_index)\n);\n\nCREATE TABLE IF NOT EXISTS scanner_events (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n session_id TEXT NOT NULL,\n source TEXT NOT NULL,\n event_type TEXT NOT NULL,\n timestamp_ms INTEGER NOT NULL,\n tool_name TEXT,\n tool_input TEXT,\n tool_output TEXT,\n content TEXT,\n metadata JSON,\n sync_id TEXT DEFAULT (hex(randomblob(8))),\n UNIQUE(session_id, source, event_type, timestamp_ms, tool_name)\n);\n\nCREATE TABLE IF NOT EXISTS scanner_file_watermarks (\n file_path TEXT PRIMARY KEY,\n byte_offset INTEGER NOT NULL DEFAULT 0,\n last_scanned_ms INTEGER NOT NULL,\n archived_size INTEGER DEFAULT 0\n);\n\n-- ── Session summaries ───────────────────────────────────────────────────────\n\nCREATE TABLE IF NOT EXISTS session_summary_deltas (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n session_id TEXT NOT NULL,\n delta_index INTEGER NOT NULL,\n created_at_ms INTEGER NOT NULL,\n from_turn INTEGER NOT NULL,\n to_turn INTEGER NOT NULL,\n content TEXT NOT NULL,\n method TEXT NOT NULL DEFAULT 'deterministic',\n UNIQUE(session_id, delta_index)\n);\n\n-- ── Model pricing ───────────────────────────────────────────────────────────\n\nCREATE TABLE IF NOT EXISTS model_pricing (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n model_id TEXT NOT NULL,\n input_per_m REAL NOT NULL,\n output_per_m REAL NOT NULL,\n cache_read_per_m REAL NOT NULL DEFAULT 0,\n cache_write_per_m REAL NOT NULL DEFAULT 0,\n updated_ms INTEGER NOT NULL\n);\n\n-- ── Config snapshots ────────────────────────────────────────────────────────\n\nCREATE TABLE IF NOT EXISTS user_config_snapshots (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n device_name TEXT NOT NULL,\n snapshot_at_ms INTEGER NOT NULL,\n content_hash TEXT NOT NULL,\n permissions JSON NOT NULL DEFAULT '{}',\n enabled_plugins JSON NOT NULL DEFAULT '[]',\n hooks JSON NOT NULL DEFAULT '[]',\n commands JSON NOT NULL DEFAULT '[]',\n rules JSON NOT NULL DEFAULT '[]',\n skills JSON NOT NULL DEFAULT '[]'\n);\n\nCREATE TABLE IF NOT EXISTS repo_config_snapshots (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n repository TEXT NOT NULL,\n cwd TEXT NOT NULL,\n session_id TEXT,\n snapshot_at_ms INTEGER NOT NULL,\n content_hash TEXT NOT NULL,\n hooks JSON NOT NULL DEFAULT '[]',\n mcp_servers JSON NOT NULL DEFAULT '[]',\n commands JSON NOT NULL DEFAULT '[]',\n agents JSON NOT NULL DEFAULT '[]',\n rules JSON NOT NULL DEFAULT '[]',\n local_hooks JSON NOT NULL DEFAULT '[]',\n local_mcp_servers JSON NOT NULL DEFAULT '[]',\n local_permissions JSON NOT NULL DEFAULT '{}',\n local_is_gitignored INTEGER NOT NULL DEFAULT 1,\n instructions JSON NOT NULL DEFAULT '[]'\n);\n\n-- ── Sync watermarks ─────────────────────────────────────────────────────────\n\nCREATE TABLE IF NOT EXISTS watermarks (\n key TEXT PRIMARY KEY,\n value INTEGER NOT NULL\n);\n\n-- ── Per-session sync state ─────────────────────────────────────────────────\n\nDROP TABLE IF EXISTS pending_session_sync;\n\nCREATE TABLE IF NOT EXISTS target_session_sync (\n session_id TEXT NOT NULL,\n target TEXT NOT NULL,\n confirmed INTEGER DEFAULT 0,\n sync_seq INTEGER DEFAULT 0,\n synced_seq INTEGER DEFAULT 0,\n wm_messages INTEGER DEFAULT 0,\n wm_tool_calls INTEGER DEFAULT 0,\n wm_scanner_turns INTEGER DEFAULT 0,\n wm_scanner_events INTEGER DEFAULT 0,\n wm_hook_events INTEGER DEFAULT 0,\n wm_otel_logs INTEGER DEFAULT 0,\n wm_otel_metrics INTEGER DEFAULT 0,\n wm_otel_spans INTEGER DEFAULT 0,\n PRIMARY KEY (session_id, target)\n);\n\n-- ── Indexes ─────────────────────────────────────────────────────────────────\n\n-- otel_logs\nCREATE INDEX IF NOT EXISTS idx_logs_session ON otel_logs(session_id);\nCREATE INDEX IF NOT EXISTS idx_logs_body ON otel_logs(body);\nCREATE INDEX IF NOT EXISTS idx_logs_ts ON otel_logs(timestamp_ns);\nCREATE INDEX IF NOT EXISTS idx_logs_prompt ON otel_logs(prompt_id);\n\n-- otel_metrics\nCREATE INDEX IF NOT EXISTS idx_metrics_session ON otel_metrics(session_id);\nCREATE INDEX IF NOT EXISTS idx_metrics_name ON otel_metrics(name);\nCREATE INDEX IF NOT EXISTS idx_metrics_ts ON otel_metrics(timestamp_ns);\n\n-- otel_spans\nCREATE INDEX IF NOT EXISTS idx_spans_session ON otel_spans(session_id);\nCREATE INDEX IF NOT EXISTS idx_spans_trace ON otel_spans(trace_id);\nCREATE INDEX IF NOT EXISTS idx_spans_start ON otel_spans(start_time_ns);\n\n-- hook_events\nCREATE INDEX IF NOT EXISTS idx_hooks_session ON hook_events(session_id);\nCREATE INDEX IF NOT EXISTS idx_hooks_type ON hook_events(event_type);\nCREATE INDEX IF NOT EXISTS idx_hooks_tool ON hook_events(tool_name);\nCREATE INDEX IF NOT EXISTS idx_hooks_ts ON hook_events(timestamp_ms);\nCREATE INDEX IF NOT EXISTS idx_hooks_file_path ON hook_events(file_path);\nCREATE INDEX IF NOT EXISTS idx_hooks_target ON hook_events(target);\n\n-- sessions\nCREATE INDEX IF NOT EXISTS idx_sessions_target ON sessions(target);\nCREATE INDEX IF NOT EXISTS idx_sessions_started ON sessions(started_at_ms);\nCREATE INDEX IF NOT EXISTS idx_sessions_sync_seq ON sessions(sync_seq);\nCREATE INDEX IF NOT EXISTS idx_sessions_project ON sessions(project);\nCREATE INDEX IF NOT EXISTS idx_sessions_machine ON sessions(machine);\nCREATE INDEX IF NOT EXISTS idx_sessions_parent ON sessions(parent_session_id)\n WHERE parent_session_id IS NOT NULL;\n\n-- session_repositories\nCREATE INDEX IF NOT EXISTS idx_session_repos_session ON session_repositories(session_id);\nCREATE INDEX IF NOT EXISTS idx_session_repos_repo ON session_repositories(repository);\n\n-- session_cwds\nCREATE INDEX IF NOT EXISTS idx_session_cwds_session ON session_cwds(session_id);\n\n-- messages\nCREATE INDEX IF NOT EXISTS idx_messages_session_ordinal ON messages(session_id, ordinal);\nCREATE INDEX IF NOT EXISTS idx_messages_session_role ON messages(session_id, role);\n\n-- tool_calls\nCREATE INDEX IF NOT EXISTS idx_tool_calls_session ON tool_calls(session_id);\nCREATE INDEX IF NOT EXISTS idx_tool_calls_category ON tool_calls(category);\nCREATE INDEX IF NOT EXISTS idx_tool_calls_skill ON tool_calls(skill_name)\n WHERE skill_name IS NOT NULL;\nCREATE INDEX IF NOT EXISTS idx_tool_calls_subagent ON tool_calls(subagent_session_id)\n WHERE subagent_session_id IS NOT NULL;\n\n-- scanner_turns\nCREATE INDEX IF NOT EXISTS idx_scanner_turns_session ON scanner_turns(session_id);\nCREATE INDEX IF NOT EXISTS idx_scanner_turns_ts ON scanner_turns(timestamp_ms);\n\n-- scanner_events\nCREATE INDEX IF NOT EXISTS idx_scanner_events_session ON scanner_events(session_id);\nCREATE INDEX IF NOT EXISTS idx_scanner_events_type ON scanner_events(event_type);\n\n-- session_summary_deltas\nCREATE INDEX IF NOT EXISTS idx_summary_deltas_session ON session_summary_deltas(session_id);\n\n-- model_pricing\nCREATE INDEX IF NOT EXISTS idx_model_pricing_model ON model_pricing(model_id, updated_ms);\n\n-- user_config_snapshots\nCREATE INDEX IF NOT EXISTS idx_user_config_ts ON user_config_snapshots(snapshot_at_ms);\nCREATE INDEX IF NOT EXISTS idx_user_config_device_hash ON user_config_snapshots(device_name, content_hash);\n\n-- repo_config_snapshots\nCREATE INDEX IF NOT EXISTS idx_repo_config_repo_ts ON repo_config_snapshots(repository, snapshot_at_ms);\nCREATE INDEX IF NOT EXISTS idx_repo_config_session ON repo_config_snapshots(session_id);\nCREATE INDEX IF NOT EXISTS idx_repo_config_repo_hash ON repo_config_snapshots(repository, content_hash);\n\n`;\n\n/**\n * Scanner data version. Increment when parser logic changes in ways that\n * affect stored data (new fields extracted, content formatting changes,\n * fork detection improvements, etc.). On startup, if the DB's user_version\n * is lower than this, a full resync is triggered automatically.\n */\nexport const SCANNER_DATA_VERSION = 1;\n\nlet _db: Database.Database | null = null;\nlet _needsResync = false;\n\nfunction registerCompressionFunctions(db: Database.Database): void {\n db.function(\"decompress\", (blob: Buffer | null) =>\n blob ? gunzipSync(blob).toString() : null,\n );\n}\n\nexport function getDb(): Database.Database {\n // If the db file was deleted (e.g. uninstall --purge) while this process\n // still holds a stale connection, drop it so we don't serve old data.\n if (_db && !fs.existsSync(config.dbPath)) {\n try {\n _db.close();\n } catch {}\n _db = null;\n }\n if (_db) return _db;\n\n // Don't auto-create the data directory — callers that need to bootstrap\n // the DB (install, initDb, hook handler) call ensureDataDir() first.\n if (!fs.existsSync(config.dataDir)) {\n throw new Error(\n `Panopticon data directory not found: ${config.dataDir}. Run \"panopticon install\" to set up.`,\n );\n }\n\n _db = new Database(config.dbPath);\n _db.pragma(\"auto_vacuum = INCREMENTAL\");\n _db.pragma(\"journal_mode = WAL\");\n _db.pragma(\"busy_timeout = 5000\");\n\n registerCompressionFunctions(_db);\n _db.exec(SCHEMA_SQL);\n\n // Check data version for resync\n const currentVersion = (_db.pragma(\"user_version\", { simple: true }) ??\n 0) as number;\n if (currentVersion < SCANNER_DATA_VERSION) {\n _needsResync = true;\n } else {\n _needsResync = false;\n }\n\n return _db;\n}\n\n/** True when the DB was opened with a stale data version. */\nexport function needsResync(): boolean {\n return _needsResync;\n}\n\n/** Mark resync as complete and stamp the current data version. */\nexport function markResyncComplete(): void {\n _needsResync = false;\n const db = getDb();\n db.pragma(`user_version = ${SCANNER_DATA_VERSION}`);\n}\n\nexport function closeDb(): void {\n if (_db) {\n _db.close();\n _db = null;\n }\n}\n"],"mappings":"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","names":[]}