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,193 @@
1
+ "use strict";Object.defineProperty(exports, "__esModule", {value: true}); function _nullishCoalesce(lhs, rhsFn) { if (lhs != null) { return lhs; } else { return rhsFn(); } } function _optionalChain(ops) { let lastAccessLHS = undefined; let value = ops[0]; let i = 1; while (i < ops.length) { const op = ops[i]; const fn = ops[i + 1]; i += 2; if ((op === 'optionalAccess' || op === 'optionalCall') && value == null) { return undefined; } if (op === 'access' || op === 'optionalAccess') { lastAccessLHS = value; value = fn(value); } else if (op === 'call' || op === 'optionalCall') { value = fn((...args) => value.call(lastAccessLHS, ...args)); lastAccessLHS = undefined; } } return value; } var _class;
2
+
3
+ var _chunkWQ5QNU5Rcjs = require('./chunk-WQ5QNU5R.cjs');
4
+
5
+
6
+
7
+
8
+
9
+
10
+
11
+ var _chunkDJQ7OMMBcjs = require('./chunk-DJQ7OMMB.cjs');
12
+
13
+ // src/core/runtime/scope.ts
14
+ var nextScopeId = 1;
15
+ function awaitAll(fibers) {
16
+ return _chunkDJQ7OMMBcjs.asyncEffect.call(void 0, (_env, cb) => {
17
+ let remaining = fibers.length;
18
+ if (remaining === 0) {
19
+ cb({ _tag: "Success", value: void 0 });
20
+ return;
21
+ }
22
+ for (const f of fibers) {
23
+ f.join(() => {
24
+ remaining -= 1;
25
+ if (remaining === 0) cb({ _tag: "Success", value: void 0 });
26
+ });
27
+ }
28
+ });
29
+ }
30
+ var Scope = (_class = class _Scope {
31
+ constructor(runtime, parentScopeId) {;_class.prototype.__init.call(this);_class.prototype.__init2.call(this);_class.prototype.__init3.call(this);_class.prototype.__init4.call(this);
32
+ this.runtime = runtime;
33
+ this.parentScopeId = parentScopeId;
34
+ this.id = nextScopeId++;
35
+ const inferredParent = _nullishCoalesce(this.parentScopeId, () => ( _optionalChain([_chunkWQ5QNU5Rcjs.getCurrentFiber.call(void 0, ), 'optionalAccess', _ => _.scopeId])));
36
+ if (this.runtime.hasActiveHooks()) {
37
+ this.runtime.emit({
38
+ type: "scope.open",
39
+ scopeId: this.id,
40
+ parentScopeId: inferredParent
41
+ });
42
+ }
43
+ }
44
+
45
+
46
+
47
+ __init() {this.closed = false}
48
+ __init2() {this.children = /* @__PURE__ */ new Set()}
49
+ __init3() {this.subScopes = /* @__PURE__ */ new Set()}
50
+ __init4() {this.finalizers = []}
51
+ /** registra un finalizer (LIFO) */
52
+ addFinalizer(f) {
53
+ if (this.closed) {
54
+ throw new Error("Trying to add finalizer to closed scope");
55
+ }
56
+ this.finalizers.push(f);
57
+ }
58
+ /** crea un sub scope (mismo runtime) */
59
+ subScope() {
60
+ if (this.closed) throw new Error("Scope closed");
61
+ const s = new _Scope(this.runtime, this.id);
62
+ this.subScopes.add(s);
63
+ return s;
64
+ }
65
+ /** ✅ fork en este scope */
66
+ fork(eff) {
67
+ if (this.closed) throw new Error("Scope closed");
68
+ const f = this.runtime.fork(eff, this.id);
69
+ this.children.add(f);
70
+ f.join(() => this.children.delete(f));
71
+ return f;
72
+ }
73
+ /** close fire-and-forget (no bloquea) */
74
+ close(exit = { _tag: "Success", value: void 0 }) {
75
+ this.runtime.fork(this.closeAsync(exit));
76
+ }
77
+ /** Emit the scope.close event if hooks are active. */
78
+ emitCloseEvent(exit) {
79
+ if (this.runtime.hasActiveHooks()) {
80
+ const status = exit._tag === "Success" ? "success" : exit.cause._tag === "Interrupt" ? "interrupted" : "failure";
81
+ this.runtime.emit({
82
+ type: "scope.close",
83
+ scopeId: this.id,
84
+ status,
85
+ error: exit._tag === "Failure" && exit.cause._tag === "Fail" ? exit.cause.error : void 0
86
+ });
87
+ }
88
+ }
89
+ /**
90
+ * Build an effect that executes finalizers in LIFO order.
91
+ *
92
+ * Optimization over the original: instead of wrapping every finalizer in
93
+ * `asyncFold(fin(exit), () => unit(), () => unit())` which creates 3 effect
94
+ * nodes per finalizer (Fold + 2 Succeed), we use a single Sync thunk per
95
+ * finalizer that catches errors inline. When the finalizer returns a
96
+ * Succeed effect (like `unit()`), the Sync thunk completes without creating
97
+ * additional effect nodes.
98
+ */
99
+ buildFinalizerEffect(exit) {
100
+ const fins = this.finalizers;
101
+ if (fins.length === 0) return _chunkDJQ7OMMBcjs.unit.call(void 0, );
102
+ let chain = _chunkDJQ7OMMBcjs.unit.call(void 0, );
103
+ for (let i = fins.length - 1; i >= 0; i--) {
104
+ const fin = fins[i];
105
+ chain = _chunkDJQ7OMMBcjs.asyncFlatMap.call(void 0, chain, () => {
106
+ let result;
107
+ try {
108
+ result = fin(exit);
109
+ } catch (e) {
110
+ return _chunkDJQ7OMMBcjs.unit.call(void 0, );
111
+ }
112
+ if (result._tag === "Succeed") {
113
+ return _chunkDJQ7OMMBcjs.unit.call(void 0, );
114
+ }
115
+ return _chunkDJQ7OMMBcjs.asyncFold.call(void 0,
116
+ result,
117
+ () => _chunkDJQ7OMMBcjs.unit.call(void 0, ),
118
+ () => _chunkDJQ7OMMBcjs.unit.call(void 0, )
119
+ );
120
+ });
121
+ }
122
+ return chain;
123
+ }
124
+ closeAsync(exit = { _tag: "Success", value: void 0 }, opts = { awaitChildren: true }) {
125
+ return _chunkDJQ7OMMBcjs.asyncFlatMap.call(void 0,
126
+ _chunkDJQ7OMMBcjs.unit.call(void 0, ),
127
+ () => _chunkDJQ7OMMBcjs.asyncEffect.call(void 0, (env, cb) => {
128
+ if (this.closed) {
129
+ cb({ _tag: "Success", value: void 0 });
130
+ return;
131
+ }
132
+ this.closed = true;
133
+ const children = Array.from(this.children);
134
+ const subScopes = Array.from(this.subScopes);
135
+ for (const child of children) {
136
+ child.interrupt();
137
+ }
138
+ const closeSubs = subScopes.reduceRight(
139
+ (acc, s) => _chunkDJQ7OMMBcjs.asyncFlatMap.call(void 0, acc, () => s.closeAsync(exit, opts)),
140
+ _chunkDJQ7OMMBcjs.unit.call(void 0, )
141
+ );
142
+ const runFinalizers = this.buildFinalizerEffect(exit);
143
+ const needsAwait = opts.awaitChildren && children.length > 0;
144
+ const awaitChildrenEff = needsAwait ? awaitAll(children) : _chunkDJQ7OMMBcjs.unit.call(void 0, );
145
+ const hasSubScopes = subScopes.length > 0;
146
+ const hasNoFinalizers = this.finalizers.length === 0;
147
+ if (!hasSubScopes && !needsAwait && hasNoFinalizers) {
148
+ this.emitCloseEvent(exit);
149
+ cb({ _tag: "Success", value: void 0 });
150
+ return;
151
+ }
152
+ const all = _chunkDJQ7OMMBcjs.asyncFlatMap.call(void 0, closeSubs, () => _chunkDJQ7OMMBcjs.asyncFlatMap.call(void 0, awaitChildrenEff, () => runFinalizers));
153
+ this.runtime.fork(all).join(() => {
154
+ this.emitCloseEvent(exit);
155
+ cb({ _tag: "Success", value: void 0 });
156
+ });
157
+ })
158
+ );
159
+ }
160
+ }, _class);
161
+ function withScopeAsync(runtime, f) {
162
+ return _chunkDJQ7OMMBcjs.asyncEffect.call(void 0, (_env, cb) => {
163
+ const scope = new Scope(runtime);
164
+ let done = false;
165
+ const completeAfterClose = (exit) => {
166
+ runtime.fork(scope.closeAsync(exit)).join(() => {
167
+ if (done) return;
168
+ done = true;
169
+ cb(exit);
170
+ });
171
+ };
172
+ const fiber = runtime.fork(f(scope));
173
+ fiber.join(completeAfterClose);
174
+ return () => {
175
+ if (done) return;
176
+ fiber.interrupt();
177
+ runtime.fork(scope.closeAsync(_chunkDJQ7OMMBcjs.Exit.failCause(_chunkDJQ7OMMBcjs.Cause.interrupt())));
178
+ };
179
+ });
180
+ }
181
+ function withScope(runtime, f) {
182
+ return withScopeAsync(runtime, (scope) => {
183
+ const out = f(scope);
184
+ if (out && typeof out === "object" && "_tag" in out) return out;
185
+ return _chunkDJQ7OMMBcjs.unit.call(void 0, );
186
+ });
187
+ }
188
+
189
+
190
+
191
+
192
+
193
+ exports.Scope = Scope; exports.withScopeAsync = withScopeAsync; exports.withScope = withScope;