brass-runtime 1.15.0 → 1.16.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 (93) hide show
  1. package/README.md +409 -137
  2. package/dist/agent/cli/main.cjs +40 -35
  3. package/dist/agent/cli/main.js +9 -4
  4. package/dist/agent/cli/main.mjs +9 -4
  5. package/dist/agent/index.cjs +8 -4
  6. package/dist/agent/index.d.ts +1 -1
  7. package/dist/agent/index.js +7 -3
  8. package/dist/agent/index.mjs +7 -3
  9. package/dist/{chunk-PPUXIH5R.js → chunk-2WC63LJK.mjs} +11 -7
  10. package/dist/chunk-3RG5ZIWI.js +10 -0
  11. package/dist/chunk-45F7OKGT.cjs +104 -0
  12. package/dist/chunk-5YOQOXEQ.cjs +2491 -0
  13. package/dist/{chunk-STVLQ3XD.cjs → chunk-7HUOJA4W.cjs} +78 -74
  14. package/dist/{chunk-BMH5AV44.js → chunk-7LVI2GIN.js} +251 -370
  15. package/dist/chunk-7TL2LHQJ.js +2491 -0
  16. package/dist/chunk-7V4KY4RL.mjs +104 -0
  17. package/dist/chunk-7XOPAB5Q.js +2143 -0
  18. package/dist/chunk-CCKHV5BT.mjs +193 -0
  19. package/dist/{chunk-AR22SXML.js → chunk-CY33PGEX.mjs} +488 -421
  20. package/dist/chunk-DJQ7OMMB.cjs +144 -0
  21. package/dist/chunk-F5EUMJL7.mjs +2143 -0
  22. package/dist/chunk-FM4W4QPL.js +193 -0
  23. package/dist/{chunk-TO7IKXYT.js → chunk-G3XGCZDQ.js} +1 -1
  24. package/dist/{chunk-BDF4AMWX.mjs → chunk-G6IQOE4P.mjs} +251 -370
  25. package/dist/chunk-GOV47PPB.mjs +552 -0
  26. package/dist/chunk-H55LI6WY.js +93 -0
  27. package/dist/chunk-IJT6RRQ5.cjs +93 -0
  28. package/dist/{chunk-ELOOF35R.mjs → chunk-J3H54ZRV.mjs} +1 -1
  29. package/dist/chunk-JF4XXPZ5.cjs +552 -0
  30. package/dist/chunk-JNFRRJYH.cjs +2143 -0
  31. package/dist/chunk-JX3LZQJH.cjs +354 -0
  32. package/dist/chunk-K2T3DV26.mjs +93 -0
  33. package/dist/chunk-KCPT2D6G.js +552 -0
  34. package/dist/chunk-MWXMNYJS.cjs +1110 -0
  35. package/dist/{chunk-VEZNF5GZ.cjs → chunk-N6VHMOWB.cjs} +130 -126
  36. package/dist/{chunk-3QMOKAS5.js → chunk-NC5SDRYE.js} +9 -5
  37. package/dist/chunk-NOYZIMUJ.mjs +144 -0
  38. package/dist/{chunk-R3R2FVLG.cjs → chunk-NYL4D7SK.cjs} +5 -5
  39. package/dist/chunk-OBGZSXTJ.cjs +10 -0
  40. package/dist/{chunk-4NHES7VK.mjs → chunk-OOGJ73B6.js} +11 -7
  41. package/dist/chunk-PNVFW245.js +144 -0
  42. package/dist/chunk-PRWCB3QL.mjs +2491 -0
  43. package/dist/{chunk-JFPU5GQI.mjs → chunk-QY5FKYEQ.js} +488 -421
  44. package/dist/chunk-ROJC3NBJ.js +104 -0
  45. package/dist/chunk-SPUEME2B.cjs +343 -0
  46. package/dist/chunk-TDVMADDN.js +343 -0
  47. package/dist/chunk-TVN5I4U6.cjs +193 -0
  48. package/dist/chunk-U5KWK3PX.mjs +343 -0
  49. package/dist/chunk-VFIUZG7J.mjs +354 -0
  50. package/dist/{chunk-TGIFUAK4.cjs → chunk-WQ5QNU5R.cjs} +459 -578
  51. package/dist/chunk-XDZOO4L5.js +354 -0
  52. package/dist/chunk-Y6FXYEAI.mjs +10 -0
  53. package/dist/{chunk-K6M7MDZ4.mjs → chunk-ZGLD4TVZ.mjs} +9 -5
  54. package/dist/client-CtFmoDvM.d.ts +645 -0
  55. package/dist/core/index.cjs +72 -4
  56. package/dist/core/index.d.ts +92 -198
  57. package/dist/core/index.js +106 -38
  58. package/dist/core/index.mjs +106 -38
  59. package/dist/{effect-CMOQKX8y.d.ts → effect-CGNl5Rqp.d.ts} +107 -1
  60. package/dist/effectRunner-3ZHAD3LE.cjs +8 -0
  61. package/dist/effectRunner-A4CHJXJI.js +8 -0
  62. package/dist/effectRunner-OPUF6QRN.mjs +8 -0
  63. package/dist/http/index.cjs +2189 -1271
  64. package/dist/http/index.d.ts +830 -270
  65. package/dist/http/index.js +2008 -1090
  66. package/dist/http/index.mjs +2008 -1090
  67. package/dist/http/testing.cjs +159 -0
  68. package/dist/http/testing.d.ts +42 -0
  69. package/dist/http/testing.js +159 -0
  70. package/dist/http/testing.mjs +159 -0
  71. package/dist/index.cjs +246 -178
  72. package/dist/index.d.ts +9 -35
  73. package/dist/index.js +120 -52
  74. package/dist/index.mjs +120 -52
  75. package/dist/observability/index.cjs +677 -0
  76. package/dist/observability/index.d.ts +79 -0
  77. package/dist/observability/index.js +677 -0
  78. package/dist/observability/index.mjs +677 -0
  79. package/dist/schedule-Fque9Abz.d.ts +70 -0
  80. package/dist/schema/index.cjs +25 -0
  81. package/dist/schema/index.d.ts +177 -0
  82. package/dist/schema/index.js +25 -0
  83. package/dist/schema/index.mjs +25 -0
  84. package/dist/server-C8hDXA74.d.ts +674 -0
  85. package/dist/{stream-FQm9h4Mg.d.ts → stream-dvSs0QS5.d.ts} +1 -1
  86. package/dist/tracer-B5tRH9H7.d.ts +230 -0
  87. package/dist/tracing-Dt9S_6V8.d.ts +148 -0
  88. package/package.json +27 -1
  89. package/dist/chunk-BDYEENHT.js +0 -224
  90. package/dist/chunk-MS34J5LY.cjs +0 -224
  91. package/dist/chunk-UMAZLXAB.mjs +0 -224
  92. package/dist/chunk-XPZNXSVN.cjs +0 -1043
  93. package/dist/tracing-DNT9jEbr.d.ts +0 -106
@@ -0,0 +1,343 @@
1
+ // src/schema/index.ts
2
+ var SchemaValidationException = class extends Error {
3
+ issues;
4
+ constructor(issues) {
5
+ super(formatIssues(issues));
6
+ this.name = "SchemaValidationException";
7
+ this.issues = issues;
8
+ }
9
+ };
10
+ var ConfigValidationError = class extends Error {
11
+ _tag = "ConfigValidationError";
12
+ configName;
13
+ issues;
14
+ constructor(configName, issues) {
15
+ super(`${configName} failed validation: ${formatIssues(issues)}`);
16
+ this.name = "ConfigValidationError";
17
+ this.configName = configName;
18
+ this.issues = issues;
19
+ }
20
+ };
21
+ var ok = (data) => ({ success: true, data });
22
+ var fail = (issues) => ({ success: false, issues });
23
+ var receivedKind = (value) => {
24
+ if (value === null) return "null";
25
+ if (value === void 0) return "undefined";
26
+ if (Array.isArray(value)) return "array";
27
+ if (typeof value === "number" && Number.isNaN(value)) return "NaN";
28
+ return typeof value;
29
+ };
30
+ var pathLabel = (path) => path.length === 0 ? "$" : path.reduce((acc, part) => typeof part === "number" ? `${acc}[${part}]` : `${acc}.${part}`, "$");
31
+ var makeSchemaIssue = (path, expected, received, message) => ({
32
+ path,
33
+ expected,
34
+ received: receivedKind(received),
35
+ message: message ?? `Expected ${expected} at ${pathLabel(path)}, received ${receivedKind(received)}`
36
+ });
37
+ function formatIssues(issues) {
38
+ if (issues.length === 0) return "Validation failed";
39
+ const preview = issues.slice(0, 3).map((issue) => `${pathLabel(issue.path)}: ${issue.message}`).join("; ");
40
+ return issues.length > 3 ? `${preview}; +${issues.length - 3} more` : preview;
41
+ }
42
+ function makeSchema(kind, isOptional, parser, name) {
43
+ const self = {
44
+ _tag: "Schema",
45
+ kind,
46
+ name,
47
+ isOptional,
48
+ _parse: parser,
49
+ safeParse: (input) => parser(input, []),
50
+ parse: (input) => {
51
+ const result = parser(input, []);
52
+ if (result.success) return result.data;
53
+ throw new SchemaValidationException(result.issues);
54
+ },
55
+ optional: () => makeSchema(
56
+ `${kind}.optional`,
57
+ true,
58
+ (input, path) => input === void 0 ? ok(void 0) : parser(input, path),
59
+ name
60
+ ),
61
+ nullable: () => makeSchema(
62
+ `${kind}.nullable`,
63
+ isOptional,
64
+ (input, path) => input === null ? ok(null) : parser(input, path),
65
+ name
66
+ ),
67
+ array: () => arraySchema(self),
68
+ refine: (predicate, message) => makeSchema(
69
+ `${kind}.refine`,
70
+ isOptional,
71
+ (input, path) => {
72
+ const result = parser(input, path);
73
+ if (!result.success) return result;
74
+ return predicate(result.data) ? result : fail([makeSchemaIssue(path, name ?? kind, result.data, message ?? `Failed refinement for ${name ?? kind}`)]);
75
+ },
76
+ name
77
+ ),
78
+ transform: (fn, expected) => makeSchema(
79
+ `${kind}.transform`,
80
+ false,
81
+ (input, path) => {
82
+ const result = parser(input, path);
83
+ if (!result.success) return result;
84
+ try {
85
+ return ok(fn(result.data));
86
+ } catch (error) {
87
+ return fail([
88
+ makeSchemaIssue(
89
+ path,
90
+ expected ?? `transform(${name ?? kind})`,
91
+ result.data,
92
+ error instanceof Error ? error.message : String(error)
93
+ )
94
+ ]);
95
+ }
96
+ },
97
+ expected ?? name
98
+ )
99
+ };
100
+ return self;
101
+ }
102
+ function stringSchema(options = {}) {
103
+ return makeSchema("string", false, (input, path) => {
104
+ if (typeof input !== "string") return fail([makeSchemaIssue(path, options.name ?? "string", input)]);
105
+ if (options.minLength !== void 0 && input.length < options.minLength) {
106
+ return fail([makeSchemaIssue(path, `string length >= ${options.minLength}`, input, `Expected at least ${options.minLength} characters`)]);
107
+ }
108
+ if (options.maxLength !== void 0 && input.length > options.maxLength) {
109
+ return fail([makeSchemaIssue(path, `string length <= ${options.maxLength}`, input, `Expected at most ${options.maxLength} characters`)]);
110
+ }
111
+ if (options.pattern && !options.pattern.test(input)) {
112
+ return fail([makeSchemaIssue(path, `string matching ${String(options.pattern)}`, input)]);
113
+ }
114
+ return ok(input);
115
+ }, options.name);
116
+ }
117
+ function nonEmptyStringSchema(options = {}) {
118
+ return stringSchema({
119
+ ...options,
120
+ minLength: 1,
121
+ name: options.name ?? "non-empty string"
122
+ });
123
+ }
124
+ var EMAIL_PATTERN = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
125
+ var UUID_PATTERN = /^[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i;
126
+ var ISO_DATE_PATTERN = /^\d{4}-\d{2}-\d{2}(?:[T ]\d{2}:\d{2}(?::\d{2}(?:\.\d{1,9})?)?(?:Z|[+-]\d{2}:\d{2})?)?$/;
127
+ function emailSchema() {
128
+ return stringSchema({ pattern: EMAIL_PATTERN, name: "email" });
129
+ }
130
+ function uuidSchema() {
131
+ return stringSchema({ pattern: UUID_PATTERN, name: "uuid" });
132
+ }
133
+ function urlSchema() {
134
+ return stringSchema({ name: "url" }).refine((value) => {
135
+ try {
136
+ new URL(value);
137
+ return true;
138
+ } catch {
139
+ return false;
140
+ }
141
+ }, "Expected valid URL");
142
+ }
143
+ function dateIsoSchema() {
144
+ return stringSchema({ pattern: ISO_DATE_PATTERN, name: "ISO date" }).refine((value) => Number.isFinite(Date.parse(value)), "Expected valid ISO date string");
145
+ }
146
+ function numberSchema(options = {}) {
147
+ const finite = options.finite ?? true;
148
+ return makeSchema("number", false, (input, path) => {
149
+ if (typeof input !== "number" || Number.isNaN(input)) return fail([makeSchemaIssue(path, options.name ?? "number", input)]);
150
+ if (finite && !Number.isFinite(input)) return fail([makeSchemaIssue(path, "finite number", input)]);
151
+ if (options.int && !Number.isInteger(input)) return fail([makeSchemaIssue(path, "integer", input)]);
152
+ if (options.min !== void 0 && input < options.min) return fail([makeSchemaIssue(path, `number >= ${options.min}`, input)]);
153
+ if (options.max !== void 0 && input > options.max) return fail([makeSchemaIssue(path, `number <= ${options.max}`, input)]);
154
+ return ok(input);
155
+ }, options.name);
156
+ }
157
+ function intSchema(options = {}) {
158
+ return numberSchema({
159
+ ...options,
160
+ int: true,
161
+ name: options.name ?? "integer"
162
+ });
163
+ }
164
+ function positiveSchema(options = {}) {
165
+ const min = Math.max(options.min ?? 0, 0);
166
+ return numberSchema({
167
+ ...options,
168
+ min,
169
+ name: options.name ?? "positive number"
170
+ }).refine((value) => value > 0, "Expected positive number");
171
+ }
172
+ function booleanSchema(name) {
173
+ return makeSchema(
174
+ "boolean",
175
+ false,
176
+ (input, path) => typeof input === "boolean" ? ok(input) : fail([makeSchemaIssue(path, name ?? "boolean", input)]),
177
+ name
178
+ );
179
+ }
180
+ function unknownSchema() {
181
+ return makeSchema("unknown", false, (input) => ok(input));
182
+ }
183
+ function anySchema() {
184
+ return makeSchema("any", false, (input) => ok(input));
185
+ }
186
+ function literalSchema(value) {
187
+ return makeSchema(
188
+ `literal(${JSON.stringify(value)})`,
189
+ false,
190
+ (input, path) => Object.is(input, value) ? ok(value) : fail([makeSchemaIssue(path, JSON.stringify(value), input)])
191
+ );
192
+ }
193
+ function enumSchema(values) {
194
+ const allowed = new Set(values);
195
+ const expected = values.map((value) => JSON.stringify(value)).join(" | ");
196
+ return makeSchema(
197
+ `enum(${expected})`,
198
+ false,
199
+ (input, path) => (typeof input === "string" || typeof input === "number") && allowed.has(input) ? ok(input) : fail([makeSchemaIssue(path, expected, input)])
200
+ );
201
+ }
202
+ function arraySchema(item) {
203
+ return makeSchema("array", false, (input, path) => {
204
+ if (!Array.isArray(input)) return fail([makeSchemaIssue(path, "array", input)]);
205
+ const out = [];
206
+ const issues = [];
207
+ input.forEach((value, index) => {
208
+ const result = item._parse(value, [...path, index]);
209
+ if (result.success) {
210
+ out.push(result.data);
211
+ } else {
212
+ issues.push(...result.issues);
213
+ }
214
+ });
215
+ return issues.length > 0 ? fail(issues) : ok(out);
216
+ });
217
+ }
218
+ function objectSchema(shape, options = {}) {
219
+ const unknownKeys = options.unknownKeys ?? "strip";
220
+ return makeSchema("object", false, (input, path) => {
221
+ if (typeof input !== "object" || input === null || Array.isArray(input)) {
222
+ return fail([makeSchemaIssue(path, options.name ?? "object", input)]);
223
+ }
224
+ const source = input;
225
+ const out = unknownKeys === "passthrough" ? { ...source } : {};
226
+ const issues = [];
227
+ const knownKeys = new Set(Object.keys(shape));
228
+ for (const [key, fieldSchema] of Object.entries(shape)) {
229
+ if (!(key in source)) {
230
+ if (!fieldSchema.isOptional) {
231
+ issues.push(makeSchemaIssue([...path, key], fieldSchema.name ?? fieldSchema.kind, void 0, "Required field is missing"));
232
+ }
233
+ continue;
234
+ }
235
+ const result = fieldSchema._parse(source[key], [...path, key]);
236
+ if (result.success) {
237
+ out[key] = result.data;
238
+ } else {
239
+ issues.push(...result.issues);
240
+ }
241
+ }
242
+ if (unknownKeys === "strict") {
243
+ for (const key of Object.keys(source)) {
244
+ if (!knownKeys.has(key)) {
245
+ issues.push(makeSchemaIssue([...path, key], "known key", source[key], "Unknown key is not allowed"));
246
+ }
247
+ }
248
+ }
249
+ return issues.length > 0 ? fail(issues) : ok(out);
250
+ }, options.name);
251
+ }
252
+ function recordSchema(valueSchema) {
253
+ return makeSchema("record", false, (input, path) => {
254
+ if (typeof input !== "object" || input === null || Array.isArray(input)) {
255
+ return fail([makeSchemaIssue(path, "record", input)]);
256
+ }
257
+ const out = {};
258
+ const issues = [];
259
+ for (const [key, value] of Object.entries(input)) {
260
+ const result = valueSchema._parse(value, [...path, key]);
261
+ if (result.success) {
262
+ out[key] = result.data;
263
+ } else {
264
+ issues.push(...result.issues);
265
+ }
266
+ }
267
+ return issues.length > 0 ? fail(issues) : ok(out);
268
+ });
269
+ }
270
+ function unionSchema(members) {
271
+ const expected = members.map((member) => member.name ?? member.kind).join(" | ");
272
+ return makeSchema("union", false, (input, path) => {
273
+ const branchIssues = [];
274
+ for (const member of members) {
275
+ const result = member._parse(input, path);
276
+ if (result.success) return ok(result.data);
277
+ branchIssues.push(result.issues);
278
+ }
279
+ const mostSpecific = branchIssues.flatMap((issues) => issues).filter((issue) => issue.path.length > path.length).sort((a, b) => b.path.length - a.path.length);
280
+ if (mostSpecific.length > 0) return fail(mostSpecific);
281
+ return fail([makeSchemaIssue(path, expected, input)]);
282
+ });
283
+ }
284
+ function customSchema(guard, expected, message) {
285
+ return makeSchema(
286
+ expected,
287
+ false,
288
+ (input, path) => guard(input) ? ok(input) : fail([makeSchemaIssue(path, expected, input, message)]),
289
+ expected
290
+ );
291
+ }
292
+ var schema = Object.freeze({
293
+ string: stringSchema,
294
+ nonEmptyString: nonEmptyStringSchema,
295
+ email: emailSchema,
296
+ url: urlSchema,
297
+ uuid: uuidSchema,
298
+ dateIso: dateIsoSchema,
299
+ number: numberSchema,
300
+ int: intSchema,
301
+ positive: positiveSchema,
302
+ boolean: booleanSchema,
303
+ literal: literalSchema,
304
+ enum: enumSchema,
305
+ array: arraySchema,
306
+ object: objectSchema,
307
+ record: recordSchema,
308
+ union: unionSchema,
309
+ optional: (inner) => inner.optional(),
310
+ nullable: (inner) => inner.nullable(),
311
+ unknown: unknownSchema,
312
+ any: anySchema,
313
+ custom: customSchema
314
+ });
315
+ var s = schema;
316
+ var Schema = schema;
317
+ function isSchema(value) {
318
+ return typeof value === "object" && value !== null && value._tag === "Schema" && typeof value.safeParse === "function" && typeof value._parse === "function";
319
+ }
320
+ function validateValue(data, validator) {
321
+ if (isSchema(validator)) return validator.safeParse(data);
322
+ const result = validator(data);
323
+ if (result.success) return ok(result.data);
324
+ return fail(result.issues ?? [makeSchemaIssue([], "valid JSON shape", data, result.error)]);
325
+ }
326
+ function parseConfig(configName, validator, value) {
327
+ const result = validateValue(value, validator);
328
+ if (result.success) return result.data;
329
+ throw new ConfigValidationError(configName, result.issues);
330
+ }
331
+
332
+ export {
333
+ SchemaValidationException,
334
+ ConfigValidationError,
335
+ makeSchemaIssue,
336
+ formatIssues,
337
+ schema,
338
+ s,
339
+ Schema,
340
+ isSchema,
341
+ validateValue,
342
+ parseConfig
343
+ };
@@ -0,0 +1,354 @@
1
+ import {
2
+ makeBoundedRingBuffer
3
+ } from "./chunk-G6IQOE4P.mjs";
4
+
5
+ // src/core/runtime/metrics.ts
6
+ var DEFAULT_BOUNDARIES = [1, 5, 10, 25, 50, 100, 250, 500, 1e3, 5e3, 1e4];
7
+ function makeMetrics() {
8
+ const counters = /* @__PURE__ */ new Map();
9
+ const gauges = /* @__PURE__ */ new Map();
10
+ const histograms = /* @__PURE__ */ new Map();
11
+ const key = (name, labels) => labels ? `${name}|${Object.entries(labels).sort().map(([k, v]) => `${k}=${v}`).join(",")}` : name;
12
+ const counter = (name, labels = {}) => {
13
+ const k = key(name, labels);
14
+ if (!counters.has(k)) counters.set(k, { labels, value: 0 });
15
+ const entry = counters.get(k);
16
+ return {
17
+ increment: (n = 1) => {
18
+ entry.value += Math.max(0, n);
19
+ },
20
+ value: () => entry.value
21
+ };
22
+ };
23
+ const gauge = (name, labels = {}) => {
24
+ const k = key(name, labels);
25
+ if (!gauges.has(k)) gauges.set(k, { labels, value: 0 });
26
+ const entry = gauges.get(k);
27
+ return {
28
+ set: (v) => {
29
+ entry.value = v;
30
+ },
31
+ increment: (n = 1) => {
32
+ entry.value += n;
33
+ },
34
+ decrement: (n = 1) => {
35
+ entry.value -= n;
36
+ },
37
+ value: () => entry.value
38
+ };
39
+ };
40
+ const histogram = (name, boundaries = DEFAULT_BOUNDARIES, labels = {}) => {
41
+ const k = key(name, labels);
42
+ if (!histograms.has(k)) {
43
+ const sorted = [...boundaries].sort((a, b) => a - b);
44
+ histograms.set(k, {
45
+ labels,
46
+ boundaries: sorted,
47
+ data: { boundaries: sorted, counts: new Array(sorted.length + 1).fill(0), sum: 0, count: 0, min: Infinity, max: -Infinity }
48
+ });
49
+ }
50
+ const entry = histograms.get(k);
51
+ return {
52
+ observe: (value, exemplar) => {
53
+ entry.data.sum += value;
54
+ entry.data.count++;
55
+ entry.data.min = Math.min(entry.data.min, value);
56
+ entry.data.max = Math.max(entry.data.max, value);
57
+ let placed = false;
58
+ let bucketIndex = entry.boundaries.length;
59
+ for (let i = 0; i < entry.boundaries.length; i++) {
60
+ if (value <= entry.boundaries[i]) {
61
+ entry.data.counts[i]++;
62
+ bucketIndex = i;
63
+ placed = true;
64
+ break;
65
+ }
66
+ }
67
+ if (!placed) entry.data.counts[entry.boundaries.length]++;
68
+ if (exemplar) {
69
+ const exemplars = entry.data.exemplars ?? new Array(entry.boundaries.length + 1).fill(void 0);
70
+ exemplars[bucketIndex] = {
71
+ ...exemplar,
72
+ value,
73
+ timestamp: exemplar.timestamp
74
+ };
75
+ entry.data.exemplars = exemplars;
76
+ }
77
+ },
78
+ buckets: () => ({ ...entry.data, counts: [...entry.data.counts], exemplars: entry.data.exemplars ? [...entry.data.exemplars] : void 0 }),
79
+ percentile: (p) => {
80
+ const target = Math.ceil(entry.data.count * (p / 100));
81
+ let cumulative = 0;
82
+ for (let i = 0; i < entry.boundaries.length; i++) {
83
+ cumulative += entry.data.counts[i];
84
+ if (cumulative >= target) return entry.boundaries[i];
85
+ }
86
+ return entry.data.max;
87
+ }
88
+ };
89
+ };
90
+ return {
91
+ counter,
92
+ gauge,
93
+ histogram,
94
+ snapshot: () => ({
95
+ counters: Array.from(counters.entries()).map(([k, v]) => ({ name: k.split("|")[0], labels: v.labels, value: v.value })),
96
+ gauges: Array.from(gauges.entries()).map(([k, v]) => ({ name: k.split("|")[0], labels: v.labels, value: v.value })),
97
+ histograms: Array.from(histograms.entries()).map(([k, v]) => ({ name: k.split("|")[0], labels: v.labels, buckets: v.data }))
98
+ }),
99
+ reset: () => {
100
+ counters.clear();
101
+ gauges.clear();
102
+ histograms.clear();
103
+ }
104
+ };
105
+ }
106
+
107
+ // src/core/runtime/events.ts
108
+ function makeRuntimeEventRecord(ev, ctx, seq) {
109
+ const wallTs = Date.now();
110
+ const ts = typeof performance !== "undefined" ? performance.now() : wallTs;
111
+ return {
112
+ ...ctx,
113
+ contextFiberId: ctx.fiberId,
114
+ contextScopeId: ctx.scopeId,
115
+ ...ev,
116
+ seq,
117
+ wallTs,
118
+ ts
119
+ };
120
+ }
121
+ function runtimeEventRecordContext(record) {
122
+ return {
123
+ fiberId: record.contextFiberId ?? record.fiberId,
124
+ scopeId: record.contextScopeId ?? record.scopeId,
125
+ traceId: record.traceId,
126
+ spanId: record.spanId,
127
+ parentSpanId: record.parentSpanId,
128
+ traceState: record.traceState,
129
+ baggage: record.baggage,
130
+ sampled: record.sampled
131
+ };
132
+ }
133
+
134
+ // src/core/runtime/eventBus.ts
135
+ function runtimeHooksToEventHandler(hooks) {
136
+ return (record) => {
137
+ hooks.emit(record, runtimeEventRecordContext(record));
138
+ };
139
+ }
140
+ var EventBus = class {
141
+ constructor(options = {}) {
142
+ this.options = options;
143
+ }
144
+ options;
145
+ seq = 1;
146
+ subs = [];
147
+ flushScheduled = false;
148
+ boundFlush = () => this.flush();
149
+ emit(ev, ctx) {
150
+ if (this.subs.length === 0) return;
151
+ const full = makeRuntimeEventRecord(ev, ctx, this.seq++);
152
+ for (const s of this.subs) {
153
+ const st = s.q.push(full);
154
+ if (st & 2 /* Dropped */) s.dropped++;
155
+ }
156
+ if (!this.flushScheduled) {
157
+ this.flushScheduled = true;
158
+ queueMicrotask(this.boundFlush);
159
+ }
160
+ }
161
+ subscribe(handler, perSubscriberCapacity = 2048) {
162
+ this.subs.push({
163
+ handler,
164
+ q: makeBoundedRingBuffer(perSubscriberCapacity, perSubscriberCapacity, this.options),
165
+ dropped: 0
166
+ });
167
+ return () => {
168
+ this.subs = this.subs.filter((s) => s.handler !== handler);
169
+ };
170
+ }
171
+ subscribeHooks(hooks, perSubscriberCapacity = 2048) {
172
+ return this.subscribe(runtimeHooksToEventHandler(hooks), perSubscriberCapacity);
173
+ }
174
+ flush(budget = 4096) {
175
+ this.flushScheduled = false;
176
+ for (const s of this.subs) {
177
+ let n = 0;
178
+ if (s.dropped > 0) {
179
+ const dropEv = makeRuntimeEventRecord(
180
+ {
181
+ type: "log",
182
+ level: "warn",
183
+ message: "eventbus.dropped",
184
+ fields: { dropped: s.dropped }
185
+ },
186
+ {},
187
+ 0
188
+ );
189
+ try {
190
+ s.handler(dropEv);
191
+ } catch {
192
+ }
193
+ s.dropped = 0;
194
+ }
195
+ while (n++ < budget) {
196
+ const ev = s.q.shift();
197
+ if (!ev) break;
198
+ try {
199
+ s.handler(ev);
200
+ } catch {
201
+ }
202
+ }
203
+ }
204
+ }
205
+ };
206
+
207
+ // src/core/runtime/tracingSink.ts
208
+ var InMemoryTracer = class {
209
+ constructor(options = {}) {
210
+ this.options = options;
211
+ }
212
+ options;
213
+ spans = /* @__PURE__ */ new Map();
214
+ // key: spanId
215
+ prunedFinishedSpans = 0;
216
+ emit(ev, ctx) {
217
+ if (ctx.sampled === false) return;
218
+ const wallTs = this.now();
219
+ const traceId = ctx.traceId ?? "no-trace";
220
+ const spanId = ctx.spanId;
221
+ if (ev.type === "fiber.start") {
222
+ if (!spanId) return;
223
+ this.spans.set(spanId, {
224
+ traceId,
225
+ spanId,
226
+ parentSpanId: ctx.parentSpanId,
227
+ traceState: ctx.traceState,
228
+ name: ev.name ?? `fiber#${ev.fiberId}`,
229
+ startWallTs: wallTs,
230
+ attrs: this.attrs({ fiberId: ev.fiberId, parentFiberId: ev.parentFiberId, scopeId: ev.scopeId }),
231
+ links: [],
232
+ events: []
233
+ });
234
+ return;
235
+ }
236
+ if (ev.type === "fiber.end") {
237
+ if (!spanId) return;
238
+ const sp = this.spans.get(spanId);
239
+ if (sp) {
240
+ sp.endWallTs = wallTs;
241
+ sp.events.push({ wallTs, name: "fiber.end", attrs: this.attrs({ status: ev.status, error: this.error(ev.error) }) });
242
+ this.pruneFinished();
243
+ }
244
+ return;
245
+ }
246
+ if (ev.type === "span.start") {
247
+ if (!spanId) return;
248
+ this.spans.set(spanId, {
249
+ traceId,
250
+ spanId,
251
+ parentSpanId: ctx.parentSpanId,
252
+ traceState: ctx.traceState,
253
+ name: ev.name,
254
+ startWallTs: wallTs,
255
+ attrs: this.attrs(ev.attributes ?? {}),
256
+ links: ev.links ?? [],
257
+ events: []
258
+ });
259
+ return;
260
+ }
261
+ if (ev.type === "span.event") {
262
+ if (!spanId) return;
263
+ const sp = this.spans.get(spanId);
264
+ if (!sp) return;
265
+ sp.events.push({ wallTs, name: ev.name, attrs: this.attrs(ev.attributes ?? {}) });
266
+ return;
267
+ }
268
+ if (ev.type === "span.end") {
269
+ if (!spanId) return;
270
+ const sp = this.spans.get(spanId);
271
+ if (sp) {
272
+ sp.endWallTs = wallTs;
273
+ sp.events.push({ wallTs, name: "span.end", attrs: this.attrs({ status: ev.status, error: this.error(ev.error), ...ev.attributes ?? {} }) });
274
+ this.pruneFinished();
275
+ }
276
+ return;
277
+ }
278
+ if (ev.type === "fiber.suspend" || ev.type === "fiber.resume" || ev.type === "scope.open" || ev.type === "scope.close") {
279
+ if (!spanId) return;
280
+ const sp = this.spans.get(spanId);
281
+ if (!sp) return;
282
+ sp.events.push({ wallTs, name: ev.type, attrs: this.attrs(ev) });
283
+ }
284
+ }
285
+ exportFinished() {
286
+ this.pruneFinished();
287
+ return Array.from(this.spans.values()).filter((s) => s.endWallTs != null);
288
+ }
289
+ pruneFinished(spanIds) {
290
+ let dropped = 0;
291
+ if (spanIds) {
292
+ for (const spanId of spanIds) {
293
+ const span = this.spans.get(spanId);
294
+ if (span?.endWallTs == null) continue;
295
+ if (this.spans.delete(spanId)) dropped++;
296
+ }
297
+ this.prunedFinishedSpans += dropped;
298
+ return dropped;
299
+ }
300
+ dropped += this.pruneExpiredFinished();
301
+ dropped += this.pruneFinishedOverLimit();
302
+ this.prunedFinishedSpans += dropped;
303
+ return dropped;
304
+ }
305
+ stats() {
306
+ return {
307
+ storedSpans: this.spans.size,
308
+ finishedSpans: Array.from(this.spans.values()).filter((s) => s.endWallTs != null).length,
309
+ prunedFinishedSpans: this.prunedFinishedSpans
310
+ };
311
+ }
312
+ attrs(attrs) {
313
+ return this.options.sanitizeAttributes?.(attrs) ?? attrs;
314
+ }
315
+ error(error) {
316
+ return error === void 0 ? void 0 : this.options.sanitizeError?.(error) ?? error;
317
+ }
318
+ now() {
319
+ return this.options.clock?.() ?? Date.now();
320
+ }
321
+ pruneExpiredFinished() {
322
+ const maxAgeMs = this.options.maxSpanAgeMs;
323
+ if (maxAgeMs === void 0 || maxAgeMs <= 0) return 0;
324
+ const now = this.now();
325
+ let dropped = 0;
326
+ for (const [spanId, span] of this.spans) {
327
+ if (span.endWallTs == null) continue;
328
+ if (now - span.endWallTs > maxAgeMs && this.spans.delete(spanId)) dropped++;
329
+ }
330
+ return dropped;
331
+ }
332
+ pruneFinishedOverLimit() {
333
+ const maxFinishedSpans = this.options.maxFinishedSpans;
334
+ if (maxFinishedSpans === void 0 || maxFinishedSpans < 0) return 0;
335
+ const finished = Array.from(this.spans.values()).filter((span) => span.endWallTs != null).sort((a, b) => (a.endWallTs ?? 0) - (b.endWallTs ?? 0));
336
+ const overflow = finished.length - maxFinishedSpans;
337
+ if (overflow <= 0) return 0;
338
+ let dropped = 0;
339
+ for (let i = 0; i < overflow; i++) {
340
+ const span = finished[i];
341
+ if (this.spans.delete(span.spanId)) dropped++;
342
+ }
343
+ return dropped;
344
+ }
345
+ };
346
+
347
+ export {
348
+ makeMetrics,
349
+ makeRuntimeEventRecord,
350
+ runtimeEventRecordContext,
351
+ runtimeHooksToEventHandler,
352
+ EventBus,
353
+ InMemoryTracer
354
+ };