abxbus 2.4.1
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.
- package/README.md +647 -0
- package/dist/cjs/async_context.d.ts +12 -0
- package/dist/cjs/async_context.js +70 -0
- package/dist/cjs/async_context.js.map +7 -0
- package/dist/cjs/base_event.d.ts +207 -0
- package/dist/cjs/base_event.js +871 -0
- package/dist/cjs/base_event.js.map +7 -0
- package/dist/cjs/bridge_jsonl.d.ts +26 -0
- package/dist/cjs/bridge_jsonl.js +170 -0
- package/dist/cjs/bridge_jsonl.js.map +7 -0
- package/dist/cjs/bridge_nats.d.ts +20 -0
- package/dist/cjs/bridge_nats.js +108 -0
- package/dist/cjs/bridge_nats.js.map +7 -0
- package/dist/cjs/bridge_postgres.d.ts +31 -0
- package/dist/cjs/bridge_postgres.js +251 -0
- package/dist/cjs/bridge_postgres.js.map +7 -0
- package/dist/cjs/bridge_redis.d.ts +34 -0
- package/dist/cjs/bridge_redis.js +175 -0
- package/dist/cjs/bridge_redis.js.map +7 -0
- package/dist/cjs/bridge_sqlite.d.ts +30 -0
- package/dist/cjs/bridge_sqlite.js +255 -0
- package/dist/cjs/bridge_sqlite.js.map +7 -0
- package/dist/cjs/bridges.d.ts +49 -0
- package/dist/cjs/bridges.js +326 -0
- package/dist/cjs/bridges.js.map +7 -0
- package/dist/cjs/event_bus.d.ts +127 -0
- package/dist/cjs/event_bus.js +1058 -0
- package/dist/cjs/event_bus.js.map +7 -0
- package/dist/cjs/event_handler.d.ts +139 -0
- package/dist/cjs/event_handler.js +299 -0
- package/dist/cjs/event_handler.js.map +7 -0
- package/dist/cjs/event_history.d.ts +45 -0
- package/dist/cjs/event_history.js +192 -0
- package/dist/cjs/event_history.js.map +7 -0
- package/dist/cjs/event_result.d.ts +86 -0
- package/dist/cjs/event_result.js +446 -0
- package/dist/cjs/event_result.js.map +7 -0
- package/dist/cjs/events_suck.d.ts +40 -0
- package/dist/cjs/events_suck.js +59 -0
- package/dist/cjs/events_suck.js.map +7 -0
- package/dist/cjs/helpers.d.ts +1 -0
- package/dist/cjs/helpers.js +84 -0
- package/dist/cjs/helpers.js.map +7 -0
- package/dist/cjs/index.d.ts +17 -0
- package/dist/cjs/index.js +54 -0
- package/dist/cjs/index.js.map +7 -0
- package/dist/cjs/lock_manager.d.ts +70 -0
- package/dist/cjs/lock_manager.js +343 -0
- package/dist/cjs/lock_manager.js.map +7 -0
- package/dist/cjs/logging.d.ts +16 -0
- package/dist/cjs/logging.js +216 -0
- package/dist/cjs/logging.js.map +7 -0
- package/dist/cjs/middlewares.d.ts +13 -0
- package/dist/cjs/middlewares.js +17 -0
- package/dist/cjs/middlewares.js.map +7 -0
- package/dist/cjs/optional_deps.d.ts +3 -0
- package/dist/cjs/optional_deps.js +64 -0
- package/dist/cjs/optional_deps.js.map +7 -0
- package/dist/cjs/package.json +5 -0
- package/dist/cjs/retry.d.ts +52 -0
- package/dist/cjs/retry.js +257 -0
- package/dist/cjs/retry.js.map +7 -0
- package/dist/cjs/timing.d.ts +3 -0
- package/dist/cjs/timing.js +76 -0
- package/dist/cjs/timing.js.map +7 -0
- package/dist/cjs/type_inference.test.d.ts +1 -0
- package/dist/cjs/types.d.ts +36 -0
- package/dist/cjs/types.js +104 -0
- package/dist/cjs/types.js.map +7 -0
- package/dist/esm/async_context.js +50 -0
- package/dist/esm/async_context.js.map +7 -0
- package/dist/esm/base_event.js +857 -0
- package/dist/esm/base_event.js.map +7 -0
- package/dist/esm/bridge_jsonl.js +150 -0
- package/dist/esm/bridge_jsonl.js.map +7 -0
- package/dist/esm/bridge_nats.js +88 -0
- package/dist/esm/bridge_nats.js.map +7 -0
- package/dist/esm/bridge_postgres.js +231 -0
- package/dist/esm/bridge_postgres.js.map +7 -0
- package/dist/esm/bridge_redis.js +155 -0
- package/dist/esm/bridge_redis.js.map +7 -0
- package/dist/esm/bridge_sqlite.js +235 -0
- package/dist/esm/bridge_sqlite.js.map +7 -0
- package/dist/esm/bridges.js +306 -0
- package/dist/esm/bridges.js.map +7 -0
- package/dist/esm/event_bus.js +1046 -0
- package/dist/esm/event_bus.js.map +7 -0
- package/dist/esm/event_handler.js +279 -0
- package/dist/esm/event_handler.js.map +7 -0
- package/dist/esm/event_history.js +172 -0
- package/dist/esm/event_history.js.map +7 -0
- package/dist/esm/event_result.js +426 -0
- package/dist/esm/event_result.js.map +7 -0
- package/dist/esm/events_suck.js +39 -0
- package/dist/esm/events_suck.js.map +7 -0
- package/dist/esm/helpers.js +64 -0
- package/dist/esm/helpers.js.map +7 -0
- package/dist/esm/index.js +47 -0
- package/dist/esm/index.js.map +7 -0
- package/dist/esm/lock_manager.js +323 -0
- package/dist/esm/lock_manager.js.map +7 -0
- package/dist/esm/logging.js +196 -0
- package/dist/esm/logging.js.map +7 -0
- package/dist/esm/middlewares.js +1 -0
- package/dist/esm/middlewares.js.map +7 -0
- package/dist/esm/optional_deps.js +44 -0
- package/dist/esm/optional_deps.js.map +7 -0
- package/dist/esm/retry.js +237 -0
- package/dist/esm/retry.js.map +7 -0
- package/dist/esm/timing.js +56 -0
- package/dist/esm/timing.js.map +7 -0
- package/dist/esm/types.js +84 -0
- package/dist/esm/types.js.map +7 -0
- package/dist/types/async_context.d.ts +12 -0
- package/dist/types/base_event.d.ts +207 -0
- package/dist/types/bridge_jsonl.d.ts +26 -0
- package/dist/types/bridge_nats.d.ts +20 -0
- package/dist/types/bridge_postgres.d.ts +31 -0
- package/dist/types/bridge_redis.d.ts +34 -0
- package/dist/types/bridge_sqlite.d.ts +30 -0
- package/dist/types/bridges.d.ts +49 -0
- package/dist/types/event_bus.d.ts +127 -0
- package/dist/types/event_handler.d.ts +139 -0
- package/dist/types/event_history.d.ts +45 -0
- package/dist/types/event_result.d.ts +86 -0
- package/dist/types/events_suck.d.ts +40 -0
- package/dist/types/helpers.d.ts +1 -0
- package/dist/types/index.d.ts +17 -0
- package/dist/types/lock_manager.d.ts +70 -0
- package/dist/types/logging.d.ts +16 -0
- package/dist/types/middlewares.d.ts +13 -0
- package/dist/types/optional_deps.d.ts +3 -0
- package/dist/types/retry.d.ts +52 -0
- package/dist/types/timing.d.ts +3 -0
- package/dist/types/type_inference.test.d.ts +1 -0
- package/dist/types/types.d.ts +36 -0
- package/package.json +87 -0
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../src/base_event.ts"],
|
|
4
|
+
"sourcesContent": ["import { z } from 'zod'\nimport { v7 as uuidv7 } from 'uuid'\n\nimport { EventBus } from './event_bus.js'\nimport { EventResult } from './event_result.js'\nimport { EventHandler, EventHandlerAbortedError, EventHandlerCancelledError, EventHandlerTimeoutError } from './event_handler.js'\nimport type { EventConcurrencyMode, EventHandlerConcurrencyMode, EventHandlerCompletionMode, Deferred } from './lock_manager.js'\nimport {\n AsyncLock,\n EVENT_CONCURRENCY_MODES,\n EVENT_HANDLER_CONCURRENCY_MODES,\n EVENT_HANDLER_COMPLETION_MODES,\n withResolvers,\n} from './lock_manager.js'\nimport { _runWithTimeout } from './timing.js'\nimport { extractZodShape, normalizeEventResultType, toJsonSchema } from './types.js'\nimport type { EventHandlerCallable, EventResultType } from './types.js'\nimport { monotonicDatetime } from './helpers.js'\n\nconst RESERVED_USER_EVENT_FIELDS = new Set(['bus', 'first', 'toString', 'toJSON', 'fromJSON'])\n\nfunction assertNoReservedUserEventFields(data: Record<string, unknown>, context: string): void {\n for (const field_name of RESERVED_USER_EVENT_FIELDS) {\n if (Object.prototype.hasOwnProperty.call(data, field_name)) {\n throw new Error(`${context} field \"${field_name}\" is reserved for EventBus runtime context and cannot be set in event payload`)\n }\n }\n}\n\nfunction assertNoUnknownEventPrefixedFields(data: Record<string, unknown>, context: string): void {\n for (const field_name of Object.keys(data)) {\n if (field_name.startsWith('event_') && !KNOWN_BASE_EVENT_FIELDS.has(field_name)) {\n throw new Error(`${context} field \"${field_name}\" starts with \"event_\" but is not a recognized BaseEvent field`)\n }\n }\n}\n\nfunction assertNoModelPrefixedFields(data: Record<string, unknown>, context: string): void {\n for (const field_name of Object.keys(data)) {\n if (field_name.startsWith('model_')) {\n throw new Error(`${context} field \"${field_name}\" starts with \"model_\" and is reserved for model internals`)\n }\n }\n}\n\nfunction compareIsoDatetime(left: string | null | undefined, right: string | null | undefined): number {\n const left_value = left ?? ''\n const right_value = right ?? ''\n if (left_value === right_value) {\n return 0\n }\n return left_value < right_value ? -1 : 1\n}\n\nexport const BaseEventSchema = z\n .object({\n event_id: z.string().uuid(),\n event_created_at: z.string().datetime(),\n event_type: z.string(),\n event_version: z.string().default('0.0.1'),\n event_timeout: z.number().positive().nullable(),\n event_slow_timeout: z.number().positive().nullable().optional(),\n event_handler_timeout: z.number().positive().nullable().optional(),\n event_handler_slow_timeout: z.number().positive().nullable().optional(),\n event_parent_id: z.string().uuid().nullable().optional(),\n event_path: z.array(z.string()).optional(),\n event_result_type: z.unknown().optional(),\n event_emitted_by_handler_id: z.string().uuid().nullable().optional(),\n event_pending_bus_count: z.number().nonnegative().optional(),\n event_status: z.enum(['pending', 'started', 'completed']).optional(),\n event_started_at: z.string().datetime().nullable().optional(),\n event_completed_at: z.string().datetime().nullable().optional(),\n event_results: z.array(z.unknown()).optional(),\n event_concurrency: z.enum(EVENT_CONCURRENCY_MODES).nullable().optional(),\n event_handler_concurrency: z.enum(EVENT_HANDLER_CONCURRENCY_MODES).nullable().optional(),\n event_handler_completion: z.enum(EVENT_HANDLER_COMPLETION_MODES).nullable().optional(),\n })\n .loose()\n\nconst KNOWN_BASE_EVENT_FIELDS = new Set(Object.keys(BaseEventSchema.shape))\n\nexport type BaseEventData = z.infer<typeof BaseEventSchema>\nexport type BaseEventJSON = BaseEventData & Record<string, unknown>\ntype BaseEventFields = Pick<\n BaseEventData,\n | 'event_id'\n | 'event_created_at'\n | 'event_type'\n | 'event_version'\n | 'event_timeout'\n | 'event_slow_timeout'\n | 'event_handler_timeout'\n | 'event_handler_slow_timeout'\n | 'event_parent_id'\n | 'event_path'\n | 'event_result_type'\n | 'event_emitted_by_handler_id'\n | 'event_pending_bus_count'\n | 'event_status'\n | 'event_started_at'\n | 'event_completed_at'\n | 'event_results'\n | 'event_concurrency'\n | 'event_handler_concurrency'\n | 'event_handler_completion'\n>\n\nexport type BaseEventInit<TFields extends Record<string, unknown>> = TFields & Partial<BaseEventFields>\n\ntype BaseEventSchemaShape = typeof BaseEventSchema.shape\n\nexport type EventSchema<TShape extends z.ZodRawShape> = z.ZodObject<BaseEventSchemaShape & TShape>\ntype EventPayload<TShape extends z.ZodRawShape> = TShape extends Record<string, never> ? {} : z.infer<z.ZodObject<TShape>>\n\ntype EventInput<TShape extends z.ZodRawShape> = z.input<EventSchema<TShape>>\nexport type EventInit<TShape extends z.ZodRawShape> = Omit<EventInput<TShape>, keyof BaseEventFields> & Partial<BaseEventFields>\n\ntype EventWithResultSchema<TResult> = BaseEvent & { __event_result_type__?: TResult }\n\ntype ResultTypeFromEventResultTypeInput<TInput> = TInput extends z.ZodTypeAny\n ? z.infer<TInput>\n : TInput extends StringConstructor\n ? string\n : TInput extends NumberConstructor\n ? number\n : TInput extends BooleanConstructor\n ? boolean\n : TInput extends ArrayConstructor\n ? unknown[]\n : TInput extends ObjectConstructor\n ? Record<string, unknown>\n : unknown\n\ntype ResultSchemaFromShape<TShape> = TShape extends { event_result_type: infer S } ? ResultTypeFromEventResultTypeInput<S> : unknown\ntype EventResultsListInclude<TEvent extends BaseEvent> = (\n result: EventResultType<TEvent> | undefined,\n event_result: EventResult<TEvent>\n) => boolean\ntype EventResultsListOptions<TEvent extends BaseEvent> = {\n timeout?: number | null\n include?: EventResultsListInclude<TEvent>\n raise_if_any?: boolean\n raise_if_none?: boolean\n}\ntype EventDoneOptions = {\n raise_if_any?: boolean\n}\ntype EventResultUpdateOptions<TEvent extends BaseEvent> = {\n eventbus?: EventBus\n status?: 'pending' | 'started' | 'completed' | 'error'\n result?: EventResultType<TEvent> | BaseEvent | undefined\n error?: unknown\n}\n\nconst EVENT_CLASS_DEFAULTS = new WeakMap<Function, Record<string, unknown>>()\nconst ROOT_EVENTBUS_ID = '00000000-0000-0000-0000-000000000000'\n\nexport type EventFactory<TShape extends z.ZodRawShape, TResult = unknown> = {\n (data: EventInit<TShape>): EventWithResultSchema<TResult> & EventPayload<TShape>\n new (data: EventInit<TShape>): EventWithResultSchema<TResult> & EventPayload<TShape>\n schema: EventSchema<TShape>\n class?: new (data: EventInit<TShape>) => EventWithResultSchema<TResult> & EventPayload<TShape>\n event_type?: string\n event_version?: string\n event_result_type?: z.ZodTypeAny\n fromJSON?: (data: unknown) => EventWithResultSchema<TResult> & EventPayload<TShape>\n}\n\ntype ZodShapeFrom<TShape extends Record<string, unknown>> = {\n [K in keyof TShape as K extends 'event_result_type' ? never : TShape[K] extends z.ZodTypeAny ? K : never]: Extract<\n TShape[K],\n z.ZodTypeAny\n >\n}\n\nexport class BaseEvent {\n // event metadata fields\n event_id!: string // unique uuidv7 identifier for the event\n event_created_at!: string\n event_type!: string // should match the class name of the event, e.g. BaseEvent.extend(\"MyEvent\").event_type === \"MyEvent\"\n event_version!: string // event schema/version tag managed by callers for migration-friendly payload handling\n event_timeout!: number | null // maximum time in seconds that the event is allowed to run before it is aborted\n event_slow_timeout?: number | null // optional per-event slow warning threshold in seconds\n event_handler_timeout?: number | null // optional per-event handler timeout override in seconds\n event_handler_slow_timeout?: number | null // optional per-event slow handler warning threshold in seconds\n event_parent_id!: string | null // id of the parent event that triggered this event, if this event was emitted during handling of another event, else null\n event_path!: string[] // list of bus labels (name#id) that the event has been dispatched to, including the current bus\n event_result_type?: z.ZodTypeAny // optional zod schema to enforce the shape of return values from handlers\n event_results!: Map<string, EventResult<this>> // map of handler ids to EventResult objects for the event\n event_emitted_by_handler_id!: string | null // if event was emitted inside a handler while it was running, this is set to the enclosing handler's handler id, else null\n event_pending_bus_count!: number // number of buses that have accepted this event and not yet finished processing or removed it from their queues (for queue-jump processing)\n event_status!: 'pending' | 'started' | 'completed' // processing status of the event as a whole, no separate 'error' state because events can not error, only individual handlers can\n event_started_at!: string | null\n event_completed_at!: string | null\n event_concurrency?: EventConcurrencyMode | null // concurrency mode for the event as a whole in relation to other events\n event_handler_concurrency?: EventHandlerConcurrencyMode | null // concurrency mode for the handlers within the event\n event_handler_completion?: EventHandlerCompletionMode | null // completion strategy: 'all' (default) waits for every handler, 'first' returns earliest non-undefined result and cancels the rest\n\n static event_type?: string // class name of the event, e.g. BaseEvent.extend(\"MyEvent\").event_type === \"MyEvent\"\n static event_version = '0.0.1'\n static schema = BaseEventSchema // zod schema for the event data fields, used to parse and validate event data when creating a new event\n\n // internal runtime state\n bus?: EventBus // shortcut to the bus that dispatched this event, for event.bus.emit(event) auto-child tracking via proxy wrapping\n _event_original?: BaseEvent // underlying event object that was dispatched, if this is a bus-scoped proxy wrapping it\n _event_dispatch_context?: unknown | null // captured AsyncLocalStorage context at dispatch site, used to restore that context when running handlers\n\n _event_completed_signal: Deferred<this> | null\n _lock_for_event_handler: AsyncLock | null\n\n get event_bus(): EventBus {\n return this.bus as EventBus\n }\n\n constructor(data: BaseEventInit<Record<string, unknown>> = {}) {\n assertNoReservedUserEventFields(data as Record<string, unknown>, 'BaseEvent')\n assertNoUnknownEventPrefixedFields(data as Record<string, unknown>, 'BaseEvent')\n assertNoModelPrefixedFields(data as Record<string, unknown>, 'BaseEvent')\n const ctor = this.constructor as typeof BaseEvent & {\n event_version?: string\n event_result_type?: z.ZodTypeAny\n }\n const ctor_defaults = EVENT_CLASS_DEFAULTS.get(ctor) ?? {}\n const merged_data = {\n ...ctor_defaults,\n ...data,\n } as BaseEventInit<Record<string, unknown>>\n const event_type = merged_data.event_type ?? ctor.event_type ?? ctor.name\n const event_version = merged_data.event_version ?? ctor.event_version ?? '0.0.1'\n const raw_event_result_type = merged_data.event_result_type ?? ctor.event_result_type\n const event_result_type = normalizeEventResultType(raw_event_result_type)\n const event_id = merged_data.event_id ?? uuidv7()\n const event_created_at = monotonicDatetime(merged_data.event_created_at)\n const event_timeout = merged_data.event_timeout ?? null\n\n const base_data = {\n ...merged_data,\n event_id,\n event_created_at,\n event_type,\n event_version,\n event_timeout,\n event_result_type,\n }\n\n const schema = ctor.schema ?? BaseEventSchema\n const parsed = schema.parse(base_data) as BaseEventData & Record<string, unknown>\n\n Object.assign(this, parsed)\n\n const parsed_path = (parsed as { event_path?: string[] }).event_path\n this.event_path = Array.isArray(parsed_path) ? [...parsed_path] : []\n\n // load event results from potentially raw objects from JSON to proper EventResult objects\n this.event_results = hydrateEventResults(this, (parsed as { event_results?: unknown }).event_results)\n this.event_pending_bus_count =\n typeof (parsed as { event_pending_bus_count?: unknown }).event_pending_bus_count === 'number'\n ? Math.max(0, Number((parsed as { event_pending_bus_count?: number }).event_pending_bus_count))\n : 0\n const parsed_status = (parsed as { event_status?: unknown }).event_status\n this.event_status =\n parsed_status === 'pending' || parsed_status === 'started' || parsed_status === 'completed' ? parsed_status : 'pending'\n\n this.event_started_at =\n parsed.event_started_at === null || parsed.event_started_at === undefined ? null : monotonicDatetime(parsed.event_started_at)\n this.event_completed_at =\n parsed.event_completed_at === null || parsed.event_completed_at === undefined ? null : monotonicDatetime(parsed.event_completed_at)\n this.event_parent_id =\n typeof (parsed as { event_parent_id?: unknown }).event_parent_id === 'string'\n ? (parsed as { event_parent_id: string }).event_parent_id\n : null\n this.event_emitted_by_handler_id =\n typeof (parsed as { event_emitted_by_handler_id?: unknown }).event_emitted_by_handler_id === 'string'\n ? (parsed as { event_emitted_by_handler_id: string }).event_emitted_by_handler_id\n : null\n\n this.event_result_type = event_result_type\n\n this._event_completed_signal = null\n this._lock_for_event_handler = null\n this._event_dispatch_context = undefined\n }\n\n // \"MyEvent#a48f\"\n toString(): string {\n return `${this.event_type}#${this.event_id.slice(-4)}`\n }\n\n // main entry point for users to define their own event types\n // BaseEvent.extend(\"MyEvent\", { some_custom_field: z.string(), event_result_type: z.string(), event_timeout: 25, ... }) -> MyEvent\n static extend<TShape extends z.ZodRawShape>(event_type: string, shape?: TShape): EventFactory<TShape, ResultSchemaFromShape<TShape>>\n static extend<TShape extends Record<string, unknown>>(\n event_type: string,\n shape?: TShape\n ): EventFactory<ZodShapeFrom<TShape>, ResultSchemaFromShape<TShape>>\n static extend<TShape extends Record<string, unknown>>(\n event_type: string,\n shape: TShape = {} as TShape\n ): EventFactory<ZodShapeFrom<TShape>, ResultSchemaFromShape<TShape>> {\n const raw_shape = shape as Record<string, unknown>\n assertNoReservedUserEventFields(raw_shape, `BaseEvent.extend(${event_type})`)\n assertNoUnknownEventPrefixedFields(raw_shape, `BaseEvent.extend(${event_type})`)\n assertNoModelPrefixedFields(raw_shape, `BaseEvent.extend(${event_type})`)\n const raw_event_result_type = raw_shape.event_result_type\n const event_result_type = normalizeEventResultType(raw_event_result_type)\n const event_version = typeof raw_shape.event_version === 'string' ? raw_shape.event_version : undefined\n const event_defaults = Object.fromEntries(\n Object.entries(raw_shape).filter(\n ([key, value]) => key !== 'event_result_type' && key !== 'event_version' && !(value instanceof z.ZodType)\n )\n )\n\n const zod_shape = extractZodShape(raw_shape)\n const full_schema = BaseEventSchema.extend(zod_shape)\n\n // create a new event class that extends BaseEvent and adds the custom fields\n class ExtendedEvent extends BaseEvent {\n static schema = full_schema as unknown as typeof BaseEvent.schema\n static event_type = event_type\n static event_version = event_version ?? BaseEvent.event_version\n static event_result_type = event_result_type\n\n constructor(data: EventInit<ZodShapeFrom<TShape>>) {\n super(data as BaseEventInit<Record<string, unknown>>)\n }\n }\n\n type FactoryResult = EventWithResultSchema<ResultSchemaFromShape<TShape>> & EventPayload<ZodShapeFrom<TShape>>\n\n function EventFactory(data: EventInit<ZodShapeFrom<TShape>>): FactoryResult {\n return new ExtendedEvent(data) as FactoryResult\n }\n\n EventFactory.schema = full_schema as EventSchema<ZodShapeFrom<TShape>>\n EventFactory.event_type = event_type\n EventFactory.event_version = event_version ?? BaseEvent.event_version\n EventFactory.event_result_type = event_result_type\n EventFactory.class = ExtendedEvent as unknown as new (\n data: EventInit<ZodShapeFrom<TShape>>\n ) => EventWithResultSchema<ResultSchemaFromShape<TShape>> & EventPayload<ZodShapeFrom<TShape>>\n EventFactory.fromJSON = (data: unknown) => (ExtendedEvent.fromJSON as (data: unknown) => FactoryResult)(data)\n EventFactory.prototype = ExtendedEvent.prototype\n EVENT_CLASS_DEFAULTS.set(ExtendedEvent, event_defaults)\n\n return EventFactory as unknown as EventFactory<ZodShapeFrom<TShape>, ResultSchemaFromShape<TShape>>\n }\n\n static fromJSON<T extends typeof BaseEvent>(this: T, data: unknown): InstanceType<T> {\n if (!data || typeof data !== 'object') {\n const schema = this.schema ?? BaseEventSchema\n const parsed = schema.parse(data)\n return new this(parsed) as InstanceType<T>\n }\n const record = { ...(data as Record<string, unknown>) }\n if (record.event_result_type !== undefined && record.event_result_type !== null) {\n record.event_result_type = normalizeEventResultType(record.event_result_type)\n }\n return new this(record as BaseEventInit<Record<string, unknown>>) as InstanceType<T>\n }\n\n static toJSONArray(events: Iterable<BaseEvent>): BaseEventJSON[] {\n return Array.from(events, (event) => {\n const original = event._event_original ?? event\n return original.toJSON()\n })\n }\n\n static fromJSONArray(data: unknown): BaseEvent[] {\n if (!Array.isArray(data)) {\n return []\n }\n return data.map((item) => BaseEvent.fromJSON(item))\n }\n\n toJSON(): BaseEventJSON {\n const record: Record<string, unknown> = {}\n for (const [key, value] of Object.entries(this as unknown as Record<string, unknown>)) {\n if (key.startsWith('_') || key === 'bus' || key === 'event_results') continue\n if (value === undefined || typeof value === 'function') continue\n record[key] = value\n }\n const event_results = Array.from(this.event_results.values()).map((result) => result.toJSON())\n\n return {\n ...record,\n event_id: this.event_id,\n event_type: this.event_type,\n event_version: this.event_version,\n event_result_type: this.event_result_type ? toJsonSchema(this.event_result_type) : this.event_result_type,\n\n // static configuration options\n event_timeout: this.event_timeout,\n event_slow_timeout: this.event_slow_timeout,\n event_concurrency: this.event_concurrency,\n event_handler_concurrency: this.event_handler_concurrency,\n event_handler_completion: this.event_handler_completion,\n event_handler_slow_timeout: this.event_handler_slow_timeout,\n event_handler_timeout: this.event_handler_timeout,\n\n // mutable parent/child/bus tracking runtime state\n event_parent_id: this.event_parent_id,\n event_path: this.event_path,\n event_emitted_by_handler_id: this.event_emitted_by_handler_id,\n event_pending_bus_count: this.event_pending_bus_count,\n\n // mutable runtime status and timestamps\n event_status: this.event_status,\n event_created_at: this.event_created_at,\n event_started_at: this.event_started_at ?? null,\n event_completed_at: this.event_completed_at ?? null,\n\n // mutable result state\n ...(event_results.length > 0 ? { event_results } : {}),\n }\n }\n\n _createSlowEventWarningTimer(): ReturnType<typeof setTimeout> | null {\n const event_slow_timeout = this.event_slow_timeout ?? this.bus?.event_slow_timeout ?? null\n const event_warn_ms = event_slow_timeout === null ? null : event_slow_timeout * 1000\n if (event_warn_ms === null) {\n return null\n }\n const name = this.bus?.name ?? 'EventBus'\n return setTimeout(() => {\n if (this.event_status === 'completed') {\n return\n }\n const running_handler_count = [...this.event_results.values()].filter((result) => result.status === 'started').length\n const started_at = this.event_started_at ?? this.event_created_at\n const elapsed_ms = Math.max(0, Date.now() - Date.parse(started_at))\n const elapsed_seconds = (elapsed_ms / 1000).toFixed(2)\n console.warn(\n `[abxbus] Slow event processing: ${name}.on(${this.event_type}#${this.event_id.slice(-4)}, ${running_handler_count} handlers) still running after ${elapsed_seconds}s`\n )\n }, event_warn_ms)\n }\n\n eventResultUpdate(handler: EventHandler | EventHandlerCallable<this>, options: EventResultUpdateOptions<this> = {}): EventResult<this> {\n const original_event = (this._event_original ?? this) as this\n let resolved_eventbus = options.eventbus\n let handler_entry: EventHandler\n\n if (handler instanceof EventHandler) {\n handler_entry = handler\n if (!resolved_eventbus && handler_entry.eventbus_id !== ROOT_EVENTBUS_ID && original_event.bus) {\n resolved_eventbus =\n original_event.bus.all_instances.findBusById(handler_entry.eventbus_id) ??\n (original_event.bus.id === handler_entry.eventbus_id ? original_event.bus : undefined)\n }\n } else {\n handler_entry = EventHandler.fromCallable({\n handler,\n event_pattern: original_event.event_type,\n eventbus_name: resolved_eventbus?.name ?? 'EventBus',\n eventbus_id: resolved_eventbus?.id ?? ROOT_EVENTBUS_ID,\n })\n }\n\n const scoped_event = resolved_eventbus ? resolved_eventbus._getEventProxyScopedToThisBus(original_event) : original_event\n const handler_id = handler_entry.id\n const existing = original_event.event_results.get(handler_id)\n const event_result: EventResult<this> =\n existing ?? (new EventResult({ event: scoped_event as this, handler: handler_entry }) as EventResult<this>)\n if (!existing) {\n original_event.event_results.set(handler_id, event_result)\n } else {\n if (existing.event !== scoped_event) {\n existing.event = scoped_event as this\n }\n if (existing.handler.id !== handler_entry.id) {\n existing.handler = handler_entry\n }\n }\n\n if (options.status !== undefined || options.result !== undefined || options.error !== undefined) {\n const update_params: Parameters<EventResult<this>['update']>[0] = {}\n if (options.status !== undefined) update_params.status = options.status\n if (options.result !== undefined) update_params.result = options.result\n if (options.error !== undefined) update_params.error = options.error\n event_result.update(update_params)\n if (event_result.status === 'started' && event_result.started_at !== null) {\n original_event._markStarted(event_result.started_at, false)\n }\n if (options.status === 'pending' || options.status === 'started') {\n original_event.event_completed_at = null\n }\n }\n\n return event_result\n }\n\n _createPendingHandlerResults(bus: EventBus): Array<{\n handler: EventHandler\n result: EventResult\n }> {\n const original_event = this._event_original ?? this\n const scoped_event = bus._getEventProxyScopedToThisBus(original_event)\n const handlers = bus._getHandlersForEvent(original_event)\n return handlers.map((entry) => {\n const handler_id = entry.id\n const existing = original_event.event_results.get(handler_id)\n const result = existing ?? new EventResult({ event: scoped_event, handler: entry })\n if (!existing) {\n original_event.event_results.set(handler_id, result)\n } else if (existing.event !== scoped_event) {\n existing.event = scoped_event\n }\n return { handler: entry, result }\n })\n }\n\n private _collectPendingResults(\n original: BaseEvent,\n pending_entries?: Array<{\n handler: EventHandler\n result: EventResult\n }>\n ): EventResult[] {\n if (pending_entries) {\n return pending_entries.map((entry) => entry.result)\n }\n if (!this.bus?.id) {\n return Array.from(original.event_results.values())\n }\n return Array.from(original.event_results.values()).filter((result) => result.eventbus_id === this.bus!.id)\n }\n\n private _isFirstModeWinningResult(entry: EventResult): boolean {\n return entry.status === 'completed' && entry.result !== undefined && entry.result !== null && !(entry.result instanceof BaseEvent)\n }\n\n private _markFirstModeWinnerIfNeeded(original: BaseEvent, entry: EventResult, first_state: { found: boolean }): void {\n if (first_state.found || !this._isFirstModeWinningResult(entry)) {\n return\n }\n first_state.found = true\n original._markRemainingFirstModeResultCancelled(entry)\n }\n\n private async _runHandlerWithLock(original: BaseEvent, entry: EventResult): Promise<void> {\n if (!this.bus) {\n throw new Error('event has no bus attached')\n }\n await this.bus.locks._runWithHandlerLock(original, this.bus.event_handler_concurrency, async (handler_lock) => {\n await entry.runHandler(handler_lock)\n })\n }\n\n // Run all pending handler results for the current bus context.\n async _runHandlers(\n pending_entries?: Array<{\n handler: EventHandler\n result: EventResult\n }>\n ): Promise<void> {\n const original = this._event_original ?? this\n const pending_results = this._collectPendingResults(original, pending_entries)\n if (pending_results.length === 0) {\n return\n }\n const resolved_completion = original.event_handler_completion ?? this.bus?.event_handler_completion ?? 'all'\n if (resolved_completion === 'first') {\n if (original._getHandlerLock(this.bus?.event_handler_concurrency) !== null) {\n for (const entry of pending_results) {\n await this._runHandlerWithLock(original, entry)\n if (!this._isFirstModeWinningResult(entry)) {\n continue\n }\n original._markRemainingFirstModeResultCancelled(entry)\n break\n }\n return\n }\n const first_state = { found: false }\n const handler_promises = pending_results.map((entry) => this._runHandlerWithLock(original, entry))\n const monitored = pending_results.map((entry, index) =>\n handler_promises[index].then(() => {\n this._markFirstModeWinnerIfNeeded(original, entry, first_state)\n })\n )\n await Promise.all(monitored)\n return\n } else {\n const handler_promises = pending_results.map((entry) => this._runHandlerWithLock(original, entry))\n await Promise.all(handler_promises)\n }\n }\n\n _getHandlerLock(default_concurrency?: EventHandlerConcurrencyMode): AsyncLock | null {\n const original = this._event_original ?? this\n const resolved = original.event_handler_concurrency ?? default_concurrency ?? original.bus?.event_handler_concurrency ?? 'serial'\n if (resolved === 'parallel') {\n return null\n }\n if (!original._lock_for_event_handler) {\n original._lock_for_event_handler = new AsyncLock(1)\n }\n return original._lock_for_event_handler\n }\n\n _setHandlerLock(lock: AsyncLock | null): void {\n const original = this._event_original ?? this\n original._lock_for_event_handler = lock\n }\n\n _getDispatchContext(): unknown | null | undefined {\n const original = this._event_original ?? this\n return original._event_dispatch_context\n }\n\n _setDispatchContext(dispatch_context: unknown | null | undefined): void {\n const original = this._event_original ?? this\n original._event_dispatch_context = dispatch_context\n }\n\n // Get parent event object from event_parent_id (checks across all buses)\n get event_parent(): BaseEvent | undefined {\n const original = this._event_original ?? this\n const parent_id = original.event_parent_id\n if (!parent_id) {\n return undefined\n }\n return original.bus?.findEventById(parent_id) ?? undefined\n }\n\n // get all direct children of this event\n get event_children(): BaseEvent[] {\n const children: BaseEvent[] = []\n const seen = new Set<string>()\n for (const result of this.event_results.values()) {\n for (const child of result.event_children) {\n if (!seen.has(child.event_id)) {\n seen.add(child.event_id)\n children.push(child)\n }\n }\n }\n return children\n }\n\n // get all children grandchildren etc. recursively\n get event_descendants(): BaseEvent[] {\n const descendants: BaseEvent[] = []\n const visited = new Set<string>()\n const root_id = this.event_id\n const stack = [...this.event_children]\n\n while (stack.length > 0) {\n const child = stack.pop()\n if (!child) {\n continue\n }\n const child_id = child.event_id\n if (child_id === root_id) {\n continue\n }\n if (visited.has(child_id)) {\n continue\n }\n visited.add(child_id)\n descendants.push(child)\n if (child.event_children.length > 0) {\n stack.push(...child.event_children)\n }\n }\n\n return descendants\n }\n\n // force-abort processing of all pending descendants of an event regardless of whether they have already started\n _cancelPendingChildProcessing(reason: unknown): void {\n const original = this._event_original ?? this\n const cancellation_cause =\n reason instanceof EventHandlerTimeoutError\n ? reason\n : reason instanceof EventHandlerCancelledError || reason instanceof EventHandlerAbortedError\n ? reason.cause instanceof Error\n ? reason.cause\n : reason\n : reason instanceof Error\n ? reason\n : new Error(String(reason))\n const visited = new Set<string>()\n const cancelChildEvent = (child: BaseEvent): void => {\n const original_child = child._event_original ?? child\n if (visited.has(original_child.event_id)) {\n return\n }\n visited.add(original_child.event_id)\n\n // Depth-first: cancel grandchildren before parent so\n // _areAllChildrenComplete() returns true when we get back up.\n for (const grandchild of original_child.event_children) {\n cancelChildEvent(grandchild)\n }\n\n original_child._markCancelled(cancellation_cause)\n\n // Force-complete the child event. In JS we can't stop running async\n // handlers, but _markCompleted() resolves the done() promise so callers\n // aren't blocked waiting for background work to finish. The background\n // handler's eventual _markCompleted/_markError is a no-op (terminal guard).\n if (original_child.event_status !== 'completed') {\n original_child._markCompleted()\n }\n }\n\n for (const child of original.event_children) {\n cancelChildEvent(child)\n }\n }\n\n // Cancel all handler results for an event except the winner, used by first() mode.\n // Cancels pending handlers immediately, aborts started handlers via _signalAbort(),\n // and cancels any child events emitted by the losing handlers.\n _markRemainingFirstModeResultCancelled(winner: EventResult): void {\n const cause = new Error('first() resolved: another handler returned a result first')\n const bus_id = winner.eventbus_id\n\n for (const result of this.event_results.values()) {\n if (result === winner) continue\n if (result.eventbus_id !== bus_id) continue\n\n if (result.status === 'pending') {\n result._markError(\n new EventHandlerCancelledError(`Cancelled: first() resolved`, {\n event_result: result,\n cause,\n })\n )\n } else if (result.status === 'started') {\n // Cancel child events emitted by this handler before aborting it\n for (const child of result.event_children) {\n const original_child = child._event_original ?? child\n original_child._cancelPendingChildProcessing(cause)\n original_child._markCancelled(cause)\n }\n\n // Abort the handler itself\n result._lock?.exitHandlerRun()\n const aborted_error = new EventHandlerAbortedError(`Aborted: first() resolved`, {\n event_result: result,\n cause,\n })\n result._markError(aborted_error)\n result._signalAbort(aborted_error)\n }\n }\n }\n\n // force-abort processing of this event regardless of whether it is pending or has already started\n _markCancelled(cause: Error): void {\n const original = this._event_original ?? this\n if (!this.bus) {\n if (original.event_status !== 'completed') {\n original._markCompleted()\n }\n return\n }\n const path = Array.isArray(original.event_path) ? original.event_path : []\n const buses_to_cancel = new Set<string>(path)\n for (const bus of this.bus.all_instances) {\n if (!buses_to_cancel.has(bus.label)) {\n continue\n }\n\n const handler_entries = original._createPendingHandlerResults(bus)\n let updated = false\n for (const entry of handler_entries) {\n if (entry.result.status === 'pending') {\n const cancelled_error = new EventHandlerCancelledError(`Cancelled pending handler due to parent error: ${cause.message}`, {\n event_result: entry.result,\n cause,\n })\n entry.result._markError(cancelled_error)\n updated = true\n } else if (entry.result.status === 'started') {\n entry.result._lock?.exitHandlerRun()\n const aborted_error = new EventHandlerAbortedError(`Aborted running handler due to parent error: ${cause.message}`, {\n event_result: entry.result,\n cause,\n })\n entry.result._markError(aborted_error)\n entry.result._signalAbort(aborted_error)\n updated = true\n }\n }\n\n const removed = bus.removeEventFromPendingQueue(original)\n\n if (removed > 0 && !bus.isEventInFlightOrQueued(original.event_id)) {\n original.event_pending_bus_count = Math.max(0, original.event_pending_bus_count - 1)\n }\n\n if (updated || removed > 0) {\n original._markCompleted(false)\n }\n }\n\n if (original.event_status !== 'completed') {\n original._markCompleted()\n }\n }\n\n _notifyEventParentsOfCompletion(): void {\n const original = this._event_original ?? this\n if (!this.bus) {\n return\n }\n const visited = new Set<string>()\n let parent_id = original.event_parent_id\n while (parent_id && !visited.has(parent_id)) {\n visited.add(parent_id)\n const parent = this.bus.findEventById(parent_id)\n if (!parent) {\n break\n }\n parent._markCompleted(false, false)\n if (parent.event_status !== 'completed') {\n break\n }\n parent_id = parent.event_parent_id\n }\n }\n\n // awaitable that triggers immediate (queue-jump) processing of the event on all buses where it is queued\n // use eventCompleted() to wait for normal queue-order completion without queue-jumping.\n done(options: EventDoneOptions = {}): Promise<this> {\n if (!this.bus) {\n return Promise.reject(new Error('event has no bus attached'))\n }\n const original = this._event_original ?? this\n const raise_if_any = options.raise_if_any ?? true\n const completion_promise =\n this.event_status === 'completed' ? Promise.resolve(original as this) : this.bus._processEventImmediately(this)\n\n if (!raise_if_any) {\n return completion_promise\n }\n\n // Always delegate to _processEventImmediately \u2014 it walks up the parent event tree\n // to determine whether we're inside a handler (works cross-bus). If no\n // ancestor handler is in-flight, it falls back to eventCompleted().\n return completion_promise.then((completed_event) => {\n const first_error = completed_event._firstProcessingError()\n if (first_error !== undefined) {\n if (first_error instanceof Error) {\n throw first_error\n }\n throw new Error(String(first_error))\n }\n return completed_event\n })\n }\n\n // returns the first non-undefined handler result value, cancelling remaining handlers\n // when any handler completes. Works with all event_handler_concurrency modes:\n // parallel: races all handlers, returns first non-undefined, aborts the rest\n // serial: runs handlers sequentially, returns first non-undefined, skips remaining\n first(): Promise<EventResultType<this> | undefined> {\n if (!this.bus) {\n return Promise.reject(new Error('event has no bus attached'))\n }\n const original = this._event_original ?? this\n original.event_handler_completion = 'first'\n return this.done({ raise_if_any: false }).then((completed_event) => {\n const first_error = completed_event._firstProcessingError({ ignore_first_mode_control_errors: true })\n if (first_error !== undefined) {\n if (first_error instanceof Error) {\n throw first_error\n }\n throw new Error(String(first_error))\n }\n const orig = completed_event._event_original ?? completed_event\n return Array.from(orig.event_results.values())\n .filter(\n (result) =>\n result.status === 'completed' && result.result !== undefined && result.result !== null && !(result.result instanceof BaseEvent)\n )\n .sort((a, b) => compareIsoDatetime(a.completed_at, b.completed_at))\n .map((result) => result.result as EventResultType<this>)\n .at(0)\n })\n }\n\n // returns handler result values in event_results insertion order.\n // equivalent to await event.done(); Array.from(event.event_results.values()).map((entry) => entry.result)\n eventResultsList(\n include: EventResultsListInclude<this>,\n options?: EventResultsListOptions<this>\n ): Promise<Array<EventResultType<this> | undefined>>\n eventResultsList(options?: EventResultsListOptions<this>): Promise<Array<EventResultType<this> | undefined>>\n async eventResultsList(\n include_or_options?: EventResultsListInclude<this> | EventResultsListOptions<this>,\n maybe_options?: EventResultsListOptions<this>\n ): Promise<Array<EventResultType<this> | undefined>> {\n const default_include: EventResultsListInclude<this> = (_result, event_result) =>\n event_result.status === 'completed' &&\n event_result.result !== undefined &&\n event_result.result !== null &&\n !(event_result.result instanceof Error) &&\n !(event_result.result instanceof BaseEvent) &&\n event_result.error === undefined\n\n let options: EventResultsListOptions<this>\n let include: EventResultsListInclude<this>\n if (typeof include_or_options === 'function') {\n options = maybe_options ?? {}\n include = include_or_options\n } else {\n options = include_or_options ?? {}\n include = options.include ?? default_include\n }\n const raise_if_any = options.raise_if_any ?? true\n const raise_if_none = options.raise_if_none ?? true\n\n const original = this._event_original ?? this\n const resolved_timeout_seconds = options.timeout ?? original.event_timeout ?? this.bus?.event_timeout ?? null\n let completed_event: this\n\n if (resolved_timeout_seconds === null) {\n completed_event = await this.done({ raise_if_any: false })\n } else {\n completed_event = await _runWithTimeout(\n resolved_timeout_seconds,\n () => new Error(`Timed out waiting for ${original.event_type} results after ${resolved_timeout_seconds}s`),\n () => this.done({ raise_if_any: false })\n )\n }\n\n const all_results: EventResult<this>[] = Array.from(completed_event.event_results.values())\n const error_results = all_results.filter((event_result) => event_result.error !== undefined || event_result.result instanceof Error)\n\n if (raise_if_any && error_results.length > 0) {\n if (error_results.length === 1) {\n const first_error = error_results[0]\n if (first_error.error instanceof Error) {\n throw first_error.error\n }\n if (first_error.result instanceof Error) {\n throw first_error.result\n }\n throw new Error(String(first_error.error ?? first_error.result))\n }\n\n const errors = error_results.map((event_result) => {\n if (event_result.error instanceof Error) {\n return event_result.error\n }\n if (event_result.result instanceof Error) {\n return event_result.result\n }\n return new Error(String(event_result.error ?? event_result.result))\n })\n throw new AggregateError(\n errors,\n `Event ${completed_event.event_type}#${completed_event.event_id.slice(-4)} had ${errors.length} handler error(s)`\n )\n }\n\n const included_results = all_results.filter((event_result) => include(event_result.result, event_result))\n if (raise_if_none && included_results.length === 0) {\n throw new Error(\n `Expected at least one handler to return a non-null result, but none did: ${completed_event.event_type}#${completed_event.event_id.slice(-4)}`\n )\n }\n\n return included_results.map((event_result) => event_result.result)\n }\n\n // awaitable that waits for the event to be processed in normal queue order by the _runloop\n eventCompleted(): Promise<this> {\n if (this.event_status === 'completed') {\n return Promise.resolve(this)\n }\n this._notifyDoneListeners()\n return this._event_completed_signal!.promise\n }\n\n _markPending(): this {\n const original = this._event_original ?? this\n original.event_status = 'pending'\n original.event_started_at = null\n original.event_completed_at = null\n original.event_results.clear()\n original.event_pending_bus_count = 0\n original._setDispatchContext(undefined)\n original._event_completed_signal = null\n original._lock_for_event_handler = null\n original.bus = undefined\n return this\n }\n\n eventReset(): this {\n const original = this._event_original ?? this\n const ctor = original.constructor as typeof BaseEvent\n const fresh_event = ctor.fromJSON(original.toJSON()) as this\n fresh_event.event_id = uuidv7()\n return fresh_event._markPending()\n }\n\n _markStarted(started_at: string | null = null, notify_hook: boolean = true): void {\n const original = this._event_original ?? this\n if (original.event_status !== 'pending') {\n return\n }\n original.event_status = 'started'\n original.event_started_at = started_at === null ? monotonicDatetime() : monotonicDatetime(started_at)\n if (notify_hook && original.bus) {\n const bus_for_hook = original.bus\n const event_for_bus = bus_for_hook._getEventProxyScopedToThisBus(original)\n void bus_for_hook.onEventChange(event_for_bus, 'started')\n }\n }\n\n _markCompleted(force: boolean = true, notify_parents: boolean = true): void {\n const original = this._event_original ?? this\n if (original.event_status === 'completed') {\n return\n }\n if (!force) {\n if (original.event_pending_bus_count > 0) {\n return\n }\n if (!original._areAllChildrenComplete()) {\n return\n }\n }\n original.event_status = 'completed'\n original.event_completed_at = monotonicDatetime()\n if (original.bus) {\n const bus_for_hook = original.bus\n const event_for_bus = bus_for_hook._getEventProxyScopedToThisBus(original)\n void bus_for_hook.onEventChange(event_for_bus, 'completed')\n }\n original._setDispatchContext(null)\n original._notifyDoneListeners()\n original._event_completed_signal!.resolve(original)\n original._event_completed_signal = null\n original.dropFromZeroHistoryBuses()\n if (notify_parents && original.bus) {\n original._notifyEventParentsOfCompletion()\n }\n }\n\n private dropFromZeroHistoryBuses(): void {\n if (!this.bus) {\n return\n }\n const original = this._event_original ?? this\n for (const bus of this.bus.all_instances) {\n if (bus.event_history.max_history_size !== 0) {\n continue\n }\n bus.removeEventFromHistory(original.event_id)\n }\n }\n\n get event_errors(): unknown[] {\n return (\n Array.from(this.event_results.values())\n // filter for events that have completed + have non-undefined error values\n .filter((event_result) => event_result.error !== undefined && event_result.completed_at !== null)\n // sort by completion time\n .sort((event_result_a, event_result_b) => compareIsoDatetime(event_result_a.completed_at, event_result_b.completed_at))\n // assemble array of flat error values\n .map((event_result) => event_result.error)\n )\n }\n\n private _isFirstModeControlError(error: unknown): boolean {\n if (!(error instanceof EventHandlerCancelledError || error instanceof EventHandlerAbortedError)) {\n return false\n }\n if (error.message.includes('first() resolved')) {\n return true\n }\n return error.cause instanceof Error && error.cause.message.includes('first() resolved')\n }\n\n _firstProcessingError(options: { ignore_first_mode_control_errors?: boolean } = {}): unknown | undefined {\n const ignore_first_mode_control_errors = options.ignore_first_mode_control_errors ?? false\n return Array.from(this.event_results.values())\n .filter((event_result) => event_result.error !== undefined && event_result.completed_at !== null)\n .filter((event_result) => (ignore_first_mode_control_errors ? !this._isFirstModeControlError(event_result.error) : true))\n .sort((event_result_a, event_result_b) => compareIsoDatetime(event_result_a.completed_at, event_result_b.completed_at))\n .map((event_result) => event_result.error)\n .at(0)\n }\n\n // Returns the first non-undefined completed handler result, sorted by completion time.\n // Useful after first() or done() to get the winning result value.\n get event_result(): EventResultType<this> | undefined {\n return Array.from(this.event_results.values())\n .filter((event_result) => event_result.completed_at !== null && event_result.result !== undefined)\n .sort((event_result_a, event_result_b) => compareIsoDatetime(event_result_a.completed_at, event_result_b.completed_at))\n .map((event_result) => event_result.result as EventResultType<this>)\n .at(0)\n }\n\n _areAllChildrenComplete(): boolean {\n return this.event_descendants.every((descendant) => descendant.event_status === 'completed')\n }\n\n private _notifyDoneListeners(): void {\n if (this._event_completed_signal) {\n return\n }\n this._event_completed_signal = withResolvers<this>()\n }\n\n // Break internal reference chains so a completed event can be GC'd when\n // Evicted from event_history. Called by EventHistory.trimEventHistory().\n _gc(): void {\n this._event_completed_signal = null\n this._setDispatchContext(null)\n this.bus = undefined\n this._lock_for_event_handler = null\n for (const result of this.event_results.values()) {\n result.event_children = []\n }\n this.event_results.clear()\n }\n}\n\nconst hydrateEventResults = <TEvent extends BaseEvent>(event: TEvent, raw_event_results: unknown): Map<string, EventResult<TEvent>> => {\n const event_results = new Map<string, EventResult<TEvent>>()\n if (!Array.isArray(raw_event_results)) {\n return event_results\n }\n for (const item of raw_event_results) {\n const result = EventResult.fromJSON(event, item)\n const map_key = typeof result.handler_id === 'string' && result.handler_id.length > 0 ? result.handler_id : result.id\n event_results.set(map_key, result)\n }\n return event_results\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": []
|
|
7
|
+
}
|
|
@@ -0,0 +1,150 @@
|
|
|
1
|
+
import { BaseEvent } from "./base_event.js";
|
|
2
|
+
import { EventBus } from "./event_bus.js";
|
|
3
|
+
const isNodeRuntime = () => {
|
|
4
|
+
const maybe_process = globalThis.process;
|
|
5
|
+
return typeof maybe_process?.versions?.node === "string";
|
|
6
|
+
};
|
|
7
|
+
const importNodeModule = async (specifier) => {
|
|
8
|
+
const dynamic_import = Function("module_name", "return import(module_name)");
|
|
9
|
+
return dynamic_import(specifier);
|
|
10
|
+
};
|
|
11
|
+
const randomSuffix = () => Math.random().toString(36).slice(2, 10);
|
|
12
|
+
class JSONLEventBridge {
|
|
13
|
+
path;
|
|
14
|
+
poll_interval;
|
|
15
|
+
name;
|
|
16
|
+
inbound_bus;
|
|
17
|
+
running;
|
|
18
|
+
byte_offset;
|
|
19
|
+
pending_line;
|
|
20
|
+
listener_task;
|
|
21
|
+
constructor(path, poll_interval = 0.25, name) {
|
|
22
|
+
this.path = path;
|
|
23
|
+
this.poll_interval = poll_interval;
|
|
24
|
+
this.name = name ?? `JSONLEventBridge_${randomSuffix()}`;
|
|
25
|
+
this.inbound_bus = new EventBus(this.name, { max_history_size: 0 });
|
|
26
|
+
this.running = false;
|
|
27
|
+
this.byte_offset = 0;
|
|
28
|
+
this.pending_line = "";
|
|
29
|
+
this.listener_task = null;
|
|
30
|
+
this.dispatch = this.dispatch.bind(this);
|
|
31
|
+
this.emit = this.emit.bind(this);
|
|
32
|
+
this.on = this.on.bind(this);
|
|
33
|
+
}
|
|
34
|
+
on(event_pattern, handler) {
|
|
35
|
+
this.ensureStarted();
|
|
36
|
+
if (typeof event_pattern === "string") {
|
|
37
|
+
this.inbound_bus.on(event_pattern, handler);
|
|
38
|
+
return;
|
|
39
|
+
}
|
|
40
|
+
this.inbound_bus.on(event_pattern, handler);
|
|
41
|
+
}
|
|
42
|
+
async emit(event) {
|
|
43
|
+
this.ensureStarted();
|
|
44
|
+
const fs = await this.loadFs();
|
|
45
|
+
await fs.promises.mkdir(this.dirname(this.path), { recursive: true });
|
|
46
|
+
const payload = JSON.stringify(event.toJSON()) + "\n";
|
|
47
|
+
await fs.promises.appendFile(this.path, payload, "utf8");
|
|
48
|
+
}
|
|
49
|
+
async dispatch(event) {
|
|
50
|
+
return this.emit(event);
|
|
51
|
+
}
|
|
52
|
+
async start() {
|
|
53
|
+
if (this.running) return;
|
|
54
|
+
const fs = await this.loadFs();
|
|
55
|
+
await fs.promises.mkdir(this.dirname(this.path), { recursive: true });
|
|
56
|
+
await fs.promises.appendFile(this.path, "", "utf8");
|
|
57
|
+
const stats = await fs.promises.stat(this.path);
|
|
58
|
+
this.byte_offset = Number(stats.size ?? 0);
|
|
59
|
+
this.pending_line = "";
|
|
60
|
+
this.running = true;
|
|
61
|
+
this.listener_task = this.listenLoop();
|
|
62
|
+
}
|
|
63
|
+
async close() {
|
|
64
|
+
this.running = false;
|
|
65
|
+
await Promise.allSettled(this.listener_task ? [this.listener_task] : []);
|
|
66
|
+
this.listener_task = null;
|
|
67
|
+
this.inbound_bus.destroy();
|
|
68
|
+
}
|
|
69
|
+
ensureStarted() {
|
|
70
|
+
if (this.running || this.listener_task) return;
|
|
71
|
+
void this.start().catch((error) => {
|
|
72
|
+
console.error("[abxbus] JSONLEventBridge failed to start", error);
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
async listenLoop() {
|
|
76
|
+
while (this.running) {
|
|
77
|
+
try {
|
|
78
|
+
await this.pollNewLines();
|
|
79
|
+
} catch {
|
|
80
|
+
}
|
|
81
|
+
await new Promise((resolve) => setTimeout(resolve, Math.max(1, this.poll_interval * 1e3)));
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
async pollNewLines() {
|
|
85
|
+
const previous_offset = this.byte_offset;
|
|
86
|
+
const { chunk, next_offset } = await this.readAppended(previous_offset);
|
|
87
|
+
this.byte_offset = next_offset;
|
|
88
|
+
if (next_offset < previous_offset) {
|
|
89
|
+
this.pending_line = "";
|
|
90
|
+
}
|
|
91
|
+
if (!chunk) return;
|
|
92
|
+
const new_lines = (this.pending_line + chunk).split("\n");
|
|
93
|
+
this.pending_line = new_lines.pop() ?? "";
|
|
94
|
+
for (const line of new_lines) {
|
|
95
|
+
const trimmed = line.trim();
|
|
96
|
+
if (!trimmed) continue;
|
|
97
|
+
try {
|
|
98
|
+
const payload = JSON.parse(trimmed);
|
|
99
|
+
await this.dispatchInboundPayload(payload);
|
|
100
|
+
} catch {
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
async dispatchInboundPayload(payload) {
|
|
105
|
+
const event = BaseEvent.fromJSON(payload).eventReset();
|
|
106
|
+
this.inbound_bus.emit(event);
|
|
107
|
+
}
|
|
108
|
+
async readAppended(offset) {
|
|
109
|
+
const fs = await this.loadFs();
|
|
110
|
+
let size = 0;
|
|
111
|
+
try {
|
|
112
|
+
const stats = await fs.promises.stat(this.path);
|
|
113
|
+
size = Number(stats.size ?? 0);
|
|
114
|
+
} catch (error) {
|
|
115
|
+
const code = error.code;
|
|
116
|
+
if (code === "ENOENT") {
|
|
117
|
+
return { chunk: "", next_offset: 0 };
|
|
118
|
+
}
|
|
119
|
+
throw error;
|
|
120
|
+
}
|
|
121
|
+
const start_offset = size < offset ? 0 : offset;
|
|
122
|
+
if (size === start_offset) {
|
|
123
|
+
return { chunk: "", next_offset: size };
|
|
124
|
+
}
|
|
125
|
+
const handle = await fs.promises.open(this.path, "r");
|
|
126
|
+
try {
|
|
127
|
+
const byte_count = size - start_offset;
|
|
128
|
+
const bytes = new Uint8Array(byte_count);
|
|
129
|
+
const { bytesRead } = await handle.read(bytes, 0, byte_count, start_offset);
|
|
130
|
+
const chunk = new TextDecoder().decode(bytes.subarray(0, Number(bytesRead ?? 0)));
|
|
131
|
+
return { chunk, next_offset: start_offset + Number(bytesRead ?? 0) };
|
|
132
|
+
} finally {
|
|
133
|
+
await handle.close();
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
dirname(path) {
|
|
137
|
+
const idx = path.lastIndexOf("/");
|
|
138
|
+
return idx >= 0 ? path.slice(0, idx) || "." : ".";
|
|
139
|
+
}
|
|
140
|
+
async loadFs() {
|
|
141
|
+
if (!isNodeRuntime()) {
|
|
142
|
+
throw new Error("JSONLEventBridge is only supported in Node.js runtimes");
|
|
143
|
+
}
|
|
144
|
+
return importNodeModule("node:fs");
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
export {
|
|
148
|
+
JSONLEventBridge
|
|
149
|
+
};
|
|
150
|
+
//# sourceMappingURL=bridge_jsonl.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../src/bridge_jsonl.ts"],
|
|
4
|
+
"sourcesContent": ["import { BaseEvent } from './base_event.js'\nimport { EventBus } from './event_bus.js'\nimport type { EventClass, EventHandlerCallable, EventPattern, UntypedEventHandlerFunction } from './types.js'\n\nconst isNodeRuntime = (): boolean => {\n const maybe_process = (globalThis as { process?: { versions?: { node?: string } } }).process\n return typeof maybe_process?.versions?.node === 'string'\n}\n\nconst importNodeModule = async (specifier: string): Promise<any> => {\n const dynamic_import = Function('module_name', 'return import(module_name)') as (module_name: string) => Promise<unknown>\n return dynamic_import(specifier) as Promise<any>\n}\n\nconst randomSuffix = (): string => Math.random().toString(36).slice(2, 10)\n\nexport class JSONLEventBridge {\n readonly path: string\n readonly poll_interval: number\n readonly name: string\n\n private readonly inbound_bus: EventBus\n private running: boolean\n private byte_offset: number\n private pending_line: string\n private listener_task: Promise<void> | null\n\n constructor(path: string, poll_interval: number = 0.25, name?: string) {\n this.path = path\n this.poll_interval = poll_interval\n this.name = name ?? `JSONLEventBridge_${randomSuffix()}`\n this.inbound_bus = new EventBus(this.name, { max_history_size: 0 })\n this.running = false\n this.byte_offset = 0\n this.pending_line = ''\n this.listener_task = null\n\n this.dispatch = this.dispatch.bind(this)\n this.emit = this.emit.bind(this)\n this.on = this.on.bind(this)\n }\n\n on<T extends BaseEvent>(event_pattern: EventClass<T>, handler: EventHandlerCallable<T>): void\n on<T extends BaseEvent>(event_pattern: string | '*', handler: UntypedEventHandlerFunction<T>): void\n on(event_pattern: EventPattern | '*', handler: EventHandlerCallable | UntypedEventHandlerFunction): void {\n this.ensureStarted()\n if (typeof event_pattern === 'string') {\n this.inbound_bus.on(event_pattern, handler as UntypedEventHandlerFunction<BaseEvent>)\n return\n }\n this.inbound_bus.on(event_pattern as EventClass<BaseEvent>, handler as EventHandlerCallable<BaseEvent>)\n }\n\n async emit<T extends BaseEvent>(event: T): Promise<void> {\n this.ensureStarted()\n const fs = await this.loadFs()\n await fs.promises.mkdir(this.dirname(this.path), { recursive: true })\n const payload = JSON.stringify(event.toJSON()) + '\\n'\n await fs.promises.appendFile(this.path, payload, 'utf8')\n }\n\n async dispatch<T extends BaseEvent>(event: T): Promise<void> {\n return this.emit(event)\n }\n\n async start(): Promise<void> {\n if (this.running) return\n const fs = await this.loadFs()\n await fs.promises.mkdir(this.dirname(this.path), { recursive: true })\n await fs.promises.appendFile(this.path, '', 'utf8')\n const stats = await fs.promises.stat(this.path)\n this.byte_offset = Number(stats.size ?? 0)\n this.pending_line = ''\n this.running = true\n this.listener_task = this.listenLoop()\n }\n\n async close(): Promise<void> {\n this.running = false\n await Promise.allSettled(this.listener_task ? [this.listener_task] : [])\n this.listener_task = null\n this.inbound_bus.destroy()\n }\n\n private ensureStarted(): void {\n if (this.running || this.listener_task) return\n void this.start().catch((error: unknown) => {\n console.error('[abxbus] JSONLEventBridge failed to start', error)\n })\n }\n\n private async listenLoop(): Promise<void> {\n while (this.running) {\n try {\n await this.pollNewLines()\n } catch {\n // Keep polling on transient errors.\n }\n await new Promise((resolve) => setTimeout(resolve, Math.max(1, this.poll_interval * 1000)))\n }\n }\n\n private async pollNewLines(): Promise<void> {\n const previous_offset = this.byte_offset\n const { chunk, next_offset } = await this.readAppended(previous_offset)\n this.byte_offset = next_offset\n if (next_offset < previous_offset) {\n this.pending_line = ''\n }\n if (!chunk) return\n\n const new_lines = (this.pending_line + chunk).split('\\n')\n this.pending_line = new_lines.pop() ?? ''\n\n for (const line of new_lines) {\n const trimmed = line.trim()\n if (!trimmed) continue\n try {\n const payload = JSON.parse(trimmed)\n await this.dispatchInboundPayload(payload)\n } catch {\n // Ignore malformed line.\n }\n }\n }\n\n private async dispatchInboundPayload(payload: unknown): Promise<void> {\n const event = BaseEvent.fromJSON(payload).eventReset()\n this.inbound_bus.emit(event)\n }\n\n private async readAppended(offset: number): Promise<{ chunk: string; next_offset: number }> {\n const fs = await this.loadFs()\n let size = 0\n try {\n const stats = await fs.promises.stat(this.path)\n size = Number(stats.size ?? 0)\n } catch (error: unknown) {\n const code = (error as { code?: string }).code\n if (code === 'ENOENT') {\n return { chunk: '', next_offset: 0 }\n }\n throw error\n }\n\n const start_offset = size < offset ? 0 : offset\n if (size === start_offset) {\n return { chunk: '', next_offset: size }\n }\n\n const handle = await fs.promises.open(this.path, 'r')\n try {\n const byte_count = size - start_offset\n const bytes = new Uint8Array(byte_count)\n const { bytesRead } = await handle.read(bytes, 0, byte_count, start_offset)\n const chunk = new TextDecoder().decode(bytes.subarray(0, Number(bytesRead ?? 0)))\n return { chunk, next_offset: start_offset + Number(bytesRead ?? 0) }\n } finally {\n await handle.close()\n }\n }\n\n private dirname(path: string): string {\n const idx = path.lastIndexOf('/')\n return idx >= 0 ? path.slice(0, idx) || '.' : '.'\n }\n\n private async loadFs(): Promise<any> {\n if (!isNodeRuntime()) {\n throw new Error('JSONLEventBridge is only supported in Node.js runtimes')\n }\n return importNodeModule('node:fs')\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": []
|
|
7
|
+
}
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
import { BaseEvent } from "./base_event.js";
|
|
2
|
+
import { EventBus } from "./event_bus.js";
|
|
3
|
+
import { assertOptionalDependencyAvailable, importOptionalDependency, isNodeRuntime } from "./optional_deps.js";
|
|
4
|
+
const randomSuffix = () => Math.random().toString(36).slice(2, 10);
|
|
5
|
+
class NATSEventBridge {
|
|
6
|
+
server;
|
|
7
|
+
subject;
|
|
8
|
+
name;
|
|
9
|
+
inbound_bus;
|
|
10
|
+
running;
|
|
11
|
+
nc;
|
|
12
|
+
sub_task;
|
|
13
|
+
constructor(server, subject, name) {
|
|
14
|
+
assertOptionalDependencyAvailable("NATSEventBridge", "nats");
|
|
15
|
+
this.server = server;
|
|
16
|
+
this.subject = subject;
|
|
17
|
+
this.name = name ?? `NATSEventBridge_${randomSuffix()}`;
|
|
18
|
+
this.inbound_bus = new EventBus(this.name, { max_history_size: 0 });
|
|
19
|
+
this.running = false;
|
|
20
|
+
this.nc = null;
|
|
21
|
+
this.sub_task = null;
|
|
22
|
+
this.dispatch = this.dispatch.bind(this);
|
|
23
|
+
this.emit = this.emit.bind(this);
|
|
24
|
+
this.on = this.on.bind(this);
|
|
25
|
+
}
|
|
26
|
+
on(event_pattern, handler) {
|
|
27
|
+
this.ensureStarted();
|
|
28
|
+
if (typeof event_pattern === "string") {
|
|
29
|
+
this.inbound_bus.on(event_pattern, handler);
|
|
30
|
+
return;
|
|
31
|
+
}
|
|
32
|
+
this.inbound_bus.on(event_pattern, handler);
|
|
33
|
+
}
|
|
34
|
+
async emit(event) {
|
|
35
|
+
this.ensureStarted();
|
|
36
|
+
if (!this.nc) await this.start();
|
|
37
|
+
const payload = JSON.stringify(event.toJSON());
|
|
38
|
+
this.nc.publish(this.subject, new TextEncoder().encode(payload));
|
|
39
|
+
}
|
|
40
|
+
async dispatch(event) {
|
|
41
|
+
return this.emit(event);
|
|
42
|
+
}
|
|
43
|
+
async start() {
|
|
44
|
+
if (this.running) return;
|
|
45
|
+
if (!isNodeRuntime()) {
|
|
46
|
+
throw new Error("NATSEventBridge is only supported in Node.js runtimes");
|
|
47
|
+
}
|
|
48
|
+
const mod = await importOptionalDependency("NATSEventBridge", "nats");
|
|
49
|
+
const connect = mod.connect;
|
|
50
|
+
this.nc = await connect({ servers: this.server });
|
|
51
|
+
const sub = this.nc.subscribe(this.subject);
|
|
52
|
+
this.running = true;
|
|
53
|
+
this.sub_task = (async () => {
|
|
54
|
+
for await (const msg of sub) {
|
|
55
|
+
try {
|
|
56
|
+
const payload = JSON.parse(new TextDecoder().decode(msg.data));
|
|
57
|
+
await this.dispatchInboundPayload(payload);
|
|
58
|
+
} catch {
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
})();
|
|
62
|
+
}
|
|
63
|
+
async close() {
|
|
64
|
+
this.running = false;
|
|
65
|
+
if (this.nc) {
|
|
66
|
+
await this.nc.drain();
|
|
67
|
+
await this.nc.close();
|
|
68
|
+
this.nc = null;
|
|
69
|
+
}
|
|
70
|
+
await Promise.allSettled(this.sub_task ? [this.sub_task] : []);
|
|
71
|
+
this.sub_task = null;
|
|
72
|
+
this.inbound_bus.destroy();
|
|
73
|
+
}
|
|
74
|
+
ensureStarted() {
|
|
75
|
+
if (this.running) return;
|
|
76
|
+
void this.start().catch((error) => {
|
|
77
|
+
console.error("[abxbus] NATSEventBridge failed to start", error);
|
|
78
|
+
});
|
|
79
|
+
}
|
|
80
|
+
async dispatchInboundPayload(payload) {
|
|
81
|
+
const event = BaseEvent.fromJSON(payload).eventReset();
|
|
82
|
+
this.inbound_bus.emit(event);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
export {
|
|
86
|
+
NATSEventBridge
|
|
87
|
+
};
|
|
88
|
+
//# sourceMappingURL=bridge_nats.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../src/bridge_nats.ts"],
|
|
4
|
+
"sourcesContent": ["import { BaseEvent } from './base_event.js'\nimport { EventBus } from './event_bus.js'\nimport { assertOptionalDependencyAvailable, importOptionalDependency, isNodeRuntime } from './optional_deps.js'\nimport type { EventClass, EventHandlerCallable, EventPattern, UntypedEventHandlerFunction } from './types.js'\n\nconst randomSuffix = (): string => Math.random().toString(36).slice(2, 10)\n\nexport class NATSEventBridge {\n readonly server: string\n readonly subject: string\n readonly name: string\n\n private readonly inbound_bus: EventBus\n private running: boolean\n private nc: any | null\n private sub_task: Promise<void> | null\n\n constructor(server: string, subject: string, name?: string) {\n assertOptionalDependencyAvailable('NATSEventBridge', 'nats')\n\n this.server = server\n this.subject = subject\n this.name = name ?? `NATSEventBridge_${randomSuffix()}`\n this.inbound_bus = new EventBus(this.name, { max_history_size: 0 })\n this.running = false\n this.nc = null\n this.sub_task = null\n\n this.dispatch = this.dispatch.bind(this)\n this.emit = this.emit.bind(this)\n this.on = this.on.bind(this)\n }\n\n on<T extends BaseEvent>(event_pattern: EventClass<T>, handler: EventHandlerCallable<T>): void\n on<T extends BaseEvent>(event_pattern: string | '*', handler: UntypedEventHandlerFunction<T>): void\n on(event_pattern: EventPattern | '*', handler: EventHandlerCallable | UntypedEventHandlerFunction): void {\n this.ensureStarted()\n if (typeof event_pattern === 'string') {\n this.inbound_bus.on(event_pattern, handler as UntypedEventHandlerFunction<BaseEvent>)\n return\n }\n this.inbound_bus.on(event_pattern as EventClass<BaseEvent>, handler as EventHandlerCallable<BaseEvent>)\n }\n\n async emit<T extends BaseEvent>(event: T): Promise<void> {\n this.ensureStarted()\n if (!this.nc) await this.start()\n\n const payload = JSON.stringify(event.toJSON())\n this.nc.publish(this.subject, new TextEncoder().encode(payload))\n }\n\n async dispatch<T extends BaseEvent>(event: T): Promise<void> {\n return this.emit(event)\n }\n\n async start(): Promise<void> {\n if (this.running) return\n if (!isNodeRuntime()) {\n throw new Error('NATSEventBridge is only supported in Node.js runtimes')\n }\n\n const mod = await importOptionalDependency('NATSEventBridge', 'nats')\n const connect = mod.connect\n this.nc = await connect({ servers: this.server })\n const sub = this.nc.subscribe(this.subject)\n\n this.running = true\n this.sub_task = (async () => {\n for await (const msg of sub) {\n try {\n const payload = JSON.parse(new TextDecoder().decode(msg.data))\n await this.dispatchInboundPayload(payload)\n } catch {\n // Ignore malformed payloads.\n }\n }\n })()\n }\n\n async close(): Promise<void> {\n this.running = false\n if (this.nc) {\n await this.nc.drain()\n await this.nc.close()\n this.nc = null\n }\n await Promise.allSettled(this.sub_task ? [this.sub_task] : [])\n this.sub_task = null\n this.inbound_bus.destroy()\n }\n\n private ensureStarted(): void {\n if (this.running) return\n void this.start().catch((error: unknown) => {\n console.error('[abxbus] NATSEventBridge failed to start', error)\n })\n }\n\n private async dispatchInboundPayload(payload: unknown): Promise<void> {\n const event = BaseEvent.fromJSON(payload).eventReset()\n this.inbound_bus.emit(event)\n }\n}\n"],
|
|
5
|
+
"mappings": "AAAA,SAAS,iBAAiB;AAC1B,SAAS,gBAAgB;AACzB,SAAS,mCAAmC,0BAA0B,qBAAqB;AAG3F,MAAM,eAAe,MAAc,KAAK,OAAO,EAAE,SAAS,EAAE,EAAE,MAAM,GAAG,EAAE;AAElE,MAAM,gBAAgB;AAAA,EAClB;AAAA,EACA;AAAA,EACA;AAAA,EAEQ;AAAA,EACT;AAAA,EACA;AAAA,EACA;AAAA,EAER,YAAY,QAAgB,SAAiB,MAAe;AAC1D,sCAAkC,mBAAmB,MAAM;AAE3D,SAAK,SAAS;AACd,SAAK,UAAU;AACf,SAAK,OAAO,QAAQ,mBAAmB,aAAa,CAAC;AACrD,SAAK,cAAc,IAAI,SAAS,KAAK,MAAM,EAAE,kBAAkB,EAAE,CAAC;AAClE,SAAK,UAAU;AACf,SAAK,KAAK;AACV,SAAK,WAAW;AAEhB,SAAK,WAAW,KAAK,SAAS,KAAK,IAAI;AACvC,SAAK,OAAO,KAAK,KAAK,KAAK,IAAI;AAC/B,SAAK,KAAK,KAAK,GAAG,KAAK,IAAI;AAAA,EAC7B;AAAA,EAIA,GAAG,eAAmC,SAAmE;AACvG,SAAK,cAAc;AACnB,QAAI,OAAO,kBAAkB,UAAU;AACrC,WAAK,YAAY,GAAG,eAAe,OAAiD;AACpF;AAAA,IACF;AACA,SAAK,YAAY,GAAG,eAAwC,OAA0C;AAAA,EACxG;AAAA,EAEA,MAAM,KAA0B,OAAyB;AACvD,SAAK,cAAc;AACnB,QAAI,CAAC,KAAK,GAAI,OAAM,KAAK,MAAM;AAE/B,UAAM,UAAU,KAAK,UAAU,MAAM,OAAO,CAAC;AAC7C,SAAK,GAAG,QAAQ,KAAK,SAAS,IAAI,YAAY,EAAE,OAAO,OAAO,CAAC;AAAA,EACjE;AAAA,EAEA,MAAM,SAA8B,OAAyB;AAC3D,WAAO,KAAK,KAAK,KAAK;AAAA,EACxB;AAAA,EAEA,MAAM,QAAuB;AAC3B,QAAI,KAAK,QAAS;AAClB,QAAI,CAAC,cAAc,GAAG;AACpB,YAAM,IAAI,MAAM,uDAAuD;AAAA,IACzE;AAEA,UAAM,MAAM,MAAM,yBAAyB,mBAAmB,MAAM;AACpE,UAAM,UAAU,IAAI;AACpB,SAAK,KAAK,MAAM,QAAQ,EAAE,SAAS,KAAK,OAAO,CAAC;AAChD,UAAM,MAAM,KAAK,GAAG,UAAU,KAAK,OAAO;AAE1C,SAAK,UAAU;AACf,SAAK,YAAY,YAAY;AAC3B,uBAAiB,OAAO,KAAK;AAC3B,YAAI;AACF,gBAAM,UAAU,KAAK,MAAM,IAAI,YAAY,EAAE,OAAO,IAAI,IAAI,CAAC;AAC7D,gBAAM,KAAK,uBAAuB,OAAO;AAAA,QAC3C,QAAQ;AAAA,QAER;AAAA,MACF;AAAA,IACF,GAAG;AAAA,EACL;AAAA,EAEA,MAAM,QAAuB;AAC3B,SAAK,UAAU;AACf,QAAI,KAAK,IAAI;AACX,YAAM,KAAK,GAAG,MAAM;AACpB,YAAM,KAAK,GAAG,MAAM;AACpB,WAAK,KAAK;AAAA,IACZ;AACA,UAAM,QAAQ,WAAW,KAAK,WAAW,CAAC,KAAK,QAAQ,IAAI,CAAC,CAAC;AAC7D,SAAK,WAAW;AAChB,SAAK,YAAY,QAAQ;AAAA,EAC3B;AAAA,EAEQ,gBAAsB;AAC5B,QAAI,KAAK,QAAS;AAClB,SAAK,KAAK,MAAM,EAAE,MAAM,CAAC,UAAmB;AAC1C,cAAQ,MAAM,4CAA4C,KAAK;AAAA,IACjE,CAAC;AAAA,EACH;AAAA,EAEA,MAAc,uBAAuB,SAAiC;AACpE,UAAM,QAAQ,UAAU,SAAS,OAAO,EAAE,WAAW;AACrD,SAAK,YAAY,KAAK,KAAK;AAAA,EAC7B;AACF;",
|
|
6
|
+
"names": []
|
|
7
|
+
}
|