@open-mercato/core 0.6.5-develop.5337.1.534b781eac → 0.6.5-develop.5382.1.f542de69af

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 (119) hide show
  1. package/.turbo/turbo-build.log +1 -1
  2. package/AGENTS.md +1 -1
  3. package/dist/modules/attachments/api/library/route.js +2 -2
  4. package/dist/modules/attachments/api/library/route.js.map +2 -2
  5. package/dist/modules/attachments/components/AttachmentContentPreview.js +9 -5
  6. package/dist/modules/attachments/components/AttachmentContentPreview.js.map +2 -2
  7. package/dist/modules/audit_logs/api/audit-logs/actions/redo/route.js +3 -2
  8. package/dist/modules/audit_logs/api/audit-logs/actions/redo/route.js.map +2 -2
  9. package/dist/modules/auth/commands/users.js +20 -14
  10. package/dist/modules/auth/commands/users.js.map +2 -2
  11. package/dist/modules/auth/data/entities.js +1 -1
  12. package/dist/modules/auth/data/entities.js.map +2 -2
  13. package/dist/modules/auth/migrations/Migration20260610120000.js +30 -0
  14. package/dist/modules/auth/migrations/Migration20260610120000.js.map +7 -0
  15. package/dist/modules/catalog/ai-tools/configuration-pack.js.map +1 -1
  16. package/dist/modules/catalog/ai-tools/prices-offers-pack.js.map +1 -1
  17. package/dist/modules/catalog/ai-tools/products-pack.js.map +1 -1
  18. package/dist/modules/catalog/ai-tools/variants-pack.js.map +1 -1
  19. package/dist/modules/communication_channels/data/entities.js.map +1 -1
  20. package/dist/modules/communication_channels/encryption.js.map +1 -1
  21. package/dist/modules/communication_channels/lib/thread-matcher.js.map +1 -1
  22. package/dist/modules/communication_channels/lib/thread-token.js.map +1 -1
  23. package/dist/modules/currencies/api/currencies/route.js +4 -3
  24. package/dist/modules/currencies/api/currencies/route.js.map +2 -2
  25. package/dist/modules/customer_accounts/api/admin/roles.js +2 -1
  26. package/dist/modules/customer_accounts/api/admin/roles.js.map +2 -2
  27. package/dist/modules/customer_accounts/events.js +1 -1
  28. package/dist/modules/customer_accounts/events.js.map +1 -1
  29. package/dist/modules/customer_accounts/lib/resolveTenantContext.js.map +1 -1
  30. package/dist/modules/customers/acl.js +1 -1
  31. package/dist/modules/customers/acl.js.map +1 -1
  32. package/dist/modules/customers/ai-tools/companies-pack.js.map +1 -1
  33. package/dist/modules/customers/ai-tools/deals-pack.js.map +1 -1
  34. package/dist/modules/customers/ai-tools/people-pack.js.map +1 -1
  35. package/dist/modules/customers/api/companies/route.js +4 -4
  36. package/dist/modules/customers/api/companies/route.js.map +2 -2
  37. package/dist/modules/customers/api/people/route.js +4 -4
  38. package/dist/modules/customers/api/people/route.js.map +2 -2
  39. package/dist/modules/customers/commands/addresses.js +5 -5
  40. package/dist/modules/customers/commands/addresses.js.map +2 -2
  41. package/dist/modules/customers/commands/comments.js +5 -5
  42. package/dist/modules/customers/commands/comments.js.map +2 -2
  43. package/dist/modules/customers/commands/deals.js +2 -2
  44. package/dist/modules/customers/commands/deals.js.map +2 -2
  45. package/dist/modules/customers/commands/entity-roles.js +2 -1
  46. package/dist/modules/customers/commands/entity-roles.js.map +2 -2
  47. package/dist/modules/customers/commands/interactions.js +8 -5
  48. package/dist/modules/customers/commands/interactions.js.map +2 -2
  49. package/dist/modules/customers/commands/shared.js +21 -6
  50. package/dist/modules/customers/commands/shared.js.map +2 -2
  51. package/dist/modules/customers/commands/tags.js +3 -3
  52. package/dist/modules/customers/commands/tags.js.map +2 -2
  53. package/dist/modules/customers/components/detail/assignableStaff.js +21 -8
  54. package/dist/modules/customers/components/detail/assignableStaff.js.map +2 -2
  55. package/dist/modules/customers/migrations/Migration20260519120000_pipeline_stage_color_tones.js.map +1 -1
  56. package/dist/modules/data_sync/api/run.js +1 -1
  57. package/dist/modules/data_sync/api/run.js.map +2 -2
  58. package/dist/modules/payment_gateways/api/transactions/route.js +2 -4
  59. package/dist/modules/payment_gateways/api/transactions/route.js.map +2 -2
  60. package/dist/modules/progress/api/jobs/[id]/route.js +7 -2
  61. package/dist/modules/progress/api/jobs/[id]/route.js.map +2 -2
  62. package/dist/modules/progress/api/jobs/route.js +1 -1
  63. package/dist/modules/progress/api/jobs/route.js.map +2 -2
  64. package/dist/modules/progress/lib/progressServiceImpl.js +8 -2
  65. package/dist/modules/progress/lib/progressServiceImpl.js.map +2 -2
  66. package/dist/modules/resources/api/resources.js +2 -3
  67. package/dist/modules/resources/api/resources.js.map +2 -2
  68. package/dist/modules/sales/api/documents/factory.js +2 -2
  69. package/dist/modules/sales/api/documents/factory.js.map +2 -2
  70. package/dist/modules/sync_excel/api/import/route.js +1 -1
  71. package/dist/modules/sync_excel/api/import/route.js.map +2 -2
  72. package/dist/modules/workflows/api/definitions/route.js +3 -2
  73. package/dist/modules/workflows/api/definitions/route.js.map +2 -2
  74. package/package.json +7 -7
  75. package/src/modules/attachments/api/library/route.ts +2 -2
  76. package/src/modules/attachments/components/AttachmentContentPreview.tsx +6 -6
  77. package/src/modules/audit_logs/api/audit-logs/actions/redo/route.ts +14 -2
  78. package/src/modules/auth/commands/users.ts +32 -15
  79. package/src/modules/auth/data/entities.ts +11 -1
  80. package/src/modules/auth/migrations/.snapshot-open-mercato.json +0 -10
  81. package/src/modules/auth/migrations/Migration20260610120000.ts +53 -0
  82. package/src/modules/catalog/ai-tools/configuration-pack.ts +1 -1
  83. package/src/modules/catalog/ai-tools/prices-offers-pack.ts +1 -1
  84. package/src/modules/catalog/ai-tools/products-pack.ts +1 -1
  85. package/src/modules/catalog/ai-tools/variants-pack.ts +1 -1
  86. package/src/modules/communication_channels/data/entities.ts +2 -2
  87. package/src/modules/communication_channels/encryption.ts +1 -1
  88. package/src/modules/communication_channels/lib/adapter.ts +1 -1
  89. package/src/modules/communication_channels/lib/thread-matcher.ts +1 -1
  90. package/src/modules/communication_channels/lib/thread-token.ts +1 -1
  91. package/src/modules/currencies/api/currencies/route.ts +4 -3
  92. package/src/modules/customer_accounts/api/admin/roles.ts +2 -1
  93. package/src/modules/customer_accounts/events.ts +1 -1
  94. package/src/modules/customer_accounts/lib/resolveTenantContext.ts +2 -2
  95. package/src/modules/customers/acl.ts +1 -1
  96. package/src/modules/customers/ai-tools/companies-pack.ts +1 -1
  97. package/src/modules/customers/ai-tools/deals-pack.ts +1 -1
  98. package/src/modules/customers/ai-tools/people-pack.ts +1 -1
  99. package/src/modules/customers/api/companies/route.ts +4 -4
  100. package/src/modules/customers/api/people/route.ts +4 -4
  101. package/src/modules/customers/commands/addresses.ts +5 -5
  102. package/src/modules/customers/commands/comments.ts +5 -5
  103. package/src/modules/customers/commands/deals.ts +2 -2
  104. package/src/modules/customers/commands/entity-roles.ts +2 -1
  105. package/src/modules/customers/commands/interactions.ts +8 -5
  106. package/src/modules/customers/commands/shared.ts +26 -4
  107. package/src/modules/customers/commands/tags.ts +3 -3
  108. package/src/modules/customers/components/detail/assignableStaff.ts +32 -8
  109. package/src/modules/customers/migrations/Migration20260519120000_pipeline_stage_color_tones.ts +1 -1
  110. package/src/modules/data_sync/api/run.ts +1 -1
  111. package/src/modules/payment_gateways/api/transactions/route.ts +2 -5
  112. package/src/modules/progress/api/jobs/[id]/route.ts +6 -1
  113. package/src/modules/progress/api/jobs/route.ts +1 -1
  114. package/src/modules/progress/lib/progressServiceImpl.ts +7 -1
  115. package/src/modules/resources/api/resources.ts +2 -3
  116. package/src/modules/sales/api/documents/factory.ts +2 -2
  117. package/src/modules/staff/AGENTS.md +1 -1
  118. package/src/modules/sync_excel/api/import/route.ts +1 -1
  119. package/src/modules/workflows/api/definitions/route.ts +3 -2
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/modules/customers/commands/interactions.ts"],
4
- "sourcesContent": ["import { z } from 'zod'\nimport { registerCommand } from '@open-mercato/shared/lib/commands'\nimport type { CommandHandler } from '@open-mercato/shared/lib/commands'\nimport {\n parseWithCustomFields,\n setCustomFieldsIfAny,\n emitCrudSideEffects,\n emitCrudUndoSideEffects,\n requireId,\n normalizeAuthorUserId,\n} from '@open-mercato/shared/lib/commands/helpers'\nimport { DefaultDataEngine, type DataEngine } from '@open-mercato/shared/lib/data/engine'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport type { CommandRuntimeContext } from '@open-mercato/shared/lib/commands'\nimport { CustomerInteraction, CustomerEntity } from '../data/entities'\nimport {\n interactionCreateSchema,\n interactionUpdateSchema,\n interactionCompleteSchema,\n interactionCancelSchema,\n type InteractionCreateInput,\n type InteractionUpdateInput,\n type InteractionCompleteInput,\n type InteractionCancelInput,\n} from '../data/validators'\nimport {\n ensureOrganizationScope,\n ensureTenantScope,\n requireTimelineParentEntity,\n extractUndoPayload,\n emitQueryIndexUpsertEvents,\n requireDealInScope,\n resolveParentResourceKind,\n} from './shared'\nimport { resolveTranslations } from '@open-mercato/shared/lib/i18n/server'\nimport { resolveRedoSnapshot } from '@open-mercato/shared/lib/commands/redo'\nimport {\n loadCustomFieldSnapshot,\n buildCustomFieldResetMap,\n} from '@open-mercato/shared/lib/commands/customFieldSnapshots'\nimport { CrudHttpError } from '@open-mercato/shared/lib/crud/errors'\nimport { enforceRecordGoneIsConflict, enforceCommandOptimisticLock } from '@open-mercato/shared/lib/crud/optimistic-lock-command'\nimport { findOneWithDecryption } from '@open-mercato/shared/lib/encryption/find'\nimport type { CrudIndexerConfig, CrudEventsConfig } from '@open-mercato/shared/lib/crud/types'\nimport { recomputeNextInteraction } from '../lib/interactionProjection'\nimport { canChangeEmailVisibility } from '../lib/visibilityFilter'\n\nconst INTERACTION_ENTITY_ID = 'customers:customer_interaction'\nconst interactionCrudIndexer: CrudIndexerConfig<CustomerInteraction> = {\n entityType: 'customers:customer_interaction' as const,\n}\n\nconst interactionCrudEvents: CrudEventsConfig = {\n module: 'customers',\n entity: 'interaction',\n persistent: true,\n buildPayload: (ctx) => ({\n id: ctx.identifiers.id,\n organizationId: ctx.identifiers.organizationId,\n tenantId: ctx.identifiers.tenantId,\n entityId:\n ctx.entity && typeof ctx.entity === 'object' && 'entity' in (ctx.entity as Record<string, unknown>)\n ? (() => {\n const entityRef = (ctx.entity as CustomerInteraction).entity\n return typeof entityRef === 'string' ? entityRef : entityRef?.id ?? null\n })()\n : null,\n interactionType:\n ctx.entity && typeof ctx.entity === 'object' && 'interactionType' in (ctx.entity as Record<string, unknown>)\n ? (ctx.entity as CustomerInteraction).interactionType\n : null,\n status:\n ctx.entity && typeof ctx.entity === 'object' && 'status' in (ctx.entity as Record<string, unknown>)\n ? (ctx.entity as CustomerInteraction).status\n : null,\n source:\n ctx.entity && typeof ctx.entity === 'object' && 'source' in (ctx.entity as Record<string, unknown>)\n ? (ctx.entity as CustomerInteraction).source ?? null\n : null,\n ...(ctx.syncOrigin ? { syncOrigin: ctx.syncOrigin } : {}),\n }),\n}\n\ntype InteractionSnapshot = {\n interaction: {\n id: string\n organizationId: string\n tenantId: string\n entityId: string\n entityKind: string | null\n dealId: string | null\n interactionType: string\n title: string | null\n body: string | null\n status: string\n scheduledAt: Date | null\n occurredAt: Date | null\n priority: number | null\n authorUserId: string | null\n ownerUserId: string | null\n appearanceIcon: string | null\n appearanceColor: string | null\n source: string | null\n durationMinutes: number | null\n location: string | null\n allDay: boolean | null\n recurrenceRule: string | null\n recurrenceEnd: Date | null\n participants: Array<{ userId: string; name?: string; email?: string; status?: string }> | null\n reminderMinutes: number | null\n visibility: string | null\n linkedEntities: Array<{ id: string; type: string; label: string }> | null\n guestPermissions: { canInviteOthers?: boolean; canModify?: boolean; canSeeList?: boolean } | null\n }\n custom?: Record<string, unknown>\n}\n\ntype InteractionUndoPayload = {\n before?: InteractionSnapshot | null\n after?: InteractionSnapshot | null\n}\n\nasync function loadInteractionSnapshot(em: EntityManager, id: string): Promise<InteractionSnapshot | null> {\n const interaction = await findOneWithDecryption(em, CustomerInteraction, { id }, { populate: ['entity'] })\n if (!interaction) return null\n const custom = await loadCustomFieldSnapshot(em, {\n entityId: INTERACTION_ENTITY_ID,\n recordId: interaction.id,\n tenantId: interaction.tenantId,\n organizationId: interaction.organizationId,\n })\n const entityRef = interaction.entity\n const entityKind = (typeof entityRef === 'object' && entityRef !== null && 'kind' in entityRef)\n ? (entityRef as { kind: string }).kind\n : null\n return {\n interaction: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n entityId: typeof entityRef === 'string' ? entityRef : entityRef.id,\n entityKind,\n dealId: interaction.dealId ?? null,\n interactionType: interaction.interactionType,\n title: interaction.title ?? null,\n body: interaction.body ?? null,\n status: interaction.status,\n scheduledAt: interaction.scheduledAt ?? null,\n occurredAt: interaction.occurredAt ?? null,\n priority: interaction.priority ?? null,\n authorUserId: interaction.authorUserId ?? null,\n ownerUserId: interaction.ownerUserId ?? null,\n appearanceIcon: interaction.appearanceIcon ?? null,\n appearanceColor: interaction.appearanceColor ?? null,\n source: interaction.source ?? null,\n durationMinutes: interaction.durationMinutes ?? null,\n location: interaction.location ?? null,\n allDay: interaction.allDay ?? null,\n recurrenceRule: interaction.recurrenceRule ?? null,\n recurrenceEnd: interaction.recurrenceEnd ?? null,\n participants: interaction.participants ?? null,\n reminderMinutes: interaction.reminderMinutes ?? null,\n visibility: interaction.visibility ?? null,\n linkedEntities: interaction.linkedEntities ?? null,\n guestPermissions: interaction.guestPermissions ?? null,\n },\n custom,\n }\n}\n\nasync function setInteractionCustomFields(\n dataEngine: DataEngine,\n interactionId: string,\n organizationId: string,\n tenantId: string,\n values: Record<string, unknown>\n) {\n if (!values || !Object.keys(values).length) return\n await setCustomFieldsIfAny({\n dataEngine,\n entityId: INTERACTION_ENTITY_ID,\n recordId: interactionId,\n organizationId,\n tenantId,\n values,\n notify: false,\n })\n}\n\nasync function emitLifecycleEvent(\n ctx: CommandRuntimeContext,\n eventId: string,\n payload: Record<string, unknown>\n): Promise<void> {\n let bus: { emitEvent(event: string, payload: unknown, options?: unknown): Promise<void> } | null = null\n try {\n bus = ctx.container.resolve('eventBus')\n } catch (err) {\n console.warn('[customers.commands.interactions] eventBus resolve failed; skipping emit', eventId, err)\n bus = null\n }\n if (!bus) return\n await bus\n .emitEvent(eventId, payload, { persistent: true })\n .catch((err) => {\n console.warn('[customers.commands.interactions] emit failed', eventId, err)\n return undefined\n })\n}\n\nasync function emitInteractionRevertedEvent(\n ctx: CommandRuntimeContext,\n interaction: InteractionSnapshot['interaction'],\n): Promise<void> {\n await emitLifecycleEvent(ctx, 'customers.interaction.reverted', {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n entityId: interaction.entityId,\n interactionType: interaction.interactionType,\n source: interaction.source ?? null,\n status: interaction.status,\n occurredAt: interaction.occurredAt?.toISOString() ?? null,\n ...(ctx.syncOrigin ? { syncOrigin: ctx.syncOrigin } : {}),\n })\n}\n\ntype InteractionIdentifiers = {\n id: string\n organizationId: string\n tenantId: string\n}\n\ntype InteractionProjectionMutation = {\n entityId: string\n nextInteractionId: string | null\n}\n\nfunction createTransactionalDataEngine(ctx: CommandRuntimeContext, em: EntityManager): DataEngine {\n const emWithConnection = em as EntityManager & { getConnection?: () => unknown }\n if (typeof emWithConnection.getConnection !== 'function') {\n return ctx.container.resolve('dataEngine') as DataEngine\n }\n return new DefaultDataEngine(em, ctx.container)\n}\n\nasync function runInTransaction<TResult>(\n em: EntityManager,\n operation: (trx: EntityManager) => Promise<TResult>,\n): Promise<TResult> {\n // Mirrors the SPEC-018 fix applied to withAtomicFlush: use explicit begin/commit/rollback\n // so the outer EntityManager stays bound to the transaction, and closures over `em` inside\n // `operation` participate in the same transaction. This avoids the em.transactional(cb)\n // hazard where the callback receives a child EM whose flushes can silently race against\n // subsequent queries on the original `em`.\n const supportsBegin =\n typeof (em as unknown as { begin?: () => Promise<void> }).begin === 'function' &&\n typeof (em as unknown as { commit?: () => Promise<void> }).commit === 'function' &&\n typeof (em as unknown as { rollback?: () => Promise<void> }).rollback === 'function'\n if (!supportsBegin) {\n return operation(em)\n }\n await em.begin()\n try {\n const result = await operation(em)\n await em.commit()\n return result\n } catch (err) {\n try {\n await em.rollback()\n } catch {\n // rollback failure should not mask the original error; intentionally swallowed\n }\n throw err\n }\n}\n\nasync function emitNextInteractionUpdatedEvent(\n ctx: CommandRuntimeContext,\n projection: InteractionProjectionMutation,\n identifiers: InteractionIdentifiers,\n): Promise<void> {\n await emitQueryIndexUpsertEvents(ctx, [{\n entityType: 'customers:customer_entity',\n recordId: projection.entityId,\n organizationId: identifiers.organizationId,\n tenantId: identifiers.tenantId,\n }])\n await emitLifecycleEvent(ctx, 'customers.next_interaction.updated', {\n id: projection.entityId,\n entityId: projection.entityId,\n nextInteractionId: projection.nextInteractionId,\n organizationId: identifiers.organizationId,\n tenantId: identifiers.tenantId,\n })\n}\n\n// \u2500\u2500\u2500 Create \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\ntype InteractionGraphValues = {\n id?: string\n organizationId: string\n tenantId: string\n entity: CustomerEntity\n interactionType: string\n title: string | null\n body: string | null\n status: string\n scheduledAt: Date | null\n occurredAt: Date | null\n priority: number | null\n authorUserId: string | null\n ownerUserId: string | null\n dealId: string | null\n source: string | null\n appearanceIcon: string | null\n appearanceColor: string | null\n durationMinutes: number | null\n location: string | null\n allDay: boolean | null\n recurrenceRule: string | null\n recurrenceEnd: Date | null\n participants: InteractionSnapshot['interaction']['participants']\n reminderMinutes: number | null\n visibility: string | null\n linkedEntities: InteractionSnapshot['interaction']['linkedEntities']\n guestPermissions: InteractionSnapshot['interaction']['guestPermissions']\n}\n\n// Single source of truth for the interaction row mapping, shared by `execute`\n// (values from validated input) and `redo` (values from the after-snapshot,\n// carrying the original id). Keeps create and id-preserving redo from drifting\n// field-by-field. Caller owns the surrounding transaction, persist, projection\n// recompute, and custom-field handling.\nfunction buildInteractionGraph(em: EntityManager, values: InteractionGraphValues): CustomerInteraction {\n return em.create(CustomerInteraction, {\n ...(values.id ? { id: values.id } : {}),\n organizationId: values.organizationId,\n tenantId: values.tenantId,\n entity: values.entity,\n interactionType: values.interactionType,\n title: values.title,\n body: values.body,\n status: values.status,\n scheduledAt: values.scheduledAt,\n occurredAt: values.occurredAt,\n priority: values.priority,\n authorUserId: values.authorUserId,\n ownerUserId: values.ownerUserId,\n dealId: values.dealId,\n source: values.source,\n appearanceIcon: values.appearanceIcon,\n appearanceColor: values.appearanceColor,\n durationMinutes: values.durationMinutes,\n location: values.location,\n allDay: values.allDay,\n recurrenceRule: values.recurrenceRule,\n recurrenceEnd: values.recurrenceEnd,\n participants: values.participants,\n reminderMinutes: values.reminderMinutes,\n visibility: values.visibility,\n linkedEntities: values.linkedEntities,\n guestPermissions: values.guestPermissions,\n createdAt: new Date(),\n updatedAt: new Date(),\n })\n}\n\nconst createInteractionCommand: CommandHandler<InteractionCreateInput, { interactionId: string; entityId: string }> = {\n id: 'customers.interactions.create',\n async execute(rawInput, ctx) {\n const { parsed, custom } = parseWithCustomFields(interactionCreateSchema, rawInput)\n\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const normalizedAuthor = normalizeAuthorUserId(parsed.authorUserId ?? null, ctx.auth)\n const { interaction, entityId, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const entity = await requireTimelineParentEntity(trx, parsed.entityId)\n ensureTenantScope(ctx, entity.tenantId)\n ensureOrganizationScope(ctx, entity.organizationId)\n\n if (parsed.dealId) {\n await requireDealInScope(trx, parsed.dealId, entity.tenantId, entity.organizationId)\n }\n\n const interaction = buildInteractionGraph(trx, {\n ...(parsed.id ? { id: parsed.id } : {}),\n organizationId: entity.organizationId,\n tenantId: entity.tenantId,\n entity,\n interactionType: parsed.interactionType,\n title: parsed.title ?? null,\n body: parsed.body ?? null,\n status: parsed.status ?? 'planned',\n scheduledAt: parsed.scheduledAt ?? null,\n occurredAt: parsed.occurredAt ?? null,\n priority: parsed.priority ?? null,\n authorUserId: normalizedAuthor,\n ownerUserId: parsed.ownerUserId ?? null,\n dealId: parsed.dealId ?? null,\n source: parsed.source ?? null,\n appearanceIcon: parsed.appearanceIcon ?? null,\n appearanceColor: parsed.appearanceColor ?? null,\n durationMinutes: parsed.durationMinutes ?? null,\n location: parsed.location ?? null,\n allDay: parsed.allDay ?? null,\n recurrenceRule: parsed.recurrenceRule ?? null,\n recurrenceEnd: parsed.recurrenceEnd ?? null,\n participants: parsed.participants ?? null,\n reminderMinutes: parsed.reminderMinutes ?? null,\n visibility: parsed.visibility ?? null,\n linkedEntities: parsed.linkedEntities ?? null,\n guestPermissions: parsed.guestPermissions ?? null,\n })\n trx.persist(interaction)\n await trx.flush()\n\n await setInteractionCustomFields(\n createTransactionalDataEngine(ctx, trx),\n interaction.id,\n entity.organizationId,\n entity.tenantId,\n custom,\n )\n\n const projection = await recomputeNextInteraction(trx, entity.id)\n\n return {\n interaction,\n entityId: entity.id,\n nextInteractionId: projection.nextInteractionId,\n }\n })\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'created',\n entity: interaction,\n identifiers: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitNextInteractionUpdatedEvent(ctx, { entityId, nextInteractionId }, {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n })\n\n return { interactionId: interaction.id, entityId }\n },\n captureAfter: async (_input, result, ctx) => {\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n return await loadInteractionSnapshot(em, result.interactionId)\n },\n buildLog: async ({ result, snapshots }) => {\n const { translate } = await resolveTranslations()\n const snapshot = snapshots.after as InteractionSnapshot | undefined\n return {\n actionLabel: translate('customers.audit.interactions.create', 'Create interaction'),\n resourceKind: 'customers.interaction',\n resourceId: result.interactionId,\n parentResourceKind: resolveParentResourceKind(snapshot?.interaction?.entityKind),\n parentResourceId: snapshot?.interaction?.entityId ?? null,\n tenantId: snapshot?.interaction.tenantId ?? null,\n organizationId: snapshot?.interaction.organizationId ?? null,\n snapshotAfter: snapshot ?? null,\n payload: {\n undo: {\n after: snapshot,\n } satisfies InteractionUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const interactionId = logEntry?.resourceId\n if (!interactionId) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const result = await runInTransaction(em, async (trx) => {\n const record = await findOneWithDecryption(trx, CustomerInteraction, { id: interactionId })\n if (!record) return null\n const entityId = typeof record.entity === 'string' ? record.entity : record.entity.id\n trx.remove(record)\n await trx.flush()\n const projection = await recomputeNextInteraction(trx, entityId)\n return {\n entityId,\n nextInteractionId: projection.nextInteractionId,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n },\n }\n })\n if (!result) return\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: result.entityId,\n nextInteractionId: result.nextInteractionId,\n }, result.identifiers)\n },\n redo: async ({ logEntry, ctx }) => {\n const after = resolveRedoSnapshot<InteractionSnapshot>(logEntry)\n if (!after) {\n throw new CrudHttpError(400, { error: '[internal] redo snapshot unavailable for interaction create' })\n }\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const entity = await requireTimelineParentEntity(trx, after.interaction.entityId)\n let interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: after.interaction.id })\n if (!interaction) {\n interaction = buildInteractionGraph(trx, {\n id: after.interaction.id,\n organizationId: after.interaction.organizationId,\n tenantId: after.interaction.tenantId,\n entity,\n interactionType: after.interaction.interactionType,\n title: after.interaction.title,\n body: after.interaction.body,\n status: after.interaction.status,\n scheduledAt: after.interaction.scheduledAt,\n occurredAt: after.interaction.occurredAt,\n priority: after.interaction.priority,\n authorUserId: after.interaction.authorUserId,\n ownerUserId: after.interaction.ownerUserId,\n dealId: after.interaction.dealId,\n source: after.interaction.source,\n appearanceIcon: after.interaction.appearanceIcon,\n appearanceColor: after.interaction.appearanceColor,\n durationMinutes: after.interaction.durationMinutes,\n location: after.interaction.location,\n allDay: after.interaction.allDay,\n recurrenceRule: after.interaction.recurrenceRule,\n recurrenceEnd: after.interaction.recurrenceEnd,\n participants: after.interaction.participants,\n reminderMinutes: after.interaction.reminderMinutes,\n visibility: after.interaction.visibility,\n linkedEntities: after.interaction.linkedEntities,\n guestPermissions: after.interaction.guestPermissions,\n })\n trx.persist(interaction)\n } else {\n interaction.deletedAt = null\n interaction.entity = entity\n interaction.interactionType = after.interaction.interactionType\n interaction.title = after.interaction.title\n interaction.body = after.interaction.body\n interaction.status = after.interaction.status\n interaction.scheduledAt = after.interaction.scheduledAt\n interaction.occurredAt = after.interaction.occurredAt\n interaction.priority = after.interaction.priority\n interaction.authorUserId = after.interaction.authorUserId\n interaction.ownerUserId = after.interaction.ownerUserId\n interaction.dealId = after.interaction.dealId\n interaction.source = after.interaction.source\n interaction.appearanceIcon = after.interaction.appearanceIcon\n interaction.appearanceColor = after.interaction.appearanceColor\n interaction.durationMinutes = after.interaction.durationMinutes\n interaction.location = after.interaction.location\n interaction.allDay = after.interaction.allDay\n interaction.recurrenceRule = after.interaction.recurrenceRule\n interaction.recurrenceEnd = after.interaction.recurrenceEnd\n interaction.participants = after.interaction.participants\n interaction.reminderMinutes = after.interaction.reminderMinutes\n interaction.visibility = after.interaction.visibility\n interaction.linkedEntities = after.interaction.linkedEntities\n interaction.guestPermissions = after.interaction.guestPermissions\n }\n await trx.flush()\n\n const projection = await recomputeNextInteraction(trx, after.interaction.entityId)\n\n const restoreValues = buildCustomFieldResetMap(after.custom, undefined)\n if (Object.keys(restoreValues).length) {\n await setCustomFieldsIfAny({\n dataEngine: createTransactionalDataEngine(ctx, trx),\n entityId: INTERACTION_ENTITY_ID,\n recordId: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n values: restoreValues,\n notify: false,\n })\n }\n\n return { interaction, nextInteractionId: projection.nextInteractionId }\n })\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'created',\n entity: interaction,\n identifiers: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: after.interaction.entityId,\n nextInteractionId,\n }, {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n })\n\n return { interactionId: interaction.id, entityId: after.interaction.entityId }\n },\n}\n\n// \u2500\u2500\u2500 Update \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nconst updateInteractionCommand: CommandHandler<InteractionUpdateInput, { interactionId: string }> = {\n id: 'customers.interactions.update',\n async prepare(rawInput, ctx) {\n const { parsed } = parseWithCustomFields(interactionUpdateSchema, rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const snapshot = await loadInteractionSnapshot(em, parsed.id)\n return snapshot ? { before: snapshot } : {}\n },\n async execute(rawInput, ctx) {\n const { parsed, custom } = parseWithCustomFields(interactionUpdateSchema, rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, entityId, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: parsed.id, deletedAt: null })\n if (!interaction) {\n enforceRecordGoneIsConflict({ resourceKind: 'customers.interaction', resourceId: parsed.id, request: ctx.request ?? null })\n throw new CrudHttpError(404, { error: 'Interaction not found' })\n }\n ensureTenantScope(ctx, interaction.tenantId)\n ensureOrganizationScope(ctx, interaction.organizationId)\n\n // Concurrent-edit guard for command-driven callers (e.g. the legacy\n // /api/customers/todos route, which bypasses the makeCrudRoute lock guard):\n // when the client opted into optimistic locking, a stale edit fails with the\n // unified 409 instead of silently overwriting (#2055). Strictly additive \u2014\n // no-op when no expected-version header is present.\n enforceCommandOptimisticLock({\n resourceKind: 'customers.interaction',\n resourceId: interaction.id,\n current: interaction.updatedAt,\n request: ctx.request ?? null,\n })\n\n // Email visibility is an access-controlled field: only the interaction's\n // author may change a\n // private email's visibility (mirrors the dedicated PATCH .../visibility\n // route). Enforce it here \u2014 the single persistence path \u2014 so the generic\n // update route (PUT /api/interactions) cannot bypass the gate. Evaluated\n // against the row's pre-mutation author/type. 404 (not 403) keeps the\n // existence-masking consistent with the dedicated route.\n if (\n parsed.visibility !== undefined &&\n interaction.interactionType === 'email' &&\n (parsed.visibility ?? null) !== (interaction.visibility ?? null)\n ) {\n const actorUserId = (ctx.auth as { sub?: string | null } | null)?.sub ?? null\n if (\n !canChangeEmailVisibility({\n interactionType: interaction.interactionType,\n currentVisibility: interaction.visibility,\n nextVisibility: parsed.visibility,\n authorUserId: interaction.authorUserId,\n actorUserId,\n // v1 strict owner-only: only the author may flip visibility; no admin\n // bypass (canChangeEmailVisibility ignores caller features in v1).\n userFeatures: undefined,\n })\n ) {\n throw new CrudHttpError(404, { error: 'Email not found' })\n }\n }\n\n if (parsed.dealId !== undefined) {\n if (parsed.dealId) {\n await requireDealInScope(trx, parsed.dealId, interaction.tenantId, interaction.organizationId)\n }\n interaction.dealId = parsed.dealId ?? null\n }\n if (parsed.interactionType !== undefined) interaction.interactionType = parsed.interactionType\n if (parsed.title !== undefined) interaction.title = parsed.title ?? null\n if (parsed.body !== undefined) interaction.body = parsed.body ?? null\n if (parsed.status !== undefined) interaction.status = parsed.status\n if (parsed.scheduledAt !== undefined) interaction.scheduledAt = parsed.scheduledAt ?? null\n if (parsed.occurredAt !== undefined) interaction.occurredAt = parsed.occurredAt ?? null\n if (parsed.priority !== undefined) interaction.priority = parsed.priority ?? null\n if (parsed.authorUserId !== undefined) interaction.authorUserId = parsed.authorUserId ?? null\n if (parsed.ownerUserId !== undefined) interaction.ownerUserId = parsed.ownerUserId ?? null\n if (parsed.appearanceIcon !== undefined) interaction.appearanceIcon = parsed.appearanceIcon ?? null\n if (parsed.appearanceColor !== undefined) interaction.appearanceColor = parsed.appearanceColor ?? null\n if (parsed.pinned !== undefined) interaction.pinned = parsed.pinned\n if (parsed.durationMinutes !== undefined) interaction.durationMinutes = parsed.durationMinutes ?? null\n if (parsed.location !== undefined) interaction.location = parsed.location ?? null\n if (parsed.allDay !== undefined) interaction.allDay = parsed.allDay ?? null\n if (parsed.recurrenceRule !== undefined) interaction.recurrenceRule = parsed.recurrenceRule ?? null\n if (parsed.recurrenceEnd !== undefined) interaction.recurrenceEnd = parsed.recurrenceEnd ?? null\n if (parsed.participants !== undefined) interaction.participants = parsed.participants ?? null\n if (parsed.reminderMinutes !== undefined) interaction.reminderMinutes = parsed.reminderMinutes ?? null\n if (parsed.visibility !== undefined) interaction.visibility = parsed.visibility ?? null\n if (parsed.linkedEntities !== undefined) interaction.linkedEntities = parsed.linkedEntities ?? null\n if (parsed.guestPermissions !== undefined) interaction.guestPermissions = parsed.guestPermissions ?? null\n\n await trx.flush()\n\n const entityId = typeof interaction.entity === 'string' ? interaction.entity : interaction.entity.id\n await setInteractionCustomFields(\n createTransactionalDataEngine(ctx, trx),\n interaction.id,\n interaction.organizationId,\n interaction.tenantId,\n custom,\n )\n\n const projection = await recomputeNextInteraction(trx, entityId)\n\n return { interaction, entityId, nextInteractionId: projection.nextInteractionId }\n })\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: interaction,\n identifiers: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitNextInteractionUpdatedEvent(ctx, { entityId, nextInteractionId }, {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n })\n\n return { interactionId: interaction.id }\n },\n captureAfter: async (_input, result, ctx) => {\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n return await loadInteractionSnapshot(em, result.interactionId)\n },\n buildLog: async ({ snapshots }) => {\n const { translate } = await resolveTranslations()\n const before = snapshots.before as InteractionSnapshot | undefined\n if (!before) return null\n const afterSnapshot = snapshots.after as InteractionSnapshot | undefined\n return {\n actionLabel: translate('customers.audit.interactions.update', 'Update interaction'),\n resourceKind: 'customers.interaction',\n resourceId: before.interaction.id,\n parentResourceKind: resolveParentResourceKind(before.interaction.entityKind),\n parentResourceId: before.interaction.entityId ?? null,\n tenantId: before.interaction.tenantId,\n organizationId: before.interaction.organizationId,\n snapshotBefore: before,\n snapshotAfter: afterSnapshot ?? null,\n payload: {\n undo: {\n before,\n after: afterSnapshot ?? null,\n } satisfies InteractionUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const payload = extractUndoPayload<InteractionUndoPayload>(logEntry)\n const before = payload?.before\n if (!before) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, nextInteractionId } = await runInTransaction(em, async (trx) => {\n let interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: before.interaction.id })\n const entity = await requireTimelineParentEntity(trx, before.interaction.entityId)\n\n if (!interaction) {\n interaction = trx.create(CustomerInteraction, {\n id: before.interaction.id,\n organizationId: before.interaction.organizationId,\n tenantId: before.interaction.tenantId,\n entity,\n interactionType: before.interaction.interactionType,\n title: before.interaction.title,\n body: before.interaction.body,\n status: before.interaction.status,\n scheduledAt: before.interaction.scheduledAt,\n occurredAt: before.interaction.occurredAt,\n priority: before.interaction.priority,\n authorUserId: before.interaction.authorUserId,\n ownerUserId: before.interaction.ownerUserId,\n dealId: before.interaction.dealId,\n source: before.interaction.source,\n appearanceIcon: before.interaction.appearanceIcon,\n appearanceColor: before.interaction.appearanceColor,\n durationMinutes: before.interaction.durationMinutes,\n location: before.interaction.location,\n allDay: before.interaction.allDay,\n recurrenceRule: before.interaction.recurrenceRule,\n recurrenceEnd: before.interaction.recurrenceEnd,\n participants: before.interaction.participants,\n reminderMinutes: before.interaction.reminderMinutes,\n visibility: before.interaction.visibility,\n linkedEntities: before.interaction.linkedEntities,\n guestPermissions: before.interaction.guestPermissions,\n createdAt: new Date(),\n updatedAt: new Date(),\n })\n trx.persist(interaction)\n } else {\n interaction.entity = entity\n interaction.interactionType = before.interaction.interactionType\n interaction.title = before.interaction.title\n interaction.body = before.interaction.body\n interaction.status = before.interaction.status\n interaction.scheduledAt = before.interaction.scheduledAt\n interaction.occurredAt = before.interaction.occurredAt\n interaction.priority = before.interaction.priority\n interaction.authorUserId = before.interaction.authorUserId\n interaction.ownerUserId = before.interaction.ownerUserId\n interaction.dealId = before.interaction.dealId\n interaction.source = before.interaction.source\n interaction.appearanceIcon = before.interaction.appearanceIcon\n interaction.appearanceColor = before.interaction.appearanceColor\n interaction.durationMinutes = before.interaction.durationMinutes\n interaction.location = before.interaction.location\n interaction.allDay = before.interaction.allDay\n interaction.recurrenceRule = before.interaction.recurrenceRule\n interaction.recurrenceEnd = before.interaction.recurrenceEnd\n interaction.participants = before.interaction.participants\n interaction.reminderMinutes = before.interaction.reminderMinutes\n interaction.visibility = before.interaction.visibility\n interaction.linkedEntities = before.interaction.linkedEntities\n interaction.guestPermissions = before.interaction.guestPermissions\n }\n\n await trx.flush()\n const projection = await recomputeNextInteraction(trx, before.interaction.entityId)\n\n const resetValues = buildCustomFieldResetMap(before.custom, payload?.after?.custom)\n if (Object.keys(resetValues).length) {\n await setCustomFieldsIfAny({\n dataEngine: createTransactionalDataEngine(ctx, trx),\n entityId: INTERACTION_ENTITY_ID,\n recordId: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n values: resetValues,\n notify: false,\n })\n }\n\n return {\n interaction,\n nextInteractionId: projection.nextInteractionId,\n }\n })\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudUndoSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: interaction,\n identifiers: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: before.interaction.entityId,\n nextInteractionId,\n }, {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n })\n },\n}\n\n// \u2500\u2500\u2500 Complete \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nconst completeInteractionCommand: CommandHandler<InteractionCompleteInput, { interactionId: string }> = {\n id: 'customers.interactions.complete',\n async prepare(rawInput, ctx) {\n const parsed = interactionCompleteSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const snapshot = await loadInteractionSnapshot(em, parsed.id)\n return snapshot ? { before: snapshot } : {}\n },\n async execute(rawInput, ctx) {\n const parsed = interactionCompleteSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, entityId, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: parsed.id, deletedAt: null })\n if (!interaction) {\n enforceRecordGoneIsConflict({ resourceKind: 'customers.interaction', resourceId: parsed.id, request: ctx.request ?? null })\n throw new CrudHttpError(404, { error: 'Interaction not found' })\n }\n ensureTenantScope(ctx, interaction.tenantId)\n ensureOrganizationScope(ctx, interaction.organizationId)\n\n enforceCommandOptimisticLock({\n resourceKind: 'customers.interaction',\n resourceId: interaction.id,\n current: interaction.updatedAt,\n request: ctx.request ?? null,\n })\n\n interaction.status = 'done'\n interaction.occurredAt = parsed.occurredAt ?? new Date()\n await trx.flush()\n\n const entityId = typeof interaction.entity === 'string' ? interaction.entity : interaction.entity.id\n const projection = await recomputeNextInteraction(trx, entityId)\n return { interaction, entityId, nextInteractionId: projection.nextInteractionId }\n })\n\n const identifiers = {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n }\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: interaction,\n identifiers,\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitLifecycleEvent(ctx, 'customers.interaction.completed', {\n ...identifiers,\n entityId,\n interactionType: interaction.interactionType,\n status: interaction.status,\n source: interaction.source ?? null,\n occurredAt: interaction.occurredAt?.toISOString() ?? null,\n ...(ctx.syncOrigin ? { syncOrigin: ctx.syncOrigin } : {}),\n })\n await emitNextInteractionUpdatedEvent(ctx, { entityId, nextInteractionId }, identifiers)\n\n return { interactionId: interaction.id }\n },\n captureAfter: async (_input, result, ctx) => {\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n return await loadInteractionSnapshot(em, result.interactionId)\n },\n buildLog: async ({ snapshots }) => {\n const { translate } = await resolveTranslations()\n const before = snapshots.before as InteractionSnapshot | undefined\n if (!before) return null\n const afterSnapshot = snapshots.after as InteractionSnapshot | undefined\n return {\n actionLabel: translate('customers.audit.interactions.complete', 'Complete interaction'),\n resourceKind: 'customers.interaction',\n resourceId: before.interaction.id,\n parentResourceKind: resolveParentResourceKind(before.interaction.entityKind),\n parentResourceId: before.interaction.entityId ?? null,\n tenantId: before.interaction.tenantId,\n organizationId: before.interaction.organizationId,\n snapshotBefore: before,\n snapshotAfter: afterSnapshot ?? null,\n payload: {\n undo: {\n before,\n after: afterSnapshot ?? null,\n } satisfies InteractionUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const payload = extractUndoPayload<InteractionUndoPayload>(logEntry)\n const before = payload?.before\n if (!before) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const result = await runInTransaction(em, async (trx) => {\n const interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: before.interaction.id })\n if (!interaction) return null\n\n interaction.status = before.interaction.status\n interaction.occurredAt = before.interaction.occurredAt\n await trx.flush()\n\n const projection = await recomputeNextInteraction(trx, before.interaction.entityId)\n return {\n interaction,\n nextInteractionId: projection.nextInteractionId,\n }\n })\n if (!result) return\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudUndoSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: result.interaction,\n identifiers: {\n id: result.interaction.id,\n organizationId: result.interaction.organizationId,\n tenantId: result.interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitInteractionRevertedEvent(ctx, before.interaction)\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: before.interaction.entityId,\n nextInteractionId: result.nextInteractionId,\n }, {\n id: result.interaction.id,\n organizationId: result.interaction.organizationId,\n tenantId: result.interaction.tenantId,\n })\n },\n}\n\n// \u2500\u2500\u2500 Cancel \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nconst cancelInteractionCommand: CommandHandler<InteractionCancelInput, { interactionId: string }> = {\n id: 'customers.interactions.cancel',\n async prepare(rawInput, ctx) {\n const parsed = interactionCancelSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const snapshot = await loadInteractionSnapshot(em, parsed.id)\n return snapshot ? { before: snapshot } : {}\n },\n async execute(rawInput, ctx) {\n const parsed = interactionCancelSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, entityId, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: parsed.id, deletedAt: null })\n if (!interaction) {\n enforceRecordGoneIsConflict({ resourceKind: 'customers.interaction', resourceId: parsed.id, request: ctx.request ?? null })\n throw new CrudHttpError(404, { error: 'Interaction not found' })\n }\n ensureTenantScope(ctx, interaction.tenantId)\n ensureOrganizationScope(ctx, interaction.organizationId)\n\n enforceCommandOptimisticLock({\n resourceKind: 'customers.interaction',\n resourceId: interaction.id,\n current: interaction.updatedAt,\n request: ctx.request ?? null,\n })\n\n interaction.status = 'canceled'\n await trx.flush()\n\n const entityId = typeof interaction.entity === 'string' ? interaction.entity : interaction.entity.id\n const projection = await recomputeNextInteraction(trx, entityId)\n return { interaction, entityId, nextInteractionId: projection.nextInteractionId }\n })\n\n const identifiers = {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n }\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: interaction,\n identifiers,\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitLifecycleEvent(ctx, 'customers.interaction.canceled', {\n ...identifiers,\n entityId,\n interactionType: interaction.interactionType,\n status: interaction.status,\n source: interaction.source ?? null,\n ...(ctx.syncOrigin ? { syncOrigin: ctx.syncOrigin } : {}),\n })\n await emitNextInteractionUpdatedEvent(ctx, { entityId, nextInteractionId }, identifiers)\n\n return { interactionId: interaction.id }\n },\n captureAfter: async (_input, result, ctx) => {\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n return await loadInteractionSnapshot(em, result.interactionId)\n },\n buildLog: async ({ snapshots }) => {\n const { translate } = await resolveTranslations()\n const before = snapshots.before as InteractionSnapshot | undefined\n if (!before) return null\n const afterSnapshot = snapshots.after as InteractionSnapshot | undefined\n return {\n actionLabel: translate('customers.audit.interactions.cancel', 'Cancel interaction'),\n resourceKind: 'customers.interaction',\n resourceId: before.interaction.id,\n parentResourceKind: resolveParentResourceKind(before.interaction.entityKind),\n parentResourceId: before.interaction.entityId ?? null,\n tenantId: before.interaction.tenantId,\n organizationId: before.interaction.organizationId,\n snapshotBefore: before,\n snapshotAfter: afterSnapshot ?? null,\n payload: {\n undo: {\n before,\n after: afterSnapshot ?? null,\n } satisfies InteractionUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const payload = extractUndoPayload<InteractionUndoPayload>(logEntry)\n const before = payload?.before\n if (!before) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const result = await runInTransaction(em, async (trx) => {\n const interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: before.interaction.id })\n if (!interaction) return null\n\n interaction.status = before.interaction.status\n await trx.flush()\n\n const projection = await recomputeNextInteraction(trx, before.interaction.entityId)\n return {\n interaction,\n nextInteractionId: projection.nextInteractionId,\n }\n })\n if (!result) return\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudUndoSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: result.interaction,\n identifiers: {\n id: result.interaction.id,\n organizationId: result.interaction.organizationId,\n tenantId: result.interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitInteractionRevertedEvent(ctx, before.interaction)\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: before.interaction.entityId,\n nextInteractionId: result.nextInteractionId,\n }, {\n id: result.interaction.id,\n organizationId: result.interaction.organizationId,\n tenantId: result.interaction.tenantId,\n })\n },\n}\n\n// \u2500\u2500\u2500 Delete \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nconst deleteInteractionCommand: CommandHandler<{ body?: Record<string, unknown>; query?: Record<string, unknown> }, { interactionId: string }> =\n {\n id: 'customers.interactions.delete',\n async prepare(input, ctx) {\n const id = requireId(input, 'Interaction id required')\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const snapshot = await loadInteractionSnapshot(em, id)\n return snapshot ? { before: snapshot } : {}\n },\n async execute(input, ctx) {\n const id = requireId(input, 'Interaction id required')\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, entityId, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const interaction = await findOneWithDecryption(trx, CustomerInteraction, { id, deletedAt: null })\n if (!interaction) {\n enforceRecordGoneIsConflict({ resourceKind: 'customers.interaction', resourceId: id, request: ctx.request ?? null })\n throw new CrudHttpError(404, { error: 'Interaction not found' })\n }\n ensureTenantScope(ctx, interaction.tenantId)\n ensureOrganizationScope(ctx, interaction.organizationId)\n\n enforceCommandOptimisticLock({\n resourceKind: 'customers.interaction',\n resourceId: interaction.id,\n current: interaction.updatedAt,\n request: ctx.request ?? null,\n })\n\n const entityId = typeof interaction.entity === 'string' ? interaction.entity : interaction.entity.id\n interaction.deletedAt = new Date()\n await trx.flush()\n\n const projection = await recomputeNextInteraction(trx, entityId)\n return { interaction, entityId, nextInteractionId: projection.nextInteractionId }\n })\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'deleted',\n entity: interaction,\n identifiers: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitNextInteractionUpdatedEvent(ctx, { entityId, nextInteractionId }, {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n })\n return { interactionId: interaction.id }\n },\n buildLog: async ({ snapshots }) => {\n const before = snapshots.before as InteractionSnapshot | undefined\n if (!before) return null\n const { translate } = await resolveTranslations()\n return {\n actionLabel: translate('customers.audit.interactions.delete', 'Delete interaction'),\n resourceKind: 'customers.interaction',\n resourceId: before.interaction.id,\n parentResourceKind: resolveParentResourceKind(before.interaction.entityKind),\n parentResourceId: before.interaction.entityId ?? null,\n tenantId: before.interaction.tenantId,\n organizationId: before.interaction.organizationId,\n snapshotBefore: before,\n payload: {\n undo: {\n before,\n } satisfies InteractionUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const payload = extractUndoPayload<InteractionUndoPayload>(logEntry)\n const before = payload?.before\n if (!before) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const entity = await requireTimelineParentEntity(trx, before.interaction.entityId)\n let interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: before.interaction.id })\n if (!interaction) {\n interaction = trx.create(CustomerInteraction, {\n id: before.interaction.id,\n organizationId: before.interaction.organizationId,\n tenantId: before.interaction.tenantId,\n entity,\n interactionType: before.interaction.interactionType,\n title: before.interaction.title,\n body: before.interaction.body,\n status: before.interaction.status,\n scheduledAt: before.interaction.scheduledAt,\n occurredAt: before.interaction.occurredAt,\n priority: before.interaction.priority,\n authorUserId: before.interaction.authorUserId,\n ownerUserId: before.interaction.ownerUserId,\n dealId: before.interaction.dealId,\n source: before.interaction.source,\n appearanceIcon: before.interaction.appearanceIcon,\n appearanceColor: before.interaction.appearanceColor,\n durationMinutes: before.interaction.durationMinutes,\n location: before.interaction.location,\n allDay: before.interaction.allDay,\n recurrenceRule: before.interaction.recurrenceRule,\n recurrenceEnd: before.interaction.recurrenceEnd,\n participants: before.interaction.participants,\n reminderMinutes: before.interaction.reminderMinutes,\n visibility: before.interaction.visibility,\n linkedEntities: before.interaction.linkedEntities,\n guestPermissions: before.interaction.guestPermissions,\n createdAt: new Date(),\n updatedAt: new Date(),\n })\n trx.persist(interaction)\n } else {\n interaction.deletedAt = null\n interaction.entity = entity\n interaction.interactionType = before.interaction.interactionType\n interaction.title = before.interaction.title\n interaction.body = before.interaction.body\n interaction.status = before.interaction.status\n interaction.scheduledAt = before.interaction.scheduledAt\n interaction.occurredAt = before.interaction.occurredAt\n interaction.priority = before.interaction.priority\n interaction.authorUserId = before.interaction.authorUserId\n interaction.ownerUserId = before.interaction.ownerUserId\n interaction.dealId = before.interaction.dealId\n interaction.source = before.interaction.source\n interaction.appearanceIcon = before.interaction.appearanceIcon\n interaction.appearanceColor = before.interaction.appearanceColor\n interaction.durationMinutes = before.interaction.durationMinutes\n interaction.location = before.interaction.location\n interaction.allDay = before.interaction.allDay\n interaction.recurrenceRule = before.interaction.recurrenceRule\n interaction.recurrenceEnd = before.interaction.recurrenceEnd\n interaction.participants = before.interaction.participants\n interaction.reminderMinutes = before.interaction.reminderMinutes\n interaction.visibility = before.interaction.visibility\n interaction.linkedEntities = before.interaction.linkedEntities\n interaction.guestPermissions = before.interaction.guestPermissions\n }\n await trx.flush()\n\n const projection = await recomputeNextInteraction(trx, before.interaction.entityId)\n\n const resetValues = buildCustomFieldResetMap(before.custom, undefined)\n if (Object.keys(resetValues).length) {\n await setCustomFieldsIfAny({\n dataEngine: createTransactionalDataEngine(ctx, trx),\n entityId: INTERACTION_ENTITY_ID,\n recordId: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n values: resetValues,\n notify: false,\n })\n }\n\n return {\n interaction,\n nextInteractionId: projection.nextInteractionId,\n }\n })\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudUndoSideEffects({\n dataEngine: de,\n action: 'created',\n entity: interaction,\n identifiers: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: before.interaction.entityId,\n nextInteractionId,\n }, {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n })\n },\n }\n\n// \u2500\u2500\u2500 Recompute Next (internal repair) \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nconst recomputeNextSchema = z.object({ entityId: z.string().min(1) })\n\nconst recomputeNextCommand: CommandHandler<{ entityId: string }, { entityId: string }> = {\n id: 'customers.interaction.recompute_next',\n async execute(rawInput, ctx) {\n const parsed = recomputeNextSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const projection = await recomputeNextInteraction(em, parsed.entityId)\n const entity = await requireTimelineParentEntity(em, parsed.entityId)\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: parsed.entityId,\n nextInteractionId: projection.nextInteractionId,\n }, {\n id: parsed.entityId,\n organizationId: entity.organizationId,\n tenantId: entity.tenantId,\n })\n return { entityId: parsed.entityId }\n },\n}\n\nregisterCommand(createInteractionCommand)\nregisterCommand(updateInteractionCommand)\nregisterCommand(completeInteractionCommand)\nregisterCommand(cancelInteractionCommand)\nregisterCommand(deleteInteractionCommand)\nregisterCommand(recomputeNextCommand)\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import { z } from 'zod'\nimport { registerCommand } from '@open-mercato/shared/lib/commands'\nimport type { CommandHandler } from '@open-mercato/shared/lib/commands'\nimport {\n parseWithCustomFields,\n setCustomFieldsIfAny,\n emitCrudSideEffects,\n emitCrudUndoSideEffects,\n requireId,\n normalizeAuthorUserId,\n} from '@open-mercato/shared/lib/commands/helpers'\nimport { DefaultDataEngine, type DataEngine } from '@open-mercato/shared/lib/data/engine'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport type { CommandRuntimeContext } from '@open-mercato/shared/lib/commands'\nimport { CustomerInteraction, CustomerEntity } from '../data/entities'\nimport {\n interactionCreateSchema,\n interactionUpdateSchema,\n interactionCompleteSchema,\n interactionCancelSchema,\n type InteractionCreateInput,\n type InteractionUpdateInput,\n type InteractionCompleteInput,\n type InteractionCancelInput,\n} from '../data/validators'\nimport {\n ensureOrganizationScope,\n ensureTenantScope,\n requireTimelineParentEntity,\n extractUndoPayload,\n emitQueryIndexUpsertEvents,\n requireDealInScope,\n resolveParentResourceKind,\n} from './shared'\nimport { resolveTranslations } from '@open-mercato/shared/lib/i18n/server'\nimport { resolveRedoSnapshot } from '@open-mercato/shared/lib/commands/redo'\nimport {\n loadCustomFieldSnapshot,\n buildCustomFieldResetMap,\n} from '@open-mercato/shared/lib/commands/customFieldSnapshots'\nimport { CrudHttpError } from '@open-mercato/shared/lib/crud/errors'\nimport { enforceRecordGoneIsConflict, enforceCommandOptimisticLock } from '@open-mercato/shared/lib/crud/optimistic-lock-command'\nimport { findOneWithDecryption } from '@open-mercato/shared/lib/encryption/find'\nimport type { CrudIndexerConfig, CrudEventsConfig } from '@open-mercato/shared/lib/crud/types'\nimport { recomputeNextInteraction } from '../lib/interactionProjection'\nimport { canChangeEmailVisibility } from '../lib/visibilityFilter'\n\nconst INTERACTION_ENTITY_ID = 'customers:customer_interaction'\nconst interactionCrudIndexer: CrudIndexerConfig<CustomerInteraction> = {\n entityType: 'customers:customer_interaction' as const,\n}\n\nconst interactionCrudEvents: CrudEventsConfig = {\n module: 'customers',\n entity: 'interaction',\n persistent: true,\n buildPayload: (ctx) => ({\n id: ctx.identifiers.id,\n organizationId: ctx.identifiers.organizationId,\n tenantId: ctx.identifiers.tenantId,\n entityId:\n ctx.entity && typeof ctx.entity === 'object' && 'entity' in (ctx.entity as Record<string, unknown>)\n ? (() => {\n const entityRef = (ctx.entity as CustomerInteraction).entity\n return typeof entityRef === 'string' ? entityRef : entityRef?.id ?? null\n })()\n : null,\n interactionType:\n ctx.entity && typeof ctx.entity === 'object' && 'interactionType' in (ctx.entity as Record<string, unknown>)\n ? (ctx.entity as CustomerInteraction).interactionType\n : null,\n status:\n ctx.entity && typeof ctx.entity === 'object' && 'status' in (ctx.entity as Record<string, unknown>)\n ? (ctx.entity as CustomerInteraction).status\n : null,\n source:\n ctx.entity && typeof ctx.entity === 'object' && 'source' in (ctx.entity as Record<string, unknown>)\n ? (ctx.entity as CustomerInteraction).source ?? null\n : null,\n ...(ctx.syncOrigin ? { syncOrigin: ctx.syncOrigin } : {}),\n }),\n}\n\ntype InteractionSnapshot = {\n interaction: {\n id: string\n organizationId: string\n tenantId: string\n entityId: string\n entityKind: string | null\n dealId: string | null\n interactionType: string\n title: string | null\n body: string | null\n status: string\n scheduledAt: Date | null\n occurredAt: Date | null\n priority: number | null\n authorUserId: string | null\n ownerUserId: string | null\n appearanceIcon: string | null\n appearanceColor: string | null\n source: string | null\n durationMinutes: number | null\n location: string | null\n allDay: boolean | null\n recurrenceRule: string | null\n recurrenceEnd: Date | null\n participants: Array<{ userId: string; name?: string; email?: string; status?: string }> | null\n reminderMinutes: number | null\n visibility: string | null\n linkedEntities: Array<{ id: string; type: string; label: string }> | null\n guestPermissions: { canInviteOthers?: boolean; canModify?: boolean; canSeeList?: boolean } | null\n }\n custom?: Record<string, unknown>\n}\n\ntype InteractionUndoPayload = {\n before?: InteractionSnapshot | null\n after?: InteractionSnapshot | null\n}\n\nasync function loadInteractionSnapshot(em: EntityManager, id: string): Promise<InteractionSnapshot | null> {\n const interaction = await findOneWithDecryption(em, CustomerInteraction, { id }, { populate: ['entity'] })\n if (!interaction) return null\n const custom = await loadCustomFieldSnapshot(em, {\n entityId: INTERACTION_ENTITY_ID,\n recordId: interaction.id,\n tenantId: interaction.tenantId,\n organizationId: interaction.organizationId,\n })\n const entityRef = interaction.entity\n const entityKind = (typeof entityRef === 'object' && entityRef !== null && 'kind' in entityRef)\n ? (entityRef as { kind: string }).kind\n : null\n return {\n interaction: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n entityId: typeof entityRef === 'string' ? entityRef : entityRef.id,\n entityKind,\n dealId: interaction.dealId ?? null,\n interactionType: interaction.interactionType,\n title: interaction.title ?? null,\n body: interaction.body ?? null,\n status: interaction.status,\n scheduledAt: interaction.scheduledAt ?? null,\n occurredAt: interaction.occurredAt ?? null,\n priority: interaction.priority ?? null,\n authorUserId: interaction.authorUserId ?? null,\n ownerUserId: interaction.ownerUserId ?? null,\n appearanceIcon: interaction.appearanceIcon ?? null,\n appearanceColor: interaction.appearanceColor ?? null,\n source: interaction.source ?? null,\n durationMinutes: interaction.durationMinutes ?? null,\n location: interaction.location ?? null,\n allDay: interaction.allDay ?? null,\n recurrenceRule: interaction.recurrenceRule ?? null,\n recurrenceEnd: interaction.recurrenceEnd ?? null,\n participants: interaction.participants ?? null,\n reminderMinutes: interaction.reminderMinutes ?? null,\n visibility: interaction.visibility ?? null,\n linkedEntities: interaction.linkedEntities ?? null,\n guestPermissions: interaction.guestPermissions ?? null,\n },\n custom,\n }\n}\n\nasync function setInteractionCustomFields(\n dataEngine: DataEngine,\n interactionId: string,\n organizationId: string,\n tenantId: string,\n values: Record<string, unknown>\n) {\n if (!values || !Object.keys(values).length) return\n await setCustomFieldsIfAny({\n dataEngine,\n entityId: INTERACTION_ENTITY_ID,\n recordId: interactionId,\n organizationId,\n tenantId,\n values,\n notify: false,\n })\n}\n\nasync function emitLifecycleEvent(\n ctx: CommandRuntimeContext,\n eventId: string,\n payload: Record<string, unknown>\n): Promise<void> {\n let bus: { emitEvent(event: string, payload: unknown, options?: unknown): Promise<void> } | null = null\n try {\n bus = ctx.container.resolve('eventBus')\n } catch (err) {\n console.warn('[customers.commands.interactions] eventBus resolve failed; skipping emit', eventId, err)\n bus = null\n }\n if (!bus) return\n await bus\n .emitEvent(eventId, payload, { persistent: true })\n .catch((err) => {\n console.warn('[customers.commands.interactions] emit failed', eventId, err)\n return undefined\n })\n}\n\nasync function emitInteractionRevertedEvent(\n ctx: CommandRuntimeContext,\n interaction: InteractionSnapshot['interaction'],\n): Promise<void> {\n await emitLifecycleEvent(ctx, 'customers.interaction.reverted', {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n entityId: interaction.entityId,\n interactionType: interaction.interactionType,\n source: interaction.source ?? null,\n status: interaction.status,\n occurredAt: interaction.occurredAt?.toISOString() ?? null,\n ...(ctx.syncOrigin ? { syncOrigin: ctx.syncOrigin } : {}),\n })\n}\n\ntype InteractionIdentifiers = {\n id: string\n organizationId: string\n tenantId: string\n}\n\ntype InteractionProjectionMutation = {\n entityId: string\n nextInteractionId: string | null\n}\n\nfunction createTransactionalDataEngine(ctx: CommandRuntimeContext, em: EntityManager): DataEngine {\n const emWithConnection = em as EntityManager & { getConnection?: () => unknown }\n if (typeof emWithConnection.getConnection !== 'function') {\n return ctx.container.resolve('dataEngine') as DataEngine\n }\n return new DefaultDataEngine(em, ctx.container)\n}\n\nasync function runInTransaction<TResult>(\n em: EntityManager,\n operation: (trx: EntityManager) => Promise<TResult>,\n): Promise<TResult> {\n // Mirrors the SPEC-018 fix applied to withAtomicFlush: use explicit begin/commit/rollback\n // so the outer EntityManager stays bound to the transaction, and closures over `em` inside\n // `operation` participate in the same transaction. This avoids the em.transactional(cb)\n // hazard where the callback receives a child EM whose flushes can silently race against\n // subsequent queries on the original `em`.\n const supportsBegin =\n typeof (em as unknown as { begin?: () => Promise<void> }).begin === 'function' &&\n typeof (em as unknown as { commit?: () => Promise<void> }).commit === 'function' &&\n typeof (em as unknown as { rollback?: () => Promise<void> }).rollback === 'function'\n if (!supportsBegin) {\n return operation(em)\n }\n await em.begin()\n try {\n const result = await operation(em)\n await em.commit()\n return result\n } catch (err) {\n try {\n await em.rollback()\n } catch {\n // rollback failure should not mask the original error; intentionally swallowed\n }\n throw err\n }\n}\n\nasync function emitNextInteractionUpdatedEvent(\n ctx: CommandRuntimeContext,\n projection: InteractionProjectionMutation,\n identifiers: InteractionIdentifiers,\n): Promise<void> {\n await emitQueryIndexUpsertEvents(ctx, [{\n entityType: 'customers:customer_entity',\n recordId: projection.entityId,\n organizationId: identifiers.organizationId,\n tenantId: identifiers.tenantId,\n }])\n await emitLifecycleEvent(ctx, 'customers.next_interaction.updated', {\n id: projection.entityId,\n entityId: projection.entityId,\n nextInteractionId: projection.nextInteractionId,\n organizationId: identifiers.organizationId,\n tenantId: identifiers.tenantId,\n })\n}\n\n// \u2500\u2500\u2500 Create \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\ntype InteractionGraphValues = {\n id?: string\n organizationId: string\n tenantId: string\n entity: CustomerEntity\n interactionType: string\n title: string | null\n body: string | null\n status: string\n scheduledAt: Date | null\n occurredAt: Date | null\n priority: number | null\n authorUserId: string | null\n ownerUserId: string | null\n dealId: string | null\n source: string | null\n appearanceIcon: string | null\n appearanceColor: string | null\n durationMinutes: number | null\n location: string | null\n allDay: boolean | null\n recurrenceRule: string | null\n recurrenceEnd: Date | null\n participants: InteractionSnapshot['interaction']['participants']\n reminderMinutes: number | null\n visibility: string | null\n linkedEntities: InteractionSnapshot['interaction']['linkedEntities']\n guestPermissions: InteractionSnapshot['interaction']['guestPermissions']\n}\n\n// Single source of truth for the interaction row mapping, shared by `execute`\n// (values from validated input) and `redo` (values from the after-snapshot,\n// carrying the original id). Keeps create and id-preserving redo from drifting\n// field-by-field. Caller owns the surrounding transaction, persist, projection\n// recompute, and custom-field handling.\nfunction buildInteractionGraph(em: EntityManager, values: InteractionGraphValues): CustomerInteraction {\n return em.create(CustomerInteraction, {\n ...(values.id ? { id: values.id } : {}),\n organizationId: values.organizationId,\n tenantId: values.tenantId,\n entity: values.entity,\n interactionType: values.interactionType,\n title: values.title,\n body: values.body,\n status: values.status,\n scheduledAt: values.scheduledAt,\n occurredAt: values.occurredAt,\n priority: values.priority,\n authorUserId: values.authorUserId,\n ownerUserId: values.ownerUserId,\n dealId: values.dealId,\n source: values.source,\n appearanceIcon: values.appearanceIcon,\n appearanceColor: values.appearanceColor,\n durationMinutes: values.durationMinutes,\n location: values.location,\n allDay: values.allDay,\n recurrenceRule: values.recurrenceRule,\n recurrenceEnd: values.recurrenceEnd,\n participants: values.participants,\n reminderMinutes: values.reminderMinutes,\n visibility: values.visibility,\n linkedEntities: values.linkedEntities,\n guestPermissions: values.guestPermissions,\n createdAt: new Date(),\n updatedAt: new Date(),\n })\n}\n\nconst createInteractionCommand: CommandHandler<InteractionCreateInput, { interactionId: string; entityId: string }> = {\n id: 'customers.interactions.create',\n async execute(rawInput, ctx) {\n const { parsed, custom } = parseWithCustomFields(interactionCreateSchema, rawInput)\n\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const normalizedAuthor = normalizeAuthorUserId(parsed.authorUserId ?? null, ctx.auth)\n const { interaction, entityId, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const entity = await requireTimelineParentEntity(trx, parsed.entityId, { tenantId: parsed.tenantId, organizationId: parsed.organizationId })\n ensureTenantScope(ctx, entity.tenantId)\n ensureOrganizationScope(ctx, entity.organizationId)\n\n if (parsed.dealId) {\n await requireDealInScope(trx, parsed.dealId, entity.tenantId, entity.organizationId)\n }\n\n const interaction = buildInteractionGraph(trx, {\n ...(parsed.id ? { id: parsed.id } : {}),\n organizationId: entity.organizationId,\n tenantId: entity.tenantId,\n entity,\n interactionType: parsed.interactionType,\n title: parsed.title ?? null,\n body: parsed.body ?? null,\n status: parsed.status ?? 'planned',\n scheduledAt: parsed.scheduledAt ?? null,\n occurredAt: parsed.occurredAt ?? null,\n priority: parsed.priority ?? null,\n authorUserId: normalizedAuthor,\n ownerUserId: parsed.ownerUserId ?? null,\n dealId: parsed.dealId ?? null,\n source: parsed.source ?? null,\n appearanceIcon: parsed.appearanceIcon ?? null,\n appearanceColor: parsed.appearanceColor ?? null,\n durationMinutes: parsed.durationMinutes ?? null,\n location: parsed.location ?? null,\n allDay: parsed.allDay ?? null,\n recurrenceRule: parsed.recurrenceRule ?? null,\n recurrenceEnd: parsed.recurrenceEnd ?? null,\n participants: parsed.participants ?? null,\n reminderMinutes: parsed.reminderMinutes ?? null,\n visibility: parsed.visibility ?? null,\n linkedEntities: parsed.linkedEntities ?? null,\n guestPermissions: parsed.guestPermissions ?? null,\n })\n trx.persist(interaction)\n await trx.flush()\n\n await setInteractionCustomFields(\n createTransactionalDataEngine(ctx, trx),\n interaction.id,\n entity.organizationId,\n entity.tenantId,\n custom,\n )\n\n const projection = await recomputeNextInteraction(trx, entity.id)\n\n return {\n interaction,\n entityId: entity.id,\n nextInteractionId: projection.nextInteractionId,\n }\n })\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'created',\n entity: interaction,\n identifiers: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitNextInteractionUpdatedEvent(ctx, { entityId, nextInteractionId }, {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n })\n\n return { interactionId: interaction.id, entityId }\n },\n captureAfter: async (_input, result, ctx) => {\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n return await loadInteractionSnapshot(em, result.interactionId)\n },\n buildLog: async ({ result, snapshots }) => {\n const { translate } = await resolveTranslations()\n const snapshot = snapshots.after as InteractionSnapshot | undefined\n return {\n actionLabel: translate('customers.audit.interactions.create', 'Create interaction'),\n resourceKind: 'customers.interaction',\n resourceId: result.interactionId,\n parentResourceKind: resolveParentResourceKind(snapshot?.interaction?.entityKind),\n parentResourceId: snapshot?.interaction?.entityId ?? null,\n tenantId: snapshot?.interaction.tenantId ?? null,\n organizationId: snapshot?.interaction.organizationId ?? null,\n snapshotAfter: snapshot ?? null,\n payload: {\n undo: {\n after: snapshot,\n } satisfies InteractionUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const interactionId = logEntry?.resourceId\n if (!interactionId) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const result = await runInTransaction(em, async (trx) => {\n const record = await findOneWithDecryption(trx, CustomerInteraction, { id: interactionId })\n if (!record) return null\n const entityId = typeof record.entity === 'string' ? record.entity : record.entity.id\n trx.remove(record)\n await trx.flush()\n const projection = await recomputeNextInteraction(trx, entityId)\n return {\n entityId,\n nextInteractionId: projection.nextInteractionId,\n identifiers: {\n id: record.id,\n organizationId: record.organizationId,\n tenantId: record.tenantId,\n },\n }\n })\n if (!result) return\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: result.entityId,\n nextInteractionId: result.nextInteractionId,\n }, result.identifiers)\n },\n redo: async ({ logEntry, ctx }) => {\n const after = resolveRedoSnapshot<InteractionSnapshot>(logEntry)\n if (!after) {\n throw new CrudHttpError(400, { error: '[internal] redo snapshot unavailable for interaction create' })\n }\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const entity = await requireTimelineParentEntity(trx, after.interaction.entityId, { tenantId: after.interaction.tenantId, organizationId: after.interaction.organizationId })\n let interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: after.interaction.id })\n if (!interaction) {\n interaction = buildInteractionGraph(trx, {\n id: after.interaction.id,\n organizationId: after.interaction.organizationId,\n tenantId: after.interaction.tenantId,\n entity,\n interactionType: after.interaction.interactionType,\n title: after.interaction.title,\n body: after.interaction.body,\n status: after.interaction.status,\n scheduledAt: after.interaction.scheduledAt,\n occurredAt: after.interaction.occurredAt,\n priority: after.interaction.priority,\n authorUserId: after.interaction.authorUserId,\n ownerUserId: after.interaction.ownerUserId,\n dealId: after.interaction.dealId,\n source: after.interaction.source,\n appearanceIcon: after.interaction.appearanceIcon,\n appearanceColor: after.interaction.appearanceColor,\n durationMinutes: after.interaction.durationMinutes,\n location: after.interaction.location,\n allDay: after.interaction.allDay,\n recurrenceRule: after.interaction.recurrenceRule,\n recurrenceEnd: after.interaction.recurrenceEnd,\n participants: after.interaction.participants,\n reminderMinutes: after.interaction.reminderMinutes,\n visibility: after.interaction.visibility,\n linkedEntities: after.interaction.linkedEntities,\n guestPermissions: after.interaction.guestPermissions,\n })\n trx.persist(interaction)\n } else {\n interaction.deletedAt = null\n interaction.entity = entity\n interaction.interactionType = after.interaction.interactionType\n interaction.title = after.interaction.title\n interaction.body = after.interaction.body\n interaction.status = after.interaction.status\n interaction.scheduledAt = after.interaction.scheduledAt\n interaction.occurredAt = after.interaction.occurredAt\n interaction.priority = after.interaction.priority\n interaction.authorUserId = after.interaction.authorUserId\n interaction.ownerUserId = after.interaction.ownerUserId\n interaction.dealId = after.interaction.dealId\n interaction.source = after.interaction.source\n interaction.appearanceIcon = after.interaction.appearanceIcon\n interaction.appearanceColor = after.interaction.appearanceColor\n interaction.durationMinutes = after.interaction.durationMinutes\n interaction.location = after.interaction.location\n interaction.allDay = after.interaction.allDay\n interaction.recurrenceRule = after.interaction.recurrenceRule\n interaction.recurrenceEnd = after.interaction.recurrenceEnd\n interaction.participants = after.interaction.participants\n interaction.reminderMinutes = after.interaction.reminderMinutes\n interaction.visibility = after.interaction.visibility\n interaction.linkedEntities = after.interaction.linkedEntities\n interaction.guestPermissions = after.interaction.guestPermissions\n }\n await trx.flush()\n\n const projection = await recomputeNextInteraction(trx, after.interaction.entityId)\n\n const restoreValues = buildCustomFieldResetMap(after.custom, undefined)\n if (Object.keys(restoreValues).length) {\n await setCustomFieldsIfAny({\n dataEngine: createTransactionalDataEngine(ctx, trx),\n entityId: INTERACTION_ENTITY_ID,\n recordId: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n values: restoreValues,\n notify: false,\n })\n }\n\n return { interaction, nextInteractionId: projection.nextInteractionId }\n })\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'created',\n entity: interaction,\n identifiers: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: after.interaction.entityId,\n nextInteractionId,\n }, {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n })\n\n return { interactionId: interaction.id, entityId: after.interaction.entityId }\n },\n}\n\n// \u2500\u2500\u2500 Update \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nconst updateInteractionCommand: CommandHandler<InteractionUpdateInput, { interactionId: string }> = {\n id: 'customers.interactions.update',\n async prepare(rawInput, ctx) {\n const { parsed } = parseWithCustomFields(interactionUpdateSchema, rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const snapshot = await loadInteractionSnapshot(em, parsed.id)\n return snapshot ? { before: snapshot } : {}\n },\n async execute(rawInput, ctx) {\n const { parsed, custom } = parseWithCustomFields(interactionUpdateSchema, rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, entityId, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: parsed.id, deletedAt: null })\n if (!interaction) {\n enforceRecordGoneIsConflict({ resourceKind: 'customers.interaction', resourceId: parsed.id, request: ctx.request ?? null })\n throw new CrudHttpError(404, { error: 'Interaction not found' })\n }\n ensureTenantScope(ctx, interaction.tenantId)\n ensureOrganizationScope(ctx, interaction.organizationId)\n\n // Concurrent-edit guard for command-driven callers (e.g. the legacy\n // /api/customers/todos route, which bypasses the makeCrudRoute lock guard):\n // when the client opted into optimistic locking, a stale edit fails with the\n // unified 409 instead of silently overwriting (#2055). Strictly additive \u2014\n // no-op when no expected-version header is present.\n enforceCommandOptimisticLock({\n resourceKind: 'customers.interaction',\n resourceId: interaction.id,\n current: interaction.updatedAt,\n request: ctx.request ?? null,\n })\n\n // Email visibility is an access-controlled field: only the interaction's\n // author may change a\n // private email's visibility (mirrors the dedicated PATCH .../visibility\n // route). Enforce it here \u2014 the single persistence path \u2014 so the generic\n // update route (PUT /api/interactions) cannot bypass the gate. Evaluated\n // against the row's pre-mutation author/type. 404 (not 403) keeps the\n // existence-masking consistent with the dedicated route.\n if (\n parsed.visibility !== undefined &&\n interaction.interactionType === 'email' &&\n (parsed.visibility ?? null) !== (interaction.visibility ?? null)\n ) {\n const actorUserId = (ctx.auth as { sub?: string | null } | null)?.sub ?? null\n if (\n !canChangeEmailVisibility({\n interactionType: interaction.interactionType,\n currentVisibility: interaction.visibility,\n nextVisibility: parsed.visibility,\n authorUserId: interaction.authorUserId,\n actorUserId,\n // v1 strict owner-only: only the author may flip visibility; no admin\n // bypass (canChangeEmailVisibility ignores caller features in v1).\n userFeatures: undefined,\n })\n ) {\n throw new CrudHttpError(404, { error: 'Email not found' })\n }\n }\n\n if (parsed.dealId !== undefined) {\n if (parsed.dealId) {\n await requireDealInScope(trx, parsed.dealId, interaction.tenantId, interaction.organizationId)\n }\n interaction.dealId = parsed.dealId ?? null\n }\n if (parsed.interactionType !== undefined) interaction.interactionType = parsed.interactionType\n if (parsed.title !== undefined) interaction.title = parsed.title ?? null\n if (parsed.body !== undefined) interaction.body = parsed.body ?? null\n if (parsed.status !== undefined) interaction.status = parsed.status\n if (parsed.scheduledAt !== undefined) interaction.scheduledAt = parsed.scheduledAt ?? null\n if (parsed.occurredAt !== undefined) interaction.occurredAt = parsed.occurredAt ?? null\n if (parsed.priority !== undefined) interaction.priority = parsed.priority ?? null\n if (parsed.authorUserId !== undefined) interaction.authorUserId = parsed.authorUserId ?? null\n if (parsed.ownerUserId !== undefined) interaction.ownerUserId = parsed.ownerUserId ?? null\n if (parsed.appearanceIcon !== undefined) interaction.appearanceIcon = parsed.appearanceIcon ?? null\n if (parsed.appearanceColor !== undefined) interaction.appearanceColor = parsed.appearanceColor ?? null\n if (parsed.pinned !== undefined) interaction.pinned = parsed.pinned\n if (parsed.durationMinutes !== undefined) interaction.durationMinutes = parsed.durationMinutes ?? null\n if (parsed.location !== undefined) interaction.location = parsed.location ?? null\n if (parsed.allDay !== undefined) interaction.allDay = parsed.allDay ?? null\n if (parsed.recurrenceRule !== undefined) interaction.recurrenceRule = parsed.recurrenceRule ?? null\n if (parsed.recurrenceEnd !== undefined) interaction.recurrenceEnd = parsed.recurrenceEnd ?? null\n if (parsed.participants !== undefined) interaction.participants = parsed.participants ?? null\n if (parsed.reminderMinutes !== undefined) interaction.reminderMinutes = parsed.reminderMinutes ?? null\n if (parsed.visibility !== undefined) interaction.visibility = parsed.visibility ?? null\n if (parsed.linkedEntities !== undefined) interaction.linkedEntities = parsed.linkedEntities ?? null\n if (parsed.guestPermissions !== undefined) interaction.guestPermissions = parsed.guestPermissions ?? null\n\n await trx.flush()\n\n const entityId = typeof interaction.entity === 'string' ? interaction.entity : interaction.entity.id\n await setInteractionCustomFields(\n createTransactionalDataEngine(ctx, trx),\n interaction.id,\n interaction.organizationId,\n interaction.tenantId,\n custom,\n )\n\n const projection = await recomputeNextInteraction(trx, entityId)\n\n return { interaction, entityId, nextInteractionId: projection.nextInteractionId }\n })\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: interaction,\n identifiers: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitNextInteractionUpdatedEvent(ctx, { entityId, nextInteractionId }, {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n })\n\n return { interactionId: interaction.id }\n },\n captureAfter: async (_input, result, ctx) => {\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n return await loadInteractionSnapshot(em, result.interactionId)\n },\n buildLog: async ({ snapshots }) => {\n const { translate } = await resolveTranslations()\n const before = snapshots.before as InteractionSnapshot | undefined\n if (!before) return null\n const afterSnapshot = snapshots.after as InteractionSnapshot | undefined\n return {\n actionLabel: translate('customers.audit.interactions.update', 'Update interaction'),\n resourceKind: 'customers.interaction',\n resourceId: before.interaction.id,\n parentResourceKind: resolveParentResourceKind(before.interaction.entityKind),\n parentResourceId: before.interaction.entityId ?? null,\n tenantId: before.interaction.tenantId,\n organizationId: before.interaction.organizationId,\n snapshotBefore: before,\n snapshotAfter: afterSnapshot ?? null,\n payload: {\n undo: {\n before,\n after: afterSnapshot ?? null,\n } satisfies InteractionUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const payload = extractUndoPayload<InteractionUndoPayload>(logEntry)\n const before = payload?.before\n if (!before) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, nextInteractionId } = await runInTransaction(em, async (trx) => {\n let interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: before.interaction.id })\n const entity = await requireTimelineParentEntity(trx, before.interaction.entityId, { tenantId: before.interaction.tenantId, organizationId: before.interaction.organizationId })\n\n if (!interaction) {\n interaction = trx.create(CustomerInteraction, {\n id: before.interaction.id,\n organizationId: before.interaction.organizationId,\n tenantId: before.interaction.tenantId,\n entity,\n interactionType: before.interaction.interactionType,\n title: before.interaction.title,\n body: before.interaction.body,\n status: before.interaction.status,\n scheduledAt: before.interaction.scheduledAt,\n occurredAt: before.interaction.occurredAt,\n priority: before.interaction.priority,\n authorUserId: before.interaction.authorUserId,\n ownerUserId: before.interaction.ownerUserId,\n dealId: before.interaction.dealId,\n source: before.interaction.source,\n appearanceIcon: before.interaction.appearanceIcon,\n appearanceColor: before.interaction.appearanceColor,\n durationMinutes: before.interaction.durationMinutes,\n location: before.interaction.location,\n allDay: before.interaction.allDay,\n recurrenceRule: before.interaction.recurrenceRule,\n recurrenceEnd: before.interaction.recurrenceEnd,\n participants: before.interaction.participants,\n reminderMinutes: before.interaction.reminderMinutes,\n visibility: before.interaction.visibility,\n linkedEntities: before.interaction.linkedEntities,\n guestPermissions: before.interaction.guestPermissions,\n createdAt: new Date(),\n updatedAt: new Date(),\n })\n trx.persist(interaction)\n } else {\n interaction.entity = entity\n interaction.interactionType = before.interaction.interactionType\n interaction.title = before.interaction.title\n interaction.body = before.interaction.body\n interaction.status = before.interaction.status\n interaction.scheduledAt = before.interaction.scheduledAt\n interaction.occurredAt = before.interaction.occurredAt\n interaction.priority = before.interaction.priority\n interaction.authorUserId = before.interaction.authorUserId\n interaction.ownerUserId = before.interaction.ownerUserId\n interaction.dealId = before.interaction.dealId\n interaction.source = before.interaction.source\n interaction.appearanceIcon = before.interaction.appearanceIcon\n interaction.appearanceColor = before.interaction.appearanceColor\n interaction.durationMinutes = before.interaction.durationMinutes\n interaction.location = before.interaction.location\n interaction.allDay = before.interaction.allDay\n interaction.recurrenceRule = before.interaction.recurrenceRule\n interaction.recurrenceEnd = before.interaction.recurrenceEnd\n interaction.participants = before.interaction.participants\n interaction.reminderMinutes = before.interaction.reminderMinutes\n interaction.visibility = before.interaction.visibility\n interaction.linkedEntities = before.interaction.linkedEntities\n interaction.guestPermissions = before.interaction.guestPermissions\n }\n\n await trx.flush()\n const projection = await recomputeNextInteraction(trx, before.interaction.entityId)\n\n const resetValues = buildCustomFieldResetMap(before.custom, payload?.after?.custom)\n if (Object.keys(resetValues).length) {\n await setCustomFieldsIfAny({\n dataEngine: createTransactionalDataEngine(ctx, trx),\n entityId: INTERACTION_ENTITY_ID,\n recordId: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n values: resetValues,\n notify: false,\n })\n }\n\n return {\n interaction,\n nextInteractionId: projection.nextInteractionId,\n }\n })\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudUndoSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: interaction,\n identifiers: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: before.interaction.entityId,\n nextInteractionId,\n }, {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n })\n },\n}\n\n// \u2500\u2500\u2500 Complete \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nconst completeInteractionCommand: CommandHandler<InteractionCompleteInput, { interactionId: string }> = {\n id: 'customers.interactions.complete',\n async prepare(rawInput, ctx) {\n const parsed = interactionCompleteSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const snapshot = await loadInteractionSnapshot(em, parsed.id)\n return snapshot ? { before: snapshot } : {}\n },\n async execute(rawInput, ctx) {\n const parsed = interactionCompleteSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, entityId, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: parsed.id, deletedAt: null })\n if (!interaction) {\n enforceRecordGoneIsConflict({ resourceKind: 'customers.interaction', resourceId: parsed.id, request: ctx.request ?? null })\n throw new CrudHttpError(404, { error: 'Interaction not found' })\n }\n ensureTenantScope(ctx, interaction.tenantId)\n ensureOrganizationScope(ctx, interaction.organizationId)\n\n enforceCommandOptimisticLock({\n resourceKind: 'customers.interaction',\n resourceId: interaction.id,\n current: interaction.updatedAt,\n request: ctx.request ?? null,\n })\n\n interaction.status = 'done'\n interaction.occurredAt = parsed.occurredAt ?? new Date()\n await trx.flush()\n\n const entityId = typeof interaction.entity === 'string' ? interaction.entity : interaction.entity.id\n const projection = await recomputeNextInteraction(trx, entityId)\n return { interaction, entityId, nextInteractionId: projection.nextInteractionId }\n })\n\n const identifiers = {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n }\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: interaction,\n identifiers,\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitLifecycleEvent(ctx, 'customers.interaction.completed', {\n ...identifiers,\n entityId,\n interactionType: interaction.interactionType,\n status: interaction.status,\n source: interaction.source ?? null,\n occurredAt: interaction.occurredAt?.toISOString() ?? null,\n ...(ctx.syncOrigin ? { syncOrigin: ctx.syncOrigin } : {}),\n })\n await emitNextInteractionUpdatedEvent(ctx, { entityId, nextInteractionId }, identifiers)\n\n return { interactionId: interaction.id }\n },\n captureAfter: async (_input, result, ctx) => {\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n return await loadInteractionSnapshot(em, result.interactionId)\n },\n buildLog: async ({ snapshots }) => {\n const { translate } = await resolveTranslations()\n const before = snapshots.before as InteractionSnapshot | undefined\n if (!before) return null\n const afterSnapshot = snapshots.after as InteractionSnapshot | undefined\n return {\n actionLabel: translate('customers.audit.interactions.complete', 'Complete interaction'),\n resourceKind: 'customers.interaction',\n resourceId: before.interaction.id,\n parentResourceKind: resolveParentResourceKind(before.interaction.entityKind),\n parentResourceId: before.interaction.entityId ?? null,\n tenantId: before.interaction.tenantId,\n organizationId: before.interaction.organizationId,\n snapshotBefore: before,\n snapshotAfter: afterSnapshot ?? null,\n payload: {\n undo: {\n before,\n after: afterSnapshot ?? null,\n } satisfies InteractionUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const payload = extractUndoPayload<InteractionUndoPayload>(logEntry)\n const before = payload?.before\n if (!before) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const result = await runInTransaction(em, async (trx) => {\n const interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: before.interaction.id })\n if (!interaction) return null\n\n interaction.status = before.interaction.status\n interaction.occurredAt = before.interaction.occurredAt\n await trx.flush()\n\n const projection = await recomputeNextInteraction(trx, before.interaction.entityId)\n return {\n interaction,\n nextInteractionId: projection.nextInteractionId,\n }\n })\n if (!result) return\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudUndoSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: result.interaction,\n identifiers: {\n id: result.interaction.id,\n organizationId: result.interaction.organizationId,\n tenantId: result.interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitInteractionRevertedEvent(ctx, before.interaction)\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: before.interaction.entityId,\n nextInteractionId: result.nextInteractionId,\n }, {\n id: result.interaction.id,\n organizationId: result.interaction.organizationId,\n tenantId: result.interaction.tenantId,\n })\n },\n}\n\n// \u2500\u2500\u2500 Cancel \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nconst cancelInteractionCommand: CommandHandler<InteractionCancelInput, { interactionId: string }> = {\n id: 'customers.interactions.cancel',\n async prepare(rawInput, ctx) {\n const parsed = interactionCancelSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const snapshot = await loadInteractionSnapshot(em, parsed.id)\n return snapshot ? { before: snapshot } : {}\n },\n async execute(rawInput, ctx) {\n const parsed = interactionCancelSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, entityId, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: parsed.id, deletedAt: null })\n if (!interaction) {\n enforceRecordGoneIsConflict({ resourceKind: 'customers.interaction', resourceId: parsed.id, request: ctx.request ?? null })\n throw new CrudHttpError(404, { error: 'Interaction not found' })\n }\n ensureTenantScope(ctx, interaction.tenantId)\n ensureOrganizationScope(ctx, interaction.organizationId)\n\n enforceCommandOptimisticLock({\n resourceKind: 'customers.interaction',\n resourceId: interaction.id,\n current: interaction.updatedAt,\n request: ctx.request ?? null,\n })\n\n interaction.status = 'canceled'\n await trx.flush()\n\n const entityId = typeof interaction.entity === 'string' ? interaction.entity : interaction.entity.id\n const projection = await recomputeNextInteraction(trx, entityId)\n return { interaction, entityId, nextInteractionId: projection.nextInteractionId }\n })\n\n const identifiers = {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n }\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: interaction,\n identifiers,\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitLifecycleEvent(ctx, 'customers.interaction.canceled', {\n ...identifiers,\n entityId,\n interactionType: interaction.interactionType,\n status: interaction.status,\n source: interaction.source ?? null,\n ...(ctx.syncOrigin ? { syncOrigin: ctx.syncOrigin } : {}),\n })\n await emitNextInteractionUpdatedEvent(ctx, { entityId, nextInteractionId }, identifiers)\n\n return { interactionId: interaction.id }\n },\n captureAfter: async (_input, result, ctx) => {\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n return await loadInteractionSnapshot(em, result.interactionId)\n },\n buildLog: async ({ snapshots }) => {\n const { translate } = await resolveTranslations()\n const before = snapshots.before as InteractionSnapshot | undefined\n if (!before) return null\n const afterSnapshot = snapshots.after as InteractionSnapshot | undefined\n return {\n actionLabel: translate('customers.audit.interactions.cancel', 'Cancel interaction'),\n resourceKind: 'customers.interaction',\n resourceId: before.interaction.id,\n parentResourceKind: resolveParentResourceKind(before.interaction.entityKind),\n parentResourceId: before.interaction.entityId ?? null,\n tenantId: before.interaction.tenantId,\n organizationId: before.interaction.organizationId,\n snapshotBefore: before,\n snapshotAfter: afterSnapshot ?? null,\n payload: {\n undo: {\n before,\n after: afterSnapshot ?? null,\n } satisfies InteractionUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const payload = extractUndoPayload<InteractionUndoPayload>(logEntry)\n const before = payload?.before\n if (!before) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const result = await runInTransaction(em, async (trx) => {\n const interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: before.interaction.id })\n if (!interaction) return null\n\n interaction.status = before.interaction.status\n await trx.flush()\n\n const projection = await recomputeNextInteraction(trx, before.interaction.entityId)\n return {\n interaction,\n nextInteractionId: projection.nextInteractionId,\n }\n })\n if (!result) return\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudUndoSideEffects({\n dataEngine: de,\n action: 'updated',\n entity: result.interaction,\n identifiers: {\n id: result.interaction.id,\n organizationId: result.interaction.organizationId,\n tenantId: result.interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitInteractionRevertedEvent(ctx, before.interaction)\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: before.interaction.entityId,\n nextInteractionId: result.nextInteractionId,\n }, {\n id: result.interaction.id,\n organizationId: result.interaction.organizationId,\n tenantId: result.interaction.tenantId,\n })\n },\n}\n\n// \u2500\u2500\u2500 Delete \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nconst deleteInteractionCommand: CommandHandler<{ body?: Record<string, unknown>; query?: Record<string, unknown> }, { interactionId: string }> =\n {\n id: 'customers.interactions.delete',\n async prepare(input, ctx) {\n const id = requireId(input, 'Interaction id required')\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const snapshot = await loadInteractionSnapshot(em, id)\n return snapshot ? { before: snapshot } : {}\n },\n async execute(input, ctx) {\n const id = requireId(input, 'Interaction id required')\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, entityId, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const interaction = await findOneWithDecryption(trx, CustomerInteraction, { id, deletedAt: null })\n if (!interaction) {\n enforceRecordGoneIsConflict({ resourceKind: 'customers.interaction', resourceId: id, request: ctx.request ?? null })\n throw new CrudHttpError(404, { error: 'Interaction not found' })\n }\n ensureTenantScope(ctx, interaction.tenantId)\n ensureOrganizationScope(ctx, interaction.organizationId)\n\n enforceCommandOptimisticLock({\n resourceKind: 'customers.interaction',\n resourceId: interaction.id,\n current: interaction.updatedAt,\n request: ctx.request ?? null,\n })\n\n const entityId = typeof interaction.entity === 'string' ? interaction.entity : interaction.entity.id\n interaction.deletedAt = new Date()\n await trx.flush()\n\n const projection = await recomputeNextInteraction(trx, entityId)\n return { interaction, entityId, nextInteractionId: projection.nextInteractionId }\n })\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudSideEffects({\n dataEngine: de,\n action: 'deleted',\n entity: interaction,\n identifiers: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitNextInteractionUpdatedEvent(ctx, { entityId, nextInteractionId }, {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n })\n return { interactionId: interaction.id }\n },\n buildLog: async ({ snapshots }) => {\n const before = snapshots.before as InteractionSnapshot | undefined\n if (!before) return null\n const { translate } = await resolveTranslations()\n return {\n actionLabel: translate('customers.audit.interactions.delete', 'Delete interaction'),\n resourceKind: 'customers.interaction',\n resourceId: before.interaction.id,\n parentResourceKind: resolveParentResourceKind(before.interaction.entityKind),\n parentResourceId: before.interaction.entityId ?? null,\n tenantId: before.interaction.tenantId,\n organizationId: before.interaction.organizationId,\n snapshotBefore: before,\n payload: {\n undo: {\n before,\n } satisfies InteractionUndoPayload,\n },\n }\n },\n undo: async ({ logEntry, ctx }) => {\n const payload = extractUndoPayload<InteractionUndoPayload>(logEntry)\n const before = payload?.before\n if (!before) return\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const { interaction, nextInteractionId } = await runInTransaction(em, async (trx) => {\n const entity = await requireTimelineParentEntity(trx, before.interaction.entityId, { tenantId: before.interaction.tenantId, organizationId: before.interaction.organizationId })\n let interaction = await findOneWithDecryption(trx, CustomerInteraction, { id: before.interaction.id })\n if (!interaction) {\n interaction = trx.create(CustomerInteraction, {\n id: before.interaction.id,\n organizationId: before.interaction.organizationId,\n tenantId: before.interaction.tenantId,\n entity,\n interactionType: before.interaction.interactionType,\n title: before.interaction.title,\n body: before.interaction.body,\n status: before.interaction.status,\n scheduledAt: before.interaction.scheduledAt,\n occurredAt: before.interaction.occurredAt,\n priority: before.interaction.priority,\n authorUserId: before.interaction.authorUserId,\n ownerUserId: before.interaction.ownerUserId,\n dealId: before.interaction.dealId,\n source: before.interaction.source,\n appearanceIcon: before.interaction.appearanceIcon,\n appearanceColor: before.interaction.appearanceColor,\n durationMinutes: before.interaction.durationMinutes,\n location: before.interaction.location,\n allDay: before.interaction.allDay,\n recurrenceRule: before.interaction.recurrenceRule,\n recurrenceEnd: before.interaction.recurrenceEnd,\n participants: before.interaction.participants,\n reminderMinutes: before.interaction.reminderMinutes,\n visibility: before.interaction.visibility,\n linkedEntities: before.interaction.linkedEntities,\n guestPermissions: before.interaction.guestPermissions,\n createdAt: new Date(),\n updatedAt: new Date(),\n })\n trx.persist(interaction)\n } else {\n interaction.deletedAt = null\n interaction.entity = entity\n interaction.interactionType = before.interaction.interactionType\n interaction.title = before.interaction.title\n interaction.body = before.interaction.body\n interaction.status = before.interaction.status\n interaction.scheduledAt = before.interaction.scheduledAt\n interaction.occurredAt = before.interaction.occurredAt\n interaction.priority = before.interaction.priority\n interaction.authorUserId = before.interaction.authorUserId\n interaction.ownerUserId = before.interaction.ownerUserId\n interaction.dealId = before.interaction.dealId\n interaction.source = before.interaction.source\n interaction.appearanceIcon = before.interaction.appearanceIcon\n interaction.appearanceColor = before.interaction.appearanceColor\n interaction.durationMinutes = before.interaction.durationMinutes\n interaction.location = before.interaction.location\n interaction.allDay = before.interaction.allDay\n interaction.recurrenceRule = before.interaction.recurrenceRule\n interaction.recurrenceEnd = before.interaction.recurrenceEnd\n interaction.participants = before.interaction.participants\n interaction.reminderMinutes = before.interaction.reminderMinutes\n interaction.visibility = before.interaction.visibility\n interaction.linkedEntities = before.interaction.linkedEntities\n interaction.guestPermissions = before.interaction.guestPermissions\n }\n await trx.flush()\n\n const projection = await recomputeNextInteraction(trx, before.interaction.entityId)\n\n const resetValues = buildCustomFieldResetMap(before.custom, undefined)\n if (Object.keys(resetValues).length) {\n await setCustomFieldsIfAny({\n dataEngine: createTransactionalDataEngine(ctx, trx),\n entityId: INTERACTION_ENTITY_ID,\n recordId: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n values: resetValues,\n notify: false,\n })\n }\n\n return {\n interaction,\n nextInteractionId: projection.nextInteractionId,\n }\n })\n\n const de = (ctx.container.resolve('dataEngine') as DataEngine)\n await emitCrudUndoSideEffects({\n dataEngine: de,\n action: 'created',\n entity: interaction,\n identifiers: {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n },\n syncOrigin: ctx.syncOrigin,\n indexer: interactionCrudIndexer,\n events: interactionCrudEvents,\n })\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: before.interaction.entityId,\n nextInteractionId,\n }, {\n id: interaction.id,\n organizationId: interaction.organizationId,\n tenantId: interaction.tenantId,\n })\n },\n }\n\n// \u2500\u2500\u2500 Recompute Next (internal repair) \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nconst recomputeNextSchema = z.object({ entityId: z.string().min(1) })\n\nconst recomputeNextCommand: CommandHandler<{ entityId: string }, { entityId: string }> = {\n id: 'customers.interaction.recompute_next',\n async execute(rawInput, ctx) {\n const parsed = recomputeNextSchema.parse(rawInput)\n const em = (ctx.container.resolve('em') as EntityManager).fork()\n const projection = await recomputeNextInteraction(em, parsed.entityId)\n const entity = await requireTimelineParentEntity(em, parsed.entityId, {\n tenantId: ctx.auth?.tenantId ?? '',\n organizationId: ctx.selectedOrganizationId ?? ctx.auth?.orgId ?? '',\n })\n await emitNextInteractionUpdatedEvent(ctx, {\n entityId: parsed.entityId,\n nextInteractionId: projection.nextInteractionId,\n }, {\n id: parsed.entityId,\n organizationId: entity.organizationId,\n tenantId: entity.tenantId,\n })\n return { entityId: parsed.entityId }\n },\n}\n\nregisterCommand(createInteractionCommand)\nregisterCommand(updateInteractionCommand)\nregisterCommand(completeInteractionCommand)\nregisterCommand(cancelInteractionCommand)\nregisterCommand(deleteInteractionCommand)\nregisterCommand(recomputeNextCommand)\n"],
5
+ "mappings": "AAAA,SAAS,SAAS;AAClB,SAAS,uBAAuB;AAEhC;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,yBAA0C;AAGnD,SAAS,2BAA2C;AACpD;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OAKK;AACP;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,2BAA2B;AACpC,SAAS,2BAA2B;AACpC;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP,SAAS,qBAAqB;AAC9B,SAAS,6BAA6B,oCAAoC;AAC1E,SAAS,6BAA6B;AAEtC,SAAS,gCAAgC;AACzC,SAAS,gCAAgC;AAEzC,MAAM,wBAAwB;AAC9B,MAAM,yBAAiE;AAAA,EACrE,YAAY;AACd;AAEA,MAAM,wBAA0C;AAAA,EAC9C,QAAQ;AAAA,EACR,QAAQ;AAAA,EACR,YAAY;AAAA,EACZ,cAAc,CAAC,SAAS;AAAA,IACtB,IAAI,IAAI,YAAY;AAAA,IACpB,gBAAgB,IAAI,YAAY;AAAA,IAChC,UAAU,IAAI,YAAY;AAAA,IAC1B,UACE,IAAI,UAAU,OAAO,IAAI,WAAW,YAAY,YAAa,IAAI,UAC5D,MAAM;AACL,YAAM,YAAa,IAAI,OAA+B;AACtD,aAAO,OAAO,cAAc,WAAW,YAAY,WAAW,MAAM;AAAA,IACtE,GAAG,IACH;AAAA,IACN,iBACE,IAAI,UAAU,OAAO,IAAI,WAAW,YAAY,qBAAsB,IAAI,SACrE,IAAI,OAA+B,kBACpC;AAAA,IACN,QACE,IAAI,UAAU,OAAO,IAAI,WAAW,YAAY,YAAa,IAAI,SAC5D,IAAI,OAA+B,SACpC;AAAA,IACN,QACE,IAAI,UAAU,OAAO,IAAI,WAAW,YAAY,YAAa,IAAI,SAC5D,IAAI,OAA+B,UAAU,OAC9C;AAAA,IACN,GAAI,IAAI,aAAa,EAAE,YAAY,IAAI,WAAW,IAAI,CAAC;AAAA,EACzD;AACF;AAyCA,eAAe,wBAAwB,IAAmB,IAAiD;AACzG,QAAM,cAAc,MAAM,sBAAsB,IAAI,qBAAqB,EAAE,GAAG,GAAG,EAAE,UAAU,CAAC,QAAQ,EAAE,CAAC;AACzG,MAAI,CAAC,YAAa,QAAO;AACzB,QAAM,SAAS,MAAM,wBAAwB,IAAI;AAAA,IAC/C,UAAU;AAAA,IACV,UAAU,YAAY;AAAA,IACtB,UAAU,YAAY;AAAA,IACtB,gBAAgB,YAAY;AAAA,EAC9B,CAAC;AACD,QAAM,YAAY,YAAY;AAC9B,QAAM,aAAc,OAAO,cAAc,YAAY,cAAc,QAAQ,UAAU,YAChF,UAA+B,OAChC;AACJ,SAAO;AAAA,IACL,aAAa;AAAA,MACX,IAAI,YAAY;AAAA,MAChB,gBAAgB,YAAY;AAAA,MAC5B,UAAU,YAAY;AAAA,MACtB,UAAU,OAAO,cAAc,WAAW,YAAY,UAAU;AAAA,MAChE;AAAA,MACA,QAAQ,YAAY,UAAU;AAAA,MAC9B,iBAAiB,YAAY;AAAA,MAC7B,OAAO,YAAY,SAAS;AAAA,MAC5B,MAAM,YAAY,QAAQ;AAAA,MAC1B,QAAQ,YAAY;AAAA,MACpB,aAAa,YAAY,eAAe;AAAA,MACxC,YAAY,YAAY,cAAc;AAAA,MACtC,UAAU,YAAY,YAAY;AAAA,MAClC,cAAc,YAAY,gBAAgB;AAAA,MAC1C,aAAa,YAAY,eAAe;AAAA,MACxC,gBAAgB,YAAY,kBAAkB;AAAA,MAC9C,iBAAiB,YAAY,mBAAmB;AAAA,MAChD,QAAQ,YAAY,UAAU;AAAA,MAC9B,iBAAiB,YAAY,mBAAmB;AAAA,MAChD,UAAU,YAAY,YAAY;AAAA,MAClC,QAAQ,YAAY,UAAU;AAAA,MAC9B,gBAAgB,YAAY,kBAAkB;AAAA,MAC9C,eAAe,YAAY,iBAAiB;AAAA,MAC5C,cAAc,YAAY,gBAAgB;AAAA,MAC1C,iBAAiB,YAAY,mBAAmB;AAAA,MAChD,YAAY,YAAY,cAAc;AAAA,MACtC,gBAAgB,YAAY,kBAAkB;AAAA,MAC9C,kBAAkB,YAAY,oBAAoB;AAAA,IACpD;AAAA,IACA;AAAA,EACF;AACF;AAEA,eAAe,2BACb,YACA,eACA,gBACA,UACA,QACA;AACA,MAAI,CAAC,UAAU,CAAC,OAAO,KAAK,MAAM,EAAE,OAAQ;AAC5C,QAAM,qBAAqB;AAAA,IACzB;AAAA,IACA,UAAU;AAAA,IACV,UAAU;AAAA,IACV;AAAA,IACA;AAAA,IACA;AAAA,IACA,QAAQ;AAAA,EACV,CAAC;AACH;AAEA,eAAe,mBACb,KACA,SACA,SACe;AACf,MAAI,MAA+F;AACnG,MAAI;AACF,UAAM,IAAI,UAAU,QAAQ,UAAU;AAAA,EACxC,SAAS,KAAK;AACZ,YAAQ,KAAK,4EAA4E,SAAS,GAAG;AACrG,UAAM;AAAA,EACR;AACA,MAAI,CAAC,IAAK;AACV,QAAM,IACH,UAAU,SAAS,SAAS,EAAE,YAAY,KAAK,CAAC,EAChD,MAAM,CAAC,QAAQ;AACd,YAAQ,KAAK,iDAAiD,SAAS,GAAG;AAC1E,WAAO;AAAA,EACT,CAAC;AACL;AAEA,eAAe,6BACb,KACA,aACe;AACf,QAAM,mBAAmB,KAAK,kCAAkC;AAAA,IAC9D,IAAI,YAAY;AAAA,IAChB,gBAAgB,YAAY;AAAA,IAC5B,UAAU,YAAY;AAAA,IACtB,UAAU,YAAY;AAAA,IACtB,iBAAiB,YAAY;AAAA,IAC7B,QAAQ,YAAY,UAAU;AAAA,IAC9B,QAAQ,YAAY;AAAA,IACpB,YAAY,YAAY,YAAY,YAAY,KAAK;AAAA,IACrD,GAAI,IAAI,aAAa,EAAE,YAAY,IAAI,WAAW,IAAI,CAAC;AAAA,EACzD,CAAC;AACH;AAaA,SAAS,8BAA8B,KAA4B,IAA+B;AAChG,QAAM,mBAAmB;AACzB,MAAI,OAAO,iBAAiB,kBAAkB,YAAY;AACxD,WAAO,IAAI,UAAU,QAAQ,YAAY;AAAA,EAC3C;AACA,SAAO,IAAI,kBAAkB,IAAI,IAAI,SAAS;AAChD;AAEA,eAAe,iBACb,IACA,WACkB;AAMlB,QAAM,gBACJ,OAAQ,GAAkD,UAAU,cACpE,OAAQ,GAAmD,WAAW,cACtE,OAAQ,GAAqD,aAAa;AAC5E,MAAI,CAAC,eAAe;AAClB,WAAO,UAAU,EAAE;AAAA,EACrB;AACA,QAAM,GAAG,MAAM;AACf,MAAI;AACF,UAAM,SAAS,MAAM,UAAU,EAAE;AACjC,UAAM,GAAG,OAAO;AAChB,WAAO;AAAA,EACT,SAAS,KAAK;AACZ,QAAI;AACF,YAAM,GAAG,SAAS;AAAA,IACpB,QAAQ;AAAA,IAER;AACA,UAAM;AAAA,EACR;AACF;AAEA,eAAe,gCACb,KACA,YACA,aACe;AACf,QAAM,2BAA2B,KAAK,CAAC;AAAA,IACrC,YAAY;AAAA,IACZ,UAAU,WAAW;AAAA,IACrB,gBAAgB,YAAY;AAAA,IAC5B,UAAU,YAAY;AAAA,EACxB,CAAC,CAAC;AACF,QAAM,mBAAmB,KAAK,sCAAsC;AAAA,IAClE,IAAI,WAAW;AAAA,IACf,UAAU,WAAW;AAAA,IACrB,mBAAmB,WAAW;AAAA,IAC9B,gBAAgB,YAAY;AAAA,IAC5B,UAAU,YAAY;AAAA,EACxB,CAAC;AACH;AAuCA,SAAS,sBAAsB,IAAmB,QAAqD;AACrG,SAAO,GAAG,OAAO,qBAAqB;AAAA,IACpC,GAAI,OAAO,KAAK,EAAE,IAAI,OAAO,GAAG,IAAI,CAAC;AAAA,IACrC,gBAAgB,OAAO;AAAA,IACvB,UAAU,OAAO;AAAA,IACjB,QAAQ,OAAO;AAAA,IACf,iBAAiB,OAAO;AAAA,IACxB,OAAO,OAAO;AAAA,IACd,MAAM,OAAO;AAAA,IACb,QAAQ,OAAO;AAAA,IACf,aAAa,OAAO;AAAA,IACpB,YAAY,OAAO;AAAA,IACnB,UAAU,OAAO;AAAA,IACjB,cAAc,OAAO;AAAA,IACrB,aAAa,OAAO;AAAA,IACpB,QAAQ,OAAO;AAAA,IACf,QAAQ,OAAO;AAAA,IACf,gBAAgB,OAAO;AAAA,IACvB,iBAAiB,OAAO;AAAA,IACxB,iBAAiB,OAAO;AAAA,IACxB,UAAU,OAAO;AAAA,IACjB,QAAQ,OAAO;AAAA,IACf,gBAAgB,OAAO;AAAA,IACvB,eAAe,OAAO;AAAA,IACtB,cAAc,OAAO;AAAA,IACrB,iBAAiB,OAAO;AAAA,IACxB,YAAY,OAAO;AAAA,IACnB,gBAAgB,OAAO;AAAA,IACvB,kBAAkB,OAAO;AAAA,IACzB,WAAW,oBAAI,KAAK;AAAA,IACpB,WAAW,oBAAI,KAAK;AAAA,EACtB,CAAC;AACH;AAEA,MAAM,2BAAgH;AAAA,EACpH,IAAI;AAAA,EACJ,MAAM,QAAQ,UAAU,KAAK;AAC3B,UAAM,EAAE,QAAQ,OAAO,IAAI,sBAAsB,yBAAyB,QAAQ;AAElF,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,mBAAmB,sBAAsB,OAAO,gBAAgB,MAAM,IAAI,IAAI;AACpF,UAAM,EAAE,aAAa,UAAU,kBAAkB,IAAI,MAAM,iBAAiB,IAAI,OAAO,QAAQ;AAC7F,YAAM,SAAS,MAAM,4BAA4B,KAAK,OAAO,UAAU,EAAE,UAAU,OAAO,UAAU,gBAAgB,OAAO,eAAe,CAAC;AAC3I,wBAAkB,KAAK,OAAO,QAAQ;AACtC,8BAAwB,KAAK,OAAO,cAAc;AAElD,UAAI,OAAO,QAAQ;AACjB,cAAM,mBAAmB,KAAK,OAAO,QAAQ,OAAO,UAAU,OAAO,cAAc;AAAA,MACrF;AAEA,YAAMA,eAAc,sBAAsB,KAAK;AAAA,QAC7C,GAAI,OAAO,KAAK,EAAE,IAAI,OAAO,GAAG,IAAI,CAAC;AAAA,QACrC,gBAAgB,OAAO;AAAA,QACvB,UAAU,OAAO;AAAA,QACjB;AAAA,QACA,iBAAiB,OAAO;AAAA,QACxB,OAAO,OAAO,SAAS;AAAA,QACvB,MAAM,OAAO,QAAQ;AAAA,QACrB,QAAQ,OAAO,UAAU;AAAA,QACzB,aAAa,OAAO,eAAe;AAAA,QACnC,YAAY,OAAO,cAAc;AAAA,QACjC,UAAU,OAAO,YAAY;AAAA,QAC7B,cAAc;AAAA,QACd,aAAa,OAAO,eAAe;AAAA,QACnC,QAAQ,OAAO,UAAU;AAAA,QACzB,QAAQ,OAAO,UAAU;AAAA,QACzB,gBAAgB,OAAO,kBAAkB;AAAA,QACzC,iBAAiB,OAAO,mBAAmB;AAAA,QAC3C,iBAAiB,OAAO,mBAAmB;AAAA,QAC3C,UAAU,OAAO,YAAY;AAAA,QAC7B,QAAQ,OAAO,UAAU;AAAA,QACzB,gBAAgB,OAAO,kBAAkB;AAAA,QACzC,eAAe,OAAO,iBAAiB;AAAA,QACvC,cAAc,OAAO,gBAAgB;AAAA,QACrC,iBAAiB,OAAO,mBAAmB;AAAA,QAC3C,YAAY,OAAO,cAAc;AAAA,QACjC,gBAAgB,OAAO,kBAAkB;AAAA,QACzC,kBAAkB,OAAO,oBAAoB;AAAA,MAC/C,CAAC;AACD,UAAI,QAAQA,YAAW;AACvB,YAAM,IAAI,MAAM;AAEhB,YAAM;AAAA,QACJ,8BAA8B,KAAK,GAAG;AAAA,QACtCA,aAAY;AAAA,QACZ,OAAO;AAAA,QACP,OAAO;AAAA,QACP;AAAA,MACF;AAEA,YAAM,aAAa,MAAM,yBAAyB,KAAK,OAAO,EAAE;AAEhE,aAAO;AAAA,QACL,aAAAA;AAAA,QACA,UAAU,OAAO;AAAA,QACjB,mBAAmB,WAAW;AAAA,MAChC;AAAA,IACF,CAAC;AAED,UAAM,KAAM,IAAI,UAAU,QAAQ,YAAY;AAC9C,UAAM,oBAAoB;AAAA,MACxB,YAAY;AAAA,MACZ,QAAQ;AAAA,MACR,QAAQ;AAAA,MACR,aAAa;AAAA,QACX,IAAI,YAAY;AAAA,QAChB,gBAAgB,YAAY;AAAA,QAC5B,UAAU,YAAY;AAAA,MACxB;AAAA,MACA,YAAY,IAAI;AAAA,MAChB,SAAS;AAAA,MACT,QAAQ;AAAA,IACV,CAAC;AACD,UAAM,gCAAgC,KAAK,EAAE,UAAU,kBAAkB,GAAG;AAAA,MAC1E,IAAI,YAAY;AAAA,MAChB,gBAAgB,YAAY;AAAA,MAC5B,UAAU,YAAY;AAAA,IACxB,CAAC;AAED,WAAO,EAAE,eAAe,YAAY,IAAI,SAAS;AAAA,EACnD;AAAA,EACA,cAAc,OAAO,QAAQ,QAAQ,QAAQ;AAC3C,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,WAAO,MAAM,wBAAwB,IAAI,OAAO,aAAa;AAAA,EAC/D;AAAA,EACA,UAAU,OAAO,EAAE,QAAQ,UAAU,MAAM;AACzC,UAAM,EAAE,UAAU,IAAI,MAAM,oBAAoB;AAChD,UAAM,WAAW,UAAU;AAC3B,WAAO;AAAA,MACL,aAAa,UAAU,uCAAuC,oBAAoB;AAAA,MAClF,cAAc;AAAA,MACd,YAAY,OAAO;AAAA,MACnB,oBAAoB,0BAA0B,UAAU,aAAa,UAAU;AAAA,MAC/E,kBAAkB,UAAU,aAAa,YAAY;AAAA,MACrD,UAAU,UAAU,YAAY,YAAY;AAAA,MAC5C,gBAAgB,UAAU,YAAY,kBAAkB;AAAA,MACxD,eAAe,YAAY;AAAA,MAC3B,SAAS;AAAA,QACP,MAAM;AAAA,UACJ,OAAO;AAAA,QACT;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EACA,MAAM,OAAO,EAAE,UAAU,IAAI,MAAM;AACjC,UAAM,gBAAgB,UAAU;AAChC,QAAI,CAAC,cAAe;AACpB,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,SAAS,MAAM,iBAAiB,IAAI,OAAO,QAAQ;AACvD,YAAM,SAAS,MAAM,sBAAsB,KAAK,qBAAqB,EAAE,IAAI,cAAc,CAAC;AAC1F,UAAI,CAAC,OAAQ,QAAO;AACpB,YAAM,WAAW,OAAO,OAAO,WAAW,WAAW,OAAO,SAAS,OAAO,OAAO;AACnF,UAAI,OAAO,MAAM;AACjB,YAAM,IAAI,MAAM;AAChB,YAAM,aAAa,MAAM,yBAAyB,KAAK,QAAQ;AAC/D,aAAO;AAAA,QACL;AAAA,QACA,mBAAmB,WAAW;AAAA,QAC9B,aAAa;AAAA,UACX,IAAI,OAAO;AAAA,UACX,gBAAgB,OAAO;AAAA,UACvB,UAAU,OAAO;AAAA,QACnB;AAAA,MACF;AAAA,IACF,CAAC;AACD,QAAI,CAAC,OAAQ;AACb,UAAM,gCAAgC,KAAK;AAAA,MACzC,UAAU,OAAO;AAAA,MACjB,mBAAmB,OAAO;AAAA,IAC5B,GAAG,OAAO,WAAW;AAAA,EACvB;AAAA,EACA,MAAM,OAAO,EAAE,UAAU,IAAI,MAAM;AACjC,UAAM,QAAQ,oBAAyC,QAAQ;AAC/D,QAAI,CAAC,OAAO;AACV,YAAM,IAAI,cAAc,KAAK,EAAE,OAAO,8DAA8D,CAAC;AAAA,IACvG;AACA,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,EAAE,aAAa,kBAAkB,IAAI,MAAM,iBAAiB,IAAI,OAAO,QAAQ;AACnF,YAAM,SAAS,MAAM,4BAA4B,KAAK,MAAM,YAAY,UAAU,EAAE,UAAU,MAAM,YAAY,UAAU,gBAAgB,MAAM,YAAY,eAAe,CAAC;AAC5K,UAAIA,eAAc,MAAM,sBAAsB,KAAK,qBAAqB,EAAE,IAAI,MAAM,YAAY,GAAG,CAAC;AACpG,UAAI,CAACA,cAAa;AAChB,QAAAA,eAAc,sBAAsB,KAAK;AAAA,UACvC,IAAI,MAAM,YAAY;AAAA,UACtB,gBAAgB,MAAM,YAAY;AAAA,UAClC,UAAU,MAAM,YAAY;AAAA,UAC5B;AAAA,UACA,iBAAiB,MAAM,YAAY;AAAA,UACnC,OAAO,MAAM,YAAY;AAAA,UACzB,MAAM,MAAM,YAAY;AAAA,UACxB,QAAQ,MAAM,YAAY;AAAA,UAC1B,aAAa,MAAM,YAAY;AAAA,UAC/B,YAAY,MAAM,YAAY;AAAA,UAC9B,UAAU,MAAM,YAAY;AAAA,UAC5B,cAAc,MAAM,YAAY;AAAA,UAChC,aAAa,MAAM,YAAY;AAAA,UAC/B,QAAQ,MAAM,YAAY;AAAA,UAC1B,QAAQ,MAAM,YAAY;AAAA,UAC1B,gBAAgB,MAAM,YAAY;AAAA,UAClC,iBAAiB,MAAM,YAAY;AAAA,UACnC,iBAAiB,MAAM,YAAY;AAAA,UACnC,UAAU,MAAM,YAAY;AAAA,UAC5B,QAAQ,MAAM,YAAY;AAAA,UAC1B,gBAAgB,MAAM,YAAY;AAAA,UAClC,eAAe,MAAM,YAAY;AAAA,UACjC,cAAc,MAAM,YAAY;AAAA,UAChC,iBAAiB,MAAM,YAAY;AAAA,UACnC,YAAY,MAAM,YAAY;AAAA,UAC9B,gBAAgB,MAAM,YAAY;AAAA,UAClC,kBAAkB,MAAM,YAAY;AAAA,QACtC,CAAC;AACD,YAAI,QAAQA,YAAW;AAAA,MACzB,OAAO;AACL,QAAAA,aAAY,YAAY;AACxB,QAAAA,aAAY,SAAS;AACrB,QAAAA,aAAY,kBAAkB,MAAM,YAAY;AAChD,QAAAA,aAAY,QAAQ,MAAM,YAAY;AACtC,QAAAA,aAAY,OAAO,MAAM,YAAY;AACrC,QAAAA,aAAY,SAAS,MAAM,YAAY;AACvC,QAAAA,aAAY,cAAc,MAAM,YAAY;AAC5C,QAAAA,aAAY,aAAa,MAAM,YAAY;AAC3C,QAAAA,aAAY,WAAW,MAAM,YAAY;AACzC,QAAAA,aAAY,eAAe,MAAM,YAAY;AAC7C,QAAAA,aAAY,cAAc,MAAM,YAAY;AAC5C,QAAAA,aAAY,SAAS,MAAM,YAAY;AACvC,QAAAA,aAAY,SAAS,MAAM,YAAY;AACvC,QAAAA,aAAY,iBAAiB,MAAM,YAAY;AAC/C,QAAAA,aAAY,kBAAkB,MAAM,YAAY;AAChD,QAAAA,aAAY,kBAAkB,MAAM,YAAY;AAChD,QAAAA,aAAY,WAAW,MAAM,YAAY;AACzC,QAAAA,aAAY,SAAS,MAAM,YAAY;AACvC,QAAAA,aAAY,iBAAiB,MAAM,YAAY;AAC/C,QAAAA,aAAY,gBAAgB,MAAM,YAAY;AAC9C,QAAAA,aAAY,eAAe,MAAM,YAAY;AAC7C,QAAAA,aAAY,kBAAkB,MAAM,YAAY;AAChD,QAAAA,aAAY,aAAa,MAAM,YAAY;AAC3C,QAAAA,aAAY,iBAAiB,MAAM,YAAY;AAC/C,QAAAA,aAAY,mBAAmB,MAAM,YAAY;AAAA,MACnD;AACA,YAAM,IAAI,MAAM;AAEhB,YAAM,aAAa,MAAM,yBAAyB,KAAK,MAAM,YAAY,QAAQ;AAEjF,YAAM,gBAAgB,yBAAyB,MAAM,QAAQ,MAAS;AACtE,UAAI,OAAO,KAAK,aAAa,EAAE,QAAQ;AACrC,cAAM,qBAAqB;AAAA,UACzB,YAAY,8BAA8B,KAAK,GAAG;AAAA,UAClD,UAAU;AAAA,UACV,UAAUA,aAAY;AAAA,UACtB,gBAAgBA,aAAY;AAAA,UAC5B,UAAUA,aAAY;AAAA,UACtB,QAAQ;AAAA,UACR,QAAQ;AAAA,QACV,CAAC;AAAA,MACH;AAEA,aAAO,EAAE,aAAAA,cAAa,mBAAmB,WAAW,kBAAkB;AAAA,IACxE,CAAC;AAED,UAAM,KAAM,IAAI,UAAU,QAAQ,YAAY;AAC9C,UAAM,oBAAoB;AAAA,MACxB,YAAY;AAAA,MACZ,QAAQ;AAAA,MACR,QAAQ;AAAA,MACR,aAAa;AAAA,QACX,IAAI,YAAY;AAAA,QAChB,gBAAgB,YAAY;AAAA,QAC5B,UAAU,YAAY;AAAA,MACxB;AAAA,MACA,YAAY,IAAI;AAAA,MAChB,SAAS;AAAA,MACT,QAAQ;AAAA,IACV,CAAC;AACD,UAAM,gCAAgC,KAAK;AAAA,MACzC,UAAU,MAAM,YAAY;AAAA,MAC5B;AAAA,IACF,GAAG;AAAA,MACD,IAAI,YAAY;AAAA,MAChB,gBAAgB,YAAY;AAAA,MAC5B,UAAU,YAAY;AAAA,IACxB,CAAC;AAED,WAAO,EAAE,eAAe,YAAY,IAAI,UAAU,MAAM,YAAY,SAAS;AAAA,EAC/E;AACF;AAIA,MAAM,2BAA8F;AAAA,EAClG,IAAI;AAAA,EACJ,MAAM,QAAQ,UAAU,KAAK;AAC3B,UAAM,EAAE,OAAO,IAAI,sBAAsB,yBAAyB,QAAQ;AAC1E,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,WAAW,MAAM,wBAAwB,IAAI,OAAO,EAAE;AAC5D,WAAO,WAAW,EAAE,QAAQ,SAAS,IAAI,CAAC;AAAA,EAC5C;AAAA,EACA,MAAM,QAAQ,UAAU,KAAK;AAC3B,UAAM,EAAE,QAAQ,OAAO,IAAI,sBAAsB,yBAAyB,QAAQ;AAClF,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,EAAE,aAAa,UAAU,kBAAkB,IAAI,MAAM,iBAAiB,IAAI,OAAO,QAAQ;AAC7F,YAAMA,eAAc,MAAM,sBAAsB,KAAK,qBAAqB,EAAE,IAAI,OAAO,IAAI,WAAW,KAAK,CAAC;AAC5G,UAAI,CAACA,cAAa;AAChB,oCAA4B,EAAE,cAAc,yBAAyB,YAAY,OAAO,IAAI,SAAS,IAAI,WAAW,KAAK,CAAC;AAC1H,cAAM,IAAI,cAAc,KAAK,EAAE,OAAO,wBAAwB,CAAC;AAAA,MACjE;AACA,wBAAkB,KAAKA,aAAY,QAAQ;AAC3C,8BAAwB,KAAKA,aAAY,cAAc;AAOvD,mCAA6B;AAAA,QAC3B,cAAc;AAAA,QACd,YAAYA,aAAY;AAAA,QACxB,SAASA,aAAY;AAAA,QACrB,SAAS,IAAI,WAAW;AAAA,MAC1B,CAAC;AASD,UACE,OAAO,eAAe,UACtBA,aAAY,oBAAoB,YAC/B,OAAO,cAAc,WAAWA,aAAY,cAAc,OAC3D;AACA,cAAM,cAAe,IAAI,MAAyC,OAAO;AACzE,YACE,CAAC,yBAAyB;AAAA,UACxB,iBAAiBA,aAAY;AAAA,UAC7B,mBAAmBA,aAAY;AAAA,UAC/B,gBAAgB,OAAO;AAAA,UACvB,cAAcA,aAAY;AAAA,UAC1B;AAAA;AAAA;AAAA,UAGA,cAAc;AAAA,QAChB,CAAC,GACD;AACA,gBAAM,IAAI,cAAc,KAAK,EAAE,OAAO,kBAAkB,CAAC;AAAA,QAC3D;AAAA,MACF;AAEA,UAAI,OAAO,WAAW,QAAW;AAC/B,YAAI,OAAO,QAAQ;AACjB,gBAAM,mBAAmB,KAAK,OAAO,QAAQA,aAAY,UAAUA,aAAY,cAAc;AAAA,QAC/F;AACA,QAAAA,aAAY,SAAS,OAAO,UAAU;AAAA,MACxC;AACA,UAAI,OAAO,oBAAoB,OAAW,CAAAA,aAAY,kBAAkB,OAAO;AAC/E,UAAI,OAAO,UAAU,OAAW,CAAAA,aAAY,QAAQ,OAAO,SAAS;AACpE,UAAI,OAAO,SAAS,OAAW,CAAAA,aAAY,OAAO,OAAO,QAAQ;AACjE,UAAI,OAAO,WAAW,OAAW,CAAAA,aAAY,SAAS,OAAO;AAC7D,UAAI,OAAO,gBAAgB,OAAW,CAAAA,aAAY,cAAc,OAAO,eAAe;AACtF,UAAI,OAAO,eAAe,OAAW,CAAAA,aAAY,aAAa,OAAO,cAAc;AACnF,UAAI,OAAO,aAAa,OAAW,CAAAA,aAAY,WAAW,OAAO,YAAY;AAC7E,UAAI,OAAO,iBAAiB,OAAW,CAAAA,aAAY,eAAe,OAAO,gBAAgB;AACzF,UAAI,OAAO,gBAAgB,OAAW,CAAAA,aAAY,cAAc,OAAO,eAAe;AACtF,UAAI,OAAO,mBAAmB,OAAW,CAAAA,aAAY,iBAAiB,OAAO,kBAAkB;AAC/F,UAAI,OAAO,oBAAoB,OAAW,CAAAA,aAAY,kBAAkB,OAAO,mBAAmB;AAClG,UAAI,OAAO,WAAW,OAAW,CAAAA,aAAY,SAAS,OAAO;AAC7D,UAAI,OAAO,oBAAoB,OAAW,CAAAA,aAAY,kBAAkB,OAAO,mBAAmB;AAClG,UAAI,OAAO,aAAa,OAAW,CAAAA,aAAY,WAAW,OAAO,YAAY;AAC7E,UAAI,OAAO,WAAW,OAAW,CAAAA,aAAY,SAAS,OAAO,UAAU;AACvE,UAAI,OAAO,mBAAmB,OAAW,CAAAA,aAAY,iBAAiB,OAAO,kBAAkB;AAC/F,UAAI,OAAO,kBAAkB,OAAW,CAAAA,aAAY,gBAAgB,OAAO,iBAAiB;AAC5F,UAAI,OAAO,iBAAiB,OAAW,CAAAA,aAAY,eAAe,OAAO,gBAAgB;AACzF,UAAI,OAAO,oBAAoB,OAAW,CAAAA,aAAY,kBAAkB,OAAO,mBAAmB;AAClG,UAAI,OAAO,eAAe,OAAW,CAAAA,aAAY,aAAa,OAAO,cAAc;AACnF,UAAI,OAAO,mBAAmB,OAAW,CAAAA,aAAY,iBAAiB,OAAO,kBAAkB;AAC/F,UAAI,OAAO,qBAAqB,OAAW,CAAAA,aAAY,mBAAmB,OAAO,oBAAoB;AAErG,YAAM,IAAI,MAAM;AAEhB,YAAMC,YAAW,OAAOD,aAAY,WAAW,WAAWA,aAAY,SAASA,aAAY,OAAO;AAClG,YAAM;AAAA,QACJ,8BAA8B,KAAK,GAAG;AAAA,QACtCA,aAAY;AAAA,QACZA,aAAY;AAAA,QACZA,aAAY;AAAA,QACZ;AAAA,MACF;AAEA,YAAM,aAAa,MAAM,yBAAyB,KAAKC,SAAQ;AAE/D,aAAO,EAAE,aAAAD,cAAa,UAAAC,WAAU,mBAAmB,WAAW,kBAAkB;AAAA,IAClF,CAAC;AAED,UAAM,KAAM,IAAI,UAAU,QAAQ,YAAY;AAC9C,UAAM,oBAAoB;AAAA,MACxB,YAAY;AAAA,MACZ,QAAQ;AAAA,MACR,QAAQ;AAAA,MACR,aAAa;AAAA,QACX,IAAI,YAAY;AAAA,QAChB,gBAAgB,YAAY;AAAA,QAC5B,UAAU,YAAY;AAAA,MACxB;AAAA,MACA,YAAY,IAAI;AAAA,MAChB,SAAS;AAAA,MACT,QAAQ;AAAA,IACV,CAAC;AACD,UAAM,gCAAgC,KAAK,EAAE,UAAU,kBAAkB,GAAG;AAAA,MAC1E,IAAI,YAAY;AAAA,MAChB,gBAAgB,YAAY;AAAA,MAC5B,UAAU,YAAY;AAAA,IACxB,CAAC;AAED,WAAO,EAAE,eAAe,YAAY,GAAG;AAAA,EACzC;AAAA,EACA,cAAc,OAAO,QAAQ,QAAQ,QAAQ;AAC3C,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,WAAO,MAAM,wBAAwB,IAAI,OAAO,aAAa;AAAA,EAC/D;AAAA,EACA,UAAU,OAAO,EAAE,UAAU,MAAM;AACjC,UAAM,EAAE,UAAU,IAAI,MAAM,oBAAoB;AAChD,UAAM,SAAS,UAAU;AACzB,QAAI,CAAC,OAAQ,QAAO;AACpB,UAAM,gBAAgB,UAAU;AAChC,WAAO;AAAA,MACL,aAAa,UAAU,uCAAuC,oBAAoB;AAAA,MAClF,cAAc;AAAA,MACd,YAAY,OAAO,YAAY;AAAA,MAC/B,oBAAoB,0BAA0B,OAAO,YAAY,UAAU;AAAA,MAC3E,kBAAkB,OAAO,YAAY,YAAY;AAAA,MACjD,UAAU,OAAO,YAAY;AAAA,MAC7B,gBAAgB,OAAO,YAAY;AAAA,MACnC,gBAAgB;AAAA,MAChB,eAAe,iBAAiB;AAAA,MAChC,SAAS;AAAA,QACP,MAAM;AAAA,UACJ;AAAA,UACA,OAAO,iBAAiB;AAAA,QAC1B;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EACA,MAAM,OAAO,EAAE,UAAU,IAAI,MAAM;AACjC,UAAM,UAAU,mBAA2C,QAAQ;AACnE,UAAM,SAAS,SAAS;AACxB,QAAI,CAAC,OAAQ;AACb,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,EAAE,aAAa,kBAAkB,IAAI,MAAM,iBAAiB,IAAI,OAAO,QAAQ;AACnF,UAAID,eAAc,MAAM,sBAAsB,KAAK,qBAAqB,EAAE,IAAI,OAAO,YAAY,GAAG,CAAC;AACrG,YAAM,SAAS,MAAM,4BAA4B,KAAK,OAAO,YAAY,UAAU,EAAE,UAAU,OAAO,YAAY,UAAU,gBAAgB,OAAO,YAAY,eAAe,CAAC;AAE/K,UAAI,CAACA,cAAa;AAChB,QAAAA,eAAc,IAAI,OAAO,qBAAqB;AAAA,UAC5C,IAAI,OAAO,YAAY;AAAA,UACvB,gBAAgB,OAAO,YAAY;AAAA,UACnC,UAAU,OAAO,YAAY;AAAA,UAC7B;AAAA,UACA,iBAAiB,OAAO,YAAY;AAAA,UACpC,OAAO,OAAO,YAAY;AAAA,UAC1B,MAAM,OAAO,YAAY;AAAA,UACzB,QAAQ,OAAO,YAAY;AAAA,UAC3B,aAAa,OAAO,YAAY;AAAA,UAChC,YAAY,OAAO,YAAY;AAAA,UAC/B,UAAU,OAAO,YAAY;AAAA,UAC7B,cAAc,OAAO,YAAY;AAAA,UACjC,aAAa,OAAO,YAAY;AAAA,UAChC,QAAQ,OAAO,YAAY;AAAA,UAC3B,QAAQ,OAAO,YAAY;AAAA,UAC3B,gBAAgB,OAAO,YAAY;AAAA,UACnC,iBAAiB,OAAO,YAAY;AAAA,UACpC,iBAAiB,OAAO,YAAY;AAAA,UACpC,UAAU,OAAO,YAAY;AAAA,UAC7B,QAAQ,OAAO,YAAY;AAAA,UAC3B,gBAAgB,OAAO,YAAY;AAAA,UACnC,eAAe,OAAO,YAAY;AAAA,UAClC,cAAc,OAAO,YAAY;AAAA,UACjC,iBAAiB,OAAO,YAAY;AAAA,UACpC,YAAY,OAAO,YAAY;AAAA,UAC/B,gBAAgB,OAAO,YAAY;AAAA,UACnC,kBAAkB,OAAO,YAAY;AAAA,UACrC,WAAW,oBAAI,KAAK;AAAA,UACpB,WAAW,oBAAI,KAAK;AAAA,QACtB,CAAC;AACD,YAAI,QAAQA,YAAW;AAAA,MACzB,OAAO;AACL,QAAAA,aAAY,SAAS;AACrB,QAAAA,aAAY,kBAAkB,OAAO,YAAY;AACjD,QAAAA,aAAY,QAAQ,OAAO,YAAY;AACvC,QAAAA,aAAY,OAAO,OAAO,YAAY;AACtC,QAAAA,aAAY,SAAS,OAAO,YAAY;AACxC,QAAAA,aAAY,cAAc,OAAO,YAAY;AAC7C,QAAAA,aAAY,aAAa,OAAO,YAAY;AAC5C,QAAAA,aAAY,WAAW,OAAO,YAAY;AAC1C,QAAAA,aAAY,eAAe,OAAO,YAAY;AAC9C,QAAAA,aAAY,cAAc,OAAO,YAAY;AAC7C,QAAAA,aAAY,SAAS,OAAO,YAAY;AACxC,QAAAA,aAAY,SAAS,OAAO,YAAY;AACxC,QAAAA,aAAY,iBAAiB,OAAO,YAAY;AAChD,QAAAA,aAAY,kBAAkB,OAAO,YAAY;AACjD,QAAAA,aAAY,kBAAkB,OAAO,YAAY;AACjD,QAAAA,aAAY,WAAW,OAAO,YAAY;AAC1C,QAAAA,aAAY,SAAS,OAAO,YAAY;AACxC,QAAAA,aAAY,iBAAiB,OAAO,YAAY;AAChD,QAAAA,aAAY,gBAAgB,OAAO,YAAY;AAC/C,QAAAA,aAAY,eAAe,OAAO,YAAY;AAC9C,QAAAA,aAAY,kBAAkB,OAAO,YAAY;AACjD,QAAAA,aAAY,aAAa,OAAO,YAAY;AAC5C,QAAAA,aAAY,iBAAiB,OAAO,YAAY;AAChD,QAAAA,aAAY,mBAAmB,OAAO,YAAY;AAAA,MACpD;AAEA,YAAM,IAAI,MAAM;AAChB,YAAM,aAAa,MAAM,yBAAyB,KAAK,OAAO,YAAY,QAAQ;AAElF,YAAM,cAAc,yBAAyB,OAAO,QAAQ,SAAS,OAAO,MAAM;AAClF,UAAI,OAAO,KAAK,WAAW,EAAE,QAAQ;AACnC,cAAM,qBAAqB;AAAA,UACzB,YAAY,8BAA8B,KAAK,GAAG;AAAA,UAClD,UAAU;AAAA,UACV,UAAUA,aAAY;AAAA,UACtB,gBAAgBA,aAAY;AAAA,UAC5B,UAAUA,aAAY;AAAA,UACtB,QAAQ;AAAA,UACR,QAAQ;AAAA,QACV,CAAC;AAAA,MACH;AAEA,aAAO;AAAA,QACL,aAAAA;AAAA,QACA,mBAAmB,WAAW;AAAA,MAChC;AAAA,IACF,CAAC;AAED,UAAM,KAAM,IAAI,UAAU,QAAQ,YAAY;AAC9C,UAAM,wBAAwB;AAAA,MAC5B,YAAY;AAAA,MACZ,QAAQ;AAAA,MACR,QAAQ;AAAA,MACR,aAAa;AAAA,QACX,IAAI,YAAY;AAAA,QAChB,gBAAgB,YAAY;AAAA,QAC5B,UAAU,YAAY;AAAA,MACxB;AAAA,MACA,YAAY,IAAI;AAAA,MAChB,SAAS;AAAA,MACT,QAAQ;AAAA,IACV,CAAC;AACD,UAAM,gCAAgC,KAAK;AAAA,MACzC,UAAU,OAAO,YAAY;AAAA,MAC7B;AAAA,IACF,GAAG;AAAA,MACD,IAAI,YAAY;AAAA,MAChB,gBAAgB,YAAY;AAAA,MAC5B,UAAU,YAAY;AAAA,IACxB,CAAC;AAAA,EACH;AACF;AAIA,MAAM,6BAAkG;AAAA,EACtG,IAAI;AAAA,EACJ,MAAM,QAAQ,UAAU,KAAK;AAC3B,UAAM,SAAS,0BAA0B,MAAM,QAAQ;AACvD,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,WAAW,MAAM,wBAAwB,IAAI,OAAO,EAAE;AAC5D,WAAO,WAAW,EAAE,QAAQ,SAAS,IAAI,CAAC;AAAA,EAC5C;AAAA,EACA,MAAM,QAAQ,UAAU,KAAK;AAC3B,UAAM,SAAS,0BAA0B,MAAM,QAAQ;AACvD,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,EAAE,aAAa,UAAU,kBAAkB,IAAI,MAAM,iBAAiB,IAAI,OAAO,QAAQ;AAC7F,YAAMA,eAAc,MAAM,sBAAsB,KAAK,qBAAqB,EAAE,IAAI,OAAO,IAAI,WAAW,KAAK,CAAC;AAC5G,UAAI,CAACA,cAAa;AAChB,oCAA4B,EAAE,cAAc,yBAAyB,YAAY,OAAO,IAAI,SAAS,IAAI,WAAW,KAAK,CAAC;AAC1H,cAAM,IAAI,cAAc,KAAK,EAAE,OAAO,wBAAwB,CAAC;AAAA,MACjE;AACA,wBAAkB,KAAKA,aAAY,QAAQ;AAC3C,8BAAwB,KAAKA,aAAY,cAAc;AAEvD,mCAA6B;AAAA,QAC3B,cAAc;AAAA,QACd,YAAYA,aAAY;AAAA,QACxB,SAASA,aAAY;AAAA,QACrB,SAAS,IAAI,WAAW;AAAA,MAC1B,CAAC;AAED,MAAAA,aAAY,SAAS;AACrB,MAAAA,aAAY,aAAa,OAAO,cAAc,oBAAI,KAAK;AACvD,YAAM,IAAI,MAAM;AAEhB,YAAMC,YAAW,OAAOD,aAAY,WAAW,WAAWA,aAAY,SAASA,aAAY,OAAO;AAClG,YAAM,aAAa,MAAM,yBAAyB,KAAKC,SAAQ;AAC/D,aAAO,EAAE,aAAAD,cAAa,UAAAC,WAAU,mBAAmB,WAAW,kBAAkB;AAAA,IAClF,CAAC;AAED,UAAM,cAAc;AAAA,MAClB,IAAI,YAAY;AAAA,MAChB,gBAAgB,YAAY;AAAA,MAC5B,UAAU,YAAY;AAAA,IACxB;AACA,UAAM,KAAM,IAAI,UAAU,QAAQ,YAAY;AAC9C,UAAM,oBAAoB;AAAA,MACxB,YAAY;AAAA,MACZ,QAAQ;AAAA,MACR,QAAQ;AAAA,MACR;AAAA,MACA,YAAY,IAAI;AAAA,MAChB,SAAS;AAAA,MACT,QAAQ;AAAA,IACV,CAAC;AACD,UAAM,mBAAmB,KAAK,mCAAmC;AAAA,MAC/D,GAAG;AAAA,MACH;AAAA,MACA,iBAAiB,YAAY;AAAA,MAC7B,QAAQ,YAAY;AAAA,MACpB,QAAQ,YAAY,UAAU;AAAA,MAC9B,YAAY,YAAY,YAAY,YAAY,KAAK;AAAA,MACrD,GAAI,IAAI,aAAa,EAAE,YAAY,IAAI,WAAW,IAAI,CAAC;AAAA,IACzD,CAAC;AACD,UAAM,gCAAgC,KAAK,EAAE,UAAU,kBAAkB,GAAG,WAAW;AAEvF,WAAO,EAAE,eAAe,YAAY,GAAG;AAAA,EACzC;AAAA,EACA,cAAc,OAAO,QAAQ,QAAQ,QAAQ;AAC3C,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,WAAO,MAAM,wBAAwB,IAAI,OAAO,aAAa;AAAA,EAC/D;AAAA,EACA,UAAU,OAAO,EAAE,UAAU,MAAM;AACjC,UAAM,EAAE,UAAU,IAAI,MAAM,oBAAoB;AAChD,UAAM,SAAS,UAAU;AACzB,QAAI,CAAC,OAAQ,QAAO;AACpB,UAAM,gBAAgB,UAAU;AAChC,WAAO;AAAA,MACL,aAAa,UAAU,yCAAyC,sBAAsB;AAAA,MACtF,cAAc;AAAA,MACd,YAAY,OAAO,YAAY;AAAA,MAC/B,oBAAoB,0BAA0B,OAAO,YAAY,UAAU;AAAA,MAC3E,kBAAkB,OAAO,YAAY,YAAY;AAAA,MACjD,UAAU,OAAO,YAAY;AAAA,MAC7B,gBAAgB,OAAO,YAAY;AAAA,MACnC,gBAAgB;AAAA,MAChB,eAAe,iBAAiB;AAAA,MAChC,SAAS;AAAA,QACP,MAAM;AAAA,UACJ;AAAA,UACA,OAAO,iBAAiB;AAAA,QAC1B;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EACA,MAAM,OAAO,EAAE,UAAU,IAAI,MAAM;AACjC,UAAM,UAAU,mBAA2C,QAAQ;AACnE,UAAM,SAAS,SAAS;AACxB,QAAI,CAAC,OAAQ;AACb,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,SAAS,MAAM,iBAAiB,IAAI,OAAO,QAAQ;AACvD,YAAM,cAAc,MAAM,sBAAsB,KAAK,qBAAqB,EAAE,IAAI,OAAO,YAAY,GAAG,CAAC;AACvG,UAAI,CAAC,YAAa,QAAO;AAEzB,kBAAY,SAAS,OAAO,YAAY;AACxC,kBAAY,aAAa,OAAO,YAAY;AAC5C,YAAM,IAAI,MAAM;AAEhB,YAAM,aAAa,MAAM,yBAAyB,KAAK,OAAO,YAAY,QAAQ;AAClF,aAAO;AAAA,QACL;AAAA,QACA,mBAAmB,WAAW;AAAA,MAChC;AAAA,IACF,CAAC;AACD,QAAI,CAAC,OAAQ;AAEb,UAAM,KAAM,IAAI,UAAU,QAAQ,YAAY;AAC9C,UAAM,wBAAwB;AAAA,MAC5B,YAAY;AAAA,MACZ,QAAQ;AAAA,MACR,QAAQ,OAAO;AAAA,MACf,aAAa;AAAA,QACX,IAAI,OAAO,YAAY;AAAA,QACvB,gBAAgB,OAAO,YAAY;AAAA,QACnC,UAAU,OAAO,YAAY;AAAA,MAC/B;AAAA,MACA,YAAY,IAAI;AAAA,MAChB,SAAS;AAAA,MACT,QAAQ;AAAA,IACV,CAAC;AACD,UAAM,6BAA6B,KAAK,OAAO,WAAW;AAC1D,UAAM,gCAAgC,KAAK;AAAA,MACzC,UAAU,OAAO,YAAY;AAAA,MAC7B,mBAAmB,OAAO;AAAA,IAC5B,GAAG;AAAA,MACD,IAAI,OAAO,YAAY;AAAA,MACvB,gBAAgB,OAAO,YAAY;AAAA,MACnC,UAAU,OAAO,YAAY;AAAA,IAC/B,CAAC;AAAA,EACH;AACF;AAIA,MAAM,2BAA8F;AAAA,EAClG,IAAI;AAAA,EACJ,MAAM,QAAQ,UAAU,KAAK;AAC3B,UAAM,SAAS,wBAAwB,MAAM,QAAQ;AACrD,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,WAAW,MAAM,wBAAwB,IAAI,OAAO,EAAE;AAC5D,WAAO,WAAW,EAAE,QAAQ,SAAS,IAAI,CAAC;AAAA,EAC5C;AAAA,EACA,MAAM,QAAQ,UAAU,KAAK;AAC3B,UAAM,SAAS,wBAAwB,MAAM,QAAQ;AACrD,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,EAAE,aAAa,UAAU,kBAAkB,IAAI,MAAM,iBAAiB,IAAI,OAAO,QAAQ;AAC7F,YAAMD,eAAc,MAAM,sBAAsB,KAAK,qBAAqB,EAAE,IAAI,OAAO,IAAI,WAAW,KAAK,CAAC;AAC5G,UAAI,CAACA,cAAa;AAChB,oCAA4B,EAAE,cAAc,yBAAyB,YAAY,OAAO,IAAI,SAAS,IAAI,WAAW,KAAK,CAAC;AAC1H,cAAM,IAAI,cAAc,KAAK,EAAE,OAAO,wBAAwB,CAAC;AAAA,MACjE;AACA,wBAAkB,KAAKA,aAAY,QAAQ;AAC3C,8BAAwB,KAAKA,aAAY,cAAc;AAEvD,mCAA6B;AAAA,QAC3B,cAAc;AAAA,QACd,YAAYA,aAAY;AAAA,QACxB,SAASA,aAAY;AAAA,QACrB,SAAS,IAAI,WAAW;AAAA,MAC1B,CAAC;AAED,MAAAA,aAAY,SAAS;AACrB,YAAM,IAAI,MAAM;AAEhB,YAAMC,YAAW,OAAOD,aAAY,WAAW,WAAWA,aAAY,SAASA,aAAY,OAAO;AAClG,YAAM,aAAa,MAAM,yBAAyB,KAAKC,SAAQ;AAC/D,aAAO,EAAE,aAAAD,cAAa,UAAAC,WAAU,mBAAmB,WAAW,kBAAkB;AAAA,IAClF,CAAC;AAED,UAAM,cAAc;AAAA,MAClB,IAAI,YAAY;AAAA,MAChB,gBAAgB,YAAY;AAAA,MAC5B,UAAU,YAAY;AAAA,IACxB;AACA,UAAM,KAAM,IAAI,UAAU,QAAQ,YAAY;AAC9C,UAAM,oBAAoB;AAAA,MACxB,YAAY;AAAA,MACZ,QAAQ;AAAA,MACR,QAAQ;AAAA,MACR;AAAA,MACA,YAAY,IAAI;AAAA,MAChB,SAAS;AAAA,MACT,QAAQ;AAAA,IACV,CAAC;AACD,UAAM,mBAAmB,KAAK,kCAAkC;AAAA,MAC9D,GAAG;AAAA,MACH;AAAA,MACA,iBAAiB,YAAY;AAAA,MAC7B,QAAQ,YAAY;AAAA,MACpB,QAAQ,YAAY,UAAU;AAAA,MAC9B,GAAI,IAAI,aAAa,EAAE,YAAY,IAAI,WAAW,IAAI,CAAC;AAAA,IACzD,CAAC;AACD,UAAM,gCAAgC,KAAK,EAAE,UAAU,kBAAkB,GAAG,WAAW;AAEvF,WAAO,EAAE,eAAe,YAAY,GAAG;AAAA,EACzC;AAAA,EACA,cAAc,OAAO,QAAQ,QAAQ,QAAQ;AAC3C,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,WAAO,MAAM,wBAAwB,IAAI,OAAO,aAAa;AAAA,EAC/D;AAAA,EACA,UAAU,OAAO,EAAE,UAAU,MAAM;AACjC,UAAM,EAAE,UAAU,IAAI,MAAM,oBAAoB;AAChD,UAAM,SAAS,UAAU;AACzB,QAAI,CAAC,OAAQ,QAAO;AACpB,UAAM,gBAAgB,UAAU;AAChC,WAAO;AAAA,MACL,aAAa,UAAU,uCAAuC,oBAAoB;AAAA,MAClF,cAAc;AAAA,MACd,YAAY,OAAO,YAAY;AAAA,MAC/B,oBAAoB,0BAA0B,OAAO,YAAY,UAAU;AAAA,MAC3E,kBAAkB,OAAO,YAAY,YAAY;AAAA,MACjD,UAAU,OAAO,YAAY;AAAA,MAC7B,gBAAgB,OAAO,YAAY;AAAA,MACnC,gBAAgB;AAAA,MAChB,eAAe,iBAAiB;AAAA,MAChC,SAAS;AAAA,QACP,MAAM;AAAA,UACJ;AAAA,UACA,OAAO,iBAAiB;AAAA,QAC1B;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EACA,MAAM,OAAO,EAAE,UAAU,IAAI,MAAM;AACjC,UAAM,UAAU,mBAA2C,QAAQ;AACnE,UAAM,SAAS,SAAS;AACxB,QAAI,CAAC,OAAQ;AACb,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,SAAS,MAAM,iBAAiB,IAAI,OAAO,QAAQ;AACvD,YAAM,cAAc,MAAM,sBAAsB,KAAK,qBAAqB,EAAE,IAAI,OAAO,YAAY,GAAG,CAAC;AACvG,UAAI,CAAC,YAAa,QAAO;AAEzB,kBAAY,SAAS,OAAO,YAAY;AACxC,YAAM,IAAI,MAAM;AAEhB,YAAM,aAAa,MAAM,yBAAyB,KAAK,OAAO,YAAY,QAAQ;AAClF,aAAO;AAAA,QACL;AAAA,QACA,mBAAmB,WAAW;AAAA,MAChC;AAAA,IACF,CAAC;AACD,QAAI,CAAC,OAAQ;AAEb,UAAM,KAAM,IAAI,UAAU,QAAQ,YAAY;AAC9C,UAAM,wBAAwB;AAAA,MAC5B,YAAY;AAAA,MACZ,QAAQ;AAAA,MACR,QAAQ,OAAO;AAAA,MACf,aAAa;AAAA,QACX,IAAI,OAAO,YAAY;AAAA,QACvB,gBAAgB,OAAO,YAAY;AAAA,QACnC,UAAU,OAAO,YAAY;AAAA,MAC/B;AAAA,MACA,YAAY,IAAI;AAAA,MAChB,SAAS;AAAA,MACT,QAAQ;AAAA,IACV,CAAC;AACD,UAAM,6BAA6B,KAAK,OAAO,WAAW;AAC1D,UAAM,gCAAgC,KAAK;AAAA,MACzC,UAAU,OAAO,YAAY;AAAA,MAC7B,mBAAmB,OAAO;AAAA,IAC5B,GAAG;AAAA,MACD,IAAI,OAAO,YAAY;AAAA,MACvB,gBAAgB,OAAO,YAAY;AAAA,MACnC,UAAU,OAAO,YAAY;AAAA,IAC/B,CAAC;AAAA,EACH;AACF;AAIA,MAAM,2BACJ;AAAA,EACE,IAAI;AAAA,EACJ,MAAM,QAAQ,OAAO,KAAK;AACxB,UAAM,KAAK,UAAU,OAAO,yBAAyB;AACrD,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,WAAW,MAAM,wBAAwB,IAAI,EAAE;AACrD,WAAO,WAAW,EAAE,QAAQ,SAAS,IAAI,CAAC;AAAA,EAC5C;AAAA,EACA,MAAM,QAAQ,OAAO,KAAK;AACxB,UAAM,KAAK,UAAU,OAAO,yBAAyB;AACrD,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,EAAE,aAAa,UAAU,kBAAkB,IAAI,MAAM,iBAAiB,IAAI,OAAO,QAAQ;AAC7F,YAAMD,eAAc,MAAM,sBAAsB,KAAK,qBAAqB,EAAE,IAAI,WAAW,KAAK,CAAC;AACjG,UAAI,CAACA,cAAa;AAChB,oCAA4B,EAAE,cAAc,yBAAyB,YAAY,IAAI,SAAS,IAAI,WAAW,KAAK,CAAC;AACnH,cAAM,IAAI,cAAc,KAAK,EAAE,OAAO,wBAAwB,CAAC;AAAA,MACjE;AACA,wBAAkB,KAAKA,aAAY,QAAQ;AAC3C,8BAAwB,KAAKA,aAAY,cAAc;AAEvD,mCAA6B;AAAA,QAC3B,cAAc;AAAA,QACd,YAAYA,aAAY;AAAA,QACxB,SAASA,aAAY;AAAA,QACrB,SAAS,IAAI,WAAW;AAAA,MAC1B,CAAC;AAED,YAAMC,YAAW,OAAOD,aAAY,WAAW,WAAWA,aAAY,SAASA,aAAY,OAAO;AAClG,MAAAA,aAAY,YAAY,oBAAI,KAAK;AACjC,YAAM,IAAI,MAAM;AAEhB,YAAM,aAAa,MAAM,yBAAyB,KAAKC,SAAQ;AAC/D,aAAO,EAAE,aAAAD,cAAa,UAAAC,WAAU,mBAAmB,WAAW,kBAAkB;AAAA,IAClF,CAAC;AAED,UAAM,KAAM,IAAI,UAAU,QAAQ,YAAY;AAC9C,UAAM,oBAAoB;AAAA,MACxB,YAAY;AAAA,MACZ,QAAQ;AAAA,MACR,QAAQ;AAAA,MACR,aAAa;AAAA,QACX,IAAI,YAAY;AAAA,QAChB,gBAAgB,YAAY;AAAA,QAC5B,UAAU,YAAY;AAAA,MACxB;AAAA,MACA,YAAY,IAAI;AAAA,MAChB,SAAS;AAAA,MACT,QAAQ;AAAA,IACV,CAAC;AACD,UAAM,gCAAgC,KAAK,EAAE,UAAU,kBAAkB,GAAG;AAAA,MAC1E,IAAI,YAAY;AAAA,MAChB,gBAAgB,YAAY;AAAA,MAC5B,UAAU,YAAY;AAAA,IACxB,CAAC;AACD,WAAO,EAAE,eAAe,YAAY,GAAG;AAAA,EACzC;AAAA,EACA,UAAU,OAAO,EAAE,UAAU,MAAM;AACjC,UAAM,SAAS,UAAU;AACzB,QAAI,CAAC,OAAQ,QAAO;AACpB,UAAM,EAAE,UAAU,IAAI,MAAM,oBAAoB;AAChD,WAAO;AAAA,MACL,aAAa,UAAU,uCAAuC,oBAAoB;AAAA,MAClF,cAAc;AAAA,MACd,YAAY,OAAO,YAAY;AAAA,MAC/B,oBAAoB,0BAA0B,OAAO,YAAY,UAAU;AAAA,MAC3E,kBAAkB,OAAO,YAAY,YAAY;AAAA,MACjD,UAAU,OAAO,YAAY;AAAA,MAC7B,gBAAgB,OAAO,YAAY;AAAA,MACnC,gBAAgB;AAAA,MAChB,SAAS;AAAA,QACP,MAAM;AAAA,UACJ;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EACA,MAAM,OAAO,EAAE,UAAU,IAAI,MAAM;AACjC,UAAM,UAAU,mBAA2C,QAAQ;AACnE,UAAM,SAAS,SAAS;AACxB,QAAI,CAAC,OAAQ;AACb,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,EAAE,aAAa,kBAAkB,IAAI,MAAM,iBAAiB,IAAI,OAAO,QAAQ;AACnF,YAAM,SAAS,MAAM,4BAA4B,KAAK,OAAO,YAAY,UAAU,EAAE,UAAU,OAAO,YAAY,UAAU,gBAAgB,OAAO,YAAY,eAAe,CAAC;AAC/K,UAAID,eAAc,MAAM,sBAAsB,KAAK,qBAAqB,EAAE,IAAI,OAAO,YAAY,GAAG,CAAC;AACrG,UAAI,CAACA,cAAa;AAChB,QAAAA,eAAc,IAAI,OAAO,qBAAqB;AAAA,UAC5C,IAAI,OAAO,YAAY;AAAA,UACvB,gBAAgB,OAAO,YAAY;AAAA,UACnC,UAAU,OAAO,YAAY;AAAA,UAC7B;AAAA,UACA,iBAAiB,OAAO,YAAY;AAAA,UACpC,OAAO,OAAO,YAAY;AAAA,UAC1B,MAAM,OAAO,YAAY;AAAA,UACzB,QAAQ,OAAO,YAAY;AAAA,UAC3B,aAAa,OAAO,YAAY;AAAA,UAChC,YAAY,OAAO,YAAY;AAAA,UAC/B,UAAU,OAAO,YAAY;AAAA,UAC7B,cAAc,OAAO,YAAY;AAAA,UACjC,aAAa,OAAO,YAAY;AAAA,UAChC,QAAQ,OAAO,YAAY;AAAA,UAC3B,QAAQ,OAAO,YAAY;AAAA,UAC3B,gBAAgB,OAAO,YAAY;AAAA,UACnC,iBAAiB,OAAO,YAAY;AAAA,UACpC,iBAAiB,OAAO,YAAY;AAAA,UACpC,UAAU,OAAO,YAAY;AAAA,UAC7B,QAAQ,OAAO,YAAY;AAAA,UAC3B,gBAAgB,OAAO,YAAY;AAAA,UACnC,eAAe,OAAO,YAAY;AAAA,UAClC,cAAc,OAAO,YAAY;AAAA,UACjC,iBAAiB,OAAO,YAAY;AAAA,UACpC,YAAY,OAAO,YAAY;AAAA,UAC/B,gBAAgB,OAAO,YAAY;AAAA,UACnC,kBAAkB,OAAO,YAAY;AAAA,UACrC,WAAW,oBAAI,KAAK;AAAA,UACpB,WAAW,oBAAI,KAAK;AAAA,QACtB,CAAC;AACD,YAAI,QAAQA,YAAW;AAAA,MACzB,OAAO;AACL,QAAAA,aAAY,YAAY;AACxB,QAAAA,aAAY,SAAS;AACrB,QAAAA,aAAY,kBAAkB,OAAO,YAAY;AACjD,QAAAA,aAAY,QAAQ,OAAO,YAAY;AACvC,QAAAA,aAAY,OAAO,OAAO,YAAY;AACtC,QAAAA,aAAY,SAAS,OAAO,YAAY;AACxC,QAAAA,aAAY,cAAc,OAAO,YAAY;AAC7C,QAAAA,aAAY,aAAa,OAAO,YAAY;AAC5C,QAAAA,aAAY,WAAW,OAAO,YAAY;AAC1C,QAAAA,aAAY,eAAe,OAAO,YAAY;AAC9C,QAAAA,aAAY,cAAc,OAAO,YAAY;AAC7C,QAAAA,aAAY,SAAS,OAAO,YAAY;AACxC,QAAAA,aAAY,SAAS,OAAO,YAAY;AACxC,QAAAA,aAAY,iBAAiB,OAAO,YAAY;AAChD,QAAAA,aAAY,kBAAkB,OAAO,YAAY;AACjD,QAAAA,aAAY,kBAAkB,OAAO,YAAY;AACjD,QAAAA,aAAY,WAAW,OAAO,YAAY;AAC1C,QAAAA,aAAY,SAAS,OAAO,YAAY;AACxC,QAAAA,aAAY,iBAAiB,OAAO,YAAY;AAChD,QAAAA,aAAY,gBAAgB,OAAO,YAAY;AAC/C,QAAAA,aAAY,eAAe,OAAO,YAAY;AAC9C,QAAAA,aAAY,kBAAkB,OAAO,YAAY;AACjD,QAAAA,aAAY,aAAa,OAAO,YAAY;AAC5C,QAAAA,aAAY,iBAAiB,OAAO,YAAY;AAChD,QAAAA,aAAY,mBAAmB,OAAO,YAAY;AAAA,MACpD;AACA,YAAM,IAAI,MAAM;AAEhB,YAAM,aAAa,MAAM,yBAAyB,KAAK,OAAO,YAAY,QAAQ;AAElF,YAAM,cAAc,yBAAyB,OAAO,QAAQ,MAAS;AACrE,UAAI,OAAO,KAAK,WAAW,EAAE,QAAQ;AACnC,cAAM,qBAAqB;AAAA,UACzB,YAAY,8BAA8B,KAAK,GAAG;AAAA,UAClD,UAAU;AAAA,UACV,UAAUA,aAAY;AAAA,UACtB,gBAAgBA,aAAY;AAAA,UAC5B,UAAUA,aAAY;AAAA,UACtB,QAAQ;AAAA,UACR,QAAQ;AAAA,QACV,CAAC;AAAA,MACH;AAEA,aAAO;AAAA,QACL,aAAAA;AAAA,QACA,mBAAmB,WAAW;AAAA,MAChC;AAAA,IACF,CAAC;AAED,UAAM,KAAM,IAAI,UAAU,QAAQ,YAAY;AAC9C,UAAM,wBAAwB;AAAA,MAC5B,YAAY;AAAA,MACZ,QAAQ;AAAA,MACR,QAAQ;AAAA,MACR,aAAa;AAAA,QACX,IAAI,YAAY;AAAA,QAChB,gBAAgB,YAAY;AAAA,QAC5B,UAAU,YAAY;AAAA,MACxB;AAAA,MACA,YAAY,IAAI;AAAA,MAChB,SAAS;AAAA,MACT,QAAQ;AAAA,IACV,CAAC;AACD,UAAM,gCAAgC,KAAK;AAAA,MACzC,UAAU,OAAO,YAAY;AAAA,MAC7B;AAAA,IACF,GAAG;AAAA,MACD,IAAI,YAAY;AAAA,MAChB,gBAAgB,YAAY;AAAA,MAC5B,UAAU,YAAY;AAAA,IACxB,CAAC;AAAA,EACH;AACF;AAIF,MAAM,sBAAsB,EAAE,OAAO,EAAE,UAAU,EAAE,OAAO,EAAE,IAAI,CAAC,EAAE,CAAC;AAEpE,MAAM,uBAAmF;AAAA,EACvF,IAAI;AAAA,EACJ,MAAM,QAAQ,UAAU,KAAK;AAC3B,UAAM,SAAS,oBAAoB,MAAM,QAAQ;AACjD,UAAM,KAAM,IAAI,UAAU,QAAQ,IAAI,EAAoB,KAAK;AAC/D,UAAM,aAAa,MAAM,yBAAyB,IAAI,OAAO,QAAQ;AACrE,UAAM,SAAS,MAAM,4BAA4B,IAAI,OAAO,UAAU;AAAA,MACpE,UAAU,IAAI,MAAM,YAAY;AAAA,MAChC,gBAAgB,IAAI,0BAA0B,IAAI,MAAM,SAAS;AAAA,IACnE,CAAC;AACD,UAAM,gCAAgC,KAAK;AAAA,MACzC,UAAU,OAAO;AAAA,MACjB,mBAAmB,WAAW;AAAA,IAChC,GAAG;AAAA,MACD,IAAI,OAAO;AAAA,MACX,gBAAgB,OAAO;AAAA,MACvB,UAAU,OAAO;AAAA,IACnB,CAAC;AACD,WAAO,EAAE,UAAU,OAAO,SAAS;AAAA,EACrC;AACF;AAEA,gBAAgB,wBAAwB;AACxC,gBAAgB,wBAAwB;AACxC,gBAAgB,0BAA0B;AAC1C,gBAAgB,wBAAwB;AACxC,gBAAgB,wBAAwB;AACxC,gBAAgB,oBAAoB;",
6
6
  "names": ["interaction", "entityId"]
