@camstack/core 0.1.13 → 0.1.15

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 (161) hide show
  1. package/dist/builtins/addon-pages-aggregator/addon-pages-aggregator.addon.js +220 -0
  2. package/dist/builtins/addon-pages-aggregator/addon-pages-aggregator.addon.js.map +1 -0
  3. package/dist/builtins/addon-pages-aggregator/addon-pages-aggregator.addon.mjs +9 -0
  4. package/dist/builtins/addon-pages-aggregator/index.js +222 -0
  5. package/dist/builtins/addon-pages-aggregator/index.js.map +1 -0
  6. package/dist/builtins/addon-pages-aggregator/index.mjs +9 -0
  7. package/dist/builtins/addon-pages-aggregator/index.mjs.map +1 -0
  8. package/dist/builtins/addon-widgets-aggregator/addon-widgets-aggregator.addon.js +200 -0
  9. package/dist/builtins/addon-widgets-aggregator/addon-widgets-aggregator.addon.js.map +1 -0
  10. package/dist/builtins/addon-widgets-aggregator/addon-widgets-aggregator.addon.mjs +9 -0
  11. package/dist/builtins/addon-widgets-aggregator/addon-widgets-aggregator.addon.mjs.map +1 -0
  12. package/dist/builtins/addon-widgets-aggregator/index.js +202 -0
  13. package/dist/builtins/addon-widgets-aggregator/index.js.map +1 -0
  14. package/dist/builtins/addon-widgets-aggregator/index.mjs +9 -0
  15. package/dist/builtins/addon-widgets-aggregator/index.mjs.map +1 -0
  16. package/dist/builtins/alerts/alerts.addon.js +443 -0
  17. package/dist/builtins/alerts/alerts.addon.js.map +1 -0
  18. package/dist/builtins/alerts/alerts.addon.mjs +9 -0
  19. package/dist/builtins/alerts/alerts.addon.mjs.map +1 -0
  20. package/dist/builtins/alerts/index.js +443 -0
  21. package/dist/builtins/alerts/index.js.map +1 -0
  22. package/dist/builtins/alerts/index.mjs +8 -0
  23. package/dist/builtins/alerts/index.mjs.map +1 -0
  24. package/dist/builtins/console-logging/index.js +242 -0
  25. package/dist/builtins/console-logging/index.js.map +1 -0
  26. package/dist/builtins/console-logging/index.mjs +11 -0
  27. package/dist/builtins/console-logging/index.mjs.map +1 -0
  28. package/dist/builtins/device-manager/device-manager.addon.js +2155 -0
  29. package/dist/builtins/device-manager/device-manager.addon.js.map +1 -0
  30. package/dist/builtins/device-manager/device-manager.addon.mjs +9 -0
  31. package/dist/builtins/device-manager/device-manager.addon.mjs.map +1 -0
  32. package/dist/builtins/device-manager/index.js +2157 -0
  33. package/dist/builtins/device-manager/index.js.map +1 -0
  34. package/dist/builtins/device-manager/index.mjs +10 -0
  35. package/dist/builtins/device-manager/index.mjs.map +1 -0
  36. package/dist/builtins/hub-forwarder/index.js +297 -0
  37. package/dist/builtins/hub-forwarder/index.js.map +1 -0
  38. package/dist/builtins/hub-forwarder/index.mjs +11 -0
  39. package/dist/builtins/hub-forwarder/index.mjs.map +1 -0
  40. package/dist/builtins/local-auth/index.js +623 -0
  41. package/dist/builtins/local-auth/index.js.map +1 -0
  42. package/dist/builtins/local-auth/index.mjs +8 -0
  43. package/dist/builtins/local-auth/index.mjs.map +1 -0
  44. package/dist/builtins/local-auth/local-auth.addon.js +623 -0
  45. package/dist/builtins/local-auth/local-auth.addon.js.map +1 -0
  46. package/dist/builtins/local-auth/local-auth.addon.mjs +9 -0
  47. package/dist/builtins/local-auth/local-auth.addon.mjs.map +1 -0
  48. package/dist/builtins/local-backup/index.js +53 -68
  49. package/dist/builtins/local-backup/index.js.map +1 -1
  50. package/dist/builtins/local-backup/index.mjs +1 -1
  51. package/dist/builtins/native-metrics/native-metrics.addon.js +898 -0
  52. package/dist/builtins/native-metrics/native-metrics.addon.js.map +1 -0
  53. package/dist/builtins/native-metrics/native-metrics.addon.mjs +7 -0
  54. package/dist/builtins/native-metrics/native-metrics.addon.mjs.map +1 -0
  55. package/dist/builtins/snapshot/index.js +504 -0
  56. package/dist/builtins/snapshot/index.js.map +1 -0
  57. package/dist/builtins/snapshot/index.mjs +477 -0
  58. package/dist/builtins/snapshot/index.mjs.map +1 -0
  59. package/dist/builtins/sqlite-storage/filesystem-storage.addon.js +16 -166
  60. package/dist/builtins/sqlite-storage/filesystem-storage.addon.js.map +1 -1
  61. package/dist/builtins/sqlite-storage/filesystem-storage.addon.mjs +1 -1
  62. package/dist/builtins/sqlite-storage/index.js +554 -621
  63. package/dist/builtins/sqlite-storage/index.js.map +1 -1
  64. package/dist/builtins/sqlite-storage/index.mjs +9 -11
  65. package/dist/builtins/sqlite-storage/sqlite-settings.addon.js +368 -130
  66. package/dist/builtins/sqlite-storage/sqlite-settings.addon.js.map +1 -1
  67. package/dist/builtins/sqlite-storage/sqlite-settings.addon.mjs +1 -1
  68. package/dist/builtins/system-config/index.js +189 -0
  69. package/dist/builtins/system-config/index.js.map +1 -0
  70. package/dist/builtins/system-config/index.mjs +10 -0
  71. package/dist/builtins/system-config/index.mjs.map +1 -0
  72. package/dist/builtins/system-config/system-config.addon.js +187 -0
  73. package/dist/builtins/system-config/system-config.addon.js.map +1 -0
  74. package/dist/builtins/system-config/system-config.addon.mjs +9 -0
  75. package/dist/builtins/system-config/system-config.addon.mjs.map +1 -0
  76. package/dist/builtins/winston-logging/index.js +185 -65
  77. package/dist/builtins/winston-logging/index.js.map +1 -1
  78. package/dist/builtins/winston-logging/index.mjs +2 -1
  79. package/dist/chunk-2CIYKDRN.mjs +1 -0
  80. package/dist/chunk-2CIYKDRN.mjs.map +1 -0
  81. package/dist/chunk-2F76X6NL.mjs +136 -0
  82. package/dist/chunk-2F76X6NL.mjs.map +1 -0
  83. package/dist/chunk-2QUFBZ7M.mjs +1 -0
  84. package/dist/chunk-2QUFBZ7M.mjs.map +1 -0
  85. package/dist/chunk-3BK2Y7GY.mjs +593 -0
  86. package/dist/chunk-3BK2Y7GY.mjs.map +1 -0
  87. package/dist/chunk-4OOHFJHT.mjs +421 -0
  88. package/dist/chunk-4OOHFJHT.mjs.map +1 -0
  89. package/dist/chunk-4XHB7IHT.mjs +809 -0
  90. package/dist/chunk-4XHB7IHT.mjs.map +1 -0
  91. package/dist/{chunk-2F3XZYRW.mjs → chunk-6M2HSSTQ.mjs} +16 -7
  92. package/dist/chunk-6M2HSSTQ.mjs.map +1 -0
  93. package/dist/{chunk-SO4LROOT.mjs → chunk-7FI7SQS7.mjs} +54 -69
  94. package/dist/chunk-7FI7SQS7.mjs.map +1 -0
  95. package/dist/chunk-ED57RCQE.mjs +171 -0
  96. package/dist/chunk-ED57RCQE.mjs.map +1 -0
  97. package/dist/chunk-FZN56HGQ.mjs +626 -0
  98. package/dist/chunk-FZN56HGQ.mjs.map +1 -0
  99. package/dist/chunk-GL4OOB25.mjs +51 -0
  100. package/dist/chunk-GL4OOB25.mjs.map +1 -0
  101. package/dist/chunk-KDG2NTDB.mjs +137 -0
  102. package/dist/chunk-KDG2NTDB.mjs.map +1 -0
  103. package/dist/chunk-NRBQWBDM.mjs +191 -0
  104. package/dist/chunk-NRBQWBDM.mjs.map +1 -0
  105. package/dist/chunk-O4V246GG.mjs +2137 -0
  106. package/dist/chunk-O4V246GG.mjs.map +1 -0
  107. package/dist/chunk-QT57H266.mjs +163 -0
  108. package/dist/chunk-QT57H266.mjs.map +1 -0
  109. package/dist/chunk-QX4RH25I.mjs +141 -0
  110. package/dist/chunk-QX4RH25I.mjs.map +1 -0
  111. package/dist/chunk-TB562PZX.mjs +86 -0
  112. package/dist/chunk-TB562PZX.mjs.map +1 -0
  113. package/dist/chunk-TDYPZXK5.mjs +1 -0
  114. package/dist/chunk-TDYPZXK5.mjs.map +1 -0
  115. package/dist/chunk-UJI4LN5P.mjs +36 -0
  116. package/dist/chunk-UJI4LN5P.mjs.map +1 -0
  117. package/dist/chunk-W6RTHQGP.mjs +1 -0
  118. package/dist/chunk-W6RTHQGP.mjs.map +1 -0
  119. package/dist/chunk-ZELBCPDC.mjs +369 -0
  120. package/dist/chunk-ZELBCPDC.mjs.map +1 -0
  121. package/dist/index.d.mts +1103 -544
  122. package/dist/index.d.ts +1103 -544
  123. package/dist/index.js +7032 -6033
  124. package/dist/index.js.map +1 -1
  125. package/dist/index.mjs +568 -2226
  126. package/dist/index.mjs.map +1 -1
  127. package/dist/resource-monitor-UZUGPIAU.mjs +9 -0
  128. package/dist/resource-monitor-UZUGPIAU.mjs.map +1 -0
  129. package/dist/storage-location-manager-HFNB3PCS.mjs +7 -0
  130. package/dist/storage-location-manager-HFNB3PCS.mjs.map +1 -0
  131. package/package.json +123 -2
  132. package/dist/builtins/local-backup/index.d.mts +0 -42
  133. package/dist/builtins/local-backup/index.d.ts +0 -42
  134. package/dist/builtins/sqlite-storage/filesystem-storage.addon.d.mts +0 -2
  135. package/dist/builtins/sqlite-storage/filesystem-storage.addon.d.ts +0 -2
  136. package/dist/builtins/sqlite-storage/index.d.mts +0 -4
  137. package/dist/builtins/sqlite-storage/index.d.ts +0 -4
  138. package/dist/builtins/sqlite-storage/sqlite-settings.addon.d.mts +0 -2
  139. package/dist/builtins/sqlite-storage/sqlite-settings.addon.d.ts +0 -2
  140. package/dist/builtins/winston-logging/index.d.mts +0 -30
  141. package/dist/builtins/winston-logging/index.d.ts +0 -30
  142. package/dist/chunk-2F3XZYRW.mjs.map +0 -1
  143. package/dist/chunk-LQFPAEQF.mjs +0 -147
  144. package/dist/chunk-LQFPAEQF.mjs.map +0 -1
  145. package/dist/chunk-R3DIIBBX.mjs +0 -532
  146. package/dist/chunk-R3DIIBBX.mjs.map +0 -1
  147. package/dist/chunk-SMNR44VG.mjs +0 -386
  148. package/dist/chunk-SMNR44VG.mjs.map +0 -1
  149. package/dist/chunk-SO4LROOT.mjs.map +0 -1
  150. package/dist/chunk-SPA4JBKN.mjs +0 -175
  151. package/dist/chunk-SPA4JBKN.mjs.map +0 -1
  152. package/dist/dist-3BY63UQ5.mjs +0 -2151
  153. package/dist/dist-3BY63UQ5.mjs.map +0 -1
  154. package/dist/filesystem-storage.addon-C42r589X.d.mts +0 -57
  155. package/dist/filesystem-storage.addon-C42r589X.d.ts +0 -57
  156. package/dist/sql-schema-CKz78rId.d.mts +0 -97
  157. package/dist/sql-schema-CKz78rId.d.ts +0 -97
  158. package/dist/sqlite-settings.addon-KwG-uKMP.d.mts +0 -79
  159. package/dist/sqlite-settings.addon-KwG-uKMP.d.ts +0 -79
  160. package/dist/storage-location-manager-KKDQNAKA.mjs +0 -7
  161. /package/dist/{storage-location-manager-KKDQNAKA.mjs.map → builtins/addon-pages-aggregator/addon-pages-aggregator.addon.mjs.map} +0 -0
@@ -0,0 +1,443 @@
1
+ "use strict";
2
+ var __defProp = Object.defineProperty;
3
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
+ var __getOwnPropNames = Object.getOwnPropertyNames;
5
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
6
+ var __export = (target, all) => {
7
+ for (var name in all)
8
+ __defProp(target, name, { get: all[name], enumerable: true });
9
+ };
10
+ var __copyProps = (to, from, except, desc) => {
11
+ if (from && typeof from === "object" || typeof from === "function") {
12
+ for (let key of __getOwnPropNames(from))
13
+ if (!__hasOwnProp.call(to, key) && key !== except)
14
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
15
+ }
16
+ return to;
17
+ };
18
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
19
+
20
+ // src/builtins/alerts/index.ts
21
+ var alerts_exports = {};
22
+ __export(alerts_exports, {
23
+ AlertCenterAddon: () => AlertCenterAddon
24
+ });
25
+ module.exports = __toCommonJS(alerts_exports);
26
+
27
+ // src/builtins/alerts/alerts.addon.ts
28
+ var import_types = require("@camstack/types");
29
+ var import_types2 = require("@camstack/types");
30
+ var CATEGORY_MAPPINGS = {
31
+ // ── Critical — always surface ──────────────────────────────────────
32
+ "addon.crashed": { severity: "error", titleTemplate: "Addon crashed" },
33
+ "addon.error": { severity: "error", titleTemplate: "Addon error" },
34
+ "process.crashed": { severity: "error", titleTemplate: "Process crashed" },
35
+ "recording.error": { severity: "error", titleTemplate: "Recording error" },
36
+ "recording.storage.critical": { severity: "warning", titleTemplate: "Storage critical" },
37
+ "notification.failed": { severity: "error", titleTemplate: "Notification failed" },
38
+ // ── Cluster — agent connect/disconnect ─────────────────────────────
39
+ "agent.online": { severity: "info", titleTemplate: "Agent connected" },
40
+ "agent.offline": { severity: "warning", titleTemplate: "Agent disconnected" },
41
+ // ── System lifecycle ──────────────────────────────────────────────
42
+ "system.boot": { severity: "info", titleTemplate: "System boot" },
43
+ "system.restarting": { severity: "warning", titleTemplate: "System restarting" },
44
+ // ── Device lifecycle (add/remove only, not settings tweaks) ───────
45
+ "device.registered": { severity: "info", titleTemplate: "Device registered" },
46
+ "device.unregistered": { severity: "warning", titleTemplate: "Device unregistered" },
47
+ // ── Package management ────────────────────────────────────────────
48
+ "addon.installed": { severity: "info", titleTemplate: "Addon installed" },
49
+ "addon.updated": { severity: "info", titleTemplate: "Addon updated" },
50
+ "addon.uninstalled": { severity: "info", titleTemplate: "Addon uninstalled" },
51
+ // ── Backup (rare, important) ──────────────────────────────────────
52
+ "backup.completed": { severity: "info", titleTemplate: "Backup completed" },
53
+ "backup.restored": { severity: "info", titleTemplate: "Backup restored" },
54
+ // ── Model downloads (in-progress merging path) ────────────────────
55
+ "model.download.progress": { severity: "info", titleTemplate: "Model download" }
56
+ };
57
+ var TRACKED_CATEGORIES = Object.keys(CATEGORY_MAPPINGS);
58
+ var RATE_LIMIT_WINDOW_MS = 6e4;
59
+ var RATE_LIMIT_MAX_PER_SOURCE = 5;
60
+ function parseAlert(record) {
61
+ return import_types2.AlertSchema.parse({ id: record.id, ...record.data });
62
+ }
63
+ var DEFAULT_CONFIG = {
64
+ maxAlerts: 500,
65
+ retentionDays: 30
66
+ };
67
+ var ALERTS_COLLECTION = "alerts";
68
+ var EVENT_SOURCE = { type: "addon", id: "alert-center" };
69
+ var AlertCenterAddon = class extends import_types2.BaseAddon {
70
+ unsubscribers = [];
71
+ constructor() {
72
+ super({ ...DEFAULT_CONFIG });
73
+ }
74
+ /**
75
+ * Per `(category, source.id)` counter used to rate-limit chatty
76
+ * events. Entries are pruned lazily when they age out of
77
+ * `RATE_LIMIT_WINDOW_MS`; see `shouldSuppressByRate()` below.
78
+ */
79
+ rateLimitCounters = /* @__PURE__ */ new Map();
80
+ async onInitialize() {
81
+ const alertsProvider = {
82
+ emit: (alert) => this.emitAlert(alert),
83
+ update: (input) => this.updateAlert(input.alertId, input.patch),
84
+ list: (filter) => this.listAlerts(filter),
85
+ getUnreadCount: () => this.getUnreadCount(),
86
+ markRead: (input) => this.markRead(input.alertId),
87
+ markAllRead: () => this.markAllRead(),
88
+ dismiss: (input) => this.dismiss(input.alertId)
89
+ };
90
+ const unsubs = [];
91
+ for (const category of TRACKED_CATEGORIES) {
92
+ const unsub = this.ctx.eventBus.subscribe(
93
+ { category },
94
+ (event) => {
95
+ void this.handleEvent(category, event);
96
+ }
97
+ );
98
+ unsubs.push(unsub);
99
+ }
100
+ this.unsubscribers = unsubs;
101
+ this.ctx.logger.info("Initialized", { meta: { trackedCategoriesCount: TRACKED_CATEGORIES.length } });
102
+ return [{ capability: import_types2.alertsCapability, provider: alertsProvider }];
103
+ }
104
+ async onShutdown() {
105
+ for (const unsub of this.unsubscribers) {
106
+ unsub();
107
+ }
108
+ this.unsubscribers = [];
109
+ }
110
+ globalSettingsSchema() {
111
+ return this.schema({
112
+ sections: [{
113
+ id: "alerts",
114
+ title: "Alert Center",
115
+ description: "Configure alert storage and retention.",
116
+ columns: 1,
117
+ fields: [
118
+ this.field({
119
+ type: "number",
120
+ key: "maxAlerts",
121
+ label: "Maximum stored alerts",
122
+ description: "Oldest read alerts are deleted when this limit is exceeded.",
123
+ min: 50,
124
+ max: 5e3,
125
+ step: 50,
126
+ default: 500
127
+ }),
128
+ this.field({
129
+ type: "number",
130
+ key: "retentionDays",
131
+ label: "Alert retention (days)",
132
+ description: "Alerts older than this are automatically removed.",
133
+ min: 1,
134
+ max: 365,
135
+ step: 1,
136
+ default: 30
137
+ })
138
+ ]
139
+ }]
140
+ });
141
+ }
142
+ // ── Public methods (called by the tRPC router) ────────────────────
143
+ async listAlerts(filter) {
144
+ const backend = this.ctx.api.settingsStore;
145
+ if (!backend) return [];
146
+ const whereClause = {};
147
+ if (filter?.unreadOnly) {
148
+ whereClause.read = false;
149
+ }
150
+ const records = await backend.query.query({ collection: ALERTS_COLLECTION, filter: {
151
+ where: Object.keys(whereClause).length > 0 ? whereClause : void 0,
152
+ orderBy: { field: "updatedAt", direction: "desc" },
153
+ limit: filter?.limit ?? 100
154
+ } });
155
+ return records.map(parseAlert);
156
+ }
157
+ async getUnreadCount() {
158
+ const backend = this.ctx.api.settingsStore;
159
+ if (!backend) return 0;
160
+ return backend.count.query({ collection: ALERTS_COLLECTION, filter: { where: { read: false } } });
161
+ }
162
+ async markRead(alertId) {
163
+ await this.mergeUpdate(alertId, { read: true, updatedAt: Date.now() });
164
+ }
165
+ async markAllRead() {
166
+ const backend = this.ctx.api.settingsStore;
167
+ if (!backend) return;
168
+ const unread = await backend.query.query({ collection: ALERTS_COLLECTION, filter: { where: { read: false } } });
169
+ const now = Date.now();
170
+ for (const record of unread) {
171
+ await this.mergeUpdate(record.id, { read: true, updatedAt: now });
172
+ }
173
+ }
174
+ async dismiss(alertId) {
175
+ await this.mergeUpdate(alertId, {
176
+ status: "dismissed",
177
+ read: true,
178
+ updatedAt: Date.now()
179
+ });
180
+ }
181
+ // ── Private: event handling ──────────────────────────────────────
182
+ async handleEvent(category, event) {
183
+ try {
184
+ if (category === "model.download.progress") {
185
+ await this.handleModelDownload(event);
186
+ return;
187
+ }
188
+ const mapping = CATEGORY_MAPPINGS[category];
189
+ if (!mapping) return;
190
+ if (this.shouldSuppressByRate(category, event)) return;
191
+ const message = this.buildMessage(category, event.data);
192
+ const alert = {
193
+ id: crypto.randomUUID(),
194
+ category,
195
+ severity: mapping.severity,
196
+ title: mapping.titleTemplate,
197
+ message,
198
+ status: "active",
199
+ read: false,
200
+ createdAt: Date.now(),
201
+ updatedAt: Date.now(),
202
+ source: event.source ? { type: event.source.type, id: String(event.source.id) } : void 0,
203
+ metadata: event.data
204
+ };
205
+ await this.emitAlert(alert);
206
+ } catch (err) {
207
+ this.ctx.logger?.warn(
208
+ "Failed to handle event",
209
+ { meta: { category, error: (0, import_types2.errMsg)(err) } }
210
+ );
211
+ }
212
+ }
213
+ async handleModelDownload(event) {
214
+ const data = event.data;
215
+ const modelId = data.modelId ?? "unknown";
216
+ const progress = data.progress ?? 0;
217
+ const stableId = `model-download:${modelId}`;
218
+ const backend = this.ctx.api.settingsStore;
219
+ if (!backend) return;
220
+ const existing = await backend.query.query({ collection: ALERTS_COLLECTION, filter: {
221
+ where: { id: stableId }
222
+ } });
223
+ if (existing.length > 0) {
224
+ const now = Date.now();
225
+ if (progress >= 100) {
226
+ await this.mergeUpdate(stableId, {
227
+ status: "completed",
228
+ progress: 100,
229
+ message: `Model "${modelId}" download completed`,
230
+ updatedAt: now
231
+ });
232
+ this.emitAlertUpdatedEvent(stableId, { status: "completed", progress: 100 });
233
+ } else if (progress < 0) {
234
+ await this.mergeUpdate(stableId, {
235
+ status: "failed",
236
+ message: `Model "${modelId}" download failed`,
237
+ updatedAt: now
238
+ });
239
+ this.emitAlertUpdatedEvent(stableId, { status: "failed" });
240
+ } else {
241
+ await this.mergeUpdate(stableId, {
242
+ progress,
243
+ message: `Downloading model "${modelId}" \u2014 ${String(Math.round(progress))}%`,
244
+ updatedAt: now
245
+ });
246
+ this.emitAlertUpdatedEvent(stableId, { progress });
247
+ }
248
+ } else {
249
+ const alert = {
250
+ id: stableId,
251
+ category: import_types.EventCategory.ModelDownloadProgress,
252
+ severity: "info",
253
+ title: "Model download",
254
+ message: `Downloading model "${modelId}" \u2014 ${String(Math.round(progress))}%`,
255
+ status: "in-progress",
256
+ progress,
257
+ read: false,
258
+ createdAt: Date.now(),
259
+ updatedAt: Date.now(),
260
+ source: event.source ? { type: event.source.type, id: String(event.source.id) } : void 0,
261
+ metadata: { modelId }
262
+ };
263
+ await this.persistAlert(alert);
264
+ this.emitAlertCreatedEvent(alert);
265
+ }
266
+ }
267
+ /**
268
+ * `[ALERT-CENTER-EVENTS]` — rate limit a `(category, source.id)`
269
+ * tuple. Returns `true` when the caller should DROP this event
270
+ * without persisting an alert, and `false` when the alert should
271
+ * proceed normally.
272
+ *
273
+ * Heuristic: the first `RATE_LIMIT_MAX_PER_SOURCE` events within a
274
+ * `RATE_LIMIT_WINDOW_MS` window go through. Once the threshold is
275
+ * crossed, subsequent events are suppressed until the window rolls
276
+ * over. When the threshold is first crossed, the suppression is
277
+ * logged once so operators know why they're not seeing the follow-
278
+ * up alerts; after the window rolls over the next event is
279
+ * un-suppressed and starts a new window.
280
+ */
281
+ shouldSuppressByRate(category, event) {
282
+ const sourceId = event.source?.id ?? "unknown";
283
+ const key = `${category}::${sourceId}`;
284
+ const now = Date.now();
285
+ const entry = this.rateLimitCounters.get(key);
286
+ if (!entry || now - entry.firstAt > RATE_LIMIT_WINDOW_MS) {
287
+ this.rateLimitCounters.set(key, { firstAt: now, count: 1, suppressedLogged: false });
288
+ return false;
289
+ }
290
+ entry.count += 1;
291
+ if (entry.count <= RATE_LIMIT_MAX_PER_SOURCE) return false;
292
+ if (!entry.suppressedLogged) {
293
+ entry.suppressedLogged = true;
294
+ this.ctx.logger?.info(
295
+ "rate-limiting suppressing remainder of window",
296
+ { meta: { category, sourceId, maxPerSource: RATE_LIMIT_MAX_PER_SOURCE, windowSeconds: RATE_LIMIT_WINDOW_MS / 1e3 } }
297
+ );
298
+ }
299
+ return true;
300
+ }
301
+ buildMessage(category, data) {
302
+ const addonId = data.addonId;
303
+ const deviceId = data.deviceId;
304
+ const error = data.error;
305
+ const processId = data.processId;
306
+ switch (category) {
307
+ case "addon.crashed":
308
+ return `Addon "${addonId ?? "unknown"}" crashed${error ? `: ${error}` : ""}`;
309
+ case "addon.error":
310
+ return `Addon "${addonId ?? "unknown"}" error${error ? `: ${error}` : ""}`;
311
+ case "addon.installed":
312
+ return `Addon "${addonId ?? "unknown"}" installed`;
313
+ case "addon.updated": {
314
+ const from = data.fromVersion;
315
+ const to = data.toVersion;
316
+ return `Addon "${addonId ?? "unknown"}" updated${from && to ? ` from ${from} to ${to}` : ""}`;
317
+ }
318
+ case "addon.uninstalled":
319
+ return `Addon "${addonId ?? "unknown"}" uninstalled`;
320
+ case "device.registered":
321
+ return `Device "${data.name ?? deviceId ?? "unknown"}" registered`;
322
+ case "device.unregistered":
323
+ return `Device "${deviceId ?? "unknown"}" unregistered`;
324
+ case "agent.online":
325
+ return `Agent "${data.agentId ?? data.nodeId ?? "unknown"}" connected`;
326
+ case "agent.offline":
327
+ return `Agent "${data.agentId ?? data.nodeId ?? "unknown"}" disconnected${data.reason ? `: ${String(data.reason)}` : ""}`;
328
+ case "backup.completed":
329
+ return `Backup completed${data.targetId ? ` (${String(data.targetId)})` : ""}`;
330
+ case "backup.restored":
331
+ return `Backup restored${data.source ? ` from ${String(data.source)}` : ""}`;
332
+ case "notification.failed":
333
+ return `Notification failed${error ? `: ${error}` : ""}`;
334
+ case "system.restarting":
335
+ return `System restarting${data.reason ? ` \u2014 ${String(data.reason)}` : ""}`;
336
+ case "recording.error":
337
+ return `Recording error on device "${deviceId ?? "unknown"}"${error ? `: ${error}` : ""}`;
338
+ case "recording.storage.critical":
339
+ return `Storage critical for device "${deviceId ?? "unknown"}"`;
340
+ case "system.boot":
341
+ return `System booted in "${data.mode ?? "unknown"}" mode`;
342
+ case "process.crashed":
343
+ return `Process "${processId ?? "unknown"}" crashed`;
344
+ default:
345
+ return `Event: ${category}`;
346
+ }
347
+ }
348
+ // ── Private: persistence ─────────────────────────────────────────
349
+ async emitAlert(alert) {
350
+ await this.persistAlert(alert);
351
+ this.emitAlertCreatedEvent(alert);
352
+ await this.enforceMaxAlerts();
353
+ await this.enforceRetention();
354
+ }
355
+ async updateAlert(alertId, patch) {
356
+ await this.mergeUpdate(alertId, { ...patch, updatedAt: Date.now() });
357
+ this.emitAlertUpdatedEvent(alertId, patch);
358
+ }
359
+ /**
360
+ * Read-merge-write: SettingsBackend.update() replaces the entire `data` blob,
361
+ * so we must read the existing record, merge the patch, and write back.
362
+ */
363
+ async mergeUpdate(id, patch) {
364
+ const backend = this.ctx.api.settingsStore;
365
+ if (!backend) return;
366
+ const raw = await backend.get.query({ collection: ALERTS_COLLECTION, key: id });
367
+ if (raw === null || raw === void 0 || typeof raw !== "object") return;
368
+ const existing = Object.fromEntries(Object.entries(raw));
369
+ await backend.set.mutate({ collection: ALERTS_COLLECTION, key: id, value: { ...existing, ...patch } });
370
+ }
371
+ async persistAlert(alert) {
372
+ const backend = this.ctx.api.settingsStore;
373
+ if (!backend) return;
374
+ const data = {
375
+ id: alert.id,
376
+ category: alert.category,
377
+ severity: alert.severity,
378
+ title: alert.title,
379
+ message: alert.message,
380
+ status: alert.status,
381
+ read: alert.read,
382
+ createdAt: alert.createdAt,
383
+ updatedAt: alert.updatedAt,
384
+ ...alert.progress !== void 0 ? { progress: alert.progress } : {},
385
+ ...alert.source ? { source: alert.source } : {},
386
+ ...alert.metadata ? { metadata: alert.metadata } : {}
387
+ };
388
+ await backend.set.mutate({ collection: ALERTS_COLLECTION, key: alert.id, value: data });
389
+ }
390
+ emitAlertCreatedEvent(alert) {
391
+ this.ctx.eventBus?.emit((0, import_types2.createEvent)("alert.created", EVENT_SOURCE, {
392
+ id: alert.id,
393
+ category: alert.category,
394
+ severity: alert.severity,
395
+ title: alert.title,
396
+ status: alert.status
397
+ }));
398
+ }
399
+ emitAlertUpdatedEvent(alertId, patch) {
400
+ const patchRecord = {};
401
+ for (const [key, value] of Object.entries(patch)) {
402
+ patchRecord[key] = value;
403
+ }
404
+ this.ctx.eventBus?.emit((0, import_types2.createEvent)("alert.updated", EVENT_SOURCE, {
405
+ alertId,
406
+ patch: patchRecord
407
+ }));
408
+ }
409
+ async enforceMaxAlerts() {
410
+ const backend = this.ctx.api.settingsStore;
411
+ if (!backend) return;
412
+ const total = await backend.count.query({ collection: ALERTS_COLLECTION });
413
+ if (total <= this.config.maxAlerts) return;
414
+ const excess = total - this.config.maxAlerts;
415
+ const oldest = await backend.query.query({ collection: ALERTS_COLLECTION, filter: {
416
+ where: { read: true },
417
+ orderBy: { field: "createdAt", direction: "asc" },
418
+ limit: excess
419
+ } });
420
+ for (const record of oldest) {
421
+ await backend.delete.mutate({ collection: ALERTS_COLLECTION, key: record.id });
422
+ }
423
+ }
424
+ async enforceRetention() {
425
+ const backend = this.ctx.api.settingsStore;
426
+ if (!backend) return;
427
+ const cutoff = Date.now() - this.config.retentionDays * 24 * 60 * 60 * 1e3;
428
+ const expired = await backend.query.query({ collection: ALERTS_COLLECTION, filter: {
429
+ where: { read: true },
430
+ orderBy: { field: "createdAt", direction: "asc" }
431
+ } });
432
+ for (const record of expired) {
433
+ if (Number(record.data["createdAt"] ?? 0) < cutoff) {
434
+ await backend.delete.mutate({ collection: ALERTS_COLLECTION, key: record.id });
435
+ }
436
+ }
437
+ }
438
+ };
439
+ // Annotate the CommonJS export names for ESM import in node:
440
+ 0 && (module.exports = {
441
+ AlertCenterAddon
442
+ });
443
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../src/builtins/alerts/index.ts","../../../src/builtins/alerts/alerts.addon.ts"],"sourcesContent":["export { AlertCenterAddon } from './alerts.addon.js'\n","import {\n EventCategory,\n type KnownEventCategory,\n type SystemEvent,\n type ProviderRegistration,\n} from '@camstack/types'\nimport type { Alert, AlertSeverity, AlertStatus } from '@camstack/types'\nimport { BaseAddon, AlertSchema, alertsCapability, createEvent , errMsg } from '@camstack/types'\n\n// ── Category → severity mapping ─────────────────────────────────────\n\ninterface CategoryMapping {\n readonly severity: AlertSeverity\n readonly titleTemplate: string\n}\n\nconst CATEGORY_MAPPINGS: Readonly<Record<string, CategoryMapping>> = {\n // ── Critical — always surface ──────────────────────────────────────\n 'addon.crashed': { severity: 'error', titleTemplate: 'Addon crashed' },\n 'addon.error': { severity: 'error', titleTemplate: 'Addon error' },\n 'process.crashed': { severity: 'error', titleTemplate: 'Process crashed' },\n 'recording.error': { severity: 'error', titleTemplate: 'Recording error' },\n 'recording.storage.critical': { severity: 'warning', titleTemplate: 'Storage critical' },\n 'notification.failed': { severity: 'error', titleTemplate: 'Notification failed' },\n\n // ── Cluster — agent connect/disconnect ─────────────────────────────\n 'agent.online': { severity: 'info', titleTemplate: 'Agent connected' },\n 'agent.offline': { severity: 'warning', titleTemplate: 'Agent disconnected' },\n\n // ── System lifecycle ──────────────────────────────────────────────\n 'system.boot': { severity: 'info', titleTemplate: 'System boot' },\n 'system.restarting': { severity: 'warning', titleTemplate: 'System restarting' },\n\n // ── Device lifecycle (add/remove only, not settings tweaks) ───────\n 'device.registered': { severity: 'info', titleTemplate: 'Device registered' },\n 'device.unregistered': { severity: 'warning', titleTemplate: 'Device unregistered' },\n\n // ── Package management ────────────────────────────────────────────\n 'addon.installed': { severity: 'info', titleTemplate: 'Addon installed' },\n 'addon.updated': { severity: 'info', titleTemplate: 'Addon updated' },\n 'addon.uninstalled': { severity: 'info', titleTemplate: 'Addon uninstalled' },\n\n // ── Backup (rare, important) ──────────────────────────────────────\n 'backup.completed': { severity: 'info', titleTemplate: 'Backup completed' },\n 'backup.restored': { severity: 'info', titleTemplate: 'Backup restored' },\n\n // ── Model downloads (in-progress merging path) ────────────────────\n 'model.download.progress': { severity: 'info', titleTemplate: 'Model download' },\n}\n\nconst TRACKED_CATEGORIES = Object.keys(CATEGORY_MAPPINGS)\n\n/**\n * `[ALERT-CENTER-EVENTS]` — rate limiting. Events that fire thousands of\n * times a minute (flaky cam reconnecting, chatty integration, …) would\n * flood the alert feed and push genuinely important alerts off the\n * visible page. We collapse bursts into a single throttled summary per\n * `(category, source.id)` tuple: the first N events within the window\n * still surface as individual alerts, but once the source crosses the\n * threshold we stop emitting alerts for it and wait out the window.\n *\n * A standalone category-count map is cleared on every window rollover.\n * Keep the threshold generous enough that legitimate bursts of ~a few\n * per minute still get through, but strict enough that a reconnect loop\n * firing every second gets muted.\n */\nconst RATE_LIMIT_WINDOW_MS = 60_000\nconst RATE_LIMIT_MAX_PER_SOURCE = 5\n\n/** Parse a raw settings record into a validated Alert. */\nfunction parseAlert(record: { id: string; data: Record<string, unknown> }): Alert {\n return AlertSchema.parse({ id: record.id, ...record.data })\n}\n\n// ── Alert Center Addon ─────────────────────────────────────────────\n\ninterface AlertCenterConfig {\n readonly maxAlerts: number\n readonly retentionDays: number\n}\n\nconst DEFAULT_CONFIG: AlertCenterConfig = {\n maxAlerts: 500,\n retentionDays: 30,\n}\n\nconst ALERTS_COLLECTION = 'alerts'\nconst EVENT_SOURCE = { type: 'addon', id: 'alert-center' } as const\n\n/**\n * Alert Center addon — core builtin that persists alerts in a\n * structured SQLite table and serves them to the admin UI.\n *\n * Registers as an `alerts` collection provider and subscribes to\n * important EventBus categories to create/update alerts automatically.\n */\nexport class AlertCenterAddon extends BaseAddon<AlertCenterConfig> {\n private unsubscribers: readonly (() => void)[] = []\n\n constructor() {\n super({ ...DEFAULT_CONFIG })\n }\n\n /**\n * Per `(category, source.id)` counter used to rate-limit chatty\n * events. Entries are pruned lazily when they age out of\n * `RATE_LIMIT_WINDOW_MS`; see `shouldSuppressByRate()` below.\n */\n private readonly rateLimitCounters = new Map<string, { firstAt: number; count: number; suppressedLogged: boolean }>()\n\n protected async onInitialize(): Promise<ProviderRegistration[]> {\n const alertsProvider = {\n emit: (alert: Alert) => this.emitAlert(alert),\n update: (input: { alertId: string; patch: Partial<Alert> }) =>\n this.updateAlert(input.alertId, input.patch),\n list: (filter?: { unreadOnly?: boolean; limit?: number }) =>\n this.listAlerts(filter),\n getUnreadCount: () => this.getUnreadCount(),\n markRead: (input: { alertId: string }) => this.markRead(input.alertId),\n markAllRead: () => this.markAllRead(),\n dismiss: (input: { alertId: string }) => this.dismiss(input.alertId),\n }\n\n const unsubs: (() => void)[] = []\n for (const category of TRACKED_CATEGORIES) {\n const unsub = this.ctx.eventBus.subscribe(\n { category: category as KnownEventCategory },\n (event: SystemEvent) => { void this.handleEvent(category, event) },\n )\n unsubs.push(unsub)\n }\n this.unsubscribers = unsubs\n\n this.ctx.logger.info('Initialized', { meta: { trackedCategoriesCount: TRACKED_CATEGORIES.length } })\n return [{ capability: alertsCapability, provider: alertsProvider }]\n }\n\n protected async onShutdown(): Promise<void> {\n for (const unsub of this.unsubscribers) {\n unsub()\n }\n this.unsubscribers = []\n }\n\n protected globalSettingsSchema() {\n return this.schema({\n sections: [{\n id: 'alerts',\n title: 'Alert Center',\n description: 'Configure alert storage and retention.',\n columns: 1,\n fields: [\n this.field({\n type: 'number', key: 'maxAlerts', label: 'Maximum stored alerts',\n description: 'Oldest read alerts are deleted when this limit is exceeded.',\n min: 50, max: 5000, step: 50, default: 500,\n }),\n this.field({\n type: 'number', key: 'retentionDays', label: 'Alert retention (days)',\n description: 'Alerts older than this are automatically removed.',\n min: 1, max: 365, step: 1, default: 30,\n }),\n ],\n }],\n })\n }\n\n // ── Public methods (called by the tRPC router) ────────────────────\n\n async listAlerts(filter?: { unreadOnly?: boolean; limit?: number }): Promise<readonly Alert[]> {\n const backend = this.ctx.api.settingsStore\n if (!backend) return []\n\n const whereClause: Record<string, unknown> = {}\n if (filter?.unreadOnly) {\n whereClause.read = false\n }\n\n const records = await backend.query.query({ collection: ALERTS_COLLECTION, filter: {\n where: Object.keys(whereClause).length > 0 ? whereClause : undefined,\n orderBy: { field: 'updatedAt', direction: 'desc' },\n limit: filter?.limit ?? 100,\n } })\n\n return records.map(parseAlert)\n }\n\n async getUnreadCount(): Promise<number> {\n const backend = this.ctx.api.settingsStore\n if (!backend) return 0\n\n return backend.count.query({ collection: ALERTS_COLLECTION, filter: { where: { read: false } } })\n }\n\n async markRead(alertId: string): Promise<void> {\n await this.mergeUpdate(alertId, { read: true, updatedAt: Date.now() })\n }\n\n async markAllRead(): Promise<void> {\n const backend = this.ctx.api.settingsStore\n if (!backend) return\n\n const unread = await backend.query.query({ collection: ALERTS_COLLECTION, filter: { where: { read: false } } })\n const now = Date.now()\n for (const record of unread) {\n await this.mergeUpdate(record.id, { read: true, updatedAt: now })\n }\n }\n\n async dismiss(alertId: string): Promise<void> {\n await this.mergeUpdate(alertId, {\n status: 'dismissed' satisfies AlertStatus,\n read: true,\n updatedAt: Date.now(),\n })\n }\n\n // ── Private: event handling ──────────────────────────────────────\n\n private async handleEvent(category: string, event: SystemEvent): Promise<void> {\n try {\n if (category === 'model.download.progress') {\n // Model download progress has its own in-progress merging path\n // that intentionally overwrites a single alert per model, so\n // rate limiting does not apply here — the merger itself is the\n // collapse mechanism.\n await this.handleModelDownload(event)\n return\n }\n\n const mapping = CATEGORY_MAPPINGS[category]\n if (!mapping) return\n\n if (this.shouldSuppressByRate(category, event)) return\n\n const message = this.buildMessage(category, event.data)\n const alert: Alert = {\n id: crypto.randomUUID(),\n category,\n severity: mapping.severity,\n title: mapping.titleTemplate,\n message,\n status: 'active',\n read: false,\n createdAt: Date.now(),\n updatedAt: Date.now(),\n source: event.source ? { type: event.source.type, id: String(event.source.id) } : undefined,\n metadata: event.data,\n }\n\n await this.emitAlert(alert)\n } catch (err) {\n this.ctx.logger?.warn(\n 'Failed to handle event',\n { meta: { category, error: errMsg(err) } },\n )\n }\n }\n\n private async handleModelDownload(event: SystemEvent): Promise<void> {\n const data = event.data\n const modelId = (data.modelId as string | undefined) ?? 'unknown'\n const progress = (data.progress as number | undefined) ?? 0\n const stableId = `model-download:${modelId}`\n\n const backend = this.ctx.api.settingsStore\n if (!backend) return\n\n // Check if an in-progress alert already exists for this model\n const existing = await backend.query.query({ collection: ALERTS_COLLECTION, filter: {\n where: { id: stableId },\n } })\n\n if (existing.length > 0) {\n // Update existing alert (read-merge-write)\n const now = Date.now()\n if (progress >= 100) {\n await this.mergeUpdate(stableId, {\n status: 'completed' satisfies AlertStatus,\n progress: 100,\n message: `Model \"${modelId}\" download completed`,\n updatedAt: now,\n })\n this.emitAlertUpdatedEvent(stableId, { status: 'completed', progress: 100 })\n } else if (progress < 0) {\n await this.mergeUpdate(stableId, {\n status: 'failed' satisfies AlertStatus,\n message: `Model \"${modelId}\" download failed`,\n updatedAt: now,\n })\n this.emitAlertUpdatedEvent(stableId, { status: 'failed' })\n } else {\n await this.mergeUpdate(stableId, {\n progress,\n message: `Downloading model \"${modelId}\" — ${String(Math.round(progress))}%`,\n updatedAt: now,\n })\n this.emitAlertUpdatedEvent(stableId, { progress })\n }\n } else {\n // Create new in-progress alert\n const alert: Alert = {\n id: stableId,\n category: EventCategory.ModelDownloadProgress,\n severity: 'info',\n title: 'Model download',\n message: `Downloading model \"${modelId}\" — ${String(Math.round(progress))}%`,\n status: 'in-progress',\n progress,\n read: false,\n createdAt: Date.now(),\n updatedAt: Date.now(),\n source: event.source ? { type: event.source.type, id: String(event.source.id) } : undefined,\n metadata: { modelId },\n }\n\n await this.persistAlert(alert)\n this.emitAlertCreatedEvent(alert)\n }\n }\n\n /**\n * `[ALERT-CENTER-EVENTS]` — rate limit a `(category, source.id)`\n * tuple. Returns `true` when the caller should DROP this event\n * without persisting an alert, and `false` when the alert should\n * proceed normally.\n *\n * Heuristic: the first `RATE_LIMIT_MAX_PER_SOURCE` events within a\n * `RATE_LIMIT_WINDOW_MS` window go through. Once the threshold is\n * crossed, subsequent events are suppressed until the window rolls\n * over. When the threshold is first crossed, the suppression is\n * logged once so operators know why they're not seeing the follow-\n * up alerts; after the window rolls over the next event is\n * un-suppressed and starts a new window.\n */\n private shouldSuppressByRate(category: string, event: SystemEvent): boolean {\n const sourceId = event.source?.id ?? 'unknown'\n const key = `${category}::${sourceId}`\n const now = Date.now()\n const entry = this.rateLimitCounters.get(key)\n\n if (!entry || now - entry.firstAt > RATE_LIMIT_WINDOW_MS) {\n // First event of a new window — always allow, start the counter.\n this.rateLimitCounters.set(key, { firstAt: now, count: 1, suppressedLogged: false })\n return false\n }\n\n entry.count += 1\n if (entry.count <= RATE_LIMIT_MAX_PER_SOURCE) return false\n\n // Over threshold — suppress the rest of this window.\n if (!entry.suppressedLogged) {\n entry.suppressedLogged = true\n this.ctx.logger?.info(\n 'rate-limiting suppressing remainder of window',\n { meta: { category, sourceId, maxPerSource: RATE_LIMIT_MAX_PER_SOURCE, windowSeconds: RATE_LIMIT_WINDOW_MS / 1000 } },\n )\n }\n return true\n }\n\n private buildMessage(category: string, data: Record<string, unknown>): string {\n const addonId = data.addonId as string | undefined\n const deviceId = data.deviceId as string | undefined\n const error = data.error as string | undefined\n const processId = data.processId as string | undefined\n\n switch (category) {\n case 'addon.crashed':\n return `Addon \"${addonId ?? 'unknown'}\" crashed${error ? `: ${error}` : ''}`\n case 'addon.error':\n return `Addon \"${addonId ?? 'unknown'}\" error${error ? `: ${error}` : ''}`\n case 'addon.installed':\n return `Addon \"${addonId ?? 'unknown'}\" installed`\n case 'addon.updated': {\n const from = data.fromVersion as string | undefined\n const to = data.toVersion as string | undefined\n return `Addon \"${addonId ?? 'unknown'}\" updated${from && to ? ` from ${from} to ${to}` : ''}`\n }\n case 'addon.uninstalled':\n return `Addon \"${addonId ?? 'unknown'}\" uninstalled`\n case 'device.registered':\n return `Device \"${data.name as string | undefined ?? deviceId ?? 'unknown'}\" registered`\n case 'device.unregistered':\n return `Device \"${deviceId ?? 'unknown'}\" unregistered`\n case 'agent.online':\n return `Agent \"${data.agentId as string | undefined ?? data.nodeId as string | undefined ?? 'unknown'}\" connected`\n case 'agent.offline':\n return `Agent \"${data.agentId as string | undefined ?? data.nodeId as string | undefined ?? 'unknown'}\" disconnected${data.reason ? `: ${String(data.reason)}` : ''}`\n case 'backup.completed':\n return `Backup completed${data.targetId ? ` (${String(data.targetId)})` : ''}`\n case 'backup.restored':\n return `Backup restored${data.source ? ` from ${String(data.source)}` : ''}`\n case 'notification.failed':\n return `Notification failed${error ? `: ${error}` : ''}`\n case 'system.restarting':\n return `System restarting${data.reason ? ` — ${String(data.reason)}` : ''}`\n case 'recording.error':\n return `Recording error on device \"${deviceId ?? 'unknown'}\"${error ? `: ${error}` : ''}`\n case 'recording.storage.critical':\n return `Storage critical for device \"${deviceId ?? 'unknown'}\"`\n case 'system.boot':\n return `System booted in \"${data.mode as string | undefined ?? 'unknown'}\" mode`\n case 'process.crashed':\n return `Process \"${processId ?? 'unknown'}\" crashed`\n default:\n return `Event: ${category}`\n }\n }\n\n // ── Private: persistence ─────────────────────────────────────────\n\n private async emitAlert(alert: Alert): Promise<void> {\n await this.persistAlert(alert)\n this.emitAlertCreatedEvent(alert)\n await this.enforceMaxAlerts()\n await this.enforceRetention()\n }\n\n private async updateAlert(alertId: string, patch: Partial<Alert>): Promise<void> {\n await this.mergeUpdate(alertId, { ...patch, updatedAt: Date.now() })\n this.emitAlertUpdatedEvent(alertId, patch)\n }\n\n /**\n * Read-merge-write: SettingsBackend.update() replaces the entire `data` blob,\n * so we must read the existing record, merge the patch, and write back.\n */\n private async mergeUpdate(id: string, patch: Record<string, unknown>): Promise<void> {\n const backend = this.ctx.api.settingsStore\n if (!backend) return\n\n const raw = await backend.get.query({ collection: ALERTS_COLLECTION, key: id })\n if (raw === null || raw === undefined || typeof raw !== 'object') return\n const existing: Record<string, unknown> = Object.fromEntries(Object.entries(raw))\n await backend.set.mutate({ collection: ALERTS_COLLECTION, key: id, value: { ...existing, ...patch } })\n }\n\n private async persistAlert(alert: Alert): Promise<void> {\n const backend = this.ctx.api.settingsStore\n if (!backend) return\n\n // Serialize alert to a plain Record for SQLite JSON storage\n const data: Record<string, unknown> = {\n id: alert.id,\n category: alert.category,\n severity: alert.severity,\n title: alert.title,\n message: alert.message,\n status: alert.status,\n read: alert.read,\n createdAt: alert.createdAt,\n updatedAt: alert.updatedAt,\n ...(alert.progress !== undefined ? { progress: alert.progress } : {}),\n ...(alert.source ? { source: alert.source } : {}),\n ...(alert.metadata ? { metadata: alert.metadata } : {}),\n }\n await backend.set.mutate({ collection: ALERTS_COLLECTION, key: alert.id, value: data })\n }\n\n private emitAlertCreatedEvent(alert: Alert): void {\n this.ctx.eventBus?.emit(createEvent('alert.created', EVENT_SOURCE, {\n id: alert.id,\n category: alert.category,\n severity: alert.severity,\n title: alert.title,\n status: alert.status,\n }))\n }\n\n private emitAlertUpdatedEvent(alertId: string, patch: Partial<Alert>): void {\n const patchRecord: Record<string, unknown> = {}\n for (const [key, value] of Object.entries(patch)) {\n patchRecord[key] = value\n }\n this.ctx.eventBus?.emit(createEvent('alert.updated', EVENT_SOURCE, {\n alertId,\n patch: patchRecord,\n }))\n }\n\n private async enforceMaxAlerts(): Promise<void> {\n const backend = this.ctx.api.settingsStore\n if (!backend) return\n\n const total = await backend.count.query({ collection: ALERTS_COLLECTION })\n if (total <= this.config.maxAlerts) return\n\n // Delete oldest read alerts\n const excess = total - this.config.maxAlerts\n const oldest = await backend.query.query({ collection: ALERTS_COLLECTION, filter: {\n where: { read: true },\n orderBy: { field: 'createdAt', direction: 'asc' },\n limit: excess,\n } })\n\n for (const record of oldest) {\n await backend.delete.mutate({ collection: ALERTS_COLLECTION, key: record.id })\n }\n }\n\n private async enforceRetention(): Promise<void> {\n const backend = this.ctx.api.settingsStore\n if (!backend) return\n\n const cutoff = Date.now() - this.config.retentionDays * 24 * 60 * 60 * 1000\n const expired = await backend.query.query({ collection: ALERTS_COLLECTION, filter: {\n where: { read: true },\n orderBy: { field: 'createdAt', direction: 'asc' },\n } })\n\n for (const record of expired) {\n if (Number(record.data['createdAt'] ?? 0) < cutoff) {\n await backend.delete.mutate({ collection: ALERTS_COLLECTION, key: record.id })\n }\n }\n }\n}\n\nexport default AlertCenterAddon\n"],"mappings":"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","names":["import_types"]}
@@ -0,0 +1,8 @@
1
+ import "../../chunk-2CIYKDRN.mjs";
2
+ import {
3
+ AlertCenterAddon
4
+ } from "../../chunk-4OOHFJHT.mjs";
5
+ export {
6
+ AlertCenterAddon
7
+ };
8
+ //# sourceMappingURL=index.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}