@contractspec/lib.metering 1.44.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.
@@ -0,0 +1,418 @@
1
+ import { defineEntity, defineEntityEnum, field, index } from "@contractspec/lib.schema";
2
+
3
+ //#region src/entities/index.ts
4
+ /**
5
+ * Aggregation type enum.
6
+ */
7
+ const AggregationTypeEnum = defineEntityEnum({
8
+ name: "AggregationType",
9
+ values: [
10
+ "COUNT",
11
+ "SUM",
12
+ "AVG",
13
+ "MAX",
14
+ "MIN",
15
+ "LAST"
16
+ ],
17
+ schema: "lssm_metering",
18
+ description: "How to aggregate metric values."
19
+ });
20
+ /**
21
+ * Reset period enum.
22
+ */
23
+ const ResetPeriodEnum = defineEntityEnum({
24
+ name: "ResetPeriod",
25
+ values: [
26
+ "NEVER",
27
+ "HOURLY",
28
+ "DAILY",
29
+ "WEEKLY",
30
+ "MONTHLY",
31
+ "YEARLY"
32
+ ],
33
+ schema: "lssm_metering",
34
+ description: "When to reset metric counters."
35
+ });
36
+ /**
37
+ * Period type enum.
38
+ */
39
+ const PeriodTypeEnum = defineEntityEnum({
40
+ name: "PeriodType",
41
+ values: [
42
+ "HOURLY",
43
+ "DAILY",
44
+ "WEEKLY",
45
+ "MONTHLY",
46
+ "YEARLY"
47
+ ],
48
+ schema: "lssm_metering",
49
+ description: "Time period for aggregation."
50
+ });
51
+ /**
52
+ * Threshold action enum.
53
+ */
54
+ const ThresholdActionEnum = defineEntityEnum({
55
+ name: "ThresholdAction",
56
+ values: [
57
+ "NONE",
58
+ "ALERT",
59
+ "WARN",
60
+ "BLOCK",
61
+ "DOWNGRADE"
62
+ ],
63
+ schema: "lssm_metering",
64
+ description: "Action to take when threshold is exceeded."
65
+ });
66
+ /**
67
+ * MetricDefinition entity - defines a trackable metric.
68
+ */
69
+ const MetricDefinitionEntity = defineEntity({
70
+ name: "MetricDefinition",
71
+ description: "Definition of a usage metric.",
72
+ schema: "lssm_metering",
73
+ map: "metric_definition",
74
+ fields: {
75
+ id: field.id({ description: "Unique identifier" }),
76
+ key: field.string({
77
+ isUnique: true,
78
+ description: "Metric key (e.g., api_calls, storage_gb)"
79
+ }),
80
+ name: field.string({ description: "Human-readable name" }),
81
+ description: field.string({
82
+ isOptional: true,
83
+ description: "Metric description"
84
+ }),
85
+ unit: field.string({ description: "Unit of measurement (calls, bytes, etc.)" }),
86
+ aggregationType: field.enum("AggregationType", {
87
+ default: "SUM",
88
+ description: "How to aggregate values"
89
+ }),
90
+ resetPeriod: field.enum("ResetPeriod", {
91
+ default: "MONTHLY",
92
+ description: "When to reset counters"
93
+ }),
94
+ precision: field.int({
95
+ default: 2,
96
+ description: "Decimal precision"
97
+ }),
98
+ orgId: field.string({
99
+ isOptional: true,
100
+ description: "Organization scope (null = global metric)"
101
+ }),
102
+ category: field.string({
103
+ isOptional: true,
104
+ description: "Category for grouping"
105
+ }),
106
+ displayOrder: field.int({
107
+ default: 0,
108
+ description: "Order for display"
109
+ }),
110
+ metadata: field.json({
111
+ isOptional: true,
112
+ description: "Additional metadata"
113
+ }),
114
+ isActive: field.boolean({
115
+ default: true,
116
+ description: "Whether metric is active"
117
+ }),
118
+ createdAt: field.createdAt(),
119
+ updatedAt: field.updatedAt(),
120
+ usageRecords: field.hasMany("UsageRecord"),
121
+ usageSummaries: field.hasMany("UsageSummary"),
122
+ thresholds: field.hasMany("UsageThreshold")
123
+ },
124
+ indexes: [
125
+ index.on(["orgId", "key"]),
126
+ index.on(["category"]),
127
+ index.on(["isActive"])
128
+ ],
129
+ enums: [AggregationTypeEnum, ResetPeriodEnum]
130
+ });
131
+ /**
132
+ * UsageRecord entity - individual usage event.
133
+ */
134
+ const UsageRecordEntity = defineEntity({
135
+ name: "UsageRecord",
136
+ description: "A single usage event.",
137
+ schema: "lssm_metering",
138
+ map: "usage_record",
139
+ fields: {
140
+ id: field.id({ description: "Unique identifier" }),
141
+ metricKey: field.string({ description: "Metric being recorded" }),
142
+ metricId: field.string({
143
+ isOptional: true,
144
+ description: "Metric ID (for FK)"
145
+ }),
146
+ subjectType: field.string({ description: "Subject type (org, user, project)" }),
147
+ subjectId: field.string({ description: "Subject identifier" }),
148
+ quantity: field.decimal({ description: "Usage quantity" }),
149
+ source: field.string({
150
+ isOptional: true,
151
+ description: "Source of usage (endpoint, feature, etc.)"
152
+ }),
153
+ resourceId: field.string({
154
+ isOptional: true,
155
+ description: "Related resource ID"
156
+ }),
157
+ resourceType: field.string({
158
+ isOptional: true,
159
+ description: "Related resource type"
160
+ }),
161
+ metadata: field.json({
162
+ isOptional: true,
163
+ description: "Additional context"
164
+ }),
165
+ idempotencyKey: field.string({
166
+ isOptional: true,
167
+ description: "Idempotency key for deduplication"
168
+ }),
169
+ timestamp: field.dateTime({ description: "When usage occurred" }),
170
+ createdAt: field.createdAt(),
171
+ aggregated: field.boolean({
172
+ default: false,
173
+ description: "Whether included in summary"
174
+ }),
175
+ aggregatedAt: field.dateTime({
176
+ isOptional: true,
177
+ description: "When aggregated"
178
+ })
179
+ },
180
+ indexes: [
181
+ index.on([
182
+ "metricKey",
183
+ "subjectType",
184
+ "subjectId",
185
+ "timestamp"
186
+ ]),
187
+ index.on([
188
+ "subjectType",
189
+ "subjectId",
190
+ "timestamp"
191
+ ]),
192
+ index.on(["timestamp"]),
193
+ index.on(["aggregated", "timestamp"]),
194
+ index.unique(["idempotencyKey"], { name: "usage_record_idempotency" })
195
+ ]
196
+ });
197
+ /**
198
+ * UsageSummary entity - pre-aggregated usage.
199
+ */
200
+ const UsageSummaryEntity = defineEntity({
201
+ name: "UsageSummary",
202
+ description: "Pre-aggregated usage summary.",
203
+ schema: "lssm_metering",
204
+ map: "usage_summary",
205
+ fields: {
206
+ id: field.id({ description: "Unique identifier" }),
207
+ metricKey: field.string({ description: "Metric key" }),
208
+ metricId: field.string({
209
+ isOptional: true,
210
+ description: "Metric ID (for FK)"
211
+ }),
212
+ subjectType: field.string({ description: "Subject type" }),
213
+ subjectId: field.string({ description: "Subject identifier" }),
214
+ periodType: field.enum("PeriodType", { description: "Period type" }),
215
+ periodStart: field.dateTime({ description: "Period start time" }),
216
+ periodEnd: field.dateTime({ description: "Period end time" }),
217
+ totalQuantity: field.decimal({ description: "Total/aggregated quantity" }),
218
+ recordCount: field.int({
219
+ default: 0,
220
+ description: "Number of records aggregated"
221
+ }),
222
+ minQuantity: field.decimal({
223
+ isOptional: true,
224
+ description: "Minimum value"
225
+ }),
226
+ maxQuantity: field.decimal({
227
+ isOptional: true,
228
+ description: "Maximum value"
229
+ }),
230
+ avgQuantity: field.decimal({
231
+ isOptional: true,
232
+ description: "Average value"
233
+ }),
234
+ metadata: field.json({
235
+ isOptional: true,
236
+ description: "Additional metadata"
237
+ }),
238
+ createdAt: field.createdAt(),
239
+ updatedAt: field.updatedAt()
240
+ },
241
+ indexes: [
242
+ index.unique([
243
+ "metricKey",
244
+ "subjectType",
245
+ "subjectId",
246
+ "periodType",
247
+ "periodStart"
248
+ ], { name: "usage_summary_unique" }),
249
+ index.on([
250
+ "subjectType",
251
+ "subjectId",
252
+ "periodType",
253
+ "periodStart"
254
+ ]),
255
+ index.on([
256
+ "metricKey",
257
+ "periodType",
258
+ "periodStart"
259
+ ])
260
+ ],
261
+ enums: [PeriodTypeEnum]
262
+ });
263
+ /**
264
+ * UsageThreshold entity - threshold configuration.
265
+ */
266
+ const UsageThresholdEntity = defineEntity({
267
+ name: "UsageThreshold",
268
+ description: "Usage threshold configuration.",
269
+ schema: "lssm_metering",
270
+ map: "usage_threshold",
271
+ fields: {
272
+ id: field.id({ description: "Unique identifier" }),
273
+ metricKey: field.string({ description: "Metric to monitor" }),
274
+ metricId: field.string({
275
+ isOptional: true,
276
+ description: "Metric ID (for FK)"
277
+ }),
278
+ subjectType: field.string({
279
+ isOptional: true,
280
+ description: "Subject type"
281
+ }),
282
+ subjectId: field.string({
283
+ isOptional: true,
284
+ description: "Subject identifier"
285
+ }),
286
+ name: field.string({ description: "Threshold name" }),
287
+ threshold: field.decimal({ description: "Threshold value" }),
288
+ warnThreshold: field.decimal({
289
+ isOptional: true,
290
+ description: "Warning threshold (e.g., 80%)"
291
+ }),
292
+ periodType: field.enum("PeriodType", {
293
+ default: "MONTHLY",
294
+ description: "Period to evaluate"
295
+ }),
296
+ action: field.enum("ThresholdAction", {
297
+ default: "ALERT",
298
+ description: "Action when exceeded"
299
+ }),
300
+ notifyEmails: field.json({
301
+ isOptional: true,
302
+ description: "Email addresses to notify"
303
+ }),
304
+ notifyWebhook: field.string({
305
+ isOptional: true,
306
+ description: "Webhook URL to call"
307
+ }),
308
+ currentValue: field.decimal({
309
+ default: 0,
310
+ description: "Current usage value"
311
+ }),
312
+ lastCheckedAt: field.dateTime({
313
+ isOptional: true,
314
+ description: "Last threshold check"
315
+ }),
316
+ lastExceededAt: field.dateTime({
317
+ isOptional: true,
318
+ description: "Last time threshold was exceeded"
319
+ }),
320
+ isActive: field.boolean({
321
+ default: true,
322
+ description: "Whether threshold is active"
323
+ }),
324
+ metadata: field.json({
325
+ isOptional: true,
326
+ description: "Additional metadata"
327
+ }),
328
+ createdAt: field.createdAt(),
329
+ updatedAt: field.updatedAt()
330
+ },
331
+ indexes: [
332
+ index.on(["metricKey"]),
333
+ index.on(["subjectType", "subjectId"]),
334
+ index.on(["isActive", "metricKey"])
335
+ ],
336
+ enums: [ThresholdActionEnum]
337
+ });
338
+ /**
339
+ * UsageAlert entity - threshold violation alerts.
340
+ */
341
+ const UsageAlertEntity = defineEntity({
342
+ name: "UsageAlert",
343
+ description: "Alert generated when threshold is exceeded.",
344
+ schema: "lssm_metering",
345
+ map: "usage_alert",
346
+ fields: {
347
+ id: field.id({ description: "Unique identifier" }),
348
+ thresholdId: field.foreignKey({ description: "Threshold that triggered alert" }),
349
+ metricKey: field.string({ description: "Metric key" }),
350
+ subjectType: field.string({
351
+ isOptional: true,
352
+ description: "Subject type"
353
+ }),
354
+ subjectId: field.string({
355
+ isOptional: true,
356
+ description: "Subject identifier"
357
+ }),
358
+ alertType: field.string({ description: "Alert type (warn, exceed, etc.)" }),
359
+ threshold: field.decimal({ description: "Threshold value" }),
360
+ actualValue: field.decimal({ description: "Actual usage value" }),
361
+ percentageUsed: field.decimal({ description: "Percentage of threshold used" }),
362
+ status: field.string({
363
+ default: "\"pending\"",
364
+ description: "Alert status (pending, acknowledged, resolved)"
365
+ }),
366
+ acknowledgedBy: field.string({
367
+ isOptional: true,
368
+ description: "User who acknowledged"
369
+ }),
370
+ acknowledgedAt: field.dateTime({
371
+ isOptional: true,
372
+ description: "When acknowledged"
373
+ }),
374
+ resolvedAt: field.dateTime({
375
+ isOptional: true,
376
+ description: "When resolved"
377
+ }),
378
+ notificationsSent: field.json({
379
+ isOptional: true,
380
+ description: "Notifications sent"
381
+ }),
382
+ triggeredAt: field.dateTime({ description: "When alert was triggered" }),
383
+ createdAt: field.createdAt(),
384
+ thresholdRelation: field.belongsTo("UsageThreshold", ["thresholdId"], ["id"], { onDelete: "Cascade" })
385
+ },
386
+ indexes: [
387
+ index.on(["thresholdId", "status"]),
388
+ index.on(["metricKey", "triggeredAt"]),
389
+ index.on(["status", "triggeredAt"])
390
+ ]
391
+ });
392
+ /**
393
+ * All metering entities for schema composition.
394
+ */
395
+ const meteringEntities = [
396
+ MetricDefinitionEntity,
397
+ UsageRecordEntity,
398
+ UsageSummaryEntity,
399
+ UsageThresholdEntity,
400
+ UsageAlertEntity
401
+ ];
402
+ /**
403
+ * Module schema contribution for metering.
404
+ */
405
+ const meteringSchemaContribution = {
406
+ moduleId: "@contractspec/lib.metering",
407
+ entities: meteringEntities,
408
+ enums: [
409
+ AggregationTypeEnum,
410
+ ResetPeriodEnum,
411
+ PeriodTypeEnum,
412
+ ThresholdActionEnum
413
+ ]
414
+ };
415
+
416
+ //#endregion
417
+ export { AggregationTypeEnum, MetricDefinitionEntity, PeriodTypeEnum, ResetPeriodEnum, ThresholdActionEnum, UsageAlertEntity, UsageRecordEntity, UsageSummaryEntity, UsageThresholdEntity, meteringEntities, meteringSchemaContribution };
418
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","names":["meteringSchemaContribution: ModuleSchemaContribution"],"sources":["../../src/entities/index.ts"],"sourcesContent":["import {\n defineEntity,\n defineEntityEnum,\n field,\n index,\n} from '@contractspec/lib.schema';\nimport type { ModuleSchemaContribution } from '@contractspec/lib.schema';\n\n/**\n * Aggregation type enum.\n */\nexport const AggregationTypeEnum = defineEntityEnum({\n name: 'AggregationType',\n values: ['COUNT', 'SUM', 'AVG', 'MAX', 'MIN', 'LAST'] as const,\n schema: 'lssm_metering',\n description: 'How to aggregate metric values.',\n});\n\n/**\n * Reset period enum.\n */\nexport const ResetPeriodEnum = defineEntityEnum({\n name: 'ResetPeriod',\n values: ['NEVER', 'HOURLY', 'DAILY', 'WEEKLY', 'MONTHLY', 'YEARLY'] as const,\n schema: 'lssm_metering',\n description: 'When to reset metric counters.',\n});\n\n/**\n * Period type enum.\n */\nexport const PeriodTypeEnum = defineEntityEnum({\n name: 'PeriodType',\n values: ['HOURLY', 'DAILY', 'WEEKLY', 'MONTHLY', 'YEARLY'] as const,\n schema: 'lssm_metering',\n description: 'Time period for aggregation.',\n});\n\n/**\n * Threshold action enum.\n */\nexport const ThresholdActionEnum = defineEntityEnum({\n name: 'ThresholdAction',\n values: ['NONE', 'ALERT', 'WARN', 'BLOCK', 'DOWNGRADE'] as const,\n schema: 'lssm_metering',\n description: 'Action to take when threshold is exceeded.',\n});\n\n/**\n * MetricDefinition entity - defines a trackable metric.\n */\nexport const MetricDefinitionEntity = defineEntity({\n name: 'MetricDefinition',\n description: 'Definition of a usage metric.',\n schema: 'lssm_metering',\n map: 'metric_definition',\n fields: {\n id: field.id({ description: 'Unique identifier' }),\n key: field.string({\n isUnique: true,\n description: 'Metric key (e.g., api_calls, storage_gb)',\n }),\n name: field.string({ description: 'Human-readable name' }),\n description: field.string({\n isOptional: true,\n description: 'Metric description',\n }),\n\n // Configuration\n unit: field.string({\n description: 'Unit of measurement (calls, bytes, etc.)',\n }),\n aggregationType: field.enum('AggregationType', {\n default: 'SUM',\n description: 'How to aggregate values',\n }),\n resetPeriod: field.enum('ResetPeriod', {\n default: 'MONTHLY',\n description: 'When to reset counters',\n }),\n\n // Precision\n precision: field.int({ default: 2, description: 'Decimal precision' }),\n\n // Scope\n orgId: field.string({\n isOptional: true,\n description: 'Organization scope (null = global metric)',\n }),\n\n // Display\n category: field.string({\n isOptional: true,\n description: 'Category for grouping',\n }),\n displayOrder: field.int({ default: 0, description: 'Order for display' }),\n\n // Metadata\n metadata: field.json({\n isOptional: true,\n description: 'Additional metadata',\n }),\n\n // Status\n isActive: field.boolean({\n default: true,\n description: 'Whether metric is active',\n }),\n\n // Timestamps\n createdAt: field.createdAt(),\n updatedAt: field.updatedAt(),\n\n // Relations\n usageRecords: field.hasMany('UsageRecord'),\n usageSummaries: field.hasMany('UsageSummary'),\n thresholds: field.hasMany('UsageThreshold'),\n },\n indexes: [\n index.on(['orgId', 'key']),\n index.on(['category']),\n index.on(['isActive']),\n ],\n enums: [AggregationTypeEnum, ResetPeriodEnum],\n});\n\n/**\n * UsageRecord entity - individual usage event.\n */\nexport const UsageRecordEntity = defineEntity({\n name: 'UsageRecord',\n description: 'A single usage event.',\n schema: 'lssm_metering',\n map: 'usage_record',\n fields: {\n id: field.id({ description: 'Unique identifier' }),\n\n // Metric reference\n metricKey: field.string({ description: 'Metric being recorded' }),\n metricId: field.string({\n isOptional: true,\n description: 'Metric ID (for FK)',\n }),\n\n // Subject\n subjectType: field.string({\n description: 'Subject type (org, user, project)',\n }),\n subjectId: field.string({ description: 'Subject identifier' }),\n\n // Usage value\n quantity: field.decimal({ description: 'Usage quantity' }),\n\n // Context\n source: field.string({\n isOptional: true,\n description: 'Source of usage (endpoint, feature, etc.)',\n }),\n resourceId: field.string({\n isOptional: true,\n description: 'Related resource ID',\n }),\n resourceType: field.string({\n isOptional: true,\n description: 'Related resource type',\n }),\n\n // Metadata\n metadata: field.json({\n isOptional: true,\n description: 'Additional context',\n }),\n\n // Idempotency\n idempotencyKey: field.string({\n isOptional: true,\n description: 'Idempotency key for deduplication',\n }),\n\n // Timestamps\n timestamp: field.dateTime({ description: 'When usage occurred' }),\n createdAt: field.createdAt(),\n\n // Aggregation status\n aggregated: field.boolean({\n default: false,\n description: 'Whether included in summary',\n }),\n aggregatedAt: field.dateTime({\n isOptional: true,\n description: 'When aggregated',\n }),\n },\n indexes: [\n index.on(['metricKey', 'subjectType', 'subjectId', 'timestamp']),\n index.on(['subjectType', 'subjectId', 'timestamp']),\n index.on(['timestamp']),\n index.on(['aggregated', 'timestamp']),\n index.unique(['idempotencyKey'], { name: 'usage_record_idempotency' }),\n ],\n});\n\n/**\n * UsageSummary entity - pre-aggregated usage.\n */\nexport const UsageSummaryEntity = defineEntity({\n name: 'UsageSummary',\n description: 'Pre-aggregated usage summary.',\n schema: 'lssm_metering',\n map: 'usage_summary',\n fields: {\n id: field.id({ description: 'Unique identifier' }),\n\n // Metric reference\n metricKey: field.string({ description: 'Metric key' }),\n metricId: field.string({\n isOptional: true,\n description: 'Metric ID (for FK)',\n }),\n\n // Subject\n subjectType: field.string({ description: 'Subject type' }),\n subjectId: field.string({ description: 'Subject identifier' }),\n\n // Period\n periodType: field.enum('PeriodType', { description: 'Period type' }),\n periodStart: field.dateTime({ description: 'Period start time' }),\n periodEnd: field.dateTime({ description: 'Period end time' }),\n\n // Aggregated values\n totalQuantity: field.decimal({ description: 'Total/aggregated quantity' }),\n recordCount: field.int({\n default: 0,\n description: 'Number of records aggregated',\n }),\n\n // Statistics (for AVG, MIN, MAX)\n minQuantity: field.decimal({\n isOptional: true,\n description: 'Minimum value',\n }),\n maxQuantity: field.decimal({\n isOptional: true,\n description: 'Maximum value',\n }),\n avgQuantity: field.decimal({\n isOptional: true,\n description: 'Average value',\n }),\n\n // Metadata\n metadata: field.json({\n isOptional: true,\n description: 'Additional metadata',\n }),\n\n // Timestamps\n createdAt: field.createdAt(),\n updatedAt: field.updatedAt(),\n },\n indexes: [\n index.unique(\n ['metricKey', 'subjectType', 'subjectId', 'periodType', 'periodStart'],\n { name: 'usage_summary_unique' }\n ),\n index.on(['subjectType', 'subjectId', 'periodType', 'periodStart']),\n index.on(['metricKey', 'periodType', 'periodStart']),\n ],\n enums: [PeriodTypeEnum],\n});\n\n/**\n * UsageThreshold entity - threshold configuration.\n */\nexport const UsageThresholdEntity = defineEntity({\n name: 'UsageThreshold',\n description: 'Usage threshold configuration.',\n schema: 'lssm_metering',\n map: 'usage_threshold',\n fields: {\n id: field.id({ description: 'Unique identifier' }),\n\n // Metric reference\n metricKey: field.string({ description: 'Metric to monitor' }),\n metricId: field.string({\n isOptional: true,\n description: 'Metric ID (for FK)',\n }),\n\n // Subject (optional - can be global)\n subjectType: field.string({\n isOptional: true,\n description: 'Subject type',\n }),\n subjectId: field.string({\n isOptional: true,\n description: 'Subject identifier',\n }),\n\n // Threshold configuration\n name: field.string({ description: 'Threshold name' }),\n threshold: field.decimal({ description: 'Threshold value' }),\n warnThreshold: field.decimal({\n isOptional: true,\n description: 'Warning threshold (e.g., 80%)',\n }),\n\n // Period\n periodType: field.enum('PeriodType', {\n default: 'MONTHLY',\n description: 'Period to evaluate',\n }),\n\n // Actions\n action: field.enum('ThresholdAction', {\n default: 'ALERT',\n description: 'Action when exceeded',\n }),\n notifyEmails: field.json({\n isOptional: true,\n description: 'Email addresses to notify',\n }),\n notifyWebhook: field.string({\n isOptional: true,\n description: 'Webhook URL to call',\n }),\n\n // Status tracking\n currentValue: field.decimal({\n default: 0,\n description: 'Current usage value',\n }),\n lastCheckedAt: field.dateTime({\n isOptional: true,\n description: 'Last threshold check',\n }),\n lastExceededAt: field.dateTime({\n isOptional: true,\n description: 'Last time threshold was exceeded',\n }),\n\n // Status\n isActive: field.boolean({\n default: true,\n description: 'Whether threshold is active',\n }),\n\n // Metadata\n metadata: field.json({\n isOptional: true,\n description: 'Additional metadata',\n }),\n\n // Timestamps\n createdAt: field.createdAt(),\n updatedAt: field.updatedAt(),\n },\n indexes: [\n index.on(['metricKey']),\n index.on(['subjectType', 'subjectId']),\n index.on(['isActive', 'metricKey']),\n ],\n enums: [ThresholdActionEnum],\n});\n\n/**\n * UsageAlert entity - threshold violation alerts.\n */\nexport const UsageAlertEntity = defineEntity({\n name: 'UsageAlert',\n description: 'Alert generated when threshold is exceeded.',\n schema: 'lssm_metering',\n map: 'usage_alert',\n fields: {\n id: field.id({ description: 'Unique identifier' }),\n\n // Threshold reference\n thresholdId: field.foreignKey({\n description: 'Threshold that triggered alert',\n }),\n\n // Context\n metricKey: field.string({ description: 'Metric key' }),\n subjectType: field.string({\n isOptional: true,\n description: 'Subject type',\n }),\n subjectId: field.string({\n isOptional: true,\n description: 'Subject identifier',\n }),\n\n // Alert details\n alertType: field.string({ description: 'Alert type (warn, exceed, etc.)' }),\n threshold: field.decimal({ description: 'Threshold value' }),\n actualValue: field.decimal({ description: 'Actual usage value' }),\n percentageUsed: field.decimal({\n description: 'Percentage of threshold used',\n }),\n\n // Status\n status: field.string({\n default: '\"pending\"',\n description: 'Alert status (pending, acknowledged, resolved)',\n }),\n acknowledgedBy: field.string({\n isOptional: true,\n description: 'User who acknowledged',\n }),\n acknowledgedAt: field.dateTime({\n isOptional: true,\n description: 'When acknowledged',\n }),\n resolvedAt: field.dateTime({\n isOptional: true,\n description: 'When resolved',\n }),\n\n // Notifications\n notificationsSent: field.json({\n isOptional: true,\n description: 'Notifications sent',\n }),\n\n // Timestamps\n triggeredAt: field.dateTime({ description: 'When alert was triggered' }),\n createdAt: field.createdAt(),\n\n // Relations\n thresholdRelation: field.belongsTo(\n 'UsageThreshold',\n ['thresholdId'],\n ['id'],\n { onDelete: 'Cascade' }\n ),\n },\n indexes: [\n index.on(['thresholdId', 'status']),\n index.on(['metricKey', 'triggeredAt']),\n index.on(['status', 'triggeredAt']),\n ],\n});\n\n/**\n * All metering entities for schema composition.\n */\nexport const meteringEntities = [\n MetricDefinitionEntity,\n UsageRecordEntity,\n UsageSummaryEntity,\n UsageThresholdEntity,\n UsageAlertEntity,\n];\n\n/**\n * Module schema contribution for metering.\n */\nexport const meteringSchemaContribution: ModuleSchemaContribution = {\n moduleId: '@contractspec/lib.metering',\n entities: meteringEntities,\n enums: [\n AggregationTypeEnum,\n ResetPeriodEnum,\n PeriodTypeEnum,\n ThresholdActionEnum,\n ],\n};\n"],"mappings":"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"}