7
7
  }
@@ -19,23 +19,38 @@ function normalizeDictionaryIcon(input) {
19
19
  return trimmed.slice(0, 48);
20
20
  }
21
21
  import { assertFound } from "@open-mercato/shared/lib/crud/errors";
22
- async function requireCustomerEntity(em, id, kind, message = "Customer entity not found") {
23
- const entity = await em.findOne(CustomerEntity, { id, deletedAt: null });
22
+ async function requireCustomerEntity(em, id, scope, kind, message = "Customer entity not found") {
23
+ const entity = await em.findOne(CustomerEntity, {
24
+ id,
25
+ deletedAt: null,
26
+ tenantId: scope.tenantId,
27
+ organizationId: scope.organizationId
28
+ });
24
29
  if (!entity) throw new CrudHttpError(404, { error: message });
25
30
  if (kind && entity.kind !== kind) {
26
31
  throw new CrudHttpError(400, { error: "Invalid entity type" });
27
32
  }
28
33
  return entity;
29
34
  }
30
- async function requireTimelineParentEntity(em, id) {
31
- const entity = await em.findOne(CustomerEntity, { id, deletedAt: null });
35
+ async function requireTimelineParentEntity(em, id, scope) {
36
+ const entity = await em.findOne(CustomerEntity, {
37
+ id,
38
+ deletedAt: null,
39
+ tenantId: scope.tenantId,
40
+ organizationId: scope.organizationId
41
+ });
32
42
  if (entity) {
33
43
  if (entity.kind !== "person" && entity.kind !== "company") {
34
44
  throw new CrudHttpError(422, { error: "entityId must reference a person or company" });
35
45
  }
36
46
  return entity;
37
47
  }
38
- const deal = await em.findOne(CustomerDeal, { id, deletedAt: null });
48
+ const deal = await em.findOne(CustomerDeal, {
49
+ id,
50
+ deletedAt: null,
51
+ tenantId: scope.tenantId,
52
+ organizationId: scope.organizationId
53
+ });
39
54
  if (deal) {
40
55
  throw new CrudHttpError(422, { error: "entityId must reference a person or company, not a deal" });
41
56
  }
@@ -83,7 +98,7 @@ async function loadEntityTagIds(em, entity) {
83
98
  }
84
99
  async function requireDealInScope(em, dealId, tenantId, organizationId) {
85
100
  if (!dealId) return null;
86
- const deal = await em.findOne(CustomerDeal, { id: dealId, deletedAt: null });
101
+ const deal = await em.findOne(CustomerDeal, { id: dealId, deletedAt: null, tenantId, organizationId });
87
102
  if (!deal) throw new CrudHttpError(400, { error: "Deal not found" });
88
103
  ensureSameScope(deal, organizationId, tenantId);
89
104
  return deal;
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../src/modules/customers/commands/shared.ts"],
4
- "sourcesContent": ["import type { EntityManager } from '@mikro-orm/postgresql'\nimport { CustomerDeal, CustomerEntity, CustomerTag, CustomerTagAssignment, CustomerDictionaryEntry, type CustomerEntityKind } from '../data/entities'\nimport { CrudHttpError } from '@open-mercato/shared/lib/crud/errors'\nimport type { CommandRuntimeContext } from '@open-mercato/shared/lib/commands'\nimport { ensureOrganizationScope, ensureSameScope } from '@open-mercato/shared/lib/commands/scope'\nimport { findWithDecryption } from '@open-mercato/shared/lib/encryption/find'\nimport type { EventBus } from '@open-mercato/events'\nexport { ensureOrganizationScope, ensureSameScope, ensureTenantScope } from '@open-mercato/shared/lib/commands/scope'\nexport { extractUndoPayload } from '@open-mercato/shared/lib/commands/undo'\n\nexport function normalizeDictionaryColor(input: unknown): string | null {\n if (typeof input !== 'string') return null\n const trimmed = input.trim()\n if (!trimmed) return null\n const hexMatch = /^#([0-9a-fA-F]{6})$/.exec(trimmed)\n if (!hexMatch) return null\n return `#${hexMatch[1].toLowerCase()}`\n}\n\nexport function normalizeDictionaryIcon(input: unknown): string | null {\n if (typeof input !== 'string') return null\n const trimmed = input.trim()\n if (!trimmed) return null\n return trimmed.slice(0, 48)\n}\n\nexport { assertFound } from '@open-mercato/shared/lib/crud/errors'\n\nexport async function requireCustomerEntity(\n em: EntityManager,\n id: string,\n kind?: CustomerEntityKind,\n message = 'Customer entity not found'\n): Promise<CustomerEntity> {\n const entity = await em.findOne(CustomerEntity, { id, deletedAt: null })\n if (!entity) throw new CrudHttpError(404, { error: message })\n if (kind && entity.kind !== kind) {\n throw new CrudHttpError(400, { error: 'Invalid entity type' })\n }\n return entity\n}\n\nexport async function requireTimelineParentEntity(\n em: EntityManager,\n id: string,\n): Promise<CustomerEntity> {\n const entity = await em.findOne(CustomerEntity, { id, deletedAt: null })\n if (entity) {\n if (entity.kind !== 'person' && entity.kind !== 'company') {\n throw new CrudHttpError(422, { error: 'entityId must reference a person or company' })\n }\n return entity\n }\n const deal = await em.findOne(CustomerDeal, { id, deletedAt: null })\n if (deal) {\n throw new CrudHttpError(422, { error: 'entityId must reference a person or company, not a deal' })\n }\n throw new CrudHttpError(404, { error: 'Customer not found' })\n}\n\nexport async function syncEntityTags(\n em: EntityManager,\n entity: CustomerEntity,\n tags: string[] | undefined | null\n): Promise<void> {\n if (tags === undefined) return\n const desired = Array.from(new Set((tags ?? []).filter((id) => typeof id === 'string')))\n const existing = await loadEntityTagIds(em, entity)\n const toRemove = existing.filter((id) => !desired.includes(id))\n if (toRemove.length) {\n await em.nativeDelete(CustomerTagAssignment, { entity, tag: { $in: toRemove } })\n }\n const toAdd = desired.filter((id) => !existing.includes(id))\n if (!toAdd.length) return\n const tagsInScope = await em.find(CustomerTag, {\n id: { $in: toAdd },\n organizationId: entity.organizationId,\n tenantId: entity.tenantId,\n })\n if (tagsInScope.length !== toAdd.length) {\n throw new CrudHttpError(400, { error: 'One or more tags not found for this scope' })\n }\n for (const tag of toAdd) {\n const assignment = em.create(CustomerTagAssignment, {\n tenantId: entity.tenantId,\n organizationId: entity.organizationId,\n tag: em.getReference(CustomerTag, tag),\n entity,\n })\n em.persist(assignment)\n }\n}\n\nexport async function loadEntityTagIds(em: EntityManager, entity: CustomerEntity): Promise<string[]> {\n const assignments = await findWithDecryption(\n em,\n CustomerTagAssignment,\n { entity },\n { populate: ['tag'] },\n { tenantId: entity.tenantId, organizationId: entity.organizationId },\n )\n return assignments.map((assignment) =>\n typeof assignment.tag === 'string' ? assignment.tag : assignment.tag.id\n )\n}\n\nexport async function requireDealInScope(\n em: EntityManager,\n dealId: string | null | undefined,\n tenantId: string,\n organizationId: string\n): Promise<CustomerDeal | null> {\n if (!dealId) return null\n const deal = await em.findOne(CustomerDeal, { id: dealId, deletedAt: null })\n if (!deal) throw new CrudHttpError(400, { error: 'Deal not found' })\n ensureSameScope(deal, organizationId, tenantId)\n return deal\n}\n\nconst DICTIONARY_KINDS = new Set([\n 'status',\n 'source',\n 'lifecycle_stage',\n 'address_type',\n 'activity_type',\n 'deal_status',\n 'pipeline_stage',\n 'job_title',\n 'industry',\n 'temperature',\n 'renewal_quarter',\n 'person_company_role',\n])\n\nconst CUSTOM_DICTIONARY_KIND_PATTERN = /^[a-z0-9]+(?:[-_][a-z0-9]+)*$/\n\nexport async function ensureDictionaryEntry(\n em: EntityManager,\n params: {\n tenantId: string\n organizationId: string\n kind: string\n value: string\n label?: string | null\n color?: string | null | undefined\n icon?: string | null | undefined\n }\n): Promise<CustomerDictionaryEntry | null> {\n const trimmed = params.value?.trim()\n if (!trimmed) return null\n if (!DICTIONARY_KINDS.has(params.kind) && !CUSTOM_DICTIONARY_KIND_PATTERN.test(params.kind)) {\n throw new CrudHttpError(400, { error: 'Unsupported dictionary kind' })\n }\n const normalized = trimmed.toLowerCase()\n const color = params.color === undefined ? undefined : normalizeDictionaryColor(params.color)\n const icon = params.icon === undefined ? undefined : normalizeDictionaryIcon(params.icon)\n const existing = await em.findOne(CustomerDictionaryEntry, {\n tenantId: params.tenantId,\n organizationId: params.organizationId,\n kind: params.kind,\n normalizedValue: normalized,\n })\n if (existing) {\n let changed = false\n if (color !== undefined) {\n if (existing.color !== color) {\n existing.color = color ?? null\n changed = true\n }\n }\n if (icon !== undefined) {\n if (existing.icon !== icon) {\n existing.icon = icon ?? null\n changed = true\n }\n }\n if (changed) {\n existing.updatedAt = new Date()\n em.persist(existing)\n }\n return existing\n }\n const entry = em.create(CustomerDictionaryEntry, {\n tenantId: params.tenantId,\n organizationId: params.organizationId,\n kind: params.kind,\n value: trimmed,\n label: params.label?.trim() || trimmed,\n normalizedValue: normalized,\n color: color ?? null,\n icon: icon ?? null,\n })\n em.persist(entry)\n return entry\n}\n\nexport function resolveParentResourceKind(entityKind: CustomerEntityKind | string | null | undefined): string | null {\n if (entityKind === 'company') return 'customers.company'\n if (entityKind === 'person') return 'customers.person'\n return null\n}\n\nexport type QueryIndexEventEntry = {\n entityType: string\n recordId: string\n tenantId: string | null\n organizationId: string | null\n}\n\ntype QueryIndexEventKind = 'delete' | 'upsert'\n\nfunction normalizeEventEntries(entries: readonly QueryIndexEventEntry[]): QueryIndexEventEntry[] {\n const map = new Map<string, QueryIndexEventEntry>()\n for (const entry of entries) {\n if (!entry || typeof entry !== 'object') continue\n const entityType = String(entry.entityType ?? '')\n const recordId = String(entry.recordId ?? '')\n if (!entityType || !recordId) continue\n const key = [\n entityType,\n recordId,\n entry.organizationId ?? '__org__',\n entry.tenantId ?? '__tenant__',\n ].join('|')\n if (!map.has(key)) {\n map.set(key, {\n entityType,\n recordId,\n organizationId: entry.organizationId ?? null,\n tenantId: entry.tenantId ?? null,\n })\n }\n }\n return Array.from(map.values())\n}\n\nasync function emitQueryIndexEvents(\n ctx: CommandRuntimeContext,\n entries: readonly QueryIndexEventEntry[],\n kind: QueryIndexEventKind,\n): Promise<void> {\n const normalized = normalizeEventEntries(entries)\n if (!normalized.length) return\n\n let bus: EventBus | null = null\n try {\n bus = ctx.container.resolve<EventBus>('eventBus')\n } catch (err) {\n console.warn('[customers.commands.shared] eventBus resolve failed; skipping query index events', err)\n bus = null\n }\n if (!bus) return\n\n const eventName = kind === 'delete' ? 'query_index.delete_one' : 'query_index.upsert_one'\n const crudAction = kind === 'delete' ? 'deleted' : 'updated'\n\n await Promise.all(\n normalized.map((entry) =>\n bus!\n .emitEvent(\n eventName,\n {\n entityType: entry.entityType,\n recordId: entry.recordId,\n organizationId: entry.organizationId ?? null,\n tenantId: entry.tenantId ?? null,\n crudAction,\n },\n {\n tenantId: entry.tenantId ?? null,\n organizationId: entry.organizationId ?? null,\n },\n )\n .catch((err) => {\n console.warn('[customers.commands.shared] query index emitEvent failed', entry, err)\n return undefined\n }),\n ),\n )\n}\n\nexport async function emitQueryIndexDeleteEvents(\n ctx: CommandRuntimeContext,\n entries: readonly QueryIndexEventEntry[],\n): Promise<void> {\n await emitQueryIndexEvents(ctx, entries, 'delete')\n}\n\nexport async function emitQueryIndexUpsertEvents(\n ctx: CommandRuntimeContext,\n entries: readonly QueryIndexEventEntry[],\n): Promise<void> {\n await emitQueryIndexEvents(ctx, entries, 'upsert')\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import type { EntityManager } from '@mikro-orm/postgresql'\nimport { CustomerDeal, CustomerEntity, CustomerTag, CustomerTagAssignment, CustomerDictionaryEntry, type CustomerEntityKind } from '../data/entities'\nimport { CrudHttpError } from '@open-mercato/shared/lib/crud/errors'\nimport type { CommandRuntimeContext } from '@open-mercato/shared/lib/commands'\nimport { ensureOrganizationScope, ensureSameScope } from '@open-mercato/shared/lib/commands/scope'\nimport { findWithDecryption } from '@open-mercato/shared/lib/encryption/find'\nimport type { EventBus } from '@open-mercato/events'\nexport { ensureOrganizationScope, ensureSameScope, ensureTenantScope } from '@open-mercato/shared/lib/commands/scope'\nexport { extractUndoPayload } from '@open-mercato/shared/lib/commands/undo'\n\nexport function normalizeDictionaryColor(input: unknown): string | null {\n if (typeof input !== 'string') return null\n const trimmed = input.trim()\n if (!trimmed) return null\n const hexMatch = /^#([0-9a-fA-F]{6})$/.exec(trimmed)\n if (!hexMatch) return null\n return `#${hexMatch[1].toLowerCase()}`\n}\n\nexport function normalizeDictionaryIcon(input: unknown): string | null {\n if (typeof input !== 'string') return null\n const trimmed = input.trim()\n if (!trimmed) return null\n return trimmed.slice(0, 48)\n}\n\nexport { assertFound } from '@open-mercato/shared/lib/crud/errors'\n\nexport type CustomerEntityScope = {\n tenantId: string\n organizationId: string\n}\n\nexport async function requireCustomerEntity(\n em: EntityManager,\n id: string,\n scope: CustomerEntityScope,\n kind?: CustomerEntityKind,\n message = 'Customer entity not found'\n): Promise<CustomerEntity> {\n const entity = await em.findOne(CustomerEntity, {\n id,\n deletedAt: null,\n tenantId: scope.tenantId,\n organizationId: scope.organizationId,\n })\n if (!entity) throw new CrudHttpError(404, { error: message })\n if (kind && entity.kind !== kind) {\n throw new CrudHttpError(400, { error: 'Invalid entity type' })\n }\n return entity\n}\n\nexport async function requireTimelineParentEntity(\n em: EntityManager,\n id: string,\n scope: CustomerEntityScope,\n): Promise<CustomerEntity> {\n const entity = await em.findOne(CustomerEntity, {\n id,\n deletedAt: null,\n tenantId: scope.tenantId,\n organizationId: scope.organizationId,\n })\n if (entity) {\n if (entity.kind !== 'person' && entity.kind !== 'company') {\n throw new CrudHttpError(422, { error: 'entityId must reference a person or company' })\n }\n return entity\n }\n const deal = await em.findOne(CustomerDeal, {\n id,\n deletedAt: null,\n tenantId: scope.tenantId,\n organizationId: scope.organizationId,\n })\n if (deal) {\n throw new CrudHttpError(422, { error: 'entityId must reference a person or company, not a deal' })\n }\n throw new CrudHttpError(404, { error: 'Customer not found' })\n}\n\nexport async function syncEntityTags(\n em: EntityManager,\n entity: CustomerEntity,\n tags: string[] | undefined | null\n): Promise<void> {\n if (tags === undefined) return\n const desired = Array.from(new Set((tags ?? []).filter((id) => typeof id === 'string')))\n const existing = await loadEntityTagIds(em, entity)\n const toRemove = existing.filter((id) => !desired.includes(id))\n if (toRemove.length) {\n await em.nativeDelete(CustomerTagAssignment, { entity, tag: { $in: toRemove } })\n }\n const toAdd = desired.filter((id) => !existing.includes(id))\n if (!toAdd.length) return\n const tagsInScope = await em.find(CustomerTag, {\n id: { $in: toAdd },\n organizationId: entity.organizationId,\n tenantId: entity.tenantId,\n })\n if (tagsInScope.length !== toAdd.length) {\n throw new CrudHttpError(400, { error: 'One or more tags not found for this scope' })\n }\n for (const tag of toAdd) {\n const assignment = em.create(CustomerTagAssignment, {\n tenantId: entity.tenantId,\n organizationId: entity.organizationId,\n tag: em.getReference(CustomerTag, tag),\n entity,\n })\n em.persist(assignment)\n }\n}\n\nexport async function loadEntityTagIds(em: EntityManager, entity: CustomerEntity): Promise<string[]> {\n const assignments = await findWithDecryption(\n em,\n CustomerTagAssignment,\n { entity },\n { populate: ['tag'] },\n { tenantId: entity.tenantId, organizationId: entity.organizationId },\n )\n return assignments.map((assignment) =>\n typeof assignment.tag === 'string' ? assignment.tag : assignment.tag.id\n )\n}\n\nexport async function requireDealInScope(\n em: EntityManager,\n dealId: string | null | undefined,\n tenantId: string,\n organizationId: string\n): Promise<CustomerDeal | null> {\n if (!dealId) return null\n const deal = await em.findOne(CustomerDeal, { id: dealId, deletedAt: null, tenantId, organizationId })\n if (!deal) throw new CrudHttpError(400, { error: 'Deal not found' })\n ensureSameScope(deal, organizationId, tenantId)\n return deal\n}\n\nconst DICTIONARY_KINDS = new Set([\n 'status',\n 'source',\n 'lifecycle_stage',\n 'address_type',\n 'activity_type',\n 'deal_status',\n 'pipeline_stage',\n 'job_title',\n 'industry',\n 'temperature',\n 'renewal_quarter',\n 'person_company_role',\n])\n\nconst CUSTOM_DICTIONARY_KIND_PATTERN = /^[a-z0-9]+(?:[-_][a-z0-9]+)*$/\n\nexport async function ensureDictionaryEntry(\n em: EntityManager,\n params: {\n tenantId: string\n organizationId: string\n kind: string\n value: string\n label?: string | null\n color?: string | null | undefined\n icon?: string | null | undefined\n }\n): Promise<CustomerDictionaryEntry | null> {\n const trimmed = params.value?.trim()\n if (!trimmed) return null\n if (!DICTIONARY_KINDS.has(params.kind) && !CUSTOM_DICTIONARY_KIND_PATTERN.test(params.kind)) {\n throw new CrudHttpError(400, { error: 'Unsupported dictionary kind' })\n }\n const normalized = trimmed.toLowerCase()\n const color = params.color === undefined ? undefined : normalizeDictionaryColor(params.color)\n const icon = params.icon === undefined ? undefined : normalizeDictionaryIcon(params.icon)\n const existing = await em.findOne(CustomerDictionaryEntry, {\n tenantId: params.tenantId,\n organizationId: params.organizationId,\n kind: params.kind,\n normalizedValue: normalized,\n })\n if (existing) {\n let changed = false\n if (color !== undefined) {\n if (existing.color !== color) {\n existing.color = color ?? null\n changed = true\n }\n }\n if (icon !== undefined) {\n if (existing.icon !== icon) {\n existing.icon = icon ?? null\n changed = true\n }\n }\n if (changed) {\n existing.updatedAt = new Date()\n em.persist(existing)\n }\n return existing\n }\n const entry = em.create(CustomerDictionaryEntry, {\n tenantId: params.tenantId,\n organizationId: params.organizationId,\n kind: params.kind,\n value: trimmed,\n label: params.label?.trim() || trimmed,\n normalizedValue: normalized,\n color: color ?? null,\n icon: icon ?? null,\n })\n em.persist(entry)\n return entry\n}\n\nexport function resolveParentResourceKind(entityKind: CustomerEntityKind | string | null | undefined): string | null {\n if (entityKind === 'company') return 'customers.company'\n if (entityKind === 'person') return 'customers.person'\n return null\n}\n\nexport type QueryIndexEventEntry = {\n entityType: string\n recordId: string\n tenantId: string | null\n organizationId: string | null\n}\n\ntype QueryIndexEventKind = 'delete' | 'upsert'\n\nfunction normalizeEventEntries(entries: readonly QueryIndexEventEntry[]): QueryIndexEventEntry[] {\n const map = new Map<string, QueryIndexEventEntry>()\n for (const entry of entries) {\n if (!entry || typeof entry !== 'object') continue\n const entityType = String(entry.entityType ?? '')\n const recordId = String(entry.recordId ?? '')\n if (!entityType || !recordId) continue\n const key = [\n entityType,\n recordId,\n entry.organizationId ?? '__org__',\n entry.tenantId ?? '__tenant__',\n ].join('|')\n if (!map.has(key)) {\n map.set(key, {\n entityType,\n recordId,\n organizationId: entry.organizationId ?? null,\n tenantId: entry.tenantId ?? null,\n })\n }\n }\n return Array.from(map.values())\n}\n\nasync function emitQueryIndexEvents(\n ctx: CommandRuntimeContext,\n entries: readonly QueryIndexEventEntry[],\n kind: QueryIndexEventKind,\n): Promise<void> {\n const normalized = normalizeEventEntries(entries)\n if (!normalized.length) return\n\n let bus: EventBus | null = null\n try {\n bus = ctx.container.resolve<EventBus>('eventBus')\n } catch (err) {\n console.warn('[customers.commands.shared] eventBus resolve failed; skipping query index events', err)\n bus = null\n }\n if (!bus) return\n\n const eventName = kind === 'delete' ? 'query_index.delete_one' : 'query_index.upsert_one'\n const crudAction = kind === 'delete' ? 'deleted' : 'updated'\n\n await Promise.all(\n normalized.map((entry) =>\n bus!\n .emitEvent(\n eventName,\n {\n entityType: entry.entityType,\n recordId: entry.recordId,\n organizationId: entry.organizationId ?? null,\n tenantId: entry.tenantId ?? null,\n crudAction,\n },\n {\n tenantId: entry.tenantId ?? null,\n organizationId: entry.organizationId ?? null,\n },\n )\n .catch((err) => {\n console.warn('[customers.commands.shared] query index emitEvent failed', entry, err)\n return undefined\n }),\n ),\n )\n}\n\nexport async function emitQueryIndexDeleteEvents(\n ctx: CommandRuntimeContext,\n entries: readonly QueryIndexEventEntry[],\n): Promise<void> {\n await emitQueryIndexEvents(ctx, entries, 'delete')\n}\n\nexport async function emitQueryIndexUpsertEvents(\n ctx: CommandRuntimeContext,\n entries: readonly QueryIndexEventEntry[],\n): Promise<void> {\n await emitQueryIndexEvents(ctx, entries, 'upsert')\n}\n"],
5
+ "mappings": "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",
6
6
  "names": ["ensureOrganizationScope", "ensureSameScope"]
7
7
  }
@@ -333,7 +333,7 @@ const assignTagCommand = {
333
333
  const em = ctx.container.resolve("em").fork();
334
334
  const tag = await em.findOne(CustomerTag, { id: parsed.tagId, tenantId: parsed.tenantId, organizationId: parsed.organizationId });
335
335
  if (!tag) throw new CrudHttpError(404, { error: "Tag not found" });
336
- const entity = await requireCustomerEntity(em, parsed.entityId, void 0, "Customer not found");
336
+ const entity = await requireCustomerEntity(em, parsed.entityId, { tenantId: parsed.tenantId, organizationId: parsed.organizationId }, void 0, "Customer not found");
337
337
  ensureSameScope(entity, parsed.organizationId, parsed.tenantId);
338
338
  const tagIds = await loadEntityTagIds(em, entity);
339
339
  if (tagIds.includes(parsed.tagId)) {
@@ -420,7 +420,7 @@ const assignTagCommand = {
420
420
  const em = ctx.container.resolve("em").fork();
421
421
  const tag = await em.findOne(CustomerTag, { id: before.tagId });
422
422
  if (!tag) throw new CrudHttpError(404, { error: "Tag not found" });
423
- const entity = await requireCustomerEntity(em, before.entityId, void 0, "Customer not found");
423
+ const entity = await requireCustomerEntity(em, before.entityId, { tenantId: before.tenantId, organizationId: before.organizationId }, void 0, "Customer not found");
424
424
  ensureSameScope(entity, before.organizationId, before.tenantId);
425
425
  let assignment = await em.findOne(CustomerTagAssignment, {
426
426
  tag,
@@ -522,7 +522,7 @@ const unassignTagCommand = {
522
522
  if (!before) return;
523
523
  const em = ctx.container.resolve("em").fork();
524
524
  const tag = await em.findOne(CustomerTag, { id: before.tagId });
525
- const entity = await requireCustomerEntity(em, before.entityId, void 0, "Customer not found");
525
+ const entity = await requireCustomerEntity(em, before.entityId, { tenantId: before.tenantId, organizationId: before.organizationId }, void 0, "Customer not found");
526
526
  ensureSameScope(entity, before.organizationId, before.tenantId);
527
527
  if (!tag) throw new CrudHttpError(404, { error: "Tag not found" });
528
528
  const existing = await em.findOne(CustomerTagAssignment, {