@open-mercato/core 0.6.4-develop.4371.1.8f3030407e → 0.6.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.turbo/turbo-build.log +2 -2
- package/AGENTS.md +37 -0
- package/dist/generated/entities/channel_ingest_dead_letter/index.js +25 -0
- package/dist/generated/entities/channel_ingest_dead_letter/index.js.map +7 -0
- package/dist/generated/entities/channel_thread_mapping/index.js +25 -0
- package/dist/generated/entities/channel_thread_mapping/index.js.map +7 -0
- package/dist/generated/entities/channel_thread_token/index.js +17 -0
- package/dist/generated/entities/channel_thread_token/index.js.map +7 -0
- package/dist/generated/entities/communication_channel/index.js +43 -0
- package/dist/generated/entities/communication_channel/index.js.map +7 -0
- package/dist/generated/entities/customer_interaction/index.js +4 -0
- package/dist/generated/entities/customer_interaction/index.js.map +2 -2
- package/dist/generated/entities/customer_settings/index.js +2 -0
- package/dist/generated/entities/customer_settings/index.js.map +2 -2
- package/dist/generated/entities/dictionary/index.js +2 -0
- package/dist/generated/entities/dictionary/index.js.map +2 -2
- package/dist/generated/entities/external_conversation/index.js +25 -0
- package/dist/generated/entities/external_conversation/index.js.map +7 -0
- package/dist/generated/entities/external_message/index.js +25 -0
- package/dist/generated/entities/external_message/index.js.map +7 -0
- package/dist/generated/entities/inbox_settings/index.js +2 -0
- package/dist/generated/entities/inbox_settings/index.js.map +2 -2
- package/dist/generated/entities/integration_credentials/index.js +3 -1
- package/dist/generated/entities/integration_credentials/index.js.map +2 -2
- package/dist/generated/entities/message/index.js +2 -0
- package/dist/generated/entities/message/index.js.map +2 -2
- package/dist/generated/entities/message_channel_link/index.js +33 -0
- package/dist/generated/entities/message_channel_link/index.js.map +7 -0
- package/dist/generated/entities/message_reaction/index.js +25 -0
- package/dist/generated/entities/message_reaction/index.js.map +7 -0
- package/dist/generated/entities/role/index.js +3 -1
- package/dist/generated/entities/role/index.js.map +2 -2
- package/dist/generated/entities/step_instance/index.js +2 -0
- package/dist/generated/entities/step_instance/index.js.map +2 -2
- package/dist/generated/entities/user/index.js +3 -1
- package/dist/generated/entities/user/index.js.map +2 -2
- package/dist/generated/entities/user_task/index.js +2 -0
- package/dist/generated/entities/user_task/index.js.map +2 -2
- package/dist/generated/entities/workflow_branch_instance/index.js +39 -0
- package/dist/generated/entities/workflow_branch_instance/index.js.map +7 -0
- package/dist/generated/entities/workflow_event/index.js +2 -0
- package/dist/generated/entities/workflow_event/index.js.map +2 -2
- package/dist/generated/entities/workflow_instance/index.js +2 -0
- package/dist/generated/entities/workflow_instance/index.js.map +2 -2
- package/dist/generated/entities.ids.generated.js +12 -0
- package/dist/generated/entities.ids.generated.js.map +2 -2
- package/dist/generated/entity-fields-registry.js +146 -0
- package/dist/generated/entity-fields-registry.js.map +2 -2
- package/dist/helpers/integration/api.js +14 -1
- package/dist/helpers/integration/api.js.map +2 -2
- package/dist/helpers/integration/authFixtures.js +2 -1
- package/dist/helpers/integration/authFixtures.js.map +2 -2
- package/dist/helpers/integration/communicationChannelsFixtures.js +58 -0
- package/dist/helpers/integration/communicationChannelsFixtures.js.map +7 -0
- package/dist/helpers/integration/crudFormFields.js +33 -0
- package/dist/helpers/integration/crudFormFields.js.map +7 -0
- package/dist/helpers/integration/crudFormPersistence.js +107 -0
- package/dist/helpers/integration/crudFormPersistence.js.map +7 -0
- package/dist/helpers/integration/currenciesFixtures.js +52 -1
- package/dist/helpers/integration/currenciesFixtures.js.map +2 -2
- package/dist/helpers/integration/customerAccountsFixtures.js +124 -0
- package/dist/helpers/integration/customerAccountsFixtures.js.map +7 -0
- package/dist/helpers/integration/dbFixtures.js +9 -1
- package/dist/helpers/integration/dbFixtures.js.map +2 -2
- package/dist/helpers/integration/inboxFixtures.js +11 -0
- package/dist/helpers/integration/inboxFixtures.js.map +2 -2
- package/dist/helpers/integration/optimisticLockUi.js +104 -0
- package/dist/helpers/integration/optimisticLockUi.js.map +7 -0
- package/dist/helpers/integration/salesFixtures.js +17 -0
- package/dist/helpers/integration/salesFixtures.js.map +2 -2
- package/dist/helpers/integration/salesUi.js +17 -9
- package/dist/helpers/integration/salesUi.js.map +2 -2
- package/dist/helpers/integration/standaloneEnv.js +58 -0
- package/dist/helpers/integration/standaloneEnv.js.map +7 -0
- package/dist/helpers/integration/undoHarness.js +177 -0
- package/dist/helpers/integration/undoHarness.js.map +7 -0
- package/dist/helpers/integration/workflowsFixtures.js +161 -0
- package/dist/helpers/integration/workflowsFixtures.js.map +2 -2
- package/dist/modules/api_keys/backend/api-keys/page.js +18 -5
- package/dist/modules/api_keys/backend/api-keys/page.js.map +2 -2
- package/dist/modules/api_keys/data/validators.js +3 -1
- package/dist/modules/api_keys/data/validators.js.map +2 -2
- package/dist/modules/attachments/components/AttachmentPartitionSettings.js +17 -9
- package/dist/modules/attachments/components/AttachmentPartitionSettings.js.map +2 -2
- package/dist/modules/attachments/lib/ocrQueue.js +6 -1
- package/dist/modules/attachments/lib/ocrQueue.js.map +2 -2
- package/dist/modules/attachments/lib/security.js +1 -1
- package/dist/modules/attachments/lib/security.js.map +2 -2
- package/dist/modules/audit_logs/api/audit-logs/actions/redo/route.js +4 -2
- package/dist/modules/audit_logs/api/audit-logs/actions/redo/route.js.map +2 -2
- package/dist/modules/audit_logs/api/audit-logs/actions/undo/route.js +3 -2
- package/dist/modules/audit_logs/api/audit-logs/actions/undo/route.js.map +2 -2
- package/dist/modules/audit_logs/data/entities.js.map +1 -1
- package/dist/modules/audit_logs/services/actionLogService.js +16 -0
- package/dist/modules/audit_logs/services/actionLogService.js.map +2 -2
- package/dist/modules/auth/api/logout.js +0 -12
- package/dist/modules/auth/api/logout.js.map +2 -2
- package/dist/modules/auth/api/roles/acl/route.js +32 -13
- package/dist/modules/auth/api/roles/acl/route.js.map +2 -2
- package/dist/modules/auth/api/roles/route.js +52 -3
- package/dist/modules/auth/api/roles/route.js.map +2 -2
- package/dist/modules/auth/api/sidebar/preferences/route.js +73 -4
- package/dist/modules/auth/api/sidebar/preferences/route.js.map +2 -2
- package/dist/modules/auth/api/users/acl/route.js +88 -34
- package/dist/modules/auth/api/users/acl/route.js.map +2 -2
- package/dist/modules/auth/api/users/consents/route.js +17 -0
- package/dist/modules/auth/api/users/consents/route.js.map +2 -2
- package/dist/modules/auth/api/users/resend-invite/route.js +18 -0
- package/dist/modules/auth/api/users/resend-invite/route.js.map +2 -2
- package/dist/modules/auth/api/users/route.js +40 -2
- package/dist/modules/auth/api/users/route.js.map +2 -2
- package/dist/modules/auth/backend/roles/[id]/edit/page.js +24 -4
- package/dist/modules/auth/backend/roles/[id]/edit/page.js.map +2 -2
- package/dist/modules/auth/backend/roles/page.js +17 -4
- package/dist/modules/auth/backend/roles/page.js.map +2 -2
- package/dist/modules/auth/backend/users/[id]/edit/page.js +72 -7
- package/dist/modules/auth/backend/users/[id]/edit/page.js.map +2 -2
- package/dist/modules/auth/backend/users/page.js +15 -2
- package/dist/modules/auth/backend/users/page.js.map +2 -2
- package/dist/modules/auth/cli.js +19 -21
- package/dist/modules/auth/cli.js.map +2 -2
- package/dist/modules/auth/commands/roles.js +78 -1
- package/dist/modules/auth/commands/roles.js.map +2 -2
- package/dist/modules/auth/commands/users.js +106 -6
- package/dist/modules/auth/commands/users.js.map +3 -3
- package/dist/modules/auth/components/AclEditor.js +3 -1
- package/dist/modules/auth/components/AclEditor.js.map +2 -2
- package/dist/modules/auth/data/entities.js +6 -0
- package/dist/modules/auth/data/entities.js.map +2 -2
- package/dist/modules/auth/frontend/reset/[token]/page.js +6 -2
- package/dist/modules/auth/frontend/reset/[token]/page.js.map +2 -2
- package/dist/modules/auth/lib/consentIntegrity.js +15 -2
- package/dist/modules/auth/lib/consentIntegrity.js.map +2 -2
- package/dist/modules/auth/lib/emailHash.js +6 -2
- package/dist/modules/auth/lib/emailHash.js.map +2 -2
- package/dist/modules/auth/lib/grantChecks.js +44 -1
- package/dist/modules/auth/lib/grantChecks.js.map +2 -2
- package/dist/modules/auth/lib/sessionIntegrity.js +11 -1
- package/dist/modules/auth/lib/sessionIntegrity.js.map +2 -2
- package/dist/modules/auth/services/authService.js +10 -19
- package/dist/modules/auth/services/authService.js.map +2 -2
- package/dist/modules/auth/services/sidebarPreferencesService.js +32 -4
- package/dist/modules/auth/services/sidebarPreferencesService.js.map +2 -2
- package/dist/modules/business_rules/api/rules/route.js +28 -0
- package/dist/modules/business_rules/api/rules/route.js.map +2 -2
- package/dist/modules/business_rules/api/sets/route.js +28 -0
- package/dist/modules/business_rules/api/sets/route.js.map +2 -2
- package/dist/modules/business_rules/backend/rules/[id]/page.js +11 -4
- package/dist/modules/business_rules/backend/rules/[id]/page.js.map +3 -3
- package/dist/modules/business_rules/backend/rules/page.js +30 -12
- package/dist/modules/business_rules/backend/rules/page.js.map +2 -2
- package/dist/modules/business_rules/backend/sets/[id]/page.js +11 -4
- package/dist/modules/business_rules/backend/sets/[id]/page.js.map +2 -2
- package/dist/modules/business_rules/backend/sets/page.js +29 -11
- package/dist/modules/business_rules/backend/sets/page.js.map +2 -2
- package/dist/modules/catalog/api/categories/route.js +2 -0
- package/dist/modules/catalog/api/categories/route.js.map +2 -2
- package/dist/modules/catalog/api/products/route.js +2 -1
- package/dist/modules/catalog/api/products/route.js.map +2 -2
- package/dist/modules/catalog/backend/catalog/categories/[id]/edit/page.js +2 -0
- package/dist/modules/catalog/backend/catalog/categories/[id]/edit/page.js.map +2 -2
- package/dist/modules/catalog/backend/catalog/products/[id]/page.js +138 -75
- package/dist/modules/catalog/backend/catalog/products/[id]/page.js.map +2 -2
- package/dist/modules/catalog/backend/catalog/products/[productId]/variants/[variantId]/page.js +87 -24
- package/dist/modules/catalog/backend/catalog/products/[productId]/variants/[variantId]/page.js.map +2 -2
- package/dist/modules/catalog/backend/catalog/products/[productId]/variants/create/page.js +18 -3
- package/dist/modules/catalog/backend/catalog/products/[productId]/variants/create/page.js.map +2 -2
- package/dist/modules/catalog/backend/catalog/products/create/page.js +4 -17
- package/dist/modules/catalog/backend/catalog/products/create/page.js.map +2 -2
- package/dist/modules/catalog/backend/catalog/products/optionSchemaClient.js.map +2 -2
- package/dist/modules/catalog/commands/categories.js +42 -1
- package/dist/modules/catalog/commands/categories.js.map +2 -2
- package/dist/modules/catalog/commands/offers.js +44 -1
- package/dist/modules/catalog/commands/offers.js.map +2 -2
- package/dist/modules/catalog/commands/optionSchemas.js +6 -1
- package/dist/modules/catalog/commands/optionSchemas.js.map +2 -2
- package/dist/modules/catalog/commands/priceKinds.js +6 -1
- package/dist/modules/catalog/commands/priceKinds.js.map +2 -2
- package/dist/modules/catalog/commands/prices.js +50 -1
- package/dist/modules/catalog/commands/prices.js.map +2 -2
- package/dist/modules/catalog/commands/productUnitConversions.js +25 -1
- package/dist/modules/catalog/commands/productUnitConversions.js.map +2 -2
- package/dist/modules/catalog/commands/products.js +84 -0
- package/dist/modules/catalog/commands/products.js.map +2 -2
- package/dist/modules/catalog/commands/variants.js +117 -31
- package/dist/modules/catalog/commands/variants.js.map +2 -2
- package/dist/modules/catalog/components/PriceKindSettings.js +14 -6
- package/dist/modules/catalog/components/PriceKindSettings.js.map +2 -2
- package/dist/modules/catalog/components/categories/CategoriesDataTable.js +9 -0
- package/dist/modules/catalog/components/categories/CategoriesDataTable.js.map +2 -2
- package/dist/modules/catalog/components/products/ProductMediaManager.js.map +2 -2
- package/dist/modules/catalog/components/products/ProductsDataTable.js +14 -3
- package/dist/modules/catalog/components/products/ProductsDataTable.js.map +2 -2
- package/dist/modules/catalog/components/products/VariantBuilder.js +38 -11
- package/dist/modules/catalog/components/products/VariantBuilder.js.map +2 -2
- package/dist/modules/catalog/components/products/productForm.js +21 -0
- package/dist/modules/catalog/components/products/productForm.js.map +2 -2
- package/dist/modules/catalog/components/products/variantForm.js +2 -1
- package/dist/modules/catalog/components/products/variantForm.js.map +2 -2
- package/dist/modules/catalog/lib/pricing.js +3 -0
- package/dist/modules/catalog/lib/pricing.js.map +2 -2
- package/dist/modules/communication_channels/acl.js +47 -0
- package/dist/modules/communication_channels/acl.js.map +7 -0
- package/dist/modules/communication_channels/api/delete/channels/[id]/route.js +133 -0
- package/dist/modules/communication_channels/api/delete/channels/[id]/route.js.map +7 -0
- package/dist/modules/communication_channels/api/delete/messages/[messageId]/reactions/[reactionId]/route.js +113 -0
- package/dist/modules/communication_channels/api/delete/messages/[messageId]/reactions/[reactionId]/route.js.map +7 -0
- package/dist/modules/communication_channels/api/get/channels/[id]/health/route.js +138 -0
- package/dist/modules/communication_channels/api/get/channels/[id]/health/route.js.map +7 -0
- package/dist/modules/communication_channels/api/get/channels/[id]/route.js +93 -0
- package/dist/modules/communication_channels/api/get/channels/[id]/route.js.map +7 -0
- package/dist/modules/communication_channels/api/get/channels/route.js +96 -0
- package/dist/modules/communication_channels/api/get/channels/route.js.map +7 -0
- package/dist/modules/communication_channels/api/get/me/channels/route.js +82 -0
- package/dist/modules/communication_channels/api/get/me/channels/route.js.map +7 -0
- package/dist/modules/communication_channels/api/get/oauth/[provider]/callback/route.js +274 -0
- package/dist/modules/communication_channels/api/get/oauth/[provider]/callback/route.js.map +7 -0
- package/dist/modules/communication_channels/api/post/channels/[id]/import-history/route.js +168 -0
- package/dist/modules/communication_channels/api/post/channels/[id]/import-history/route.js.map +7 -0
- package/dist/modules/communication_channels/api/post/channels/[id]/poll-now/route.js +143 -0
- package/dist/modules/communication_channels/api/post/channels/[id]/poll-now/route.js.map +7 -0
- package/dist/modules/communication_channels/api/post/channels/[id]/push/register/route.js +127 -0
- package/dist/modules/communication_channels/api/post/channels/[id]/push/register/route.js.map +7 -0
- package/dist/modules/communication_channels/api/post/channels/[id]/set-primary/route.js +99 -0
- package/dist/modules/communication_channels/api/post/channels/[id]/set-primary/route.js.map +7 -0
- package/dist/modules/communication_channels/api/post/channels/[id]/test-send/route.js +197 -0
- package/dist/modules/communication_channels/api/post/channels/[id]/test-send/route.js.map +7 -0
- package/dist/modules/communication_channels/api/post/channels/connect/credentials/route.js +124 -0
- package/dist/modules/communication_channels/api/post/channels/connect/credentials/route.js.map +7 -0
- package/dist/modules/communication_channels/api/post/messages/[messageId]/reactions/route.js +120 -0
- package/dist/modules/communication_channels/api/post/messages/[messageId]/reactions/route.js.map +7 -0
- package/dist/modules/communication_channels/api/post/oauth/[provider]/initiate/route.js +157 -0
- package/dist/modules/communication_channels/api/post/oauth/[provider]/initiate/route.js.map +7 -0
- package/dist/modules/communication_channels/api/post/send-as-user/route.js +115 -0
- package/dist/modules/communication_channels/api/post/send-as-user/route.js.map +7 -0
- package/dist/modules/communication_channels/api/post/test-seed/route.js +238 -0
- package/dist/modules/communication_channels/api/post/test-seed/route.js.map +7 -0
- package/dist/modules/communication_channels/api/post/webhook/[provider]/route.js +175 -0
- package/dist/modules/communication_channels/api/post/webhook/[provider]/route.js.map +7 -0
- package/dist/modules/communication_channels/api/post/webhooks/gmail/route.js +123 -0
- package/dist/modules/communication_channels/api/post/webhooks/gmail/route.js.map +7 -0
- package/dist/modules/communication_channels/api/put/threads/[threadId]/assign/route.js +117 -0
- package/dist/modules/communication_channels/api/put/threads/[threadId]/assign/route.js.map +7 -0
- package/dist/modules/communication_channels/backend/communication_channels/channels/[id]/page.js +180 -0
- package/dist/modules/communication_channels/backend/communication_channels/channels/[id]/page.js.map +7 -0
- package/dist/modules/communication_channels/backend/communication_channels/channels/[id]/page.meta.js +36 -0
- package/dist/modules/communication_channels/backend/communication_channels/channels/[id]/page.meta.js.map +7 -0
- package/dist/modules/communication_channels/backend/communication_channels/channels/page.js +107 -0
- package/dist/modules/communication_channels/backend/communication_channels/channels/page.js.map +7 -0
- package/dist/modules/communication_channels/backend/communication_channels/channels/page.meta.js +38 -0
- package/dist/modules/communication_channels/backend/communication_channels/channels/page.meta.js.map +7 -0
- package/dist/modules/communication_channels/backend/profile/communication-channels/page.js +732 -0
- package/dist/modules/communication_channels/backend/profile/communication-channels/page.js.map +7 -0
- package/dist/modules/communication_channels/backend/profile/communication-channels/page.meta.js +38 -0
- package/dist/modules/communication_channels/backend/profile/communication-channels/page.meta.js.map +7 -0
- package/dist/modules/communication_channels/commands/connect-credential-channel.js +154 -0
- package/dist/modules/communication_channels/commands/connect-credential-channel.js.map +7 -0
- package/dist/modules/communication_channels/commands/delete-channel.js +137 -0
- package/dist/modules/communication_channels/commands/delete-channel.js.map +7 -0
- package/dist/modules/communication_channels/commands/deliver-outbound-message.js +400 -0
- package/dist/modules/communication_channels/commands/deliver-outbound-message.js.map +7 -0
- package/dist/modules/communication_channels/commands/disconnect-channel.js +163 -0
- package/dist/modules/communication_channels/commands/disconnect-channel.js.map +7 -0
- package/dist/modules/communication_channels/commands/ingest-inbound-message.js +413 -0
- package/dist/modules/communication_channels/commands/ingest-inbound-message.js.map +7 -0
- package/dist/modules/communication_channels/commands/interceptors.js +68 -0
- package/dist/modules/communication_channels/commands/interceptors.js.map +7 -0
- package/dist/modules/communication_channels/commands/process-inbound-reaction.js +198 -0
- package/dist/modules/communication_channels/commands/process-inbound-reaction.js.map +7 -0
- package/dist/modules/communication_channels/commands/push-register.js +146 -0
- package/dist/modules/communication_channels/commands/push-register.js.map +7 -0
- package/dist/modules/communication_channels/commands/push-renew.js +23 -0
- package/dist/modules/communication_channels/commands/push-renew.js.map +7 -0
- package/dist/modules/communication_channels/commands/push-unregister.js +108 -0
- package/dist/modules/communication_channels/commands/push-unregister.js.map +7 -0
- package/dist/modules/communication_channels/commands/queue-import-history.js +113 -0
- package/dist/modules/communication_channels/commands/queue-import-history.js.map +7 -0
- package/dist/modules/communication_channels/commands/reassign-conversation.js +193 -0
- package/dist/modules/communication_channels/commands/reassign-conversation.js.map +7 -0
- package/dist/modules/communication_channels/commands/set-primary-channel.js +115 -0
- package/dist/modules/communication_channels/commands/set-primary-channel.js.map +7 -0
- package/dist/modules/communication_channels/commands/toggle-outbound-reaction.js +260 -0
- package/dist/modules/communication_channels/commands/toggle-outbound-reaction.js.map +7 -0
- package/dist/modules/communication_channels/data/enrichers.js +286 -0
- package/dist/modules/communication_channels/data/enrichers.js.map +7 -0
- package/dist/modules/communication_channels/data/entities.js +447 -0
- package/dist/modules/communication_channels/data/entities.js.map +7 -0
- package/dist/modules/communication_channels/data/extensions.js +67 -0
- package/dist/modules/communication_channels/data/extensions.js.map +7 -0
- package/dist/modules/communication_channels/data/validators.js +123 -0
- package/dist/modules/communication_channels/data/validators.js.map +7 -0
- package/dist/modules/communication_channels/di.js +35 -0
- package/dist/modules/communication_channels/di.js.map +7 -0
- package/dist/modules/communication_channels/encryption.js +12 -0
- package/dist/modules/communication_channels/encryption.js.map +7 -0
- package/dist/modules/communication_channels/events.js +124 -0
- package/dist/modules/communication_channels/events.js.map +7 -0
- package/dist/modules/communication_channels/index.js +20 -0
- package/dist/modules/communication_channels/index.js.map +7 -0
- package/dist/modules/communication_channels/lib/access-control.js +43 -0
- package/dist/modules/communication_channels/lib/access-control.js.map +7 -0
- package/dist/modules/communication_channels/lib/adapter-compat.js +36 -0
- package/dist/modules/communication_channels/lib/adapter-compat.js.map +7 -0
- package/dist/modules/communication_channels/lib/adapter-registry-singleton.js +22 -0
- package/dist/modules/communication_channels/lib/adapter-registry-singleton.js.map +7 -0
- package/dist/modules/communication_channels/lib/adapter.js +1 -0
- package/dist/modules/communication_channels/lib/adapter.js.map +7 -0
- package/dist/modules/communication_channels/lib/connect-channel.js +95 -0
- package/dist/modules/communication_channels/lib/connect-channel.js.map +7 -0
- package/dist/modules/communication_channels/lib/contact-resolver.js +79 -0
- package/dist/modules/communication_channels/lib/contact-resolver.js.map +7 -0
- package/dist/modules/communication_channels/lib/credential-refresh.js +97 -0
- package/dist/modules/communication_channels/lib/credential-refresh.js.map +7 -0
- package/dist/modules/communication_channels/lib/dead-letter.js +62 -0
- package/dist/modules/communication_channels/lib/dead-letter.js.map +7 -0
- package/dist/modules/communication_channels/lib/email-capabilities.js +47 -0
- package/dist/modules/communication_channels/lib/email-capabilities.js.map +7 -0
- package/dist/modules/communication_channels/lib/email-contact.js +14 -0
- package/dist/modules/communication_channels/lib/email-contact.js.map +7 -0
- package/dist/modules/communication_channels/lib/email-mime.js +299 -0
- package/dist/modules/communication_channels/lib/email-mime.js.map +7 -0
- package/dist/modules/communication_channels/lib/error-classification.js +101 -0
- package/dist/modules/communication_channels/lib/error-classification.js.map +7 -0
- package/dist/modules/communication_channels/lib/gmail-pubsub-jwt.js +185 -0
- package/dist/modules/communication_channels/lib/gmail-pubsub-jwt.js.map +7 -0
- package/dist/modules/communication_channels/lib/mutation-guards.js +114 -0
- package/dist/modules/communication_channels/lib/mutation-guards.js.map +7 -0
- package/dist/modules/communication_channels/lib/oauth-client-config.js +32 -0
- package/dist/modules/communication_channels/lib/oauth-client-config.js.map +7 -0
- package/dist/modules/communication_channels/lib/oauth-state.js +128 -0
- package/dist/modules/communication_channels/lib/oauth-state.js.map +7 -0
- package/dist/modules/communication_channels/lib/oauth-token.js +45 -0
- package/dist/modules/communication_channels/lib/oauth-token.js.map +7 -0
- package/dist/modules/communication_channels/lib/pg-errors.js +11 -0
- package/dist/modules/communication_channels/lib/pg-errors.js.map +7 -0
- package/dist/modules/communication_channels/lib/provider-health.js +24 -0
- package/dist/modules/communication_channels/lib/provider-health.js.map +7 -0
- package/dist/modules/communication_channels/lib/push-state.js +19 -0
- package/dist/modules/communication_channels/lib/push-state.js.map +7 -0
- package/dist/modules/communication_channels/lib/queue.js +54 -0
- package/dist/modules/communication_channels/lib/queue.js.map +7 -0
- package/dist/modules/communication_channels/lib/reaction-processor-types.js +5 -0
- package/dist/modules/communication_channels/lib/reaction-processor-types.js.map +7 -0
- package/dist/modules/communication_channels/lib/reaction-semantics.js +11 -0
- package/dist/modules/communication_channels/lib/reaction-semantics.js.map +7 -0
- package/dist/modules/communication_channels/lib/registry.js +67 -0
- package/dist/modules/communication_channels/lib/registry.js.map +7 -0
- package/dist/modules/communication_channels/lib/route-mutation-guard.js +43 -0
- package/dist/modules/communication_channels/lib/route-mutation-guard.js.map +7 -0
- package/dist/modules/communication_channels/lib/sanitize-channel-html.js +96 -0
- package/dist/modules/communication_channels/lib/sanitize-channel-html.js.map +7 -0
- package/dist/modules/communication_channels/lib/send-as-user.js +194 -0
- package/dist/modules/communication_channels/lib/send-as-user.js.map +7 -0
- package/dist/modules/communication_channels/lib/system-user.js +22 -0
- package/dist/modules/communication_channels/lib/system-user.js.map +7 -0
- package/dist/modules/communication_channels/lib/test-seed.js +68 -0
- package/dist/modules/communication_channels/lib/test-seed.js.map +7 -0
- package/dist/modules/communication_channels/lib/thread-matcher.js +263 -0
- package/dist/modules/communication_channels/lib/thread-matcher.js.map +7 -0
- package/dist/modules/communication_channels/lib/thread-token.js +219 -0
- package/dist/modules/communication_channels/lib/thread-token.js.map +7 -0
- package/dist/modules/communication_channels/lib/use-connect-channel.js +61 -0
- package/dist/modules/communication_channels/lib/use-connect-channel.js.map +7 -0
- package/dist/modules/communication_channels/migrations/Migration20260526134719_communication_channels.js +50 -0
- package/dist/modules/communication_channels/migrations/Migration20260526134719_communication_channels.js.map +7 -0
- package/dist/modules/communication_channels/migrations/Migration20260527195446_communication_channels.js +19 -0
- package/dist/modules/communication_channels/migrations/Migration20260527195446_communication_channels.js.map +7 -0
- package/dist/modules/communication_channels/migrations/Migration20260529231848_communication_channels.js +13 -0
- package/dist/modules/communication_channels/migrations/Migration20260529231848_communication_channels.js.map +7 -0
- package/dist/modules/communication_channels/migrations/Migration20260531120000_communication_channels.js +17 -0
- package/dist/modules/communication_channels/migrations/Migration20260531120000_communication_channels.js.map +7 -0
- package/dist/modules/communication_channels/notifications.client.js +51 -0
- package/dist/modules/communication_channels/notifications.client.js.map +7 -0
- package/dist/modules/communication_channels/notifications.handlers.js +53 -0
- package/dist/modules/communication_channels/notifications.handlers.js.map +7 -0
- package/dist/modules/communication_channels/notifications.js +56 -0
- package/dist/modules/communication_channels/notifications.js.map +7 -0
- package/dist/modules/communication_channels/setup.js +105 -0
- package/dist/modules/communication_channels/setup.js.map +7 -0
- package/dist/modules/communication_channels/subscribers/channel-requires-reauth-notification.js +71 -0
- package/dist/modules/communication_channels/subscribers/channel-requires-reauth-notification.js.map +7 -0
- package/dist/modules/communication_channels/subscribers/outbound-bridge.js +103 -0
- package/dist/modules/communication_channels/subscribers/outbound-bridge.js.map +7 -0
- package/dist/modules/communication_channels/subscribers/user-deleted-cascade.js +51 -0
- package/dist/modules/communication_channels/subscribers/user-deleted-cascade.js.map +7 -0
- package/dist/modules/communication_channels/widgets/components.js +7 -0
- package/dist/modules/communication_channels/widgets/components.js.map +7 -0
- package/dist/modules/communication_channels/widgets/injection/channel-badge/widget.client.js +18 -0
- package/dist/modules/communication_channels/widgets/injection/channel-badge/widget.client.js.map +7 -0
- package/dist/modules/communication_channels/widgets/injection/channel-badge/widget.js +30 -0
- package/dist/modules/communication_channels/widgets/injection/channel-badge/widget.js.map +7 -0
- package/dist/modules/communication_channels/widgets/injection/channel-info-panel/widget.client.js +185 -0
- package/dist/modules/communication_channels/widgets/injection/channel-info-panel/widget.client.js.map +7 -0
- package/dist/modules/communication_channels/widgets/injection/channel-info-panel/widget.js +17 -0
- package/dist/modules/communication_channels/widgets/injection/channel-info-panel/widget.js.map +7 -0
- package/dist/modules/communication_channels/widgets/injection/channel-payload-renderer/widget.client.js +44 -0
- package/dist/modules/communication_channels/widgets/injection/channel-payload-renderer/widget.client.js.map +7 -0
- package/dist/modules/communication_channels/widgets/injection/channel-payload-renderer/widget.js +17 -0
- package/dist/modules/communication_channels/widgets/injection/channel-payload-renderer/widget.js.map +7 -0
- package/dist/modules/communication_channels/widgets/injection/profile-channels-menu/widget.js +23 -0
- package/dist/modules/communication_channels/widgets/injection/profile-channels-menu/widget.js.map +7 -0
- package/dist/modules/communication_channels/widgets/injection/reaction-bar/widget.client.js +141 -0
- package/dist/modules/communication_channels/widgets/injection/reaction-bar/widget.client.js.map +7 -0
- package/dist/modules/communication_channels/widgets/injection/reaction-bar/widget.js +17 -0
- package/dist/modules/communication_channels/widgets/injection/reaction-bar/widget.js.map +7 -0
- package/dist/modules/communication_channels/widgets/injection-table.js +38 -0
- package/dist/modules/communication_channels/widgets/injection-table.js.map +7 -0
- package/dist/modules/communication_channels/widgets/notifications/ChannelRequiresReauthRenderer.js +25 -0
- package/dist/modules/communication_channels/widgets/notifications/ChannelRequiresReauthRenderer.js.map +7 -0
- package/dist/modules/communication_channels/widgets/notifications/MessageReceivedRenderer.js +19 -0
- package/dist/modules/communication_channels/widgets/notifications/MessageReceivedRenderer.js.map +7 -0
- package/dist/modules/communication_channels/widgets/notifications/index.js +7 -0
- package/dist/modules/communication_channels/widgets/notifications/index.js.map +7 -0
- package/dist/modules/communication_channels/workers/channel-import-history.js +185 -0
- package/dist/modules/communication_channels/workers/channel-import-history.js.map +7 -0
- package/dist/modules/communication_channels/workers/gmail-history-sync.js +154 -0
- package/dist/modules/communication_channels/workers/gmail-history-sync.js.map +7 -0
- package/dist/modules/communication_channels/workers/gmail-renew-watch.js +95 -0
- package/dist/modules/communication_channels/workers/gmail-renew-watch.js.map +7 -0
- package/dist/modules/communication_channels/workers/inbound-processor.js +56 -0
- package/dist/modules/communication_channels/workers/inbound-processor.js.map +7 -0
- package/dist/modules/communication_channels/workers/outbound-delivery.js +85 -0
- package/dist/modules/communication_channels/workers/outbound-delivery.js.map +7 -0
- package/dist/modules/communication_channels/workers/poll-channel.js +240 -0
- package/dist/modules/communication_channels/workers/poll-channel.js.map +7 -0
- package/dist/modules/communication_channels/workers/poll-tick.js +132 -0
- package/dist/modules/communication_channels/workers/poll-tick.js.map +7 -0
- package/dist/modules/communication_channels/workers/reaction-processor.js +192 -0
- package/dist/modules/communication_channels/workers/reaction-processor.js.map +7 -0
- package/dist/modules/configs/api/openapi.js +11 -2
- package/dist/modules/configs/api/openapi.js.map +2 -2
- package/dist/modules/currencies/backend/currencies/[id]/page.js +6 -3
- package/dist/modules/currencies/backend/currencies/[id]/page.js.map +2 -2
- package/dist/modules/currencies/backend/currencies/page.js +27 -11
- package/dist/modules/currencies/backend/currencies/page.js.map +2 -2
- package/dist/modules/currencies/backend/exchange-rates/[id]/page.js +1 -0
- package/dist/modules/currencies/backend/exchange-rates/[id]/page.js.map +2 -2
- package/dist/modules/currencies/backend/exchange-rates/page.js +21 -7
- package/dist/modules/currencies/backend/exchange-rates/page.js.map +2 -2
- package/dist/modules/currencies/commands/currencies.js +37 -16
- package/dist/modules/currencies/commands/currencies.js.map +2 -2
- package/dist/modules/currencies/commands/exchange-rates.js +19 -9
- package/dist/modules/currencies/commands/exchange-rates.js.map +2 -2
- package/dist/modules/currencies/components/CurrencyFetchingConfig.js +26 -19
- package/dist/modules/currencies/components/CurrencyFetchingConfig.js.map +2 -2
- package/dist/modules/customer_accounts/api/admin/roles/[id].js +28 -5
- package/dist/modules/customer_accounts/api/admin/roles/[id].js.map +2 -2
- package/dist/modules/customer_accounts/api/admin/roles.js +4 -2
- package/dist/modules/customer_accounts/api/admin/roles.js.map +2 -2
- package/dist/modules/customer_accounts/api/admin/users/[id].js +39 -5
- package/dist/modules/customer_accounts/api/admin/users/[id].js.map +2 -2
- package/dist/modules/customer_accounts/api/admin/users-invite.js +17 -1
- package/dist/modules/customer_accounts/api/admin/users-invite.js.map +2 -2
- package/dist/modules/customer_accounts/api/admin/users.js +60 -23
- package/dist/modules/customer_accounts/api/admin/users.js.map +2 -2
- package/dist/modules/customer_accounts/api/login.js +11 -4
- package/dist/modules/customer_accounts/api/login.js.map +2 -2
- package/dist/modules/customer_accounts/api/portal/users-invite.js +17 -1
- package/dist/modules/customer_accounts/api/portal/users-invite.js.map +2 -2
- package/dist/modules/customer_accounts/api/signup.js +3 -1
- package/dist/modules/customer_accounts/api/signup.js.map +2 -2
- package/dist/modules/customer_accounts/backend/customer_accounts/roles/[id]/page.js +16 -8
- package/dist/modules/customer_accounts/backend/customer_accounts/roles/[id]/page.js.map +2 -2
- package/dist/modules/customer_accounts/backend/customer_accounts/roles/page.js +19 -5
- package/dist/modules/customer_accounts/backend/customer_accounts/roles/page.js.map +2 -2
- package/dist/modules/customer_accounts/backend/customer_accounts/settings/domain/page.js +8 -4
- package/dist/modules/customer_accounts/backend/customer_accounts/settings/domain/page.js.map +2 -2
- package/dist/modules/customer_accounts/backend/customer_accounts/users/[id]/page.js +29 -18
- package/dist/modules/customer_accounts/backend/customer_accounts/users/[id]/page.js.map +2 -2
- package/dist/modules/customer_accounts/backend/customer_accounts/users/page.js +27 -11
- package/dist/modules/customer_accounts/backend/customer_accounts/users/page.js.map +2 -2
- package/dist/modules/customer_accounts/data/validators.js +2 -1
- package/dist/modules/customer_accounts/data/validators.js.map +2 -2
- package/dist/modules/customer_accounts/lib/customerEntityOwnership.js +22 -0
- package/dist/modules/customer_accounts/lib/customerEntityOwnership.js.map +7 -0
- package/dist/modules/customer_accounts/lib/rateLimiter.js +14 -0
- package/dist/modules/customer_accounts/lib/rateLimiter.js.map +2 -2
- package/dist/modules/customer_accounts/lib/resolveTenantContext.js +44 -1
- package/dist/modules/customer_accounts/lib/resolveTenantContext.js.map +3 -3
- package/dist/modules/customer_accounts/services/customerInvitationService.js +28 -1
- package/dist/modules/customer_accounts/services/customerInvitationService.js.map +2 -2
- package/dist/modules/customer_accounts/services/customerUserService.js +13 -7
- package/dist/modules/customer_accounts/services/customerUserService.js.map +2 -2
- package/dist/modules/customer_accounts/subscribers/autoLinkCrmReverse.js +3 -3
- package/dist/modules/customer_accounts/subscribers/autoLinkCrmReverse.js.map +2 -2
- package/dist/modules/customers/acl.js +18 -0
- package/dist/modules/customers/acl.js.map +2 -2
- package/dist/modules/customers/api/activities/route.js +32 -10
- package/dist/modules/customers/api/activities/route.js.map +2 -2
- package/dist/modules/customers/api/addresses/route.js +2 -0
- package/dist/modules/customers/api/addresses/route.js.map +2 -2
- package/dist/modules/customers/api/companies/[id]/route.js +22 -9
- package/dist/modules/customers/api/companies/[id]/route.js.map +2 -2
- package/dist/modules/customers/api/companies/route.js +18 -7
- package/dist/modules/customers/api/companies/route.js.map +2 -2
- package/dist/modules/customers/api/deals/route.js +2 -0
- package/dist/modules/customers/api/deals/route.js.map +2 -2
- package/dist/modules/customers/api/dictionaries/[kind]/route.js +22 -38
- package/dist/modules/customers/api/dictionaries/[kind]/route.js.map +2 -2
- package/dist/modules/customers/api/dictionaries/cache.js +2 -1
- package/dist/modules/customers/api/dictionaries/cache.js.map +2 -2
- package/dist/modules/customers/api/dictionaries/context.js +1 -0
- package/dist/modules/customers/api/dictionaries/context.js.map +2 -2
- package/dist/modules/customers/api/interactions/[id]/visibility/route.js +151 -0
- package/dist/modules/customers/api/interactions/[id]/visibility/route.js.map +7 -0
- package/dist/modules/customers/api/interactions/counts/route.js +6 -0
- package/dist/modules/customers/api/interactions/counts/route.js.map +2 -2
- package/dist/modules/customers/api/interactions/route.js +33 -12
- package/dist/modules/customers/api/interactions/route.js.map +2 -2
- package/dist/modules/customers/api/people/[id]/email-threads/route.js +82 -0
- package/dist/modules/customers/api/people/[id]/email-threads/route.js.map +7 -0
- package/dist/modules/customers/api/people/[id]/emails/route.js +157 -0
- package/dist/modules/customers/api/people/[id]/emails/route.js.map +7 -0
- package/dist/modules/customers/api/people/[id]/route.js +20 -10
- package/dist/modules/customers/api/people/[id]/route.js.map +2 -2
- package/dist/modules/customers/api/people/route.js +16 -7
- package/dist/modules/customers/api/people/route.js.map +2 -2
- package/dist/modules/customers/api/pipeline-stages/route.js +0 -13
- package/dist/modules/customers/api/pipeline-stages/route.js.map +2 -2
- package/dist/modules/customers/api/settings/dictionary-sort-modes/route.js +223 -0
- package/dist/modules/customers/api/settings/dictionary-sort-modes/route.js.map +7 -0
- package/dist/modules/customers/api/todos/route.js +29 -11
- package/dist/modules/customers/api/todos/route.js.map +2 -2
- package/dist/modules/customers/backend/config/customers/deals/page.js.map +2 -2
- package/dist/modules/customers/backend/config/customers/pipeline-stages/page.js +34 -21
- package/dist/modules/customers/backend/config/customers/pipeline-stages/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/companies/[id]/page.js +45 -27
- package/dist/modules/customers/backend/customers/companies/[id]/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/companies/page.js +10 -1
- package/dist/modules/customers/backend/customers/companies/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/companies-v2/[id]/page.js +22 -5
- package/dist/modules/customers/backend/customers/companies-v2/[id]/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealFormHandlers.js +30 -8
- package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealFormHandlers.js.map +2 -2
- package/dist/modules/customers/backend/customers/deals/[id]/page.js +1 -0
- package/dist/modules/customers/backend/customers/deals/[id]/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/deals/page.js +26 -7
- package/dist/modules/customers/backend/customers/deals/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/deals/pipeline/page.js +62 -39
- package/dist/modules/customers/backend/customers/deals/pipeline/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/people/[id]/page.js +41 -26
- package/dist/modules/customers/backend/customers/people/[id]/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/people/page.js +10 -1
- package/dist/modules/customers/backend/customers/people/page.js.map +2 -2
- package/dist/modules/customers/backend/customers/people-v2/[id]/page.js +60 -23
- package/dist/modules/customers/backend/customers/people-v2/[id]/page.js.map +2 -2
- package/dist/modules/customers/commands/activities.js +19 -0
- package/dist/modules/customers/commands/activities.js.map +2 -2
- package/dist/modules/customers/commands/addresses.js +98 -14
- package/dist/modules/customers/commands/addresses.js.map +2 -2
- package/dist/modules/customers/commands/comments.js +30 -1
- package/dist/modules/customers/commands/comments.js.map +2 -2
- package/dist/modules/customers/commands/companies.js +113 -1
- package/dist/modules/customers/commands/companies.js.map +2 -2
- package/dist/modules/customers/commands/deals.js +196 -77
- package/dist/modules/customers/commands/deals.js.map +2 -2
- package/dist/modules/customers/commands/dictionaries.js +44 -0
- package/dist/modules/customers/commands/dictionaries.js.map +2 -2
- package/dist/modules/customers/commands/entity-roles.js +47 -0
- package/dist/modules/customers/commands/entity-roles.js.map +2 -2
- package/dist/modules/customers/commands/interactions.js +200 -8
- package/dist/modules/customers/commands/interactions.js.map +2 -2
- package/dist/modules/customers/commands/labels.js +83 -1
- package/dist/modules/customers/commands/labels.js.map +2 -2
- package/dist/modules/customers/commands/people.js +167 -13
- package/dist/modules/customers/commands/people.js.map +2 -2
- package/dist/modules/customers/commands/personCompanyLinks.js +141 -69
- package/dist/modules/customers/commands/personCompanyLinks.js.map +2 -2
- package/dist/modules/customers/commands/pipeline-stages.js +13 -11
- package/dist/modules/customers/commands/pipeline-stages.js.map +3 -3
- package/dist/modules/customers/commands/settings.js +31 -0
- package/dist/modules/customers/commands/settings.js.map +2 -2
- package/dist/modules/customers/commands/tags.js +56 -1
- package/dist/modules/customers/commands/tags.js.map +2 -2
- package/dist/modules/customers/commands/todos.js +19 -0
- package/dist/modules/customers/commands/todos.js.map +2 -2
- package/dist/modules/customers/components/AddressFormatSettings.js.map +2 -2
- package/dist/modules/customers/components/CustomersConfigurationSections.js +3 -0
- package/dist/modules/customers/components/CustomersConfigurationSections.js.map +2 -2
- package/dist/modules/customers/components/DictionarySettings.js +20 -13
- package/dist/modules/customers/components/DictionarySettings.js.map +2 -2
- package/dist/modules/customers/components/DictionarySortSettings.js +214 -0
- package/dist/modules/customers/components/DictionarySortSettings.js.map +7 -0
- package/dist/modules/customers/components/PipelineSettings.js +38 -23
- package/dist/modules/customers/components/PipelineSettings.js.map +2 -2
- package/dist/modules/customers/components/detail/ActivityCard.js +32 -0
- package/dist/modules/customers/components/detail/ActivityCard.js.map +2 -2
- package/dist/modules/customers/components/detail/ActivityForm.js +1 -0
- package/dist/modules/customers/components/detail/ActivityForm.js.map +2 -2
- package/dist/modules/customers/components/detail/ActivityTimeline.js +1 -1
- package/dist/modules/customers/components/detail/ActivityTimeline.js.map +2 -2
- package/dist/modules/customers/components/detail/AddressesSection.js +4 -0
- package/dist/modules/customers/components/detail/AddressesSection.js.map +2 -2
- package/dist/modules/customers/components/detail/AnnualRevenueField.js +1 -0
- package/dist/modules/customers/components/detail/AnnualRevenueField.js.map +2 -2
- package/dist/modules/customers/components/detail/CompanyPeopleSection.js +28 -22
- package/dist/modules/customers/components/detail/CompanyPeopleSection.js.map +2 -2
- package/dist/modules/customers/components/detail/ComposeEmailDialog.js +242 -0
- package/dist/modules/customers/components/detail/ComposeEmailDialog.js.map +7 -0
- package/dist/modules/customers/components/detail/DealForm.js +3 -1
- package/dist/modules/customers/components/detail/DealForm.js.map +2 -2
- package/dist/modules/customers/components/detail/DealsSection.js +46 -24
- package/dist/modules/customers/components/detail/DealsSection.js.map +2 -2
- package/dist/modules/customers/components/detail/EmailCardActions.js +184 -0
- package/dist/modules/customers/components/detail/EmailCardActions.js.map +7 -0
- package/dist/modules/customers/components/detail/EmailReplyForwardActions.js +52 -0
- package/dist/modules/customers/components/detail/EmailReplyForwardActions.js.map +7 -0
- package/dist/modules/customers/components/detail/EntityTagsDialog.js +7 -0
- package/dist/modules/customers/components/detail/EntityTagsDialog.js.map +2 -2
- package/dist/modules/customers/components/detail/ManageTagsDialog.js +34 -22
- package/dist/modules/customers/components/detail/ManageTagsDialog.js.map +2 -2
- package/dist/modules/customers/components/detail/PersonCompaniesSection.js +41 -29
- package/dist/modules/customers/components/detail/PersonCompaniesSection.js.map +2 -2
- package/dist/modules/customers/components/detail/PersonDetailTabs.js +7 -1
- package/dist/modules/customers/components/detail/PersonDetailTabs.js.map +2 -2
- package/dist/modules/customers/components/detail/PersonEmailThreadsTab.js +366 -0
- package/dist/modules/customers/components/detail/PersonEmailThreadsTab.js.map +7 -0
- package/dist/modules/customers/components/detail/RoleAssignmentRow.js +14 -8
- package/dist/modules/customers/components/detail/RoleAssignmentRow.js.map +2 -2
- package/dist/modules/customers/components/detail/ScheduleActivityDialog.js +14 -6
- package/dist/modules/customers/components/detail/ScheduleActivityDialog.js.map +2 -2
- package/dist/modules/customers/components/detail/create/DealCurrencyField.js +1 -0
- package/dist/modules/customers/components/detail/create/DealCurrencyField.js.map +2 -2
- package/dist/modules/customers/components/detail/hooks/useCustomerDictionary.js +2 -1
- package/dist/modules/customers/components/detail/hooks/useCustomerDictionary.js.map +2 -2
- package/dist/modules/customers/components/detail/hooks/useInteractionMutations.js +29 -13
- package/dist/modules/customers/components/detail/hooks/useInteractionMutations.js.map +2 -2
- package/dist/modules/customers/components/detail/hooks/useInteractions.js +77 -35
- package/dist/modules/customers/components/detail/hooks/useInteractions.js.map +2 -2
- package/dist/modules/customers/components/detail/hooks/usePersonTasks.js +25 -17
- package/dist/modules/customers/components/detail/hooks/usePersonTasks.js.map +2 -2
- package/dist/modules/customers/components/detail/schedule/useScheduleFormState.js.map +2 -2
- package/dist/modules/customers/components/formConfig.js +62 -12
- package/dist/modules/customers/components/formConfig.js.map +2 -2
- package/dist/modules/customers/data/enrichers.js +133 -2
- package/dist/modules/customers/data/enrichers.js.map +2 -2
- package/dist/modules/customers/data/entities.js +21 -0
- package/dist/modules/customers/data/entities.js.map +2 -2
- package/dist/modules/customers/data/extensions.js +16 -0
- package/dist/modules/customers/data/extensions.js.map +7 -0
- package/dist/modules/customers/data/guards.js +66 -0
- package/dist/modules/customers/data/guards.js.map +7 -0
- package/dist/modules/customers/data/validators.js +33 -8
- package/dist/modules/customers/data/validators.js.map +2 -2
- package/dist/modules/customers/di.js +37 -0
- package/dist/modules/customers/di.js.map +2 -2
- package/dist/modules/customers/encryption.js +11 -0
- package/dist/modules/customers/encryption.js.map +2 -2
- package/dist/modules/customers/events.js +4 -1
- package/dist/modules/customers/events.js.map +2 -2
- package/dist/modules/customers/lib/detailTenantScope.js +15 -0
- package/dist/modules/customers/lib/detailTenantScope.js.map +7 -0
- package/dist/modules/customers/lib/findPeopleByAddresses.js +64 -0
- package/dist/modules/customers/lib/findPeopleByAddresses.js.map +7 -0
- package/dist/modules/customers/lib/kysely.js.map +2 -2
- package/dist/modules/customers/lib/link-channel-message-handler.js +303 -0
- package/dist/modules/customers/lib/link-channel-message-handler.js.map +7 -0
- package/dist/modules/customers/lib/personCompanyLinkTable.js +15 -1
- package/dist/modules/customers/lib/personCompanyLinkTable.js.map +2 -2
- package/dist/modules/customers/lib/personEmailThreads.js +205 -0
- package/dist/modules/customers/lib/personEmailThreads.js.map +7 -0
- package/dist/modules/customers/lib/todoCompatibility.js +11 -0
- package/dist/modules/customers/lib/todoCompatibility.js.map +2 -2
- package/dist/modules/customers/lib/visibilityFilter.js +51 -0
- package/dist/modules/customers/lib/visibilityFilter.js.map +7 -0
- package/dist/modules/customers/migrations/Migration20260527012240_customers.js +20 -0
- package/dist/modules/customers/migrations/Migration20260527012240_customers.js.map +7 -0
- package/dist/modules/customers/migrations/Migration20260602202147_customers.js +13 -0
- package/dist/modules/customers/migrations/Migration20260602202147_customers.js.map +7 -0
- package/dist/modules/customers/search.js +12 -6
- package/dist/modules/customers/search.js.map +2 -2
- package/dist/modules/customers/setup.js +2 -1
- package/dist/modules/customers/setup.js.map +2 -2
- package/dist/modules/customers/subscribers/link-channel-message-received.js +12 -0
- package/dist/modules/customers/subscribers/link-channel-message-received.js.map +7 -0
- package/dist/modules/customers/subscribers/link-channel-message-sent.js +12 -0
- package/dist/modules/customers/subscribers/link-channel-message-sent.js.map +7 -0
- package/dist/modules/dashboards/components/WidgetVisibilityEditor.js.map +2 -2
- package/dist/modules/data_sync/api/options.js +4 -4
- package/dist/modules/data_sync/api/options.js.map +2 -2
- package/dist/modules/data_sync/api/schedules/route.js +9 -1
- package/dist/modules/data_sync/api/schedules/route.js.map +2 -2
- package/dist/modules/data_sync/backend/data-sync/page.js +17 -8
- package/dist/modules/data_sync/backend/data-sync/page.js.map +2 -2
- package/dist/modules/data_sync/components/IntegrationScheduleTab.js +43 -22
- package/dist/modules/data_sync/components/IntegrationScheduleTab.js.map +2 -2
- package/dist/modules/data_sync/lib/sync-schedule-service.js +9 -0
- package/dist/modules/data_sync/lib/sync-schedule-service.js.map +2 -2
- package/dist/modules/dictionaries/api/[dictionaryId]/entries/[entryId]/route.js +8 -1
- package/dist/modules/dictionaries/api/[dictionaryId]/entries/[entryId]/route.js.map +2 -2
- package/dist/modules/dictionaries/api/[dictionaryId]/entries/route.js +12 -5
- package/dist/modules/dictionaries/api/[dictionaryId]/entries/route.js.map +2 -2
- package/dist/modules/dictionaries/api/[dictionaryId]/route.js +25 -1
- package/dist/modules/dictionaries/api/[dictionaryId]/route.js.map +2 -2
- package/dist/modules/dictionaries/api/openapi.js +2 -0
- package/dist/modules/dictionaries/api/openapi.js.map +2 -2
- package/dist/modules/dictionaries/api/route.js +7 -0
- package/dist/modules/dictionaries/api/route.js.map +2 -2
- package/dist/modules/dictionaries/commands/entry-operations.js +8 -0
- package/dist/modules/dictionaries/commands/entry-operations.js.map +2 -2
- package/dist/modules/dictionaries/components/DictionariesManager.js +95 -16
- package/dist/modules/dictionaries/components/DictionariesManager.js.map +2 -2
- package/dist/modules/dictionaries/components/DictionaryEntriesEditor.js +29 -16
- package/dist/modules/dictionaries/components/DictionaryEntriesEditor.js.map +2 -2
- package/dist/modules/dictionaries/components/DictionaryEntrySelect.js +51 -11
- package/dist/modules/dictionaries/components/DictionaryEntrySelect.js.map +2 -2
- package/dist/modules/dictionaries/components/DictionarySelectControl.js +9 -1
- package/dist/modules/dictionaries/components/DictionarySelectControl.js.map +2 -2
- package/dist/modules/dictionaries/components/dictionaryAppearance.js +2 -1
- package/dist/modules/dictionaries/components/dictionaryAppearance.js.map +2 -2
- package/dist/modules/dictionaries/components/hooks/useDictionaryEntries.js +2 -1
- package/dist/modules/dictionaries/components/hooks/useDictionaryEntries.js.map +2 -2
- package/dist/modules/dictionaries/data/entities.js +5 -0
- package/dist/modules/dictionaries/data/entities.js.map +2 -2
- package/dist/modules/dictionaries/data/validators.js +3 -1
- package/dist/modules/dictionaries/data/validators.js.map +2 -2
- package/dist/modules/dictionaries/fields/dictionary.js +2 -2
- package/dist/modules/dictionaries/fields/dictionary.js.map +1 -1
- package/dist/modules/dictionaries/lib/entrySort.js +61 -0
- package/dist/modules/dictionaries/lib/entrySort.js.map +7 -0
- package/dist/modules/dictionaries/migrations/Migration20260602202147_dictionaries.js +13 -0
- package/dist/modules/dictionaries/migrations/Migration20260602202147_dictionaries.js.map +7 -0
- package/dist/modules/directory/api/get/organizations/lookup.js +3 -7
- package/dist/modules/directory/api/get/organizations/lookup.js.map +2 -2
- package/dist/modules/directory/api/organizations/route.js +41 -16
- package/dist/modules/directory/api/organizations/route.js.map +2 -2
- package/dist/modules/directory/backend/directory/organizations/[id]/edit/page.js +2 -0
- package/dist/modules/directory/backend/directory/organizations/[id]/edit/page.js.map +2 -2
- package/dist/modules/directory/backend/directory/organizations/page.js +18 -5
- package/dist/modules/directory/backend/directory/organizations/page.js.map +2 -2
- package/dist/modules/directory/backend/directory/tenants/[id]/edit/page.js +7 -3
- package/dist/modules/directory/backend/directory/tenants/[id]/edit/page.js.map +2 -2
- package/dist/modules/directory/backend/directory/tenants/page.js +17 -4
- package/dist/modules/directory/backend/directory/tenants/page.js.map +2 -2
- package/dist/modules/directory/commands/organizations.js +70 -2
- package/dist/modules/directory/commands/organizations.js.map +2 -2
- package/dist/modules/directory/commands/tenants.js +16 -3
- package/dist/modules/directory/commands/tenants.js.map +2 -2
- package/dist/modules/entities/api/records.js +107 -7
- package/dist/modules/entities/api/records.js.map +2 -2
- package/dist/modules/entities/backend/entities/user/[entityId]/records/[recordId]/page.js +1 -0
- package/dist/modules/entities/backend/entities/user/[entityId]/records/[recordId]/page.js.map +2 -2
- package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js +8 -4
- package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js.map +2 -2
- package/dist/modules/entities/components/UserEntitiesTable.js +9 -0
- package/dist/modules/entities/components/UserEntitiesTable.js.map +2 -2
- package/dist/modules/entities/lib/entityAcl.js +82 -0
- package/dist/modules/entities/lib/entityAcl.js.map +7 -0
- package/dist/modules/entities/lib/helpers.js +79 -65
- package/dist/modules/entities/lib/helpers.js.map +2 -2
- package/dist/modules/feature_toggles/api/global/[id]/override/route.js +15 -10
- package/dist/modules/feature_toggles/api/global/[id]/override/route.js.map +2 -2
- package/dist/modules/feature_toggles/api/global/[id]/route.js +2 -1
- package/dist/modules/feature_toggles/api/global/[id]/route.js.map +2 -2
- package/dist/modules/feature_toggles/api/overrides/route.js +20 -7
- package/dist/modules/feature_toggles/api/overrides/route.js.map +2 -2
- package/dist/modules/feature_toggles/backend/feature-toggles/global/[id]/edit/page.js +21 -16
- package/dist/modules/feature_toggles/backend/feature-toggles/global/[id]/edit/page.js.map +2 -2
- package/dist/modules/feature_toggles/commands/global.js +127 -8
- package/dist/modules/feature_toggles/commands/global.js.map +2 -2
- package/dist/modules/feature_toggles/commands/overrides.js +8 -6
- package/dist/modules/feature_toggles/commands/overrides.js.map +2 -2
- package/dist/modules/feature_toggles/components/FeatureToggleOverrideCard.js +20 -12
- package/dist/modules/feature_toggles/components/FeatureToggleOverrideCard.js.map +2 -2
- package/dist/modules/feature_toggles/components/FeatureTogglesTable.js +15 -2
- package/dist/modules/feature_toggles/components/FeatureTogglesTable.js.map +2 -2
- package/dist/modules/feature_toggles/components/formConfig.js +2 -1
- package/dist/modules/feature_toggles/components/formConfig.js.map +2 -2
- package/dist/modules/feature_toggles/components/overrideFormConfig.js +5 -1
- package/dist/modules/feature_toggles/components/overrideFormConfig.js.map +2 -2
- package/dist/modules/feature_toggles/data/validators.js +7 -4
- package/dist/modules/feature_toggles/data/validators.js.map +2 -2
- package/dist/modules/feature_toggles/lib/queries.js +5 -5
- package/dist/modules/feature_toggles/lib/queries.js.map +2 -2
- package/dist/modules/feature_toggles/lib/utils.js.map +2 -2
- package/dist/modules/inbox_ops/api/proposals/[id]/replies/[replyId]/send/route.js +8 -1
- package/dist/modules/inbox_ops/api/proposals/[id]/replies/[replyId]/send/route.js.map +2 -2
- package/dist/modules/inbox_ops/api/settings/route.js +23 -2
- package/dist/modules/inbox_ops/api/settings/route.js.map +2 -2
- package/dist/modules/inbox_ops/api/webhook/inbound.js +53 -14
- package/dist/modules/inbox_ops/api/webhook/inbound.js.map +2 -2
- package/dist/modules/inbox_ops/backend/inbox-ops/settings/page.js +13 -8
- package/dist/modules/inbox_ops/backend/inbox-ops/settings/page.js.map +2 -2
- package/dist/modules/inbox_ops/components/proposals/EditActionDialog.js +9 -4
- package/dist/modules/inbox_ops/components/proposals/EditActionDialog.js.map +2 -2
- package/dist/modules/inbox_ops/data/entities.js +3 -0
- package/dist/modules/inbox_ops/data/entities.js.map +2 -2
- package/dist/modules/inbox_ops/data/validators.js +5 -1
- package/dist/modules/inbox_ops/data/validators.js.map +2 -2
- package/dist/modules/inbox_ops/encryption.js +9 -0
- package/dist/modules/inbox_ops/encryption.js.map +2 -2
- package/dist/modules/inbox_ops/lib/executionEngine.js +6 -1
- package/dist/modules/inbox_ops/lib/executionEngine.js.map +2 -2
- package/dist/modules/inbox_ops/lib/executionHelpers.js +2 -7
- package/dist/modules/inbox_ops/lib/executionHelpers.js.map +2 -2
- package/dist/modules/inbox_ops/lib/priceValidator.js +35 -7
- package/dist/modules/inbox_ops/lib/priceValidator.js.map +2 -2
- package/dist/modules/inbox_ops/lib/rateLimiter.js +58 -23
- package/dist/modules/inbox_ops/lib/rateLimiter.js.map +2 -2
- package/dist/modules/inbox_ops/lib/translationProvider.js +33 -5
- package/dist/modules/inbox_ops/lib/translationProvider.js.map +2 -2
- package/dist/modules/inbox_ops/migrations/Migration20260607205834.js +13 -0
- package/dist/modules/inbox_ops/migrations/Migration20260607205834.js.map +7 -0
- package/dist/modules/inbox_ops/subscribers/extractionWorker.js +2 -1
- package/dist/modules/inbox_ops/subscribers/extractionWorker.js.map +2 -2
- package/dist/modules/integrations/api/[id]/credentials/route.js +11 -0
- package/dist/modules/integrations/api/[id]/credentials/route.js.map +2 -2
- package/dist/modules/integrations/backend/integrations/[id]/page.js +8 -0
- package/dist/modules/integrations/backend/integrations/[id]/page.js.map +2 -2
- package/dist/modules/integrations/backend/integrations/bundle/[id]/page.js +18 -11
- package/dist/modules/integrations/backend/integrations/bundle/[id]/page.js.map +2 -2
- package/dist/modules/integrations/backend/integrations/page.js +12 -8
- package/dist/modules/integrations/backend/integrations/page.js.map +2 -2
- package/dist/modules/integrations/data/entities.js +8 -1
- package/dist/modules/integrations/data/entities.js.map +2 -2
- package/dist/modules/integrations/lib/credentials-field-validation.js +32 -0
- package/dist/modules/integrations/lib/credentials-field-validation.js.map +7 -0
- package/dist/modules/integrations/lib/credentials-service.js +29 -14
- package/dist/modules/integrations/lib/credentials-service.js.map +2 -2
- package/dist/modules/integrations/migrations/Migration20260526154136_integrations.js +15 -0
- package/dist/modules/integrations/migrations/Migration20260526154136_integrations.js.map +7 -0
- package/dist/modules/messages/api/route.js +2 -2
- package/dist/modules/messages/api/route.js.map +2 -2
- package/dist/modules/messages/api/token/[token]/route.js +1 -1
- package/dist/modules/messages/api/token/[token]/route.js.map +2 -2
- package/dist/modules/messages/commands/actions.js +9 -7
- package/dist/modules/messages/commands/actions.js.map +2 -2
- package/dist/modules/messages/commands/messages.js +83 -18
- package/dist/modules/messages/commands/messages.js.map +2 -2
- package/dist/modules/messages/commands/tokens.js +1 -1
- package/dist/modules/messages/commands/tokens.js.map +2 -2
- package/dist/modules/messages/components/ComposeMessagePageClient.js +24 -13
- package/dist/modules/messages/components/ComposeMessagePageClient.js.map +2 -2
- package/dist/modules/messages/components/MessageDetailPageClient.js +39 -2
- package/dist/modules/messages/components/MessageDetailPageClient.js.map +2 -2
- package/dist/modules/messages/components/MessagesInboxPageClient.js +10 -0
- package/dist/modules/messages/components/MessagesInboxPageClient.js.map +2 -2
- package/dist/modules/messages/components/message-detail/hooks/useMessageDetailsActions.js +2 -2
- package/dist/modules/messages/components/message-detail/hooks/useMessageDetailsActions.js.map +2 -2
- package/dist/modules/messages/components/message-detail/utils.js +17 -0
- package/dist/modules/messages/components/message-detail/utils.js.map +2 -2
- package/dist/modules/messages/data/entities.js +8 -1
- package/dist/modules/messages/data/entities.js.map +2 -2
- package/dist/modules/messages/data/validators.js +5 -1
- package/dist/modules/messages/data/validators.js.map +2 -2
- package/dist/modules/messages/lib/actions.js +5 -3
- package/dist/modules/messages/lib/actions.js.map +2 -2
- package/dist/modules/messages/migrations/Migration20260531130000.js +15 -0
- package/dist/modules/messages/migrations/Migration20260531130000.js.map +7 -0
- package/dist/modules/messages/widgets/injection-table.js +7 -0
- package/dist/modules/messages/widgets/injection-table.js.map +7 -0
- package/dist/modules/notifications/api/[id]/action/route.js +12 -2
- package/dist/modules/notifications/api/[id]/action/route.js.map +2 -2
- package/dist/modules/notifications/api/route.js +17 -4
- package/dist/modules/notifications/api/route.js.map +2 -2
- package/dist/modules/notifications/lib/notificationService.js +26 -21
- package/dist/modules/notifications/lib/notificationService.js.map +2 -2
- package/dist/modules/notifications/lib/routeHelpers.js +46 -8
- package/dist/modules/notifications/lib/routeHelpers.js.map +2 -2
- package/dist/modules/payment_gateways/api/status/route.js +6 -0
- package/dist/modules/payment_gateways/api/status/route.js.map +2 -2
- package/dist/modules/perspectives/api/[tableId]/route.js +39 -30
- package/dist/modules/perspectives/api/[tableId]/route.js.map +2 -2
- package/dist/modules/perspectives/services/perspectiveService.js +7 -0
- package/dist/modules/perspectives/services/perspectiveService.js.map +2 -2
- package/dist/modules/planner/backend/planner/availability-rulesets/[id]/page.js +6 -14
- package/dist/modules/planner/backend/planner/availability-rulesets/[id]/page.js.map +3 -3
- package/dist/modules/planner/backend/planner/availability-rulesets/page.js +4 -2
- package/dist/modules/planner/backend/planner/availability-rulesets/page.js.map +2 -2
- package/dist/modules/planner/commands/availability-rule-sets.js +20 -1
- package/dist/modules/planner/commands/availability-rule-sets.js.map +2 -2
- package/dist/modules/planner/commands/availability.js +9 -0
- package/dist/modules/planner/commands/availability.js.map +2 -2
- package/dist/modules/planner/components/AvailabilityRuleSetForm.js +3 -1
- package/dist/modules/planner/components/AvailabilityRuleSetForm.js.map +2 -2
- package/dist/modules/planner/components/AvailabilityRulesEditor.js +52 -13
- package/dist/modules/planner/components/AvailabilityRulesEditor.js.map +2 -2
- package/dist/modules/planner/components/AvailabilitySchedule.js +9 -5
- package/dist/modules/planner/components/AvailabilitySchedule.js.map +2 -2
- package/dist/modules/portal/frontend/[orgSlug]/portal/login/page.js +15 -14
- package/dist/modules/portal/frontend/[orgSlug]/portal/login/page.js.map +2 -2
- package/dist/modules/portal/frontend/[orgSlug]/portal/page.js +14 -21
- package/dist/modules/portal/frontend/[orgSlug]/portal/page.js.map +2 -2
- package/dist/modules/portal/frontend/[orgSlug]/portal/reset-password/page.js +11 -1
- package/dist/modules/portal/frontend/[orgSlug]/portal/reset-password/page.js.map +2 -2
- package/dist/modules/portal/frontend/[orgSlug]/portal/signup/page.js +15 -5
- package/dist/modules/portal/frontend/[orgSlug]/portal/signup/page.js.map +2 -2
- package/dist/modules/progress/api/jobs/[id]/route.js +7 -1
- package/dist/modules/progress/api/jobs/[id]/route.js.map +2 -2
- package/dist/modules/query_index/api/openapi.js +2 -1
- package/dist/modules/query_index/api/openapi.js.map +2 -2
- package/dist/modules/query_index/api/reindex.js +4 -0
- package/dist/modules/query_index/api/reindex.js.map +2 -2
- package/dist/modules/query_index/data/entities.js +5 -1
- package/dist/modules/query_index/data/entities.js.map +2 -2
- package/dist/modules/query_index/lib/engine.js +48 -15
- package/dist/modules/query_index/lib/engine.js.map +2 -2
- package/dist/modules/query_index/lib/indexer.js +50 -24
- package/dist/modules/query_index/lib/indexer.js.map +2 -2
- package/dist/modules/query_index/lib/jobs.js +28 -0
- package/dist/modules/query_index/lib/jobs.js.map +2 -2
- package/dist/modules/query_index/lib/reindexer.js +24 -15
- package/dist/modules/query_index/lib/reindexer.js.map +2 -2
- package/dist/modules/query_index/lib/search-tokens.js +3 -3
- package/dist/modules/query_index/lib/search-tokens.js.map +2 -2
- package/dist/modules/query_index/migrations/Migration20260606205453_query_index.js +35 -0
- package/dist/modules/query_index/migrations/Migration20260606205453_query_index.js.map +7 -0
- package/dist/modules/query_index/subscribers/delete_one.js +28 -15
- package/dist/modules/query_index/subscribers/delete_one.js.map +2 -2
- package/dist/modules/query_index/subscribers/upsert_one.js +31 -13
- package/dist/modules/query_index/subscribers/upsert_one.js.map +2 -2
- package/dist/modules/resources/backend/resources/resource-types/[id]/edit/page.js +18 -2
- package/dist/modules/resources/backend/resources/resource-types/[id]/edit/page.js.map +2 -2
- package/dist/modules/resources/backend/resources/resource-types/page.js +4 -2
- package/dist/modules/resources/backend/resources/resource-types/page.js.map +2 -2
- package/dist/modules/resources/backend/resources/resources/[id]/page.js +31 -4
- package/dist/modules/resources/backend/resources/resources/[id]/page.js.map +2 -2
- package/dist/modules/resources/backend/resources/resources/page.js +17 -4
- package/dist/modules/resources/backend/resources/resources/page.js.map +2 -2
- package/dist/modules/resources/commands/activities.js +64 -0
- package/dist/modules/resources/commands/activities.js.map +2 -2
- package/dist/modules/resources/commands/comments.js +21 -1
- package/dist/modules/resources/commands/comments.js.map +2 -2
- package/dist/modules/resources/commands/resource-types.js +22 -1
- package/dist/modules/resources/commands/resource-types.js.map +2 -2
- package/dist/modules/resources/commands/resources.js +87 -0
- package/dist/modules/resources/commands/resources.js.map +2 -2
- package/dist/modules/resources/commands/tag-assignments.js +53 -0
- package/dist/modules/resources/commands/tag-assignments.js.map +2 -2
- package/dist/modules/resources/components/ResourceCrudForm.js +90 -50
- package/dist/modules/resources/components/ResourceCrudForm.js.map +2 -2
- package/dist/modules/resources/components/ResourceTypeCrudForm.js +2 -1
- package/dist/modules/resources/components/ResourceTypeCrudForm.js.map +2 -2
- package/dist/modules/sales/api/documents/factory.js +7 -2
- package/dist/modules/sales/api/documents/factory.js.map +2 -2
- package/dist/modules/sales/api/shipping-methods/route.js +4 -0
- package/dist/modules/sales/api/shipping-methods/route.js.map +2 -2
- package/dist/modules/sales/api/tax-rates/route.js +4 -0
- package/dist/modules/sales/api/tax-rates/route.js.map +2 -2
- package/dist/modules/sales/backend/sales/channels/[channelId]/edit/page.js +3 -1
- package/dist/modules/sales/backend/sales/channels/[channelId]/edit/page.js.map +2 -2
- package/dist/modules/sales/backend/sales/channels/offers/page.js +13 -4
- package/dist/modules/sales/backend/sales/channels/offers/page.js.map +2 -2
- package/dist/modules/sales/backend/sales/channels/page.js +16 -4
- package/dist/modules/sales/backend/sales/channels/page.js.map +2 -2
- package/dist/modules/sales/backend/sales/documents/[id]/page.js +70 -22
- package/dist/modules/sales/backend/sales/documents/[id]/page.js.map +2 -2
- package/dist/modules/sales/backend/sales/documents/create/page.js.map +2 -2
- package/dist/modules/sales/commands/configuration.js +112 -7
- package/dist/modules/sales/commands/configuration.js.map +2 -2
- package/dist/modules/sales/commands/documentAddresses.js +181 -2
- package/dist/modules/sales/commands/documentAddresses.js.map +2 -2
- package/dist/modules/sales/commands/documents.js +46 -14
- package/dist/modules/sales/commands/documents.js.map +2 -2
- package/dist/modules/sales/commands/notes.js +39 -1
- package/dist/modules/sales/commands/notes.js.map +2 -2
- package/dist/modules/sales/commands/payments.js +121 -14
- package/dist/modules/sales/commands/payments.js.map +2 -2
- package/dist/modules/sales/commands/returns.js +178 -2
- package/dist/modules/sales/commands/returns.js.map +2 -2
- package/dist/modules/sales/commands/shared.js +15 -0
- package/dist/modules/sales/commands/shared.js.map +2 -2
- package/dist/modules/sales/commands/shipments.js +48 -6
- package/dist/modules/sales/commands/shipments.js.map +2 -2
- package/dist/modules/sales/components/AdjustmentKindSettings.js +19 -11
- package/dist/modules/sales/components/AdjustmentKindSettings.js.map +2 -2
- package/dist/modules/sales/components/DocumentNumberSettings.js.map +2 -2
- package/dist/modules/sales/components/OrderEditingSettings.js.map +2 -2
- package/dist/modules/sales/components/PaymentMethodsSettings.js +12 -4
- package/dist/modules/sales/components/PaymentMethodsSettings.js.map +2 -2
- package/dist/modules/sales/components/ShippingMethodsSettings.js +12 -4
- package/dist/modules/sales/components/ShippingMethodsSettings.js.map +2 -2
- package/dist/modules/sales/components/StatusSettings.js +18 -11
- package/dist/modules/sales/components/StatusSettings.js.map +2 -2
- package/dist/modules/sales/components/TaxRatesSettings.js +12 -4
- package/dist/modules/sales/components/TaxRatesSettings.js.map +2 -2
- package/dist/modules/sales/components/channels/ChannelOfferForm.js +65 -18
- package/dist/modules/sales/components/channels/ChannelOfferForm.js.map +2 -2
- package/dist/modules/sales/components/channels/SalesChannelOffersPanel.js +11 -4
- package/dist/modules/sales/components/channels/SalesChannelOffersPanel.js.map +2 -2
- package/dist/modules/sales/components/documents/AddressesSection.js +101 -36
- package/dist/modules/sales/components/documents/AddressesSection.js.map +2 -2
- package/dist/modules/sales/components/documents/AdjustmentDialog.js +37 -10
- package/dist/modules/sales/components/documents/AdjustmentDialog.js.map +2 -2
- package/dist/modules/sales/components/documents/AdjustmentsSection.js +43 -23
- package/dist/modules/sales/components/documents/AdjustmentsSection.js.map +2 -2
- package/dist/modules/sales/components/documents/ItemsSection.js +23 -14
- package/dist/modules/sales/components/documents/ItemsSection.js.map +2 -2
- package/dist/modules/sales/components/documents/LineItemDialog.js +70 -28
- package/dist/modules/sales/components/documents/LineItemDialog.js.map +2 -2
- package/dist/modules/sales/components/documents/PaymentDialog.js +29 -14
- package/dist/modules/sales/components/documents/PaymentDialog.js.map +2 -2
- package/dist/modules/sales/components/documents/PaymentsSection.js +20 -10
- package/dist/modules/sales/components/documents/PaymentsSection.js.map +2 -2
- package/dist/modules/sales/components/documents/ReturnDialog.js +26 -17
- package/dist/modules/sales/components/documents/ReturnDialog.js.map +2 -2
- package/dist/modules/sales/components/documents/ReturnsSection.js +3 -1
- package/dist/modules/sales/components/documents/ReturnsSection.js.map +2 -2
- package/dist/modules/sales/components/documents/SalesDocumentForm.js +1 -0
- package/dist/modules/sales/components/documents/SalesDocumentForm.js.map +2 -2
- package/dist/modules/sales/components/documents/SalesDocumentsTable.js +11 -6
- package/dist/modules/sales/components/documents/SalesDocumentsTable.js.map +2 -2
- package/dist/modules/sales/components/documents/ShipmentDialog.js +102 -17
- package/dist/modules/sales/components/documents/ShipmentDialog.js.map +2 -2
- package/dist/modules/sales/components/documents/ShipmentsSection.js +19 -10
- package/dist/modules/sales/components/documents/ShipmentsSection.js.map +2 -2
- package/dist/modules/sales/components/documents/optimisticLock.js +27 -0
- package/dist/modules/sales/components/documents/optimisticLock.js.map +7 -0
- package/dist/modules/sales/di.js +18 -0
- package/dist/modules/sales/di.js.map +2 -2
- package/dist/modules/sales/frontend/quote/[token]/page.js +11 -1
- package/dist/modules/sales/frontend/quote/[token]/page.js.map +2 -2
- package/dist/modules/sales/lib/calculations.js +18 -1
- package/dist/modules/sales/lib/calculations.js.map +2 -2
- package/dist/modules/sales/lib/dictionaries.js +2 -2
- package/dist/modules/sales/lib/dictionaries.js.map +2 -2
- package/dist/modules/sales/lib/makeStatusDictionaryRoute.js +2 -0
- package/dist/modules/sales/lib/makeStatusDictionaryRoute.js.map +2 -2
- package/dist/modules/shipping_carriers/api/cancel/route.js +2 -2
- package/dist/modules/shipping_carriers/api/cancel/route.js.map +2 -2
- package/dist/modules/shipping_carriers/lib/status-sync.js +8 -1
- package/dist/modules/shipping_carriers/lib/status-sync.js.map +2 -2
- package/dist/modules/staff/api/job-histories.js +11 -2
- package/dist/modules/staff/api/job-histories.js.map +2 -2
- package/dist/modules/staff/api/timesheets/time-entries/route.js +11 -4
- package/dist/modules/staff/api/timesheets/time-entries/route.js.map +2 -2
- package/dist/modules/staff/backend/staff/leave-requests/[id]/page.js +13 -8
- package/dist/modules/staff/backend/staff/leave-requests/[id]/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/leave-requests/page.js +9 -1
- package/dist/modules/staff/backend/staff/leave-requests/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/my-leave-requests/[id]/page.js +2 -1
- package/dist/modules/staff/backend/staff/my-leave-requests/[id]/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/profile/create/page.js +1 -1
- package/dist/modules/staff/backend/staff/profile/create/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/team-members/[id]/page.js +7 -4
- package/dist/modules/staff/backend/staff/team-members/[id]/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/team-members/page.js +17 -27
- package/dist/modules/staff/backend/staff/team-members/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/team-roles/[id]/edit/page.js +48 -2
- package/dist/modules/staff/backend/staff/team-roles/[id]/edit/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/team-roles/page.js +17 -35
- package/dist/modules/staff/backend/staff/team-roles/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js +5 -2
- package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/teams/page.js +24 -14
- package/dist/modules/staff/backend/staff/teams/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/timesheets/page.js +4 -1
- package/dist/modules/staff/backend/staff/timesheets/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/timesheets/projects/[id]/edit/page.js +18 -3
- package/dist/modules/staff/backend/staff/timesheets/projects/[id]/edit/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/timesheets/projects/[id]/page.js +35 -5
- package/dist/modules/staff/backend/staff/timesheets/projects/[id]/page.js.map +2 -2
- package/dist/modules/staff/backend/staff/timesheets/projects/page.js +12 -3
- package/dist/modules/staff/backend/staff/timesheets/projects/page.js.map +2 -2
- package/dist/modules/staff/commands/activities.js +53 -0
- package/dist/modules/staff/commands/activities.js.map +2 -2
- package/dist/modules/staff/commands/addresses.js +70 -0
- package/dist/modules/staff/commands/addresses.js.map +2 -2
- package/dist/modules/staff/commands/comments.js +21 -1
- package/dist/modules/staff/commands/comments.js.map +2 -2
- package/dist/modules/staff/commands/job-histories.js +88 -3
- package/dist/modules/staff/commands/job-histories.js.map +2 -2
- package/dist/modules/staff/commands/leave-requests.js +39 -1
- package/dist/modules/staff/commands/leave-requests.js.map +2 -2
- package/dist/modules/staff/commands/tag-assignments.js +33 -0
- package/dist/modules/staff/commands/tag-assignments.js.map +2 -2
- package/dist/modules/staff/commands/team-members.js +43 -0
- package/dist/modules/staff/commands/team-members.js.map +2 -2
- package/dist/modules/staff/commands/team-roles.js +23 -0
- package/dist/modules/staff/commands/team-roles.js.map +2 -2
- package/dist/modules/staff/commands/teams.js +23 -0
- package/dist/modules/staff/commands/teams.js.map +2 -2
- package/dist/modules/staff/commands/timesheets-entries.js +31 -1
- package/dist/modules/staff/commands/timesheets-entries.js.map +2 -2
- package/dist/modules/staff/commands/timesheets-projects.js +56 -2
- package/dist/modules/staff/commands/timesheets-projects.js.map +2 -2
- package/dist/modules/staff/components/LeaveRequestForm.js +2 -0
- package/dist/modules/staff/components/LeaveRequestForm.js.map +2 -2
- package/dist/modules/staff/components/TeamForm.js +2 -1
- package/dist/modules/staff/components/TeamForm.js.map +2 -2
- package/dist/modules/staff/components/TeamMemberForm.js +35 -4
- package/dist/modules/staff/components/TeamMemberForm.js.map +3 -3
- package/dist/modules/staff/components/TeamRoleForm.js +4 -3
- package/dist/modules/staff/components/TeamRoleForm.js.map +2 -2
- package/dist/modules/staff/components/detail/JobHistorySection.js +20 -7
- package/dist/modules/staff/components/detail/JobHistorySection.js.map +2 -2
- package/dist/modules/staff/data/validators.js +7 -1
- package/dist/modules/staff/data/validators.js.map +2 -2
- package/dist/modules/staff/lib/leaveRequestHelpers.js +2 -1
- package/dist/modules/staff/lib/leaveRequestHelpers.js.map +2 -2
- package/dist/modules/translations/api/[entityType]/[entityId]/route.js +9 -1
- package/dist/modules/translations/api/[entityType]/[entityId]/route.js.map +2 -2
- package/dist/modules/translations/components/TranslationManager.js +12 -8
- package/dist/modules/translations/components/TranslationManager.js.map +2 -2
- package/dist/modules/workflows/acl.js +102 -17
- package/dist/modules/workflows/acl.js.map +2 -2
- package/dist/modules/workflows/api/definitions/[id]/route.js +106 -0
- package/dist/modules/workflows/api/definitions/[id]/route.js.map +2 -2
- package/dist/modules/workflows/backend/definitions/[id]/page.js +11 -3
- package/dist/modules/workflows/backend/definitions/[id]/page.js.map +2 -2
- package/dist/modules/workflows/backend/definitions/page.js +33 -13
- package/dist/modules/workflows/backend/definitions/page.js.map +2 -2
- package/dist/modules/workflows/backend/definitions/visual-editor/page.js +29 -16
- package/dist/modules/workflows/backend/definitions/visual-editor/page.js.map +2 -2
- package/dist/modules/workflows/components/NodeEditDialog.js +3 -1
- package/dist/modules/workflows/components/NodeEditDialog.js.map +2 -2
- package/dist/modules/workflows/components/WorkflowGraphImpl.js +4 -2
- package/dist/modules/workflows/components/WorkflowGraphImpl.js.map +2 -2
- package/dist/modules/workflows/components/formConfig.js +26 -13
- package/dist/modules/workflows/components/formConfig.js.map +2 -2
- package/dist/modules/workflows/components/nodes/ParallelForkNode.js +49 -0
- package/dist/modules/workflows/components/nodes/ParallelForkNode.js.map +7 -0
- package/dist/modules/workflows/components/nodes/ParallelJoinNode.js +49 -0
- package/dist/modules/workflows/components/nodes/ParallelJoinNode.js.map +7 -0
- package/dist/modules/workflows/components/nodes/index.js +4 -0
- package/dist/modules/workflows/components/nodes/index.js.map +2 -2
- package/dist/modules/workflows/data/entities.js +81 -0
- package/dist/modules/workflows/data/entities.js.map +2 -2
- package/dist/modules/workflows/data/validators.js +146 -1
- package/dist/modules/workflows/data/validators.js.map +2 -2
- package/dist/modules/workflows/di.js +12 -0
- package/dist/modules/workflows/di.js.map +2 -2
- package/dist/modules/workflows/events.js +7 -1
- package/dist/modules/workflows/events.js.map +2 -2
- package/dist/modules/workflows/lib/activity-executor.js +4 -2
- package/dist/modules/workflows/lib/activity-executor.js.map +2 -2
- package/dist/modules/workflows/lib/activity-queue-types.js.map +2 -2
- package/dist/modules/workflows/lib/event-logger.js +2 -0
- package/dist/modules/workflows/lib/event-logger.js.map +2 -2
- package/dist/modules/workflows/lib/execution-token.js +98 -0
- package/dist/modules/workflows/lib/execution-token.js.map +7 -0
- package/dist/modules/workflows/lib/node-type-icons.js +14 -5
- package/dist/modules/workflows/lib/node-type-icons.js.map +2 -2
- package/dist/modules/workflows/lib/parallel-handler.js +364 -0
- package/dist/modules/workflows/lib/parallel-handler.js.map +7 -0
- package/dist/modules/workflows/lib/signal-handler.js +63 -1
- package/dist/modules/workflows/lib/signal-handler.js.map +2 -2
- package/dist/modules/workflows/lib/step-handler.js +74 -30
- package/dist/modules/workflows/lib/step-handler.js.map +2 -2
- package/dist/modules/workflows/lib/task-handler.js +26 -0
- package/dist/modules/workflows/lib/task-handler.js.map +2 -2
- package/dist/modules/workflows/lib/timer-handler.js +26 -1
- package/dist/modules/workflows/lib/timer-handler.js.map +2 -2
- package/dist/modules/workflows/lib/transition-handler.js +33 -21
- package/dist/modules/workflows/lib/transition-handler.js.map +2 -2
- package/dist/modules/workflows/lib/workflow-executor.js +54 -1
- package/dist/modules/workflows/lib/workflow-executor.js.map +2 -2
- package/dist/modules/workflows/migrations/Migration20260602120000.js +24 -0
- package/dist/modules/workflows/migrations/Migration20260602120000.js.map +7 -0
- package/dist/modules/workflows/workers/workflow-activities.worker.js +8 -4
- package/dist/modules/workflows/workers/workflow-activities.worker.js.map +2 -2
- package/generated/entities/channel_ingest_dead_letter/index.ts +11 -0
- package/generated/entities/channel_thread_mapping/index.ts +11 -0
- package/generated/entities/channel_thread_token/index.ts +7 -0
- package/generated/entities/communication_channel/index.ts +20 -0
- package/generated/entities/customer_interaction/index.ts +2 -0
- package/generated/entities/customer_settings/index.ts +1 -0
- package/generated/entities/dictionary/index.ts +1 -0
- package/generated/entities/external_conversation/index.ts +11 -0
- package/generated/entities/external_message/index.ts +11 -0
- package/generated/entities/inbox_settings/index.ts +1 -0
- package/generated/entities/integration_credentials/index.ts +1 -0
- package/generated/entities/message/index.ts +1 -0
- package/generated/entities/message_channel_link/index.ts +15 -0
- package/generated/entities/message_reaction/index.ts +11 -0
- package/generated/entities/role/index.ts +1 -0
- package/generated/entities/step_instance/index.ts +1 -0
- package/generated/entities/user/index.ts +1 -0
- package/generated/entities/user_task/index.ts +1 -0
- package/generated/entities/workflow_branch_instance/index.ts +18 -0
- package/generated/entities/workflow_event/index.ts +1 -0
- package/generated/entities/workflow_instance/index.ts +1 -0
- package/generated/entities.ids.generated.ts +12 -0
- package/generated/entity-fields-registry.ts +146 -0
- package/jest.config.cjs +4 -0
- package/jest.mocks/entities.ids.generated.js +9 -0
- package/jest.setup.ts +17 -0
- package/package.json +17 -15
- package/src/helpers/integration/api.ts +15 -2
- package/src/helpers/integration/authFixtures.ts +4 -1
- package/src/helpers/integration/communicationChannelsFixtures.ts +130 -0
- package/src/helpers/integration/crudFormFields.ts +48 -0
- package/src/helpers/integration/crudFormPersistence.ts +178 -0
- package/src/helpers/integration/currenciesFixtures.ts +60 -0
- package/src/helpers/integration/customerAccountsFixtures.ts +256 -0
- package/src/helpers/integration/dbFixtures.ts +15 -1
- package/src/helpers/integration/inboxFixtures.ts +36 -0
- package/src/helpers/integration/optimisticLockUi.ts +172 -0
- package/src/helpers/integration/salesFixtures.ts +29 -0
- package/src/helpers/integration/salesUi.ts +19 -12
- package/src/helpers/integration/standaloneEnv.ts +62 -0
- package/src/helpers/integration/undoHarness.ts +243 -0
- package/src/helpers/integration/workflowsFixtures.ts +231 -0
- package/src/modules/api_keys/backend/api-keys/page.tsx +18 -5
- package/src/modules/api_keys/data/validators.ts +3 -1
- package/src/modules/attachments/components/AttachmentPartitionSettings.tsx +19 -9
- package/src/modules/attachments/lib/ocrQueue.ts +9 -1
- package/src/modules/attachments/lib/security.ts +1 -1
- package/src/modules/audit_logs/api/audit-logs/actions/redo/route.ts +3 -1
- package/src/modules/audit_logs/api/audit-logs/actions/undo/route.ts +13 -2
- package/src/modules/audit_logs/data/entities.ts +1 -1
- package/src/modules/audit_logs/services/actionLogService.ts +18 -0
- package/src/modules/auth/api/logout.ts +0 -12
- package/src/modules/auth/api/roles/acl/route.ts +37 -11
- package/src/modules/auth/api/roles/route.ts +56 -2
- package/src/modules/auth/api/sidebar/preferences/route.ts +83 -3
- package/src/modules/auth/api/users/acl/route.ts +94 -33
- package/src/modules/auth/api/users/consents/route.ts +20 -0
- package/src/modules/auth/api/users/resend-invite/route.ts +20 -0
- package/src/modules/auth/api/users/route.ts +42 -1
- package/src/modules/auth/backend/roles/[id]/edit/page.tsx +29 -4
- package/src/modules/auth/backend/roles/page.tsx +17 -4
- package/src/modules/auth/backend/users/[id]/edit/page.tsx +92 -6
- package/src/modules/auth/backend/users/page.tsx +15 -2
- package/src/modules/auth/cli.ts +20 -21
- package/src/modules/auth/commands/roles.ts +81 -1
- package/src/modules/auth/commands/users.ts +117 -6
- package/src/modules/auth/components/AclEditor.tsx +10 -1
- package/src/modules/auth/data/entities.ts +7 -1
- package/src/modules/auth/frontend/reset/[token]/page.tsx +7 -2
- package/src/modules/auth/lib/consentIntegrity.ts +18 -2
- package/src/modules/auth/lib/emailHash.ts +5 -1
- package/src/modules/auth/lib/grantChecks.ts +55 -1
- package/src/modules/auth/lib/sessionIntegrity.ts +18 -1
- package/src/modules/auth/services/authService.ts +10 -19
- package/src/modules/auth/services/sidebarPreferencesService.ts +38 -4
- package/src/modules/business_rules/api/rules/route.ts +30 -0
- package/src/modules/business_rules/api/sets/route.ts +30 -0
- package/src/modules/business_rules/backend/rules/[id]/page.tsx +16 -4
- package/src/modules/business_rules/backend/rules/page.tsx +29 -12
- package/src/modules/business_rules/backend/sets/[id]/page.tsx +16 -4
- package/src/modules/business_rules/backend/sets/page.tsx +28 -11
- package/src/modules/catalog/AGENTS.md +10 -0
- package/src/modules/catalog/api/categories/route.ts +3 -0
- package/src/modules/catalog/api/products/route.ts +4 -0
- package/src/modules/catalog/backend/catalog/categories/[id]/edit/page.tsx +5 -0
- package/src/modules/catalog/backend/catalog/products/[id]/page.tsx +174 -91
- package/src/modules/catalog/backend/catalog/products/[productId]/variants/[variantId]/page.tsx +121 -34
- package/src/modules/catalog/backend/catalog/products/[productId]/variants/create/page.tsx +22 -5
- package/src/modules/catalog/backend/catalog/products/create/page.tsx +5 -28
- package/src/modules/catalog/backend/catalog/products/optionSchemaClient.ts +2 -0
- package/src/modules/catalog/commands/categories.ts +42 -0
- package/src/modules/catalog/commands/offers.ts +44 -0
- package/src/modules/catalog/commands/optionSchemas.ts +10 -0
- package/src/modules/catalog/commands/priceKinds.ts +5 -0
- package/src/modules/catalog/commands/prices.ts +50 -0
- package/src/modules/catalog/commands/productUnitConversions.ts +32 -0
- package/src/modules/catalog/commands/products.ts +87 -0
- package/src/modules/catalog/commands/variants.ts +118 -32
- package/src/modules/catalog/components/PriceKindSettings.tsx +21 -7
- package/src/modules/catalog/components/categories/CategoriesDataTable.tsx +9 -0
- package/src/modules/catalog/components/products/ProductMediaManager.tsx +2 -0
- package/src/modules/catalog/components/products/ProductsDataTable.tsx +16 -4
- package/src/modules/catalog/components/products/VariantBuilder.tsx +48 -9
- package/src/modules/catalog/components/products/productForm.ts +41 -0
- package/src/modules/catalog/components/products/variantForm.ts +10 -0
- package/src/modules/catalog/lib/pricing.ts +8 -0
- package/src/modules/communication_channels/acl.ts +43 -0
- package/src/modules/communication_channels/api/delete/channels/[id]/route.ts +163 -0
- package/src/modules/communication_channels/api/delete/messages/[messageId]/reactions/[reactionId]/route.ts +143 -0
- package/src/modules/communication_channels/api/get/channels/[id]/health/route.ts +173 -0
- package/src/modules/communication_channels/api/get/channels/[id]/route.ts +111 -0
- package/src/modules/communication_channels/api/get/channels/route.ts +109 -0
- package/src/modules/communication_channels/api/get/me/channels/route.ts +100 -0
- package/src/modules/communication_channels/api/get/oauth/[provider]/callback/route.ts +355 -0
- package/src/modules/communication_channels/api/post/channels/[id]/import-history/route.ts +206 -0
- package/src/modules/communication_channels/api/post/channels/[id]/poll-now/route.ts +174 -0
- package/src/modules/communication_channels/api/post/channels/[id]/push/register/route.ts +158 -0
- package/src/modules/communication_channels/api/post/channels/[id]/set-primary/route.ts +114 -0
- package/src/modules/communication_channels/api/post/channels/[id]/test-send/route.ts +241 -0
- package/src/modules/communication_channels/api/post/channels/connect/credentials/route.ts +134 -0
- package/src/modules/communication_channels/api/post/messages/[messageId]/reactions/route.ts +143 -0
- package/src/modules/communication_channels/api/post/oauth/[provider]/initiate/route.ts +192 -0
- package/src/modules/communication_channels/api/post/send-as-user/route.ts +125 -0
- package/src/modules/communication_channels/api/post/test-seed/route.ts +294 -0
- package/src/modules/communication_channels/api/post/webhook/[provider]/route.ts +227 -0
- package/src/modules/communication_channels/api/post/webhooks/gmail/route.ts +161 -0
- package/src/modules/communication_channels/api/put/threads/[threadId]/assign/route.ts +132 -0
- package/src/modules/communication_channels/backend/communication_channels/channels/[id]/page.meta.ts +34 -0
- package/src/modules/communication_channels/backend/communication_channels/channels/[id]/page.tsx +250 -0
- package/src/modules/communication_channels/backend/communication_channels/channels/page.meta.ts +36 -0
- package/src/modules/communication_channels/backend/communication_channels/channels/page.tsx +137 -0
- package/src/modules/communication_channels/backend/profile/communication-channels/page.meta.ts +36 -0
- package/src/modules/communication_channels/backend/profile/communication-channels/page.tsx +912 -0
- package/src/modules/communication_channels/commands/connect-credential-channel.ts +243 -0
- package/src/modules/communication_channels/commands/delete-channel.ts +193 -0
- package/src/modules/communication_channels/commands/deliver-outbound-message.ts +579 -0
- package/src/modules/communication_channels/commands/disconnect-channel.ts +241 -0
- package/src/modules/communication_channels/commands/ingest-inbound-message.ts +602 -0
- package/src/modules/communication_channels/commands/interceptors.ts +104 -0
- package/src/modules/communication_channels/commands/process-inbound-reaction.ts +265 -0
- package/src/modules/communication_channels/commands/push-register.ts +203 -0
- package/src/modules/communication_channels/commands/push-renew.ts +49 -0
- package/src/modules/communication_channels/commands/push-unregister.ts +168 -0
- package/src/modules/communication_channels/commands/queue-import-history.ts +180 -0
- package/src/modules/communication_channels/commands/reassign-conversation.ts +273 -0
- package/src/modules/communication_channels/commands/set-primary-channel.ts +157 -0
- package/src/modules/communication_channels/commands/toggle-outbound-reaction.ts +347 -0
- package/src/modules/communication_channels/data/enrichers.ts +413 -0
- package/src/modules/communication_channels/data/entities.ts +546 -0
- package/src/modules/communication_channels/data/extensions.ts +76 -0
- package/src/modules/communication_channels/data/validators.ts +138 -0
- package/src/modules/communication_channels/di.ts +40 -0
- package/src/modules/communication_channels/encryption.ts +44 -0
- package/src/modules/communication_channels/events.ts +122 -0
- package/src/modules/communication_channels/i18n/de.json +138 -0
- package/src/modules/communication_channels/i18n/en.json +138 -0
- package/src/modules/communication_channels/i18n/es.json +138 -0
- package/src/modules/communication_channels/i18n/pl.json +138 -0
- package/src/modules/communication_channels/index.ts +19 -0
- package/src/modules/communication_channels/lib/access-control.ts +110 -0
- package/src/modules/communication_channels/lib/adapter-compat.ts +57 -0
- package/src/modules/communication_channels/lib/adapter-registry-singleton.ts +35 -0
- package/src/modules/communication_channels/lib/adapter.ts +605 -0
- package/src/modules/communication_channels/lib/connect-channel.ts +163 -0
- package/src/modules/communication_channels/lib/contact-resolver.ts +162 -0
- package/src/modules/communication_channels/lib/credential-refresh.ts +197 -0
- package/src/modules/communication_channels/lib/dead-letter.ts +87 -0
- package/src/modules/communication_channels/lib/email-capabilities.ts +60 -0
- package/src/modules/communication_channels/lib/email-contact.ts +17 -0
- package/src/modules/communication_channels/lib/email-mime.ts +501 -0
- package/src/modules/communication_channels/lib/error-classification.ts +144 -0
- package/src/modules/communication_channels/lib/gmail-pubsub-jwt.ts +278 -0
- package/src/modules/communication_channels/lib/mutation-guards.ts +215 -0
- package/src/modules/communication_channels/lib/oauth-client-config.ts +79 -0
- package/src/modules/communication_channels/lib/oauth-state.ts +228 -0
- package/src/modules/communication_channels/lib/oauth-token.ts +81 -0
- package/src/modules/communication_channels/lib/pg-errors.ts +12 -0
- package/src/modules/communication_channels/lib/provider-health.ts +47 -0
- package/src/modules/communication_channels/lib/push-state.ts +38 -0
- package/src/modules/communication_channels/lib/queue.ts +66 -0
- package/src/modules/communication_channels/lib/reaction-processor-types.ts +51 -0
- package/src/modules/communication_channels/lib/reaction-semantics.ts +48 -0
- package/src/modules/communication_channels/lib/registry.ts +99 -0
- package/src/modules/communication_channels/lib/route-mutation-guard.ts +68 -0
- package/src/modules/communication_channels/lib/sanitize-channel-html.ts +129 -0
- package/src/modules/communication_channels/lib/send-as-user.ts +284 -0
- package/src/modules/communication_channels/lib/system-user.ts +74 -0
- package/src/modules/communication_channels/lib/test-seed.ts +140 -0
- package/src/modules/communication_channels/lib/thread-matcher.ts +430 -0
- package/src/modules/communication_channels/lib/thread-token.ts +355 -0
- package/src/modules/communication_channels/lib/use-connect-channel.ts +73 -0
- package/src/modules/communication_channels/migrations/.snapshot-open-mercato.json +2142 -0
- package/src/modules/communication_channels/migrations/Migration20260526134719_communication_channels.ts +55 -0
- package/src/modules/communication_channels/migrations/Migration20260527195446_communication_channels.ts +20 -0
- package/src/modules/communication_channels/migrations/Migration20260529231848_communication_channels.ts +13 -0
- package/src/modules/communication_channels/migrations/Migration20260531120000_communication_channels.ts +24 -0
- package/src/modules/communication_channels/notifications.client.ts +50 -0
- package/src/modules/communication_channels/notifications.handlers.ts +86 -0
- package/src/modules/communication_channels/notifications.ts +52 -0
- package/src/modules/communication_channels/setup.ts +158 -0
- package/src/modules/communication_channels/subscribers/channel-requires-reauth-notification.ts +118 -0
- package/src/modules/communication_channels/subscribers/outbound-bridge.ts +175 -0
- package/src/modules/communication_channels/subscribers/user-deleted-cascade.ts +100 -0
- package/src/modules/communication_channels/widgets/components.ts +36 -0
- package/src/modules/communication_channels/widgets/injection/channel-badge/widget.client.tsx +38 -0
- package/src/modules/communication_channels/widgets/injection/channel-badge/widget.ts +51 -0
- package/src/modules/communication_channels/widgets/injection/channel-info-panel/widget.client.tsx +278 -0
- package/src/modules/communication_channels/widgets/injection/channel-info-panel/widget.ts +24 -0
- package/src/modules/communication_channels/widgets/injection/channel-payload-renderer/widget.client.tsx +63 -0
- package/src/modules/communication_channels/widgets/injection/channel-payload-renderer/widget.ts +29 -0
- package/src/modules/communication_channels/widgets/injection/profile-channels-menu/widget.ts +34 -0
- package/src/modules/communication_channels/widgets/injection/reaction-bar/widget.client.tsx +177 -0
- package/src/modules/communication_channels/widgets/injection/reaction-bar/widget.ts +26 -0
- package/src/modules/communication_channels/widgets/injection-table.ts +47 -0
- package/src/modules/communication_channels/widgets/notifications/ChannelRequiresReauthRenderer.tsx +48 -0
- package/src/modules/communication_channels/widgets/notifications/MessageReceivedRenderer.tsx +45 -0
- package/src/modules/communication_channels/widgets/notifications/index.ts +2 -0
- package/src/modules/communication_channels/workers/channel-import-history.ts +252 -0
- package/src/modules/communication_channels/workers/gmail-history-sync.ts +223 -0
- package/src/modules/communication_channels/workers/gmail-renew-watch.ts +141 -0
- package/src/modules/communication_channels/workers/inbound-processor.ts +114 -0
- package/src/modules/communication_channels/workers/outbound-delivery.ts +155 -0
- package/src/modules/communication_channels/workers/poll-channel.ts +391 -0
- package/src/modules/communication_channels/workers/poll-tick.ts +210 -0
- package/src/modules/communication_channels/workers/reaction-processor.ts +264 -0
- package/src/modules/configs/api/openapi.ts +12 -2
- package/src/modules/currencies/backend/currencies/[id]/page.tsx +13 -6
- package/src/modules/currencies/backend/currencies/page.tsx +26 -11
- package/src/modules/currencies/backend/exchange-rates/[id]/page.tsx +3 -0
- package/src/modules/currencies/backend/exchange-rates/page.tsx +21 -12
- package/src/modules/currencies/commands/currencies.ts +42 -18
- package/src/modules/currencies/commands/exchange-rates.ts +18 -8
- package/src/modules/currencies/components/CurrencyFetchingConfig.tsx +31 -21
- package/src/modules/currencies/i18n/de.json +1 -0
- package/src/modules/currencies/i18n/en.json +1 -0
- package/src/modules/currencies/i18n/es.json +1 -0
- package/src/modules/currencies/i18n/pl.json +1 -0
- package/src/modules/customer_accounts/api/admin/roles/[id].ts +35 -5
- package/src/modules/customer_accounts/api/admin/roles.ts +2 -0
- package/src/modules/customer_accounts/api/admin/users/[id].ts +57 -5
- package/src/modules/customer_accounts/api/admin/users-invite.ts +17 -0
- package/src/modules/customer_accounts/api/admin/users.ts +74 -23
- package/src/modules/customer_accounts/api/login.ts +21 -4
- package/src/modules/customer_accounts/api/portal/users-invite.ts +17 -0
- package/src/modules/customer_accounts/api/signup.ts +3 -1
- package/src/modules/customer_accounts/backend/customer_accounts/roles/[id]/page.tsx +34 -20
- package/src/modules/customer_accounts/backend/customer_accounts/roles/page.tsx +20 -6
- package/src/modules/customer_accounts/backend/customer_accounts/settings/domain/page.tsx +11 -4
- package/src/modules/customer_accounts/backend/customer_accounts/users/[id]/page.tsx +28 -17
- package/src/modules/customer_accounts/backend/customer_accounts/users/page.tsx +27 -11
- package/src/modules/customer_accounts/data/validators.ts +1 -0
- package/src/modules/customer_accounts/lib/customerEntityOwnership.ts +39 -0
- package/src/modules/customer_accounts/lib/rateLimiter.ts +14 -0
- package/src/modules/customer_accounts/lib/resolveTenantContext.ts +61 -4
- package/src/modules/customer_accounts/services/customerInvitationService.ts +35 -2
- package/src/modules/customer_accounts/services/customerUserService.ts +13 -7
- package/src/modules/customer_accounts/subscribers/autoLinkCrmReverse.ts +3 -3
- package/src/modules/customers/AGENTS.md +8 -2
- package/src/modules/customers/acl.ts +18 -0
- package/src/modules/customers/agentic/standalone-guide.md +1 -1
- package/src/modules/customers/api/activities/route.ts +42 -21
- package/src/modules/customers/api/addresses/route.ts +2 -0
- package/src/modules/customers/api/companies/[id]/route.ts +26 -3
- package/src/modules/customers/api/companies/route.ts +18 -5
- package/src/modules/customers/api/deals/route.ts +3 -0
- package/src/modules/customers/api/dictionaries/[kind]/route.ts +21 -46
- package/src/modules/customers/api/dictionaries/cache.ts +3 -1
- package/src/modules/customers/api/dictionaries/context.ts +1 -1
- package/src/modules/customers/api/interactions/[id]/visibility/route.ts +179 -0
- package/src/modules/customers/api/interactions/counts/route.ts +10 -0
- package/src/modules/customers/api/interactions/route.ts +58 -11
- package/src/modules/customers/api/people/[id]/email-threads/route.ts +92 -0
- package/src/modules/customers/api/people/[id]/emails/route.ts +184 -0
- package/src/modules/customers/api/people/[id]/route.ts +26 -8
- package/src/modules/customers/api/people/route.ts +16 -5
- package/src/modules/customers/api/pipeline-stages/route.ts +0 -14
- package/src/modules/customers/api/settings/dictionary-sort-modes/route.ts +258 -0
- package/src/modules/customers/api/todos/route.ts +29 -11
- package/src/modules/customers/backend/config/customers/deals/page.tsx +1 -0
- package/src/modules/customers/backend/config/customers/pipeline-stages/page.tsx +36 -21
- package/src/modules/customers/backend/customers/companies/[id]/page.tsx +52 -27
- package/src/modules/customers/backend/customers/companies/page.tsx +11 -1
- package/src/modules/customers/backend/customers/companies-v2/[id]/page.tsx +27 -5
- package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealFormHandlers.ts +39 -7
- package/src/modules/customers/backend/customers/deals/[id]/page.tsx +1 -0
- package/src/modules/customers/backend/customers/deals/page.tsx +27 -7
- package/src/modules/customers/backend/customers/deals/pipeline/page.tsx +64 -39
- package/src/modules/customers/backend/customers/people/[id]/page.tsx +46 -26
- package/src/modules/customers/backend/customers/people/page.tsx +11 -1
- package/src/modules/customers/backend/customers/people-v2/[id]/page.tsx +95 -25
- package/src/modules/customers/commands/activities.ts +28 -0
- package/src/modules/customers/commands/addresses.ts +100 -14
- package/src/modules/customers/commands/comments.ts +30 -0
- package/src/modules/customers/commands/companies.ts +121 -1
- package/src/modules/customers/commands/deals.ts +241 -101
- package/src/modules/customers/commands/dictionaries.ts +45 -0
- package/src/modules/customers/commands/entity-roles.ts +49 -0
- package/src/modules/customers/commands/interactions.ts +264 -8
- package/src/modules/customers/commands/labels.ts +85 -0
- package/src/modules/customers/commands/people.ts +208 -13
- package/src/modules/customers/commands/personCompanyLinks.ts +148 -70
- package/src/modules/customers/commands/pipeline-stages.ts +16 -16
- package/src/modules/customers/commands/settings.ts +41 -0
- package/src/modules/customers/commands/tags.ts +59 -0
- package/src/modules/customers/commands/todos.ts +28 -0
- package/src/modules/customers/components/AddressFormatSettings.tsx +1 -0
- package/src/modules/customers/components/CustomersConfigurationSections.tsx +3 -0
- package/src/modules/customers/components/DictionarySettings.tsx +18 -13
- package/src/modules/customers/components/DictionarySortSettings.tsx +268 -0
- package/src/modules/customers/components/PipelineSettings.tsx +42 -21
- package/src/modules/customers/components/detail/ActivityCard.tsx +48 -0
- package/src/modules/customers/components/detail/ActivityForm.tsx +1 -0
- package/src/modules/customers/components/detail/ActivityTimeline.tsx +3 -3
- package/src/modules/customers/components/detail/AddressesSection.tsx +4 -0
- package/src/modules/customers/components/detail/AnnualRevenueField.tsx +1 -0
- package/src/modules/customers/components/detail/CompanyPeopleSection.tsx +2 -0
- package/src/modules/customers/components/detail/ComposeEmailDialog.tsx +329 -0
- package/src/modules/customers/components/detail/DealForm.tsx +3 -1
- package/src/modules/customers/components/detail/DealsSection.tsx +30 -0
- package/src/modules/customers/components/detail/EmailCardActions.tsx +263 -0
- package/src/modules/customers/components/detail/EmailReplyForwardActions.tsx +53 -0
- package/src/modules/customers/components/detail/EntityTagsDialog.tsx +7 -0
- package/src/modules/customers/components/detail/ManageTagsDialog.tsx +4 -0
- package/src/modules/customers/components/detail/PersonCompaniesSection.tsx +4 -0
- package/src/modules/customers/components/detail/PersonDetailTabs.tsx +8 -1
- package/src/modules/customers/components/detail/PersonEmailThreadsTab.tsx +448 -0
- package/src/modules/customers/components/detail/RoleAssignmentRow.tsx +2 -0
- package/src/modules/customers/components/detail/ScheduleActivityDialog.tsx +23 -7
- package/src/modules/customers/components/detail/create/DealCurrencyField.tsx +1 -0
- package/src/modules/customers/components/detail/hooks/useCustomerDictionary.ts +1 -0
- package/src/modules/customers/components/detail/hooks/useInteractionMutations.ts +25 -15
- package/src/modules/customers/components/detail/hooks/useInteractions.ts +76 -35
- package/src/modules/customers/components/detail/hooks/usePersonTasks.ts +30 -17
- package/src/modules/customers/components/detail/schedule/useScheduleFormState.ts +2 -0
- package/src/modules/customers/components/detail/types.ts +1 -0
- package/src/modules/customers/components/formConfig.tsx +134 -30
- package/src/modules/customers/data/enrichers.ts +252 -1
- package/src/modules/customers/data/entities.ts +51 -2
- package/src/modules/customers/data/extensions.ts +26 -0
- package/src/modules/customers/data/guards.ts +67 -0
- package/src/modules/customers/data/validators.ts +46 -13
- package/src/modules/customers/di.ts +66 -0
- package/src/modules/customers/encryption.ts +11 -0
- package/src/modules/customers/events.ts +4 -0
- package/src/modules/customers/i18n/de.json +50 -0
- package/src/modules/customers/i18n/en.json +50 -0
- package/src/modules/customers/i18n/es.json +50 -0
- package/src/modules/customers/i18n/pl.json +50 -0
- package/src/modules/customers/lib/detailTenantScope.ts +36 -0
- package/src/modules/customers/lib/findPeopleByAddresses.ts +107 -0
- package/src/modules/customers/lib/kysely.ts +16 -0
- package/src/modules/customers/lib/link-channel-message-handler.ts +571 -0
- package/src/modules/customers/lib/personCompanyLinkTable.ts +39 -0
- package/src/modules/customers/lib/personEmailThreads.ts +325 -0
- package/src/modules/customers/lib/todoCompatibility.ts +14 -0
- package/src/modules/customers/lib/visibilityFilter.ts +152 -0
- package/src/modules/customers/migrations/.snapshot-open-mercato.json +392 -0
- package/src/modules/customers/migrations/Migration20260527012240_customers.ts +23 -0
- package/src/modules/customers/migrations/Migration20260602202147_customers.ts +13 -0
- package/src/modules/customers/search.ts +6 -0
- package/src/modules/customers/setup.ts +1 -0
- package/src/modules/customers/subscribers/link-channel-message-received.ts +21 -0
- package/src/modules/customers/subscribers/link-channel-message-sent.ts +21 -0
- package/src/modules/dashboards/components/WidgetVisibilityEditor.tsx +2 -0
- package/src/modules/data_sync/api/options.ts +7 -4
- package/src/modules/data_sync/api/schedules/route.ts +9 -1
- package/src/modules/data_sync/backend/data-sync/page.tsx +18 -5
- package/src/modules/data_sync/components/IntegrationScheduleTab.tsx +46 -19
- package/src/modules/data_sync/i18n/de.json +1 -0
- package/src/modules/data_sync/i18n/en.json +1 -0
- package/src/modules/data_sync/i18n/es.json +1 -0
- package/src/modules/data_sync/i18n/pl.json +1 -0
- package/src/modules/data_sync/lib/sync-schedule-service.ts +11 -0
- package/src/modules/dictionaries/api/[dictionaryId]/entries/[entryId]/route.ts +8 -1
- package/src/modules/dictionaries/api/[dictionaryId]/entries/route.ts +12 -5
- package/src/modules/dictionaries/api/[dictionaryId]/route.ts +31 -0
- package/src/modules/dictionaries/api/openapi.ts +2 -1
- package/src/modules/dictionaries/api/route.ts +7 -0
- package/src/modules/dictionaries/commands/entry-operations.ts +19 -0
- package/src/modules/dictionaries/components/DictionariesManager.tsx +125 -14
- package/src/modules/dictionaries/components/DictionaryEntriesEditor.tsx +31 -18
- package/src/modules/dictionaries/components/DictionaryEntrySelect.tsx +56 -9
- package/src/modules/dictionaries/components/DictionarySelectControl.tsx +9 -0
- package/src/modules/dictionaries/components/dictionaryAppearance.tsx +5 -1
- package/src/modules/dictionaries/components/hooks/useDictionaryEntries.ts +1 -0
- package/src/modules/dictionaries/data/entities.ts +6 -0
- package/src/modules/dictionaries/data/validators.ts +2 -0
- package/src/modules/dictionaries/fields/dictionary.tsx +2 -2
- package/src/modules/dictionaries/i18n/de.json +9 -0
- package/src/modules/dictionaries/i18n/en.json +9 -0
- package/src/modules/dictionaries/i18n/es.json +9 -0
- package/src/modules/dictionaries/i18n/pl.json +9 -0
- package/src/modules/dictionaries/lib/entrySort.ts +73 -0
- package/src/modules/dictionaries/migrations/.snapshot-open-mercato.json +44 -0
- package/src/modules/dictionaries/migrations/Migration20260602202147_dictionaries.ts +13 -0
- package/src/modules/directory/api/get/organizations/lookup.ts +0 -8
- package/src/modules/directory/api/organizations/route.ts +40 -16
- package/src/modules/directory/backend/directory/organizations/[id]/edit/page.tsx +8 -0
- package/src/modules/directory/backend/directory/organizations/page.tsx +18 -5
- package/src/modules/directory/backend/directory/tenants/[id]/edit/page.tsx +16 -5
- package/src/modules/directory/backend/directory/tenants/page.tsx +16 -4
- package/src/modules/directory/commands/organizations.ts +74 -4
- package/src/modules/directory/commands/tenants.ts +14 -1
- package/src/modules/entities/api/records.ts +141 -6
- package/src/modules/entities/backend/entities/user/[entityId]/records/[recordId]/page.tsx +7 -0
- package/src/modules/entities/backend/entities/user/[entityId]/records/page.tsx +8 -4
- package/src/modules/entities/components/UserEntitiesTable.tsx +8 -0
- package/src/modules/entities/lib/entityAcl.ts +108 -0
- package/src/modules/entities/lib/helpers.ts +40 -5
- package/src/modules/feature_toggles/api/global/[id]/override/route.ts +18 -13
- package/src/modules/feature_toggles/api/global/[id]/route.ts +1 -0
- package/src/modules/feature_toggles/api/overrides/route.ts +26 -9
- package/src/modules/feature_toggles/backend/feature-toggles/global/[id]/edit/page.tsx +42 -15
- package/src/modules/feature_toggles/commands/global.ts +135 -10
- package/src/modules/feature_toggles/commands/overrides.ts +8 -6
- package/src/modules/feature_toggles/components/FeatureToggleOverrideCard.tsx +22 -12
- package/src/modules/feature_toggles/components/FeatureTogglesTable.tsx +15 -2
- package/src/modules/feature_toggles/components/formConfig.tsx +2 -1
- package/src/modules/feature_toggles/components/overrideFormConfig.tsx +10 -1
- package/src/modules/feature_toggles/data/validators.ts +11 -3
- package/src/modules/feature_toggles/lib/queries.ts +5 -5
- package/src/modules/feature_toggles/lib/utils.ts +3 -3
- package/src/modules/inbox_ops/api/proposals/[id]/replies/[replyId]/send/route.ts +10 -1
- package/src/modules/inbox_ops/api/settings/route.ts +25 -0
- package/src/modules/inbox_ops/api/webhook/inbound.ts +99 -18
- package/src/modules/inbox_ops/backend/inbox-ops/settings/page.tsx +15 -10
- package/src/modules/inbox_ops/components/proposals/EditActionDialog.tsx +9 -4
- package/src/modules/inbox_ops/data/entities.ts +7 -0
- package/src/modules/inbox_ops/data/validators.ts +4 -0
- package/src/modules/inbox_ops/encryption.ts +9 -0
- package/src/modules/inbox_ops/i18n/de.json +2 -0
- package/src/modules/inbox_ops/i18n/en.json +2 -0
- package/src/modules/inbox_ops/i18n/es.json +2 -0
- package/src/modules/inbox_ops/i18n/pl.json +2 -0
- package/src/modules/inbox_ops/lib/executionEngine.ts +6 -1
- package/src/modules/inbox_ops/lib/executionHelpers.ts +8 -7
- package/src/modules/inbox_ops/lib/priceValidator.ts +82 -7
- package/src/modules/inbox_ops/lib/rateLimiter.ts +84 -28
- package/src/modules/inbox_ops/lib/translationProvider.ts +37 -5
- package/src/modules/inbox_ops/migrations/.snapshot-open-mercato.json +16 -0
- package/src/modules/inbox_ops/migrations/.snapshot-openmercato.json +16 -0
- package/src/modules/inbox_ops/migrations/Migration20260607205834.ts +13 -0
- package/src/modules/inbox_ops/subscribers/extractionWorker.ts +5 -2
- package/src/modules/integrations/AGENTS.md +9 -0
- package/src/modules/integrations/api/[id]/credentials/route.ts +12 -0
- package/src/modules/integrations/backend/integrations/[id]/page.tsx +13 -0
- package/src/modules/integrations/backend/integrations/bundle/[id]/page.tsx +20 -11
- package/src/modules/integrations/backend/integrations/page.tsx +13 -8
- package/src/modules/integrations/data/entities.ts +21 -1
- package/src/modules/integrations/i18n/de.json +3 -0
- package/src/modules/integrations/i18n/en.json +3 -0
- package/src/modules/integrations/i18n/es.json +3 -0
- package/src/modules/integrations/i18n/pl.json +3 -0
- package/src/modules/integrations/lib/credentials-field-validation.ts +53 -0
- package/src/modules/integrations/lib/credentials-service.ts +49 -13
- package/src/modules/integrations/migrations/.snapshot-open-mercato.json +26 -1
- package/src/modules/integrations/migrations/Migration20260526154136_integrations.ts +15 -0
- package/src/modules/messages/api/route.ts +2 -2
- package/src/modules/messages/api/token/[token]/route.ts +1 -3
- package/src/modules/messages/commands/actions.ts +8 -6
- package/src/modules/messages/commands/messages.ts +128 -23
- package/src/modules/messages/commands/tokens.ts +1 -3
- package/src/modules/messages/components/ComposeMessagePageClient.tsx +17 -0
- package/src/modules/messages/components/MessageDetailPageClient.tsx +43 -0
- package/src/modules/messages/components/MessagesInboxPageClient.tsx +12 -0
- package/src/modules/messages/components/message-detail/hooks/useMessageDetailsActions.ts +6 -2
- package/src/modules/messages/components/message-detail/utils.ts +18 -0
- package/src/modules/messages/data/entities.ts +11 -0
- package/src/modules/messages/data/validators.ts +8 -1
- package/src/modules/messages/lib/actions.ts +9 -3
- package/src/modules/messages/migrations/.snapshot-open-mercato.json +18 -0
- package/src/modules/messages/migrations/Migration20260531130000.ts +15 -0
- package/src/modules/messages/widgets/injection-table.ts +29 -0
- package/src/modules/notifications/api/[id]/action/route.ts +13 -2
- package/src/modules/notifications/api/route.ts +17 -4
- package/src/modules/notifications/lib/notificationService.ts +31 -21
- package/src/modules/notifications/lib/routeHelpers.ts +49 -8
- package/src/modules/payment_gateways/api/status/route.ts +6 -0
- package/src/modules/perspectives/api/[tableId]/route.ts +11 -2
- package/src/modules/perspectives/services/perspectiveService.ts +13 -1
- package/src/modules/planner/backend/planner/availability-rulesets/[id]/page.tsx +16 -14
- package/src/modules/planner/backend/planner/availability-rulesets/page.tsx +6 -3
- package/src/modules/planner/commands/availability-rule-sets.ts +19 -0
- package/src/modules/planner/commands/availability.ts +9 -0
- package/src/modules/planner/components/AvailabilityRuleSetForm.tsx +4 -1
- package/src/modules/planner/components/AvailabilityRulesEditor.tsx +74 -17
- package/src/modules/planner/components/AvailabilitySchedule.tsx +22 -7
- package/src/modules/portal/frontend/[orgSlug]/portal/login/page.tsx +15 -16
- package/src/modules/portal/frontend/[orgSlug]/portal/page.tsx +11 -31
- package/src/modules/portal/frontend/[orgSlug]/portal/reset-password/page.tsx +8 -3
- package/src/modules/portal/frontend/[orgSlug]/portal/signup/page.tsx +12 -9
- package/src/modules/progress/api/jobs/[id]/route.ts +7 -0
- package/src/modules/query_index/api/openapi.ts +2 -1
- package/src/modules/query_index/api/reindex.ts +4 -0
- package/src/modules/query_index/data/entities.ts +10 -0
- package/src/modules/query_index/lib/engine.ts +66 -6
- package/src/modules/query_index/lib/indexer.ts +71 -24
- package/src/modules/query_index/lib/jobs.ts +57 -0
- package/src/modules/query_index/lib/reindexer.ts +41 -15
- package/src/modules/query_index/lib/search-tokens.ts +4 -4
- package/src/modules/query_index/migrations/.snapshot-open-mercato.json +9 -0
- package/src/modules/query_index/migrations/Migration20260606205453_query_index.ts +39 -0
- package/src/modules/query_index/subscribers/delete_one.ts +36 -16
- package/src/modules/query_index/subscribers/upsert_one.ts +44 -15
- package/src/modules/resources/backend/resources/resource-types/[id]/edit/page.tsx +28 -3
- package/src/modules/resources/backend/resources/resource-types/page.tsx +6 -3
- package/src/modules/resources/backend/resources/resources/[id]/page.tsx +62 -4
- package/src/modules/resources/backend/resources/resources/page.tsx +23 -4
- package/src/modules/resources/commands/activities.ts +67 -0
- package/src/modules/resources/commands/comments.ts +20 -0
- package/src/modules/resources/commands/resource-types.ts +21 -0
- package/src/modules/resources/commands/resources.ts +89 -0
- package/src/modules/resources/commands/tag-assignments.ts +55 -0
- package/src/modules/resources/components/ResourceCrudForm.tsx +103 -56
- package/src/modules/resources/components/ResourceTypeCrudForm.tsx +3 -1
- package/src/modules/resources/i18n/de.json +1 -0
- package/src/modules/resources/i18n/en.json +1 -0
- package/src/modules/resources/i18n/es.json +1 -0
- package/src/modules/resources/i18n/pl.json +1 -0
- package/src/modules/sales/api/documents/factory.ts +13 -1
- package/src/modules/sales/api/shipping-methods/route.ts +4 -0
- package/src/modules/sales/api/tax-rates/route.ts +4 -0
- package/src/modules/sales/backend/sales/channels/[channelId]/edit/page.tsx +6 -0
- package/src/modules/sales/backend/sales/channels/offers/page.tsx +10 -4
- package/src/modules/sales/backend/sales/channels/page.tsx +19 -4
- package/src/modules/sales/backend/sales/documents/[id]/page.tsx +75 -20
- package/src/modules/sales/backend/sales/documents/create/page.tsx +2 -0
- package/src/modules/sales/commands/configuration.ts +108 -2
- package/src/modules/sales/commands/documentAddresses.ts +226 -4
- package/src/modules/sales/commands/documents.ts +41 -9
- package/src/modules/sales/commands/notes.ts +39 -0
- package/src/modules/sales/commands/payments.ts +160 -14
- package/src/modules/sales/commands/returns.ts +194 -3
- package/src/modules/sales/commands/shared.ts +36 -0
- package/src/modules/sales/commands/shipments.ts +63 -6
- package/src/modules/sales/components/AdjustmentKindSettings.tsx +20 -11
- package/src/modules/sales/components/DocumentNumberSettings.tsx +1 -0
- package/src/modules/sales/components/OrderEditingSettings.tsx +1 -0
- package/src/modules/sales/components/PaymentMethodsSettings.tsx +12 -4
- package/src/modules/sales/components/ShippingMethodsSettings.tsx +12 -4
- package/src/modules/sales/components/StatusSettings.tsx +20 -11
- package/src/modules/sales/components/TaxRatesSettings.tsx +12 -5
- package/src/modules/sales/components/channels/ChannelOfferForm.tsx +85 -20
- package/src/modules/sales/components/channels/SalesChannelOffersPanel.tsx +10 -4
- package/src/modules/sales/components/documents/AddressesSection.tsx +116 -48
- package/src/modules/sales/components/documents/AdjustmentDialog.tsx +61 -18
- package/src/modules/sales/components/documents/AdjustmentsSection.tsx +50 -25
- package/src/modules/sales/components/documents/ItemsSection.tsx +25 -14
- package/src/modules/sales/components/documents/LineItemDialog.tsx +107 -36
- package/src/modules/sales/components/documents/PaymentDialog.tsx +33 -14
- package/src/modules/sales/components/documents/PaymentsSection.tsx +22 -10
- package/src/modules/sales/components/documents/ReturnDialog.tsx +28 -17
- package/src/modules/sales/components/documents/ReturnsSection.tsx +4 -1
- package/src/modules/sales/components/documents/SalesDocumentForm.tsx +1 -0
- package/src/modules/sales/components/documents/SalesDocumentsTable.tsx +12 -5
- package/src/modules/sales/components/documents/ShipmentDialog.tsx +132 -22
- package/src/modules/sales/components/documents/ShipmentsSection.tsx +20 -10
- package/src/modules/sales/components/documents/optimisticLock.ts +34 -0
- package/src/modules/sales/components/documents/shipmentTypes.ts +1 -0
- package/src/modules/sales/di.ts +35 -0
- package/src/modules/sales/frontend/quote/[token]/page.tsx +8 -1
- package/src/modules/sales/i18n/de.json +4 -0
- package/src/modules/sales/i18n/en.json +4 -0
- package/src/modules/sales/i18n/es.json +4 -0
- package/src/modules/sales/i18n/pl.json +4 -0
- package/src/modules/sales/lib/calculations.ts +32 -4
- package/src/modules/sales/lib/dictionaries.ts +3 -2
- package/src/modules/sales/lib/makeStatusDictionaryRoute.ts +2 -0
- package/src/modules/shipping_carriers/api/cancel/route.ts +2 -2
- package/src/modules/shipping_carriers/lib/status-sync.ts +19 -0
- package/src/modules/staff/api/job-histories.ts +12 -2
- package/src/modules/staff/api/timesheets/time-entries/route.ts +16 -4
- package/src/modules/staff/backend/staff/leave-requests/[id]/page.tsx +12 -7
- package/src/modules/staff/backend/staff/leave-requests/page.tsx +8 -1
- package/src/modules/staff/backend/staff/my-leave-requests/[id]/page.tsx +2 -0
- package/src/modules/staff/backend/staff/profile/create/page.tsx +1 -1
- package/src/modules/staff/backend/staff/team-members/[id]/page.tsx +16 -5
- package/src/modules/staff/backend/staff/team-members/page.tsx +18 -25
- package/src/modules/staff/backend/staff/team-roles/[id]/edit/page.tsx +61 -6
- package/src/modules/staff/backend/staff/team-roles/page.tsx +22 -34
- package/src/modules/staff/backend/staff/teams/[id]/edit/page.tsx +13 -3
- package/src/modules/staff/backend/staff/teams/page.tsx +24 -15
- package/src/modules/staff/backend/staff/timesheets/page.tsx +10 -1
- package/src/modules/staff/backend/staff/timesheets/projects/[id]/edit/page.tsx +22 -3
- package/src/modules/staff/backend/staff/timesheets/projects/[id]/page.tsx +37 -7
- package/src/modules/staff/backend/staff/timesheets/projects/page.tsx +9 -3
- package/src/modules/staff/commands/activities.ts +56 -0
- package/src/modules/staff/commands/addresses.ts +72 -0
- package/src/modules/staff/commands/comments.ts +25 -0
- package/src/modules/staff/commands/job-histories.ts +92 -3
- package/src/modules/staff/commands/leave-requests.ts +40 -0
- package/src/modules/staff/commands/tag-assignments.ts +35 -0
- package/src/modules/staff/commands/team-members.ts +45 -0
- package/src/modules/staff/commands/team-roles.ts +24 -0
- package/src/modules/staff/commands/teams.ts +24 -0
- package/src/modules/staff/commands/timesheets-entries.ts +31 -0
- package/src/modules/staff/commands/timesheets-projects.ts +56 -0
- package/src/modules/staff/components/LeaveRequestForm.tsx +3 -0
- package/src/modules/staff/components/TeamForm.tsx +3 -1
- package/src/modules/staff/components/TeamMemberForm.tsx +36 -3
- package/src/modules/staff/components/TeamRoleForm.tsx +5 -3
- package/src/modules/staff/components/detail/JobHistorySection.tsx +28 -6
- package/src/modules/staff/data/validators.ts +6 -0
- package/src/modules/staff/i18n/de.json +2 -0
- package/src/modules/staff/i18n/en.json +2 -0
- package/src/modules/staff/i18n/es.json +2 -0
- package/src/modules/staff/i18n/pl.json +2 -0
- package/src/modules/staff/lib/leaveRequestHelpers.ts +4 -0
- package/src/modules/translations/api/[entityType]/[entityId]/route.ts +9 -1
- package/src/modules/translations/components/TranslationManager.tsx +13 -8
- package/src/modules/workflows/acl.ts +102 -17
- package/src/modules/workflows/api/definitions/[id]/route.ts +112 -0
- package/src/modules/workflows/backend/definitions/[id]/page.tsx +20 -4
- package/src/modules/workflows/backend/definitions/page.tsx +34 -15
- package/src/modules/workflows/backend/definitions/visual-editor/page.tsx +29 -16
- package/src/modules/workflows/components/NodeEditDialog.tsx +2 -0
- package/src/modules/workflows/components/WorkflowGraphImpl.tsx +3 -1
- package/src/modules/workflows/components/formConfig.tsx +37 -17
- package/src/modules/workflows/components/nodes/ParallelForkNode.tsx +66 -0
- package/src/modules/workflows/components/nodes/ParallelJoinNode.tsx +66 -0
- package/src/modules/workflows/components/nodes/index.ts +6 -0
- package/src/modules/workflows/data/entities.ts +109 -0
- package/src/modules/workflows/data/validators.ts +223 -0
- package/src/modules/workflows/di.ts +20 -0
- package/src/modules/workflows/events.ts +7 -0
- package/src/modules/workflows/i18n/de.json +13 -0
- package/src/modules/workflows/i18n/en.json +13 -0
- package/src/modules/workflows/i18n/es.json +13 -0
- package/src/modules/workflows/i18n/pl.json +13 -0
- package/src/modules/workflows/lib/activity-executor.ts +8 -2
- package/src/modules/workflows/lib/activity-queue-types.ts +3 -0
- package/src/modules/workflows/lib/event-logger.ts +3 -0
- package/src/modules/workflows/lib/execution-token.ts +166 -0
- package/src/modules/workflows/lib/node-type-icons.ts +11 -2
- package/src/modules/workflows/lib/parallel-handler.ts +575 -0
- package/src/modules/workflows/lib/signal-handler.ts +72 -1
- package/src/modules/workflows/lib/step-handler.ts +94 -34
- package/src/modules/workflows/lib/task-handler.ts +32 -0
- package/src/modules/workflows/lib/timer-handler.ts +30 -1
- package/src/modules/workflows/lib/transition-handler.ts +56 -24
- package/src/modules/workflows/lib/workflow-executor.ts +70 -1
- package/src/modules/workflows/migrations/.snapshot-open-mercato.json +263 -0
- package/src/modules/workflows/migrations/Migration20260602120000.ts +25 -0
- package/src/modules/workflows/workers/workflow-activities.worker.ts +9 -4
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/modules/inbox_ops/data/validators.ts"],
|
|
4
|
-
"sourcesContent": ["import { z } from 'zod'\n\nconst uuid = () => z.string().uuid()\nconst coerceNumericString = z.preprocess(\n (val) => (typeof val === 'number' ? String(val) : val),\n z.string().regex(/^\\d+(\\.\\d+)?$/),\n)\n\n// ---------------------------------------------------------------------------\n// Action Payload Schemas\n// ---------------------------------------------------------------------------\n\nconst addressSchema = z.object({\n line1: z.string().trim().max(500).optional(),\n line2: z.string().trim().max(500).optional(),\n city: z.string().trim().max(200).optional(),\n state: z.string().trim().max(200).optional(),\n postalCode: z.string().trim().max(50).optional(),\n country: z.string().trim().max(200).optional(),\n company: z.string().trim().max(300).optional(),\n contactName: z.string().trim().max(300).optional(),\n}).optional()\n\nexport const orderPayloadSchema = z.object({\n customerEntityId: uuid().optional(),\n customerName: z.string().trim().min(1).max(300),\n customerEmail: z.string().trim().email().max(320).optional(),\n channelId: uuid().optional(),\n currencyCode: z.string().trim().length(3),\n taxRateId: uuid().optional(),\n lineItems: z.array(z.object({\n productName: z.string().trim().min(1).max(300),\n productId: uuid().optional(),\n variantId: uuid().optional(),\n sku: z.string().trim().max(100).optional(),\n quantity: coerceNumericString,\n unitPrice: coerceNumericString.optional(),\n catalogPrice: z.string().optional(),\n kind: z.enum(['product', 'service']).default('product'),\n description: z.string().trim().max(2000).optional(),\n })).min(1).max(100),\n requestedDeliveryDate: z.string().optional(),\n notes: z.string().trim().max(4000).optional(),\n customerReference: z.string().trim().max(200).optional(),\n shippingAddress: addressSchema,\n billingAddress: addressSchema,\n shippingAddressId: uuid().optional(),\n billingAddressId: uuid().optional(),\n})\n\nexport const updateOrderPayloadSchema = z\n .object({\n orderId: uuid().optional(),\n orderNumber: z.string().trim().max(100).optional(),\n quantityChanges: z.array(z.object({\n lineItemName: z.string().trim().min(1).max(300),\n lineItemId: uuid().optional(),\n oldQuantity: z.string().optional(),\n newQuantity: z.string().regex(/^\\d+(\\.\\d+)?$/),\n })).optional(),\n deliveryDateChange: z.object({\n oldDate: z.string().optional(),\n newDate: z.string(),\n }).optional(),\n noteAdditions: z.array(z.string().trim().max(4000)).optional(),\n })\n .refine((value) => Boolean(value.orderId || value.orderNumber), {\n message: 'order_reference_required',\n })\n\nexport const updateShipmentPayloadSchema = z\n .object({\n orderId: uuid().optional(),\n orderNumber: z.string().trim().max(100).optional(),\n trackingNumbers: z.array(z.string().trim().max(200)).optional(),\n carrierName: z.string().trim().max(200).optional(),\n statusLabel: z.string().trim().min(1).max(200),\n shippedAt: z.string().optional(),\n deliveredAt: z.string().optional(),\n estimatedDelivery: z.string().optional(),\n notes: z.string().trim().max(4000).optional(),\n })\n .refine((value) => Boolean(value.orderId || value.orderNumber), {\n message: 'order_reference_required',\n })\n\nconst lowercaseContactType = z.preprocess(\n (val) => (typeof val === 'string' ? val.toLowerCase() : val),\n z.enum(['person', 'company']),\n)\n\nexport const createContactPayloadSchema = z.object({\n type: lowercaseContactType,\n name: z.string().trim().min(1).max(300),\n email: z.string().trim().email().max(320).optional(),\n phone: z.string().trim().max(50).optional(),\n companyName: z.string().trim().max(300).optional(),\n role: z.string().trim().max(150).optional(),\n source: z.literal('inbox_ops').default('inbox_ops'),\n})\n\nexport const linkContactPayloadSchema = z.object({\n emailAddress: z.string().trim().email().max(320),\n contactId: uuid(),\n contactType: lowercaseContactType,\n contactName: z.string().trim().min(1).max(300),\n})\n\nexport const createProductPayloadSchema = z.object({\n title: z.string().trim().min(1).max(255),\n sku: z.string().trim().max(100).optional(),\n unitPrice: coerceNumericString.optional(),\n currencyCode: z.string().trim().length(3).optional(),\n kind: z.enum(['product', 'service']).default('product'),\n description: z.string().trim().max(4000).optional(),\n})\n\nexport const logActivityPayloadSchema = z.object({\n contactId: uuid().optional(),\n contactType: lowercaseContactType,\n contactName: z.string().trim().min(1).max(300),\n activityType: z.enum(['email', 'call', 'meeting', 'note']),\n subject: z.string().trim().min(1).max(200),\n body: z.string().trim().max(8000),\n})\n\nexport const draftReplyPayloadSchema = z.object({\n to: z.string().trim().email().max(320),\n toName: z.string().trim().max(300).optional().nullable(),\n replyTo: z.string().trim().email().max(320).optional().nullable(),\n subject: z.string().trim().min(1).max(500),\n body: z.string().trim().min(1).max(10000),\n inReplyToMessageId: z.string().trim().max(500).optional().nullable(),\n references: z.array(z.string().trim().max(500)).optional().nullable(),\n context: z.string().trim().max(4000).optional().nullable(),\n})\n\n// ---------------------------------------------------------------------------\n// Category\n// ---------------------------------------------------------------------------\n\nexport const inboxProposalCategoryEnum = z.enum([\n 'rfq',\n 'order',\n 'order_update',\n 'complaint',\n 'shipping_update',\n 'inquiry',\n 'payment',\n 'other',\n])\n\nexport const ALL_CATEGORIES = inboxProposalCategoryEnum.options\n\nconst proposalCategoryFilterSchema = z.string()\n .trim()\n .max(200)\n .refine((value) => {\n const categories = value.split(',').map((category) => category.trim()).filter(Boolean)\n return (\n categories.length > 0 &&\n categories.every((category) => inboxProposalCategoryEnum.safeParse(category).success)\n )\n }, {\n message: `Category filter must contain only: ${ALL_CATEGORIES.join(', ')}`,\n })\n\nexport const categorizeProposalSchema = z.object({\n category: inboxProposalCategoryEnum,\n})\n\n// ---------------------------------------------------------------------------\n// LLM Extraction Output Schema\n// ---------------------------------------------------------------------------\n\nexport const extractedParticipantSchema = z.object({\n name: z.string(),\n email: z.string(),\n role: z.enum(['buyer', 'seller', 'logistics', 'finance', 'other']),\n})\n\nexport const extractedActionSchema = z.object({\n actionType: z.enum([\n 'create_order',\n 'create_quote',\n 'update_order',\n 'update_shipment',\n 'create_contact',\n 'create_product',\n 'link_contact',\n 'log_activity',\n 'draft_reply',\n ]),\n description: z.string(),\n confidence: z.number(),\n requiredFeature: z.string().optional(),\n payloadJson: z.string().describe('JSON-encoded payload object for this action'),\n})\n\nexport const extractedDiscrepancySchema = z.object({\n type: z.enum([\n 'price_mismatch',\n 'quantity_mismatch',\n 'unknown_contact',\n 'currency_mismatch',\n 'date_conflict',\n 'product_not_found',\n 'duplicate_order',\n 'other',\n ]),\n severity: z.enum(['warning', 'error']),\n description: z.string(),\n expectedValue: z.string().optional(),\n foundValue: z.string().optional(),\n actionIndex: z.number().optional(),\n})\n\nexport const extractionOutputSchema = z.object({\n summary: z.string(),\n category: inboxProposalCategoryEnum.optional(),\n participants: z.array(extractedParticipantSchema),\n proposedActions: z.array(extractedActionSchema),\n discrepancies: z.array(extractedDiscrepancySchema),\n draftReplies: z.array(z.object({\n to: z.string(),\n toName: z.string().optional(),\n subject: z.string(),\n body: z.string(),\n context: z.string().optional(),\n })),\n confidence: z.number(),\n detectedLanguage: z.string().optional(),\n possiblyIncomplete: z.boolean().optional(),\n})\n\nexport type ExtractionOutput = z.infer<typeof extractionOutputSchema>\nexport type OrderPayload = z.infer<typeof orderPayloadSchema>\nexport type UpdateOrderPayload = z.infer<typeof updateOrderPayloadSchema>\nexport type UpdateShipmentPayload = z.infer<typeof updateShipmentPayloadSchema>\nexport type CreateContactPayload = z.infer<typeof createContactPayloadSchema>\nexport type CreateProductPayload = z.infer<typeof createProductPayloadSchema>\nexport type LinkContactPayload = z.infer<typeof linkContactPayloadSchema>\nexport type LogActivityPayload = z.infer<typeof logActivityPayloadSchema>\nexport type DraftReplyPayload = z.infer<typeof draftReplyPayloadSchema>\n\n// ---------------------------------------------------------------------------\n// Translation / Settings Schemas\n// ---------------------------------------------------------------------------\n\nexport const translateProposalSchema = z.object({\n targetLocale: z.enum(['en', 'de', 'es', 'pl']),\n})\n\nexport const updateSettingsSchema = z.object({\n workingLanguage: z.enum(['en', 'de', 'es', 'pl']).optional(),\n isActive: z.boolean().optional(),\n})\n\n// ---------------------------------------------------------------------------\n// API Query Schemas\n// ---------------------------------------------------------------------------\n\nexport const proposalListQuerySchema = z.object({\n status: z.enum(['pending', 'partial', 'accepted', 'rejected']).optional(),\n category: proposalCategoryFilterSchema.optional(),\n search: z.string().trim().max(200).optional(),\n page: z.coerce.number().int().min(1).default(1),\n pageSize: z.coerce.number().int().min(1).max(100).default(25),\n})\n\nexport const emailListQuerySchema = z.object({\n status: z.enum(['received', 'processing', 'processed', 'needs_review', 'failed']).optional(),\n page: z.coerce.number().int().min(1).default(1),\n pageSize: z.coerce.number().int().min(1).max(100).default(25),\n})\n\nexport const actionEditSchema = z.object({\n payload: z.record(z.string(), z.unknown()),\n})\n\nconst ACTION_PAYLOAD_SCHEMAS: Record<string, z.ZodType> = {\n create_order: orderPayloadSchema,\n create_quote: orderPayloadSchema,\n update_order: updateOrderPayloadSchema,\n update_shipment: updateShipmentPayloadSchema,\n create_contact: createContactPayloadSchema,\n create_product: createProductPayloadSchema,\n link_contact: linkContactPayloadSchema,\n log_activity: logActivityPayloadSchema,\n draft_reply: draftReplyPayloadSchema,\n}\n\nexport function validateActionPayloadForType(\n actionType: string,\n payload: Record<string, unknown>,\n): { success: true } | { success: false; error: string } {\n const schema = ACTION_PAYLOAD_SCHEMAS[actionType]\n if (!schema) {\n return { success: true }\n }\n\n const result = schema.safeParse(payload)\n if (!result.success) {\n const issues = result.error.issues.map((i) => `${i.path.join('.')}: ${i.message}`).join('; ')\n return { success: false, error: `Invalid payload for ${actionType}: ${issues}` }\n }\n return { success: true }\n}\n"],
|
|
5
|
-
"mappings": "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
|
|
4
|
+
"sourcesContent": ["import { z } from 'zod'\n\nconst uuid = () => z.string().uuid()\nconst coerceNumericString = z.preprocess(\n (val) => (typeof val === 'number' ? String(val) : val),\n z.string().regex(/^\\d+(\\.\\d+)?$/),\n)\n\n// ---------------------------------------------------------------------------\n// Action Payload Schemas\n// ---------------------------------------------------------------------------\n\nconst addressSchema = z.object({\n line1: z.string().trim().max(500).optional(),\n line2: z.string().trim().max(500).optional(),\n city: z.string().trim().max(200).optional(),\n state: z.string().trim().max(200).optional(),\n postalCode: z.string().trim().max(50).optional(),\n country: z.string().trim().max(200).optional(),\n company: z.string().trim().max(300).optional(),\n contactName: z.string().trim().max(300).optional(),\n}).optional()\n\nexport const orderPayloadSchema = z.object({\n customerEntityId: uuid().optional(),\n customerName: z.string().trim().min(1).max(300),\n customerEmail: z.string().trim().email().max(320).optional(),\n channelId: uuid().optional(),\n currencyCode: z.string().trim().length(3),\n taxRateId: uuid().optional(),\n lineItems: z.array(z.object({\n productName: z.string().trim().min(1).max(300),\n productId: uuid().optional(),\n variantId: uuid().optional(),\n sku: z.string().trim().max(100).optional(),\n quantity: coerceNumericString,\n unitPrice: coerceNumericString.optional(),\n catalogPrice: z.string().optional(),\n kind: z.enum(['product', 'service']).default('product'),\n description: z.string().trim().max(2000).optional(),\n })).min(1).max(100),\n requestedDeliveryDate: z.string().optional(),\n notes: z.string().trim().max(4000).optional(),\n customerReference: z.string().trim().max(200).optional(),\n shippingAddress: addressSchema,\n billingAddress: addressSchema,\n shippingAddressId: uuid().optional(),\n billingAddressId: uuid().optional(),\n})\n\nexport const updateOrderPayloadSchema = z\n .object({\n orderId: uuid().optional(),\n orderNumber: z.string().trim().max(100).optional(),\n quantityChanges: z.array(z.object({\n lineItemName: z.string().trim().min(1).max(300),\n lineItemId: uuid().optional(),\n oldQuantity: z.string().optional(),\n newQuantity: z.string().regex(/^\\d+(\\.\\d+)?$/),\n })).optional(),\n deliveryDateChange: z.object({\n oldDate: z.string().optional(),\n newDate: z.string(),\n }).optional(),\n noteAdditions: z.array(z.string().trim().max(4000)).optional(),\n })\n .refine((value) => Boolean(value.orderId || value.orderNumber), {\n message: 'order_reference_required',\n })\n\nexport const updateShipmentPayloadSchema = z\n .object({\n orderId: uuid().optional(),\n orderNumber: z.string().trim().max(100).optional(),\n trackingNumbers: z.array(z.string().trim().max(200)).optional(),\n carrierName: z.string().trim().max(200).optional(),\n statusLabel: z.string().trim().min(1).max(200),\n shippedAt: z.string().optional(),\n deliveredAt: z.string().optional(),\n estimatedDelivery: z.string().optional(),\n notes: z.string().trim().max(4000).optional(),\n })\n .refine((value) => Boolean(value.orderId || value.orderNumber), {\n message: 'order_reference_required',\n })\n\nconst lowercaseContactType = z.preprocess(\n (val) => (typeof val === 'string' ? val.toLowerCase() : val),\n z.enum(['person', 'company']),\n)\n\nexport const createContactPayloadSchema = z.object({\n type: lowercaseContactType,\n name: z.string().trim().min(1).max(300),\n email: z.string().trim().email().max(320).optional(),\n phone: z.string().trim().max(50).optional(),\n companyName: z.string().trim().max(300).optional(),\n role: z.string().trim().max(150).optional(),\n source: z.literal('inbox_ops').default('inbox_ops'),\n})\n\nexport const linkContactPayloadSchema = z.object({\n emailAddress: z.string().trim().email().max(320),\n contactId: uuid(),\n contactType: lowercaseContactType,\n contactName: z.string().trim().min(1).max(300),\n})\n\nexport const createProductPayloadSchema = z.object({\n title: z.string().trim().min(1).max(255),\n sku: z.string().trim().max(100).optional(),\n unitPrice: coerceNumericString.optional(),\n currencyCode: z.string().trim().length(3).optional(),\n kind: z.enum(['product', 'service']).default('product'),\n description: z.string().trim().max(4000).optional(),\n})\n\nexport const logActivityPayloadSchema = z.object({\n contactId: uuid().optional(),\n contactType: lowercaseContactType,\n contactName: z.string().trim().min(1).max(300),\n activityType: z.enum(['email', 'call', 'meeting', 'note']),\n subject: z.string().trim().min(1).max(200),\n body: z.string().trim().max(8000),\n})\n\nexport const draftReplyPayloadSchema = z.object({\n to: z.string().trim().email().max(320),\n toName: z.string().trim().max(300).optional().nullable(),\n replyTo: z.string().trim().email().max(320).optional().nullable(),\n subject: z.string().trim().min(1).max(500),\n body: z.string().trim().min(1).max(10000),\n inReplyToMessageId: z.string().trim().max(500).optional().nullable(),\n references: z.array(z.string().trim().max(500)).optional().nullable(),\n context: z.string().trim().max(4000).optional().nullable(),\n})\n\n// ---------------------------------------------------------------------------\n// Category\n// ---------------------------------------------------------------------------\n\nexport const inboxProposalCategoryEnum = z.enum([\n 'rfq',\n 'order',\n 'order_update',\n 'complaint',\n 'shipping_update',\n 'inquiry',\n 'payment',\n 'other',\n])\n\nexport const ALL_CATEGORIES = inboxProposalCategoryEnum.options\n\nconst proposalCategoryFilterSchema = z.string()\n .trim()\n .max(200)\n .refine((value) => {\n const categories = value.split(',').map((category) => category.trim()).filter(Boolean)\n return (\n categories.length > 0 &&\n categories.every((category) => inboxProposalCategoryEnum.safeParse(category).success)\n )\n }, {\n message: `Category filter must contain only: ${ALL_CATEGORIES.join(', ')}`,\n })\n\nexport const categorizeProposalSchema = z.object({\n category: inboxProposalCategoryEnum,\n})\n\n// ---------------------------------------------------------------------------\n// LLM Extraction Output Schema\n// ---------------------------------------------------------------------------\n\nexport const extractedParticipantSchema = z.object({\n name: z.string(),\n email: z.string(),\n role: z.enum(['buyer', 'seller', 'logistics', 'finance', 'other']),\n})\n\nexport const extractedActionSchema = z.object({\n actionType: z.enum([\n 'create_order',\n 'create_quote',\n 'update_order',\n 'update_shipment',\n 'create_contact',\n 'create_product',\n 'link_contact',\n 'log_activity',\n 'draft_reply',\n ]),\n description: z.string(),\n confidence: z.number(),\n requiredFeature: z.string().optional(),\n payloadJson: z.string().describe('JSON-encoded payload object for this action'),\n})\n\nexport const extractedDiscrepancySchema = z.object({\n type: z.enum([\n 'price_mismatch',\n 'quantity_mismatch',\n 'unknown_contact',\n 'currency_mismatch',\n 'date_conflict',\n 'product_not_found',\n 'duplicate_order',\n 'other',\n ]),\n severity: z.enum(['warning', 'error']),\n description: z.string(),\n expectedValue: z.string().optional(),\n foundValue: z.string().optional(),\n actionIndex: z.number().optional(),\n})\n\nexport const extractionOutputSchema = z.object({\n summary: z.string(),\n category: inboxProposalCategoryEnum.optional(),\n participants: z.array(extractedParticipantSchema),\n proposedActions: z.array(extractedActionSchema),\n discrepancies: z.array(extractedDiscrepancySchema),\n draftReplies: z.array(z.object({\n to: z.string(),\n toName: z.string().optional(),\n subject: z.string(),\n body: z.string(),\n context: z.string().optional(),\n })),\n confidence: z.number(),\n detectedLanguage: z.string().optional(),\n possiblyIncomplete: z.boolean().optional(),\n})\n\nexport type ExtractionOutput = z.infer<typeof extractionOutputSchema>\nexport type OrderPayload = z.infer<typeof orderPayloadSchema>\nexport type UpdateOrderPayload = z.infer<typeof updateOrderPayloadSchema>\nexport type UpdateShipmentPayload = z.infer<typeof updateShipmentPayloadSchema>\nexport type CreateContactPayload = z.infer<typeof createContactPayloadSchema>\nexport type CreateProductPayload = z.infer<typeof createProductPayloadSchema>\nexport type LinkContactPayload = z.infer<typeof linkContactPayloadSchema>\nexport type LogActivityPayload = z.infer<typeof logActivityPayloadSchema>\nexport type DraftReplyPayload = z.infer<typeof draftReplyPayloadSchema>\n\n// ---------------------------------------------------------------------------\n// Translation / Settings Schemas\n// ---------------------------------------------------------------------------\n\nexport const translateProposalSchema = z.object({\n targetLocale: z.enum(['en', 'de', 'es', 'pl']),\n})\n\nexport const updateSettingsSchema = z.object({\n workingLanguage: z.enum(['en', 'de', 'es', 'pl']).optional(),\n isActive: z.boolean().optional(),\n // Per-tenant inbound-webhook secret. Write-only: it is never returned by the\n // settings GET. A non-empty value binds custom-provider webhook signatures for\n // this inbox to this secret; null/empty clears it and reverts to the global key.\n webhookSecret: z.string().trim().min(16).max(256).nullable().optional(),\n})\n\n// ---------------------------------------------------------------------------\n// API Query Schemas\n// ---------------------------------------------------------------------------\n\nexport const proposalListQuerySchema = z.object({\n status: z.enum(['pending', 'partial', 'accepted', 'rejected']).optional(),\n category: proposalCategoryFilterSchema.optional(),\n search: z.string().trim().max(200).optional(),\n page: z.coerce.number().int().min(1).default(1),\n pageSize: z.coerce.number().int().min(1).max(100).default(25),\n})\n\nexport const emailListQuerySchema = z.object({\n status: z.enum(['received', 'processing', 'processed', 'needs_review', 'failed']).optional(),\n page: z.coerce.number().int().min(1).default(1),\n pageSize: z.coerce.number().int().min(1).max(100).default(25),\n})\n\nexport const actionEditSchema = z.object({\n payload: z.record(z.string(), z.unknown()),\n})\n\nconst ACTION_PAYLOAD_SCHEMAS: Record<string, z.ZodType> = {\n create_order: orderPayloadSchema,\n create_quote: orderPayloadSchema,\n update_order: updateOrderPayloadSchema,\n update_shipment: updateShipmentPayloadSchema,\n create_contact: createContactPayloadSchema,\n create_product: createProductPayloadSchema,\n link_contact: linkContactPayloadSchema,\n log_activity: logActivityPayloadSchema,\n draft_reply: draftReplyPayloadSchema,\n}\n\nexport function validateActionPayloadForType(\n actionType: string,\n payload: Record<string, unknown>,\n): { success: true } | { success: false; error: string } {\n const schema = ACTION_PAYLOAD_SCHEMAS[actionType]\n if (!schema) {\n return { success: true }\n }\n\n const result = schema.safeParse(payload)\n if (!result.success) {\n const issues = result.error.issues.map((i) => `${i.path.join('.')}: ${i.message}`).join('; ')\n return { success: false, error: `Invalid payload for ${actionType}: ${issues}` }\n }\n return { success: true }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,4 +1,13 @@
|
|
|
1
1
|
const defaultEncryptionMaps = [
|
|
2
|
+
{
|
|
3
|
+
// `webhook_secret` is a tenant-scoped signing credential, never used in a
|
|
4
|
+
// WHERE/ILIKE lookup (it is loaded via the settings row, then compared with
|
|
5
|
+
// timingSafeEqual), so it is safe — and required — to encrypt at rest.
|
|
6
|
+
entityId: "inbox_ops:inbox_settings",
|
|
7
|
+
fields: [
|
|
8
|
+
{ field: "webhook_secret" }
|
|
9
|
+
]
|
|
10
|
+
},
|
|
2
11
|
{
|
|
3
12
|
entityId: "inbox_ops:inbox_email",
|
|
4
13
|
fields: [
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/modules/inbox_ops/encryption.ts"],
|
|
4
|
-
"sourcesContent": ["import type { ModuleEncryptionMap } from '@open-mercato/shared/modules/encryption'\n\n// Message body, extracted business data, and correspondent identities all\n// count as tenant PII. Columns routed through WHERE/ILIKE lookups or UNIQUE\n// indexes (`inbox_settings.inbox_address`, `inbox_emails.message_id`,\n// `in_reply_to`, `references`, `*.metadata`) are intentionally left plaintext\n// for now \u2014 encrypting them requires paired `*_hash` columns plus rewriting\n// the inbound-webhook lookups, which is out of scope for this fix.\nexport const defaultEncryptionMaps: ModuleEncryptionMap[] = [\n {\n entityId: 'inbox_ops:inbox_email',\n fields: [\n { field: 'subject' },\n { field: 'raw_text' },\n { field: 'raw_html' },\n { field: 'cleaned_text' },\n { field: 'thread_messages' },\n { field: 'forwarded_by_address' },\n { field: 'forwarded_by_name' },\n { field: 'to_address' },\n { field: 'reply_to' },\n { field: 'processing_error' },\n ],\n },\n {\n entityId: 'inbox_ops:inbox_proposal',\n fields: [\n { field: 'summary' },\n { field: 'participants' },\n { field: 'translations' },\n ],\n },\n {\n entityId: 'inbox_ops:inbox_proposal_action',\n fields: [\n { field: 'description' },\n { field: 'payload' },\n { field: 'execution_error' },\n ],\n },\n {\n entityId: 'inbox_ops:inbox_discrepancy',\n fields: [\n { field: 'description' },\n { field: 'expected_value' },\n { field: 'found_value' },\n ],\n },\n]\n\nexport default defaultEncryptionMaps\n"],
|
|
5
|
-
"mappings": "AAQO,MAAM,wBAA+C;AAAA,EAC1D;AAAA,IACE,UAAU;AAAA,IACV,QAAQ;AAAA,MACN,EAAE,OAAO,UAAU;AAAA,MACnB,EAAE,OAAO,WAAW;AAAA,MACpB,EAAE,OAAO,WAAW;AAAA,MACpB,EAAE,OAAO,eAAe;AAAA,MACxB,EAAE,OAAO,kBAAkB;AAAA,MAC3B,EAAE,OAAO,uBAAuB;AAAA,MAChC,EAAE,OAAO,oBAAoB;AAAA,MAC7B,EAAE,OAAO,aAAa;AAAA,MACtB,EAAE,OAAO,WAAW;AAAA,MACpB,EAAE,OAAO,mBAAmB;AAAA,IAC9B;AAAA,EACF;AAAA,EACA;AAAA,IACE,UAAU;AAAA,IACV,QAAQ;AAAA,MACN,EAAE,OAAO,UAAU;AAAA,MACnB,EAAE,OAAO,eAAe;AAAA,MACxB,EAAE,OAAO,eAAe;AAAA,IAC1B;AAAA,EACF;AAAA,EACA;AAAA,IACE,UAAU;AAAA,IACV,QAAQ;AAAA,MACN,EAAE,OAAO,cAAc;AAAA,MACvB,EAAE,OAAO,UAAU;AAAA,MACnB,EAAE,OAAO,kBAAkB;AAAA,IAC7B;AAAA,EACF;AAAA,EACA;AAAA,IACE,UAAU;AAAA,IACV,QAAQ;AAAA,MACN,EAAE,OAAO,cAAc;AAAA,MACvB,EAAE,OAAO,iBAAiB;AAAA,MAC1B,EAAE,OAAO,cAAc;AAAA,IACzB;AAAA,EACF;AACF;AAEA,IAAO,qBAAQ;",
|
|
4
|
+
"sourcesContent": ["import type { ModuleEncryptionMap } from '@open-mercato/shared/modules/encryption'\n\n// Message body, extracted business data, and correspondent identities all\n// count as tenant PII. Columns routed through WHERE/ILIKE lookups or UNIQUE\n// indexes (`inbox_settings.inbox_address`, `inbox_emails.message_id`,\n// `in_reply_to`, `references`, `*.metadata`) are intentionally left plaintext\n// for now \u2014 encrypting them requires paired `*_hash` columns plus rewriting\n// the inbound-webhook lookups, which is out of scope for this fix.\nexport const defaultEncryptionMaps: ModuleEncryptionMap[] = [\n {\n // `webhook_secret` is a tenant-scoped signing credential, never used in a\n // WHERE/ILIKE lookup (it is loaded via the settings row, then compared with\n // timingSafeEqual), so it is safe \u2014 and required \u2014 to encrypt at rest.\n entityId: 'inbox_ops:inbox_settings',\n fields: [\n { field: 'webhook_secret' },\n ],\n },\n {\n entityId: 'inbox_ops:inbox_email',\n fields: [\n { field: 'subject' },\n { field: 'raw_text' },\n { field: 'raw_html' },\n { field: 'cleaned_text' },\n { field: 'thread_messages' },\n { field: 'forwarded_by_address' },\n { field: 'forwarded_by_name' },\n { field: 'to_address' },\n { field: 'reply_to' },\n { field: 'processing_error' },\n ],\n },\n {\n entityId: 'inbox_ops:inbox_proposal',\n fields: [\n { field: 'summary' },\n { field: 'participants' },\n { field: 'translations' },\n ],\n },\n {\n entityId: 'inbox_ops:inbox_proposal_action',\n fields: [\n { field: 'description' },\n { field: 'payload' },\n { field: 'execution_error' },\n ],\n },\n {\n entityId: 'inbox_ops:inbox_discrepancy',\n fields: [\n { field: 'description' },\n { field: 'expected_value' },\n { field: 'found_value' },\n ],\n },\n]\n\nexport default defaultEncryptionMaps\n"],
|
|
5
|
+
"mappings": "AAQO,MAAM,wBAA+C;AAAA,EAC1D;AAAA;AAAA;AAAA;AAAA,IAIE,UAAU;AAAA,IACV,QAAQ;AAAA,MACN,EAAE,OAAO,iBAAiB;AAAA,IAC5B;AAAA,EACF;AAAA,EACA;AAAA,IACE,UAAU;AAAA,IACV,QAAQ;AAAA,MACN,EAAE,OAAO,UAAU;AAAA,MACnB,EAAE,OAAO,WAAW;AAAA,MACpB,EAAE,OAAO,WAAW;AAAA,MACpB,EAAE,OAAO,eAAe;AAAA,MACxB,EAAE,OAAO,kBAAkB;AAAA,MAC3B,EAAE,OAAO,uBAAuB;AAAA,MAChC,EAAE,OAAO,oBAAoB;AAAA,MAC7B,EAAE,OAAO,aAAa;AAAA,MACtB,EAAE,OAAO,WAAW;AAAA,MACpB,EAAE,OAAO,mBAAmB;AAAA,IAC9B;AAAA,EACF;AAAA,EACA;AAAA,IACE,UAAU;AAAA,IACV,QAAQ;AAAA,MACN,EAAE,OAAO,UAAU;AAAA,MACnB,EAAE,OAAO,eAAe;AAAA,MACxB,EAAE,OAAO,eAAe;AAAA,IAC1B;AAAA,EACF;AAAA,EACA;AAAA,IACE,UAAU;AAAA,IACV,QAAQ;AAAA,MACN,EAAE,OAAO,cAAc;AAAA,MACvB,EAAE,OAAO,UAAU;AAAA,MACnB,EAAE,OAAO,kBAAkB;AAAA,IAC7B;AAAA,EACF;AAAA,EACA;AAAA,IACE,UAAU;AAAA,IACV,QAAQ;AAAA,MACN,EAAE,OAAO,cAAc;AAAA,MACvB,EAAE,OAAO,iBAAiB;AAAA,MAC1B,EAAE,OAAO,cAAc;AAAA,IACzB;AAAA,EACF;AACF;AAEA,IAAO,qBAAQ;",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -306,7 +306,12 @@ async function resolveUnknownContactDiscrepanciesInProposal(em, proposalId, cont
|
|
|
306
306
|
}
|
|
307
307
|
async function ensureUserCanExecuteAction(action, ctx) {
|
|
308
308
|
const requiredFeature = getRequiredFeatureForAction(action);
|
|
309
|
-
if (!requiredFeature)
|
|
309
|
+
if (!requiredFeature || requiredFeature.trim().length === 0) {
|
|
310
|
+
throw new ExecutionError(
|
|
311
|
+
`No required feature is mapped for action type "${action.actionType}"; refusing to execute`,
|
|
312
|
+
403
|
|
313
|
+
);
|
|
314
|
+
}
|
|
310
315
|
const rbacService = ctx.container.resolve("rbacService");
|
|
311
316
|
if (!rbacService || typeof rbacService.userHasAllFeatures !== "function") {
|
|
312
317
|
throw new ExecutionError("Unable to verify permissions for action execution", 503);
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/modules/inbox_ops/lib/executionEngine.ts"],
|
|
4
|
-
"sourcesContent": ["import type { EntityManager } from '@mikro-orm/postgresql'\nimport type { EntityClass } from '@mikro-orm/core'\nimport type { AwilixContainer } from 'awilix'\nimport type { EventBus } from '@open-mercato/events/types'\nimport type { AuthContext } from '@open-mercato/shared/lib/auth/server'\nimport { findOneWithDecryption, findWithDecryption } from '@open-mercato/shared/lib/encryption/find'\nimport type { InboxActionExecutionContext } from '@open-mercato/shared/modules/inbox-actions'\nimport { InboxProposal, InboxProposalAction, InboxDiscrepancy } from '../data/entities'\nimport type { InboxActionStatus, InboxActionType, InboxProposalStatus } from '../data/entities'\nimport { REQUIRED_FEATURES_MAP } from './constants'\nimport { formatZodErrors } from './validation'\nimport {\n clearLatestCommandLogEntry,\n ExecutionError,\n executeCommand,\n readLatestCommandLogEntry,\n} from './executionHelpers'\n\ninterface CommonEntityFields {\n tenantId?: string\n organizationId?: string\n deletedAt?: Date | null\n createdAt?: Date\n}\n\nexport interface CrossModuleEntities {\n CustomerEntity: EntityClass<CommonEntityFields & { id: string; kind: string; displayName: string; primaryEmail?: string | null }>\n SalesOrder: EntityClass<CommonEntityFields & { id: string; orderNumber: string; currencyCode: string; comments?: string | null; customerReference?: string | null }>\n SalesShipment: EntityClass<CommonEntityFields & { id: string; order: unknown }>\n SalesChannel: EntityClass<CommonEntityFields & { id: string; name: string; currencyCode?: string; metadata?: Record<string, unknown> | null }>\n Dictionary: EntityClass<CommonEntityFields & { id: string; key: string }>\n DictionaryEntry: EntityClass<CommonEntityFields & { id: string; label: string; value: string; normalizedValue?: string | null; dictionary: unknown }>\n}\n\ninterface ExecutionContext {\n em: EntityManager\n userId: string\n tenantId: string\n organizationId: string\n eventBus?: EventBus | null\n container: AwilixContainer\n auth?: AuthContext\n entities?: CrossModuleEntities\n}\n\ninterface ExecutionResult {\n success: boolean\n createdEntityId?: string | null\n createdEntityType?: string | null\n operationLogEntry?: {\n id?: string | null\n undoToken?: string | null\n commandId?: string | null\n actionLabel?: string | null\n resourceKind?: string | null\n resourceId?: string | null\n createdAt?: Date | string | null\n } | null\n error?: string\n statusCode?: number\n}\n\ninterface TypeExecutionResult {\n createdEntityId?: string | null\n createdEntityType?: string | null\n matchedEntityId?: string | null\n matchedEntityType?: string | null\n operationLogEntry?: ExecutionResult['operationLogEntry']\n}\n\nconst ACTION_EXECUTABLE_STATUSES: InboxActionStatus[] = ['pending', 'failed']\n\nexport async function executeAction(\n action: InboxProposalAction,\n ctx: ExecutionContext,\n): Promise<ExecutionResult> {\n const em = ctx.em.fork()\n\n try {\n await ensureUserCanExecuteAction(action, ctx)\n } catch (err) {\n const message = err instanceof Error ? err.message : 'Unable to verify permissions'\n const statusCode = err instanceof ExecutionError ? err.statusCode : 503\n return { success: false, error: message, statusCode }\n }\n\n const claimed = await em.nativeUpdate(\n InboxProposalAction,\n {\n id: action.id,\n proposalId: action.proposalId,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n status: { $in: ACTION_EXECUTABLE_STATUSES },\n deletedAt: null,\n },\n {\n status: 'processing',\n executionError: null,\n },\n )\n\n if (claimed === 0) {\n return { success: false, error: 'Action already processed', statusCode: 409 }\n }\n\n const freshAction = await findOneWithDecryption(\n em,\n InboxProposalAction,\n { id: action.id, deletedAt: null },\n undefined,\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n if (!freshAction) {\n return { success: false, error: 'Action not found', statusCode: 404 }\n }\n\n try {\n const result = await executeByType(freshAction, ctx)\n\n freshAction.status = 'executed'\n freshAction.executedAt = new Date()\n freshAction.executedByUserId = ctx.userId\n freshAction.createdEntityId = result.createdEntityId || null\n freshAction.createdEntityType = result.createdEntityType || null\n if (result.matchedEntityId !== undefined) {\n freshAction.matchedEntityId = result.matchedEntityId\n }\n if (result.matchedEntityType !== undefined) {\n freshAction.matchedEntityType = result.matchedEntityType\n }\n freshAction.executionError = null\n\n await em.flush()\n const encScope = { tenantId: ctx.tenantId, organizationId: ctx.organizationId }\n await resolveActionDiscrepancies(em, freshAction.id, encScope)\n\n // After create_contact or link_contact, resolve unknown_contact discrepancies\n // on ALL other actions in the same proposal that reference the same email\n if (freshAction.actionType === 'create_contact' || freshAction.actionType === 'link_contact') {\n const payload = freshAction.payload as Record<string, unknown> | null\n const contactEmail =\n typeof payload?.email === 'string' ? payload.email\n : typeof payload?.emailAddress === 'string' ? payload.emailAddress\n : null\n if (contactEmail) {\n await resolveUnknownContactDiscrepanciesInProposal(\n em, freshAction.proposalId, contactEmail, encScope,\n )\n }\n }\n\n await recalculateProposalStatus(em, freshAction.proposalId, encScope)\n\n if (ctx.eventBus) {\n await ctx.eventBus.emit('inbox_ops.action.executed', {\n actionId: freshAction.id,\n proposalId: freshAction.proposalId,\n actionType: freshAction.actionType,\n createdEntityId: result.createdEntityId || null,\n createdEntityType: result.createdEntityType || null,\n executedByUserId: ctx.userId,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n })\n }\n\n return { success: true, ...result, statusCode: 200 }\n } catch (err) {\n const message = err instanceof Error ? err.message : String(err)\n const statusCode = err instanceof ExecutionError ? err.statusCode : 500\n\n freshAction.status = 'failed'\n freshAction.executionError = message\n freshAction.executedAt = new Date()\n freshAction.executedByUserId = ctx.userId\n await em.flush()\n\n await recalculateProposalStatus(em, freshAction.proposalId, { tenantId: ctx.tenantId, organizationId: ctx.organizationId })\n\n if (ctx.eventBus) {\n await ctx.eventBus.emit('inbox_ops.action.failed', {\n actionId: freshAction.id,\n proposalId: freshAction.proposalId,\n actionType: freshAction.actionType,\n error: freshAction.executionError,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n })\n }\n\n return { success: false, error: freshAction.executionError || 'Unknown error', statusCode }\n }\n}\n\nexport async function rejectAction(\n action: InboxProposalAction,\n ctx: ExecutionContext,\n): Promise<void> {\n const em = ctx.em.fork()\n const rejectedAt = new Date()\n const claimed = await em.nativeUpdate(\n InboxProposalAction,\n {\n id: action.id,\n proposalId: action.proposalId,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n status: { $in: ACTION_EXECUTABLE_STATUSES },\n deletedAt: null,\n },\n {\n status: 'rejected',\n executedAt: rejectedAt,\n executedByUserId: ctx.userId,\n },\n )\n if (claimed === 0) return\n\n const freshAction = await findOneWithDecryption(\n em,\n InboxProposalAction,\n { id: action.id, deletedAt: null },\n undefined,\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n if (!freshAction) return\n\n const encScope = { tenantId: ctx.tenantId, organizationId: ctx.organizationId }\n await resolveActionDiscrepancies(em, freshAction.id, encScope)\n await recalculateProposalStatus(em, freshAction.proposalId, encScope)\n\n if (ctx.eventBus) {\n await ctx.eventBus.emit('inbox_ops.action.rejected', {\n actionId: freshAction.id,\n proposalId: freshAction.proposalId,\n actionType: freshAction.actionType,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n })\n }\n}\n\nexport async function rejectProposal(\n proposalId: string,\n ctx: ExecutionContext,\n): Promise<void> {\n const em = ctx.em.fork()\n const rejectedAt = new Date()\n\n await em.nativeUpdate(\n InboxProposalAction,\n {\n proposalId,\n status: { $in: ACTION_EXECUTABLE_STATUSES },\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n {\n status: 'rejected',\n executedAt: rejectedAt,\n executedByUserId: ctx.userId,\n },\n )\n\n await em.nativeUpdate(\n InboxDiscrepancy,\n {\n proposalId,\n resolved: false,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n },\n { resolved: true },\n )\n\n await recalculateProposalStatus(em, proposalId, { tenantId: ctx.tenantId, organizationId: ctx.organizationId })\n\n if (ctx.eventBus) {\n await ctx.eventBus.emit('inbox_ops.proposal.rejected', {\n proposalId,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n })\n }\n}\n\nexport async function acceptAllActions(\n proposalId: string,\n ctx: ExecutionContext,\n): Promise<{ results: ExecutionResult[]; stoppedOnFailure: boolean }> {\n const em = ctx.em.fork()\n const actions = await findWithDecryption(\n em,\n InboxProposalAction,\n {\n proposalId,\n status: 'pending',\n deletedAt: null,\n },\n { orderBy: { sortOrder: 'ASC' } },\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n\n const results: ExecutionResult[] = []\n let stoppedOnFailure = false\n\n for (const action of actions) {\n const result = await executeAction(action, ctx)\n results.push(result)\n\n if (!result.success) {\n stoppedOnFailure = true\n break\n }\n }\n\n return { results, stoppedOnFailure }\n}\n\n/**\n * Normalize common LLM payload issues before action-specific normalization.\n */\nfunction normalizeCommonPayloadFields(\n payload: Record<string, unknown>,\n actionType: string,\n): Record<string, unknown> {\n // Lowercase contact type fields (LLM often outputs \"Person\" / \"Company\")\n if (typeof payload.type === 'string') {\n payload.type = payload.type.toLowerCase()\n }\n if (typeof payload.contactType === 'string') {\n payload.contactType = payload.contactType.toLowerCase()\n }\n\n // Normalize link_contact field names (LLM may use various alternatives)\n if (actionType === 'link_contact') {\n if (!payload.emailAddress) {\n const alt = payload.email ?? payload.contactEmail\n if (typeof alt === 'string') payload.emailAddress = alt\n }\n if (!payload.contactId) {\n const alt = payload.id ?? payload.matchedId ?? payload.matchedContactId\n if (typeof alt === 'string') payload.contactId = alt\n }\n if (!payload.contactType) {\n const alt = payload.type ?? payload.kind ?? payload.matchedType ?? payload.matchedContactType\n if (typeof alt === 'string') payload.contactType = alt.toLowerCase()\n }\n if (!payload.contactName) {\n const alt = payload.name ?? payload.displayName\n if (typeof alt === 'string') payload.contactName = alt\n }\n }\n\n return payload\n}\n\n/**\n * Adapt the internal ExecutionContext to the shared InboxActionExecutionContext\n * for use by registered action handlers.\n */\nfunction adaptContext(ctx: ExecutionContext): InboxActionExecutionContext {\n return {\n ...ctx,\n executeCommand: <TInput, TResult>(commandId: string, input: TInput) =>\n executeCommand<TInput, TResult>(ctx, commandId, input),\n resolveEntityClass: <T>(key: string) =>\n resolveEntityClassInternal(ctx, key) as (new (...args: unknown[]) => T) | null,\n }\n}\n\nasync function executeByType(\n action: InboxProposalAction,\n ctx: ExecutionContext,\n): Promise<TypeExecutionResult> {\n // Lazy-load the generated registry to avoid circular imports at module load time\n const { getInboxAction } = await import('@/.mercato/generated/inbox-actions.generated')\n const definition = getInboxAction(action.actionType)\n if (!definition) {\n throw new ExecutionError(`Unknown action type: ${action.actionType}`, 400)\n }\n\n let payload = { ...(action.payload as Record<string, unknown>) }\n\n // Common normalization (lowercase enums, field aliases)\n payload = normalizeCommonPayloadFields(payload, action.actionType)\n\n // Action-specific normalization from the registered handler\n const actionCtx = adaptContext(ctx)\n clearLatestCommandLogEntry(actionCtx)\n if (definition.normalizePayload) {\n payload = await definition.normalizePayload(payload, actionCtx)\n }\n\n const parsed = definition.payloadSchema.safeParse(payload)\n if (!parsed.success) {\n throw new ExecutionError(\n `Invalid ${action.actionType} payload: ${formatZodErrors(parsed.error)}`,\n 400,\n )\n }\n\n const result = await definition.execute(\n { id: action.id, proposalId: action.proposalId, payload: parsed.data },\n actionCtx,\n )\n\n return {\n ...result,\n operationLogEntry: readLatestCommandLogEntry(actionCtx),\n }\n}\n\nasync function resolveUnknownContactDiscrepanciesInProposal(\n em: EntityManager,\n proposalId: string,\n contactEmail: string,\n scope: { tenantId: string; organizationId: string },\n): Promise<void> {\n if (!contactEmail) return\n\n const discrepancies = await findWithDecryption(\n em,\n InboxDiscrepancy,\n {\n proposalId,\n type: 'unknown_contact',\n resolved: false,\n tenantId: scope.tenantId,\n organizationId: scope.organizationId,\n },\n undefined,\n scope,\n )\n\n const normalizedEmail = contactEmail.trim().toLowerCase()\n const matching = discrepancies.filter((d) => {\n const foundValue = (d.foundValue || '').trim().toLowerCase()\n return foundValue === normalizedEmail\n })\n\n for (const d of matching) {\n d.resolved = true\n }\n\n if (matching.length > 0) {\n await em.flush()\n }\n}\n\nasync function ensureUserCanExecuteAction(action: InboxProposalAction, ctx: ExecutionContext): Promise<void> {\n const requiredFeature = getRequiredFeatureForAction(action)\n if (!requiredFeature) return\n\n const rbacService = ctx.container.resolve('rbacService') as {\n userHasAllFeatures: (\n userId: string,\n features: string[],\n scope: { tenantId: string; organizationId: string },\n ) => Promise<boolean>\n }\n\n if (!rbacService || typeof rbacService.userHasAllFeatures !== 'function') {\n throw new ExecutionError('Unable to verify permissions for action execution', 503)\n }\n\n const hasFeature = await rbacService.userHasAllFeatures(\n ctx.userId,\n [requiredFeature],\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n\n if (!hasFeature) {\n throw new ExecutionError(`Insufficient permissions: ${requiredFeature} required`, 403)\n }\n}\n\nfunction resolveEntityClassInternal(\n ctx: ExecutionContext,\n key: string,\n): unknown {\n const fromEntities = (ctx.entities as Record<string, unknown> | undefined)?.[key]\n if (fromEntities) return fromEntities\n try { return ctx.container.resolve(key) } catch { return null }\n}\n\n// Re-export splitPersonName for backward compat\nexport { splitPersonName } from './contactValidation'\n\nasync function resolveActionDiscrepancies(\n em: EntityManager,\n actionId: string,\n scope: { tenantId: string; organizationId: string },\n): Promise<void> {\n const discrepancies = await findWithDecryption(\n em,\n InboxDiscrepancy,\n { actionId, resolved: false },\n undefined,\n scope,\n )\n for (const discrepancy of discrepancies) {\n discrepancy.resolved = true\n }\n if (discrepancies.length > 0) {\n await em.flush()\n }\n}\n\nexport async function recalculateProposalStatus(\n em: EntityManager,\n proposalId: string,\n scope: { tenantId: string; organizationId: string },\n): Promise<void> {\n const proposal = await findOneWithDecryption(\n em,\n InboxProposal,\n { id: proposalId, deletedAt: null },\n undefined,\n scope,\n )\n if (!proposal) return\n\n const actions = await findWithDecryption(\n em,\n InboxProposalAction,\n { proposalId, deletedAt: null },\n undefined,\n scope,\n )\n\n if (actions.length === 0) {\n proposal.status = 'pending'\n await em.flush()\n return\n }\n\n const statuses = actions.map((action) => action.status)\n const allAcceptedOrExecuted = statuses.every((status) => status === 'accepted' || status === 'executed')\n const allRejected = statuses.every((status) => status === 'rejected')\n const allPending = statuses.every((status) => status === 'pending')\n\n let newStatus: InboxProposalStatus\n if (allAcceptedOrExecuted) {\n newStatus = 'accepted'\n } else if (allRejected) {\n newStatus = 'rejected'\n } else if (allPending) {\n newStatus = 'pending'\n } else {\n newStatus = 'partial'\n }\n\n if (proposal.status !== newStatus) {\n proposal.status = newStatus\n await em.flush()\n }\n}\n\nexport function getRequiredFeature(actionType: InboxActionType): string {\n return REQUIRED_FEATURES_MAP[actionType]\n}\n\nfunction getRequiredFeatureForAction(action: InboxProposalAction): string {\n if (action.actionType === 'create_contact') {\n const payload = action.payload as Record<string, unknown> | null\n if (payload?.type === 'company') {\n return 'customers.companies.manage'\n }\n }\n return REQUIRED_FEATURES_MAP[action.actionType]\n}\n"],
|
|
5
|
-
"mappings": "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
|
|
4
|
+
"sourcesContent": ["import type { EntityManager } from '@mikro-orm/postgresql'\nimport type { EntityClass } from '@mikro-orm/core'\nimport type { AwilixContainer } from 'awilix'\nimport type { EventBus } from '@open-mercato/events/types'\nimport type { AuthContext } from '@open-mercato/shared/lib/auth/server'\nimport { findOneWithDecryption, findWithDecryption } from '@open-mercato/shared/lib/encryption/find'\nimport type { InboxActionExecutionContext } from '@open-mercato/shared/modules/inbox-actions'\nimport { InboxProposal, InboxProposalAction, InboxDiscrepancy } from '../data/entities'\nimport type { InboxActionStatus, InboxActionType, InboxProposalStatus } from '../data/entities'\nimport { REQUIRED_FEATURES_MAP } from './constants'\nimport { formatZodErrors } from './validation'\nimport {\n clearLatestCommandLogEntry,\n ExecutionError,\n executeCommand,\n readLatestCommandLogEntry,\n} from './executionHelpers'\n\ninterface CommonEntityFields {\n tenantId?: string\n organizationId?: string\n deletedAt?: Date | null\n createdAt?: Date\n}\n\nexport interface CrossModuleEntities {\n CustomerEntity: EntityClass<CommonEntityFields & { id: string; kind: string; displayName: string; primaryEmail?: string | null }>\n SalesOrder: EntityClass<CommonEntityFields & { id: string; orderNumber: string; currencyCode: string; comments?: string | null; customerReference?: string | null }>\n SalesShipment: EntityClass<CommonEntityFields & { id: string; order: unknown }>\n SalesChannel: EntityClass<CommonEntityFields & { id: string; name: string; currencyCode?: string; metadata?: Record<string, unknown> | null }>\n Dictionary: EntityClass<CommonEntityFields & { id: string; key: string }>\n DictionaryEntry: EntityClass<CommonEntityFields & { id: string; label: string; value: string; normalizedValue?: string | null; dictionary: unknown }>\n}\n\ninterface ExecutionContext {\n em: EntityManager\n userId: string\n tenantId: string\n organizationId: string\n eventBus?: EventBus | null\n container: AwilixContainer\n auth?: AuthContext\n entities?: CrossModuleEntities\n}\n\ninterface ExecutionResult {\n success: boolean\n createdEntityId?: string | null\n createdEntityType?: string | null\n operationLogEntry?: {\n id?: string | null\n undoToken?: string | null\n commandId?: string | null\n actionLabel?: string | null\n resourceKind?: string | null\n resourceId?: string | null\n createdAt?: Date | string | null\n } | null\n error?: string\n statusCode?: number\n}\n\ninterface TypeExecutionResult {\n createdEntityId?: string | null\n createdEntityType?: string | null\n matchedEntityId?: string | null\n matchedEntityType?: string | null\n operationLogEntry?: ExecutionResult['operationLogEntry']\n}\n\nconst ACTION_EXECUTABLE_STATUSES: InboxActionStatus[] = ['pending', 'failed']\n\nexport async function executeAction(\n action: InboxProposalAction,\n ctx: ExecutionContext,\n): Promise<ExecutionResult> {\n const em = ctx.em.fork()\n\n try {\n await ensureUserCanExecuteAction(action, ctx)\n } catch (err) {\n const message = err instanceof Error ? err.message : 'Unable to verify permissions'\n const statusCode = err instanceof ExecutionError ? err.statusCode : 503\n return { success: false, error: message, statusCode }\n }\n\n const claimed = await em.nativeUpdate(\n InboxProposalAction,\n {\n id: action.id,\n proposalId: action.proposalId,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n status: { $in: ACTION_EXECUTABLE_STATUSES },\n deletedAt: null,\n },\n {\n status: 'processing',\n executionError: null,\n },\n )\n\n if (claimed === 0) {\n return { success: false, error: 'Action already processed', statusCode: 409 }\n }\n\n const freshAction = await findOneWithDecryption(\n em,\n InboxProposalAction,\n { id: action.id, deletedAt: null },\n undefined,\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n if (!freshAction) {\n return { success: false, error: 'Action not found', statusCode: 404 }\n }\n\n try {\n const result = await executeByType(freshAction, ctx)\n\n freshAction.status = 'executed'\n freshAction.executedAt = new Date()\n freshAction.executedByUserId = ctx.userId\n freshAction.createdEntityId = result.createdEntityId || null\n freshAction.createdEntityType = result.createdEntityType || null\n if (result.matchedEntityId !== undefined) {\n freshAction.matchedEntityId = result.matchedEntityId\n }\n if (result.matchedEntityType !== undefined) {\n freshAction.matchedEntityType = result.matchedEntityType\n }\n freshAction.executionError = null\n\n await em.flush()\n const encScope = { tenantId: ctx.tenantId, organizationId: ctx.organizationId }\n await resolveActionDiscrepancies(em, freshAction.id, encScope)\n\n // After create_contact or link_contact, resolve unknown_contact discrepancies\n // on ALL other actions in the same proposal that reference the same email\n if (freshAction.actionType === 'create_contact' || freshAction.actionType === 'link_contact') {\n const payload = freshAction.payload as Record<string, unknown> | null\n const contactEmail =\n typeof payload?.email === 'string' ? payload.email\n : typeof payload?.emailAddress === 'string' ? payload.emailAddress\n : null\n if (contactEmail) {\n await resolveUnknownContactDiscrepanciesInProposal(\n em, freshAction.proposalId, contactEmail, encScope,\n )\n }\n }\n\n await recalculateProposalStatus(em, freshAction.proposalId, encScope)\n\n if (ctx.eventBus) {\n await ctx.eventBus.emit('inbox_ops.action.executed', {\n actionId: freshAction.id,\n proposalId: freshAction.proposalId,\n actionType: freshAction.actionType,\n createdEntityId: result.createdEntityId || null,\n createdEntityType: result.createdEntityType || null,\n executedByUserId: ctx.userId,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n })\n }\n\n return { success: true, ...result, statusCode: 200 }\n } catch (err) {\n const message = err instanceof Error ? err.message : String(err)\n const statusCode = err instanceof ExecutionError ? err.statusCode : 500\n\n freshAction.status = 'failed'\n freshAction.executionError = message\n freshAction.executedAt = new Date()\n freshAction.executedByUserId = ctx.userId\n await em.flush()\n\n await recalculateProposalStatus(em, freshAction.proposalId, { tenantId: ctx.tenantId, organizationId: ctx.organizationId })\n\n if (ctx.eventBus) {\n await ctx.eventBus.emit('inbox_ops.action.failed', {\n actionId: freshAction.id,\n proposalId: freshAction.proposalId,\n actionType: freshAction.actionType,\n error: freshAction.executionError,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n })\n }\n\n return { success: false, error: freshAction.executionError || 'Unknown error', statusCode }\n }\n}\n\nexport async function rejectAction(\n action: InboxProposalAction,\n ctx: ExecutionContext,\n): Promise<void> {\n const em = ctx.em.fork()\n const rejectedAt = new Date()\n const claimed = await em.nativeUpdate(\n InboxProposalAction,\n {\n id: action.id,\n proposalId: action.proposalId,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n status: { $in: ACTION_EXECUTABLE_STATUSES },\n deletedAt: null,\n },\n {\n status: 'rejected',\n executedAt: rejectedAt,\n executedByUserId: ctx.userId,\n },\n )\n if (claimed === 0) return\n\n const freshAction = await findOneWithDecryption(\n em,\n InboxProposalAction,\n { id: action.id, deletedAt: null },\n undefined,\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n if (!freshAction) return\n\n const encScope = { tenantId: ctx.tenantId, organizationId: ctx.organizationId }\n await resolveActionDiscrepancies(em, freshAction.id, encScope)\n await recalculateProposalStatus(em, freshAction.proposalId, encScope)\n\n if (ctx.eventBus) {\n await ctx.eventBus.emit('inbox_ops.action.rejected', {\n actionId: freshAction.id,\n proposalId: freshAction.proposalId,\n actionType: freshAction.actionType,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n })\n }\n}\n\nexport async function rejectProposal(\n proposalId: string,\n ctx: ExecutionContext,\n): Promise<void> {\n const em = ctx.em.fork()\n const rejectedAt = new Date()\n\n await em.nativeUpdate(\n InboxProposalAction,\n {\n proposalId,\n status: { $in: ACTION_EXECUTABLE_STATUSES },\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n {\n status: 'rejected',\n executedAt: rejectedAt,\n executedByUserId: ctx.userId,\n },\n )\n\n await em.nativeUpdate(\n InboxDiscrepancy,\n {\n proposalId,\n resolved: false,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n },\n { resolved: true },\n )\n\n await recalculateProposalStatus(em, proposalId, { tenantId: ctx.tenantId, organizationId: ctx.organizationId })\n\n if (ctx.eventBus) {\n await ctx.eventBus.emit('inbox_ops.proposal.rejected', {\n proposalId,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n })\n }\n}\n\nexport async function acceptAllActions(\n proposalId: string,\n ctx: ExecutionContext,\n): Promise<{ results: ExecutionResult[]; stoppedOnFailure: boolean }> {\n const em = ctx.em.fork()\n const actions = await findWithDecryption(\n em,\n InboxProposalAction,\n {\n proposalId,\n status: 'pending',\n deletedAt: null,\n },\n { orderBy: { sortOrder: 'ASC' } },\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n\n const results: ExecutionResult[] = []\n let stoppedOnFailure = false\n\n for (const action of actions) {\n const result = await executeAction(action, ctx)\n results.push(result)\n\n if (!result.success) {\n stoppedOnFailure = true\n break\n }\n }\n\n return { results, stoppedOnFailure }\n}\n\n/**\n * Normalize common LLM payload issues before action-specific normalization.\n */\nfunction normalizeCommonPayloadFields(\n payload: Record<string, unknown>,\n actionType: string,\n): Record<string, unknown> {\n // Lowercase contact type fields (LLM often outputs \"Person\" / \"Company\")\n if (typeof payload.type === 'string') {\n payload.type = payload.type.toLowerCase()\n }\n if (typeof payload.contactType === 'string') {\n payload.contactType = payload.contactType.toLowerCase()\n }\n\n // Normalize link_contact field names (LLM may use various alternatives)\n if (actionType === 'link_contact') {\n if (!payload.emailAddress) {\n const alt = payload.email ?? payload.contactEmail\n if (typeof alt === 'string') payload.emailAddress = alt\n }\n if (!payload.contactId) {\n const alt = payload.id ?? payload.matchedId ?? payload.matchedContactId\n if (typeof alt === 'string') payload.contactId = alt\n }\n if (!payload.contactType) {\n const alt = payload.type ?? payload.kind ?? payload.matchedType ?? payload.matchedContactType\n if (typeof alt === 'string') payload.contactType = alt.toLowerCase()\n }\n if (!payload.contactName) {\n const alt = payload.name ?? payload.displayName\n if (typeof alt === 'string') payload.contactName = alt\n }\n }\n\n return payload\n}\n\n/**\n * Adapt the internal ExecutionContext to the shared InboxActionExecutionContext\n * for use by registered action handlers.\n */\nfunction adaptContext(ctx: ExecutionContext): InboxActionExecutionContext {\n return {\n ...ctx,\n executeCommand: <TInput, TResult>(commandId: string, input: TInput) =>\n executeCommand<TInput, TResult>(ctx, commandId, input),\n resolveEntityClass: <T>(key: string) =>\n resolveEntityClassInternal(ctx, key) as (new (...args: unknown[]) => T) | null,\n }\n}\n\nasync function executeByType(\n action: InboxProposalAction,\n ctx: ExecutionContext,\n): Promise<TypeExecutionResult> {\n // Lazy-load the generated registry to avoid circular imports at module load time\n const { getInboxAction } = await import('@/.mercato/generated/inbox-actions.generated')\n const definition = getInboxAction(action.actionType)\n if (!definition) {\n throw new ExecutionError(`Unknown action type: ${action.actionType}`, 400)\n }\n\n let payload = { ...(action.payload as Record<string, unknown>) }\n\n // Common normalization (lowercase enums, field aliases)\n payload = normalizeCommonPayloadFields(payload, action.actionType)\n\n // Action-specific normalization from the registered handler\n const actionCtx = adaptContext(ctx)\n clearLatestCommandLogEntry(actionCtx)\n if (definition.normalizePayload) {\n payload = await definition.normalizePayload(payload, actionCtx)\n }\n\n const parsed = definition.payloadSchema.safeParse(payload)\n if (!parsed.success) {\n throw new ExecutionError(\n `Invalid ${action.actionType} payload: ${formatZodErrors(parsed.error)}`,\n 400,\n )\n }\n\n const result = await definition.execute(\n { id: action.id, proposalId: action.proposalId, payload: parsed.data },\n actionCtx,\n )\n\n return {\n ...result,\n operationLogEntry: readLatestCommandLogEntry(actionCtx),\n }\n}\n\nasync function resolveUnknownContactDiscrepanciesInProposal(\n em: EntityManager,\n proposalId: string,\n contactEmail: string,\n scope: { tenantId: string; organizationId: string },\n): Promise<void> {\n if (!contactEmail) return\n\n const discrepancies = await findWithDecryption(\n em,\n InboxDiscrepancy,\n {\n proposalId,\n type: 'unknown_contact',\n resolved: false,\n tenantId: scope.tenantId,\n organizationId: scope.organizationId,\n },\n undefined,\n scope,\n )\n\n const normalizedEmail = contactEmail.trim().toLowerCase()\n const matching = discrepancies.filter((d) => {\n const foundValue = (d.foundValue || '').trim().toLowerCase()\n return foundValue === normalizedEmail\n })\n\n for (const d of matching) {\n d.resolved = true\n }\n\n if (matching.length > 0) {\n await em.flush()\n }\n}\n\nasync function ensureUserCanExecuteAction(action: InboxProposalAction, ctx: ExecutionContext): Promise<void> {\n const requiredFeature = getRequiredFeatureForAction(action)\n if (!requiredFeature || requiredFeature.trim().length === 0) {\n throw new ExecutionError(\n `No required feature is mapped for action type \"${action.actionType}\"; refusing to execute`,\n 403,\n )\n }\n\n const rbacService = ctx.container.resolve('rbacService') as {\n userHasAllFeatures: (\n userId: string,\n features: string[],\n scope: { tenantId: string; organizationId: string },\n ) => Promise<boolean>\n }\n\n if (!rbacService || typeof rbacService.userHasAllFeatures !== 'function') {\n throw new ExecutionError('Unable to verify permissions for action execution', 503)\n }\n\n const hasFeature = await rbacService.userHasAllFeatures(\n ctx.userId,\n [requiredFeature],\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n\n if (!hasFeature) {\n throw new ExecutionError(`Insufficient permissions: ${requiredFeature} required`, 403)\n }\n}\n\nfunction resolveEntityClassInternal(\n ctx: ExecutionContext,\n key: string,\n): unknown {\n const fromEntities = (ctx.entities as Record<string, unknown> | undefined)?.[key]\n if (fromEntities) return fromEntities\n try { return ctx.container.resolve(key) } catch { return null }\n}\n\n// Re-export splitPersonName for backward compat\nexport { splitPersonName } from './contactValidation'\n\nasync function resolveActionDiscrepancies(\n em: EntityManager,\n actionId: string,\n scope: { tenantId: string; organizationId: string },\n): Promise<void> {\n const discrepancies = await findWithDecryption(\n em,\n InboxDiscrepancy,\n { actionId, resolved: false },\n undefined,\n scope,\n )\n for (const discrepancy of discrepancies) {\n discrepancy.resolved = true\n }\n if (discrepancies.length > 0) {\n await em.flush()\n }\n}\n\nexport async function recalculateProposalStatus(\n em: EntityManager,\n proposalId: string,\n scope: { tenantId: string; organizationId: string },\n): Promise<void> {\n const proposal = await findOneWithDecryption(\n em,\n InboxProposal,\n { id: proposalId, deletedAt: null },\n undefined,\n scope,\n )\n if (!proposal) return\n\n const actions = await findWithDecryption(\n em,\n InboxProposalAction,\n { proposalId, deletedAt: null },\n undefined,\n scope,\n )\n\n if (actions.length === 0) {\n proposal.status = 'pending'\n await em.flush()\n return\n }\n\n const statuses = actions.map((action) => action.status)\n const allAcceptedOrExecuted = statuses.every((status) => status === 'accepted' || status === 'executed')\n const allRejected = statuses.every((status) => status === 'rejected')\n const allPending = statuses.every((status) => status === 'pending')\n\n let newStatus: InboxProposalStatus\n if (allAcceptedOrExecuted) {\n newStatus = 'accepted'\n } else if (allRejected) {\n newStatus = 'rejected'\n } else if (allPending) {\n newStatus = 'pending'\n } else {\n newStatus = 'partial'\n }\n\n if (proposal.status !== newStatus) {\n proposal.status = newStatus\n await em.flush()\n }\n}\n\nexport function getRequiredFeature(actionType: InboxActionType): string {\n return REQUIRED_FEATURES_MAP[actionType]\n}\n\nfunction getRequiredFeatureForAction(action: InboxProposalAction): string {\n if (action.actionType === 'create_contact') {\n const payload = action.payload as Record<string, unknown> | null\n if (payload?.type === 'company') {\n return 'customers.companies.manage'\n }\n }\n return REQUIRED_FEATURES_MAP[action.actionType]\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -13,12 +13,7 @@ function readLatestCommandLogEntry(ctx) {
|
|
|
13
13
|
}
|
|
14
14
|
function hasSuperAdminAccess(auth) {
|
|
15
15
|
if (!auth || typeof auth !== "object") return false;
|
|
16
|
-
|
|
17
|
-
const roles = auth.roles;
|
|
18
|
-
if (!Array.isArray(roles)) return false;
|
|
19
|
-
return roles.some(
|
|
20
|
-
(role) => typeof role === "string" && role.trim().toLowerCase() === "superadmin"
|
|
21
|
-
);
|
|
16
|
+
return auth.isSuperAdmin === true;
|
|
22
17
|
}
|
|
23
18
|
class ExecutionError extends Error {
|
|
24
19
|
constructor(message, statusCode = 400) {
|
|
@@ -56,7 +51,7 @@ async function executeCommand(ctx, commandId, input) {
|
|
|
56
51
|
return result;
|
|
57
52
|
}
|
|
58
53
|
async function userHasFeature(ctx, feature) {
|
|
59
|
-
if (!feature) return
|
|
54
|
+
if (!feature || feature.trim().length === 0) return false;
|
|
60
55
|
if (hasSuperAdminAccess(ctx.auth)) return true;
|
|
61
56
|
try {
|
|
62
57
|
const rbacService = ctx.container.resolve("rbacService");
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/modules/inbox_ops/lib/executionHelpers.ts"],
|
|
4
|
-
"sourcesContent": ["import type { EntityManager } from '@mikro-orm/postgresql'\nimport type { EntityClass } from '@mikro-orm/core'\nimport type { AwilixContainer } from 'awilix'\nimport type { EventBus } from '@open-mercato/events/types'\nimport type { AuthContext } from '@open-mercato/shared/lib/auth/server'\nimport type { CommandBus, CommandRuntimeContext } from '@open-mercato/shared/lib/commands'\nimport { findOneWithDecryption, findWithDecryption } from '@open-mercato/shared/lib/encryption/find'\nimport type { InboxActionExecutionContext } from '@open-mercato/shared/modules/inbox-actions'\nimport type { CrossModuleEntities } from './executionEngine'\nexport { formatZodErrors } from './validation'\n\n// ---------------------------------------------------------------------------\n// Context type used by helper functions (concrete types for ORM/DI access)\n// ---------------------------------------------------------------------------\n\nexport interface ExecutionHelperContext {\n em: EntityManager\n userId: string\n tenantId: string\n organizationId: string\n eventBus?: EventBus | null\n container: AwilixContainer\n auth?: AuthContext\n entities?: CrossModuleEntities\n}\n\ntype OperationLogEntryLike = {\n id?: string | null\n undoToken?: string | null\n commandId?: string | null\n actionLabel?: string | null\n resourceKind?: string | null\n resourceId?: string | null\n createdAt?: Date | string | null\n}\n\nconst LAST_OPERATION_LOG_ENTRY = Symbol('inboxOps:lastOperationLogEntry')\n\ntype OperationLogCarrier = {\n [LAST_OPERATION_LOG_ENTRY]?: OperationLogEntryLike | null\n}\n\n/**\n * Cast InboxActionExecutionContext (from shared) to the concrete helper context.\n * The inbox-actions.ts handlers receive InboxActionExecutionContext but helpers\n * need concrete EntityManager / AwilixContainer types.\n */\nexport function asHelperContext(ctx: InboxActionExecutionContext): ExecutionHelperContext {\n return ctx as unknown as ExecutionHelperContext\n}\n\nexport function clearLatestCommandLogEntry(ctx: object) {\n ;(ctx as OperationLogCarrier)[LAST_OPERATION_LOG_ENTRY] = null\n}\n\nexport function readLatestCommandLogEntry(ctx: object): OperationLogEntryLike | null {\n return (ctx as OperationLogCarrier)[LAST_OPERATION_LOG_ENTRY] ?? null\n}\n\ninterface FeatureCheckingRbacService {\n userHasAllFeatures: (\n userId: string,\n features: string[],\n scope: { tenantId: string; organizationId: string },\n ) => Promise<boolean>\n}\n\nfunction hasSuperAdminAccess(auth: ExecutionHelperContext['auth']): boolean {\n if (!auth || typeof auth !== 'object') return false\n if ((auth as Record<string, unknown>).isSuperAdmin === true) return true\n const roles = (auth as Record<string, unknown>).roles\n if (!Array.isArray(roles)) return false\n return roles.some(\n (role) => typeof role === 'string' && role.trim().toLowerCase() === 'superadmin',\n )\n}\n\n// ---------------------------------------------------------------------------\n// Error\n// ---------------------------------------------------------------------------\n\nexport class ExecutionError extends Error {\n statusCode: number\n\n constructor(message: string, statusCode = 400) {\n super(message)\n this.statusCode = statusCode\n }\n}\n\n// ---------------------------------------------------------------------------\n// Command execution\n// ---------------------------------------------------------------------------\n\nexport async function executeCommand<TInput, TResult>(\n ctx: ExecutionHelperContext,\n commandId: string,\n input: TInput,\n): Promise<TResult> {\n const commandBus = ctx.container.resolve('commandBus') as CommandBus\n if (!commandBus || typeof commandBus.execute !== 'function') {\n throw new ExecutionError('Command bus is not available', 503)\n }\n\n const auth =\n ctx.auth ??\n ({\n sub: ctx.userId,\n userId: ctx.userId,\n tenantId: ctx.tenantId,\n orgId: ctx.organizationId,\n isSuperAdmin: false,\n } satisfies Exclude<AuthContext, null>)\n\n const commandContext: CommandRuntimeContext = {\n container: ctx.container,\n auth,\n organizationScope: null,\n selectedOrganizationId: ctx.organizationId,\n organizationIds: [ctx.organizationId],\n }\n\n const { result, logEntry } = await commandBus.execute<TInput, TResult>(commandId, {\n input,\n ctx: commandContext,\n })\n\n if (logEntry?.undoToken && logEntry.id && logEntry.commandId) {\n ;(ctx as ExecutionHelperContext & OperationLogCarrier)[LAST_OPERATION_LOG_ENTRY] = logEntry\n }\n\n return result\n}\n\nexport async function userHasFeature(\n ctx: ExecutionHelperContext,\n feature: string,\n): Promise<boolean> {\n if (!feature) return true\n if (hasSuperAdminAccess(ctx.auth)) return true\n\n try {\n const rbacService = ctx.container.resolve('rbacService') as FeatureCheckingRbacService\n if (!rbacService || typeof rbacService.userHasAllFeatures !== 'function') {\n return false\n }\n return rbacService.userHasAllFeatures(\n ctx.userId,\n [feature],\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n } catch {\n return false\n }\n}\n\n// ---------------------------------------------------------------------------\n// Entity resolution\n// ---------------------------------------------------------------------------\n\nexport function resolveEntityClass<K extends keyof CrossModuleEntities>(\n ctx: ExecutionHelperContext,\n key: K,\n): CrossModuleEntities[K] | null {\n const fromEntities = ctx.entities?.[key]\n if (fromEntities) return fromEntities\n try { return ctx.container.resolve(key) } catch { return null }\n}\n\n// ---------------------------------------------------------------------------\n// Source metadata\n// ---------------------------------------------------------------------------\n\nexport function buildSourceMetadata(actionId: string, proposalId: string): Record<string, unknown> {\n return {\n source: 'inbox_ops',\n inboxOpsActionId: actionId,\n inboxOpsProposalId: proposalId,\n }\n}\n\n// ---------------------------------------------------------------------------\n// Order resolution\n// ---------------------------------------------------------------------------\n\nexport async function resolveOrderByReference(\n ctx: ExecutionHelperContext,\n orderId?: string,\n orderNumber?: string,\n): Promise<{ id: string; orderNumber: string; currencyCode: string; comments?: string | null }> {\n const SalesOrderClass = resolveEntityClass(ctx, 'SalesOrder')\n if (!SalesOrderClass) {\n throw new ExecutionError('Sales module entities not available', 503)\n }\n\n const where: Record<string, unknown> = {\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n }\n if (orderId) {\n where.id = orderId\n } else if (orderNumber && orderNumber.trim().length > 0) {\n where.orderNumber = orderNumber.trim()\n } else {\n throw new ExecutionError('Order reference is required', 400)\n }\n\n const order = await findOneWithDecryption(\n ctx.em,\n SalesOrderClass,\n where,\n undefined,\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n if (!order) {\n throw new ExecutionError('Referenced order not found', 404)\n }\n return order\n}\n\n// ---------------------------------------------------------------------------\n// Channel resolution\n// ---------------------------------------------------------------------------\n\nexport async function resolveFirstChannelId(ctx: ExecutionHelperContext): Promise<string | null> {\n const SalesChannelClass = resolveEntityClass(ctx, 'SalesChannel')\n if (!SalesChannelClass) return null\n\n try {\n const channel = await findOneWithDecryption(\n ctx.em,\n SalesChannelClass,\n {\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n { orderBy: { name: 'ASC' } },\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n return channel?.id ?? null\n } catch {\n return null\n }\n}\n\nexport async function resolveChannelCurrency(\n ctx: ExecutionHelperContext,\n channelId: string | null,\n): Promise<string | null> {\n const SalesChannelClass = resolveEntityClass(ctx, 'SalesChannel')\n if (!SalesChannelClass) return null\n\n try {\n const where: Record<string, unknown> = {\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n }\n if (channelId) where.id = channelId\n const channel = await findOneWithDecryption(\n ctx.em,\n SalesChannelClass,\n where,\n channelId ? undefined : { orderBy: { name: 'ASC' } },\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n return channel?.currencyCode ?? null\n } catch {\n return null\n }\n}\n\nexport async function resolveEffectiveDocumentKind(\n ctx: ExecutionHelperContext,\n channelId: string,\n): Promise<'order' | 'quote'> {\n const SalesChannelClass = resolveEntityClass(ctx, 'SalesChannel')\n if (!SalesChannelClass) return 'order'\n\n const channel = await findOneWithDecryption(\n ctx.em,\n SalesChannelClass,\n {\n id: channelId,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n undefined,\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n if (!channel) return 'order'\n\n const metadata = channel.metadata as Record<string, unknown> | null\n if (metadata?.quotesRequired === true) {\n return 'quote'\n }\n return 'order'\n}\n\n// ---------------------------------------------------------------------------\n// Shipment status resolution\n// ---------------------------------------------------------------------------\n\nconst SALES_SHIPMENT_STATUS_DICTIONARY_KEY = 'sales.shipment_status'\n\nexport async function resolveShipmentStatusEntryId(\n ctx: ExecutionHelperContext,\n statusLabel: string,\n): Promise<string | null> {\n const DictionaryClass = resolveEntityClass(ctx, 'Dictionary')\n const DictionaryEntryClass = resolveEntityClass(ctx, 'DictionaryEntry')\n if (!DictionaryClass || !DictionaryEntryClass) return null\n\n const encryptionScope = { tenantId: ctx.tenantId, organizationId: ctx.organizationId }\n\n const dictionary = await findOneWithDecryption(\n ctx.em,\n DictionaryClass,\n {\n key: SALES_SHIPMENT_STATUS_DICTIONARY_KEY,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n undefined,\n encryptionScope,\n )\n if (!dictionary) return null\n\n const entries = await findWithDecryption(\n ctx.em,\n DictionaryEntryClass,\n {\n dictionary: dictionary.id,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n },\n undefined,\n encryptionScope,\n )\n if (!entries.length) return null\n\n const normalizedTarget = normalizeDictionaryToken(statusLabel)\n const loweredTarget = statusLabel.trim().toLowerCase()\n\n const match = entries.find((entry) => {\n const label = entry.label.trim().toLowerCase()\n const value = entry.value.trim().toLowerCase()\n return (\n entry.normalizedValue === normalizedTarget ||\n label === loweredTarget ||\n value === loweredTarget\n )\n })\n\n return match?.id ?? null\n}\n\n// ---------------------------------------------------------------------------\n// Customer / contact resolution\n// ---------------------------------------------------------------------------\n\nexport async function resolveCustomerEntityIdByEmail(\n ctx: ExecutionHelperContext,\n email: string,\n): Promise<string | null> {\n const normalized = email.trim().toLowerCase()\n if (!normalized) return null\n\n const CustomerEntityClass = resolveEntityClass(ctx, 'CustomerEntity')\n if (!CustomerEntityClass) return null\n\n const entity = await findOneWithDecryption(\n ctx.em,\n CustomerEntityClass,\n {\n primaryEmail: normalized,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n undefined,\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n if (entity) return entity.id\n\n const candidates = await findWithDecryption(\n ctx.em,\n CustomerEntityClass,\n {\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n { limit: 100, orderBy: { createdAt: 'DESC' } },\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n const match = candidates.find(\n (e) => e.primaryEmail && e.primaryEmail.toLowerCase() === normalized,\n )\n return match?.id ?? null\n}\n\nexport async function resolveContactIdByNameAndType(\n ctx: ExecutionHelperContext,\n contactName: string,\n contactType: string,\n): Promise<string | null> {\n const CustomerEntityClass = resolveEntityClass(ctx, 'CustomerEntity')\n if (!CustomerEntityClass) return null\n\n const normalized = contactName.trim()\n if (!normalized) return null\n\n const entity = await findOneWithDecryption(\n ctx.em,\n CustomerEntityClass,\n {\n displayName: normalized,\n kind: contactType,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n undefined,\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n\n return entity?.id ?? null\n}\n\n// ---------------------------------------------------------------------------\n// Order line items\n// ---------------------------------------------------------------------------\n\nexport interface OrderLineItem {\n id: string\n name?: string | null\n}\n\nexport async function loadOrderLineItems(\n ctx: ExecutionHelperContext,\n orderId: string,\n): Promise<OrderLineItem[]> {\n try {\n const result = await executeCommand<Record<string, unknown>, { lines?: OrderLineItem[] }>(\n ctx,\n 'sales.orders.lines.list',\n { orderId, organizationId: ctx.organizationId, tenantId: ctx.tenantId },\n )\n return result.lines ?? []\n } catch {\n return []\n }\n}\n\nexport function matchLineItemByName(\n orderLines: OrderLineItem[],\n lineItemName: string,\n): string | null {\n const target = lineItemName.trim().toLowerCase()\n if (!target) return null\n\n const exact = orderLines.find((l) => (l.name || '').trim().toLowerCase() === target)\n if (exact) return exact.id\n\n const partial = orderLines.find((l) => {\n const name = (l.name || '').trim().toLowerCase()\n return name.includes(target) || target.includes(name)\n })\n return partial?.id ?? null\n}\n\n// ---------------------------------------------------------------------------\n// Data normalization utilities\n// ---------------------------------------------------------------------------\n\nexport function normalizeAddressSnapshot(\n address: Record<string, unknown>,\n): Record<string, unknown> {\n return {\n addressLine1: address.line1 ?? address.addressLine1 ?? '',\n addressLine2: address.line2 ?? address.addressLine2 ?? null,\n companyName: address.company ?? address.companyName ?? null,\n name: address.contactName ?? address.name ?? null,\n city: address.city ?? null,\n region: address.state ?? address.region ?? null,\n postalCode: address.postalCode ?? null,\n country: address.country ?? null,\n }\n}\n\nexport function parseDateToken(value?: string | null): Date | undefined {\n if (!value) return undefined\n const parsed = new Date(value)\n if (Number.isNaN(parsed.getTime())) return undefined\n return parsed\n}\n\nexport function parseNumberToken(value: string, fieldName: string): number {\n const parsed = Number(value)\n if (!Number.isFinite(parsed)) {\n throw new ExecutionError(`Invalid numeric value for ${fieldName}`, 400)\n }\n return parsed\n}\n\nexport function normalizeDictionaryToken(value: string): string {\n return value.trim().toLowerCase().replace(/[\\s-]+/g, '_')\n}\n\n// ---------------------------------------------------------------------------\n// Product discrepancy resolution (used by catalog inbox action handler)\n// ---------------------------------------------------------------------------\n\nexport async function resolveProductDiscrepanciesInProposal(\n em: EntityManager,\n proposalId: string,\n productTitle: string,\n productId: string,\n scope: { tenantId: string; organizationId: string },\n): Promise<void> {\n const { InboxDiscrepancy, InboxProposalAction } = await import('../data/entities')\n\n const discrepancies = await findWithDecryption(\n em,\n InboxDiscrepancy,\n {\n proposalId,\n type: 'product_not_found',\n resolved: false,\n tenantId: scope.tenantId,\n organizationId: scope.organizationId,\n },\n undefined,\n scope,\n )\n\n const normalizedTitle = productTitle.toLowerCase().trim()\n const matchingDiscrepancies = discrepancies.filter((d) => {\n const foundValue = (d.foundValue || '').toLowerCase().trim()\n return foundValue === normalizedTitle\n })\n\n if (matchingDiscrepancies.length === 0) return\n\n // Phase 1: flush scalar mutations before any queries to avoid UoW tracking loss (SPEC-018)\n for (const discrepancy of matchingDiscrepancies) {\n discrepancy.resolved = true\n }\n await em.flush()\n\n // Phase 2: update line item product IDs (involves findOneWithDecryption queries)\n const actionIds = matchingDiscrepancies\n .map((d) => d.actionId)\n .filter((id): id is string => !!id)\n\n for (const actionId of actionIds) {\n const action = await findOneWithDecryption(\n em,\n InboxProposalAction,\n { id: actionId, deletedAt: null },\n undefined,\n scope,\n )\n if (!action) continue\n\n const payload = action.payload as Record<string, unknown>\n const lineItems = Array.isArray(payload?.lineItems)\n ? (payload.lineItems as Record<string, unknown>[])\n : []\n\n let updated = false\n for (const item of lineItems) {\n if (item.productId) continue\n const itemName = (typeof item.productName === 'string' ? item.productName : '').toLowerCase().trim()\n if (itemName === normalizedTitle) {\n item.productId = productId\n updated = true\n break\n }\n }\n\n if (updated) {\n action.payload = { ...payload, lineItems }\n }\n }\n\n if (actionIds.length > 0) {\n await em.flush()\n }\n}\n"],
|
|
5
|
-
"mappings": "AAMA,SAAS,uBAAuB,0BAA0B;AAG1D,SAAS,uBAAuB;AA2BhC,MAAM,2BAA2B,uBAAO,gCAAgC;AAWjE,SAAS,gBAAgB,KAA0D;AACxF,SAAO;AACT;AAEO,SAAS,2BAA2B,KAAa;AACtD;AAAC,EAAC,IAA4B,wBAAwB,IAAI;AAC5D;AAEO,SAAS,0BAA0B,KAA2C;AACnF,SAAQ,IAA4B,wBAAwB,KAAK;AACnE;
|
|
4
|
+
"sourcesContent": ["import type { EntityManager } from '@mikro-orm/postgresql'\nimport type { EntityClass } from '@mikro-orm/core'\nimport type { AwilixContainer } from 'awilix'\nimport type { EventBus } from '@open-mercato/events/types'\nimport type { AuthContext } from '@open-mercato/shared/lib/auth/server'\nimport type { CommandBus, CommandRuntimeContext } from '@open-mercato/shared/lib/commands'\nimport { findOneWithDecryption, findWithDecryption } from '@open-mercato/shared/lib/encryption/find'\nimport type { InboxActionExecutionContext } from '@open-mercato/shared/modules/inbox-actions'\nimport type { CrossModuleEntities } from './executionEngine'\nexport { formatZodErrors } from './validation'\n\n// ---------------------------------------------------------------------------\n// Context type used by helper functions (concrete types for ORM/DI access)\n// ---------------------------------------------------------------------------\n\nexport interface ExecutionHelperContext {\n em: EntityManager\n userId: string\n tenantId: string\n organizationId: string\n eventBus?: EventBus | null\n container: AwilixContainer\n auth?: AuthContext\n entities?: CrossModuleEntities\n}\n\ntype OperationLogEntryLike = {\n id?: string | null\n undoToken?: string | null\n commandId?: string | null\n actionLabel?: string | null\n resourceKind?: string | null\n resourceId?: string | null\n createdAt?: Date | string | null\n}\n\nconst LAST_OPERATION_LOG_ENTRY = Symbol('inboxOps:lastOperationLogEntry')\n\ntype OperationLogCarrier = {\n [LAST_OPERATION_LOG_ENTRY]?: OperationLogEntryLike | null\n}\n\n/**\n * Cast InboxActionExecutionContext (from shared) to the concrete helper context.\n * The inbox-actions.ts handlers receive InboxActionExecutionContext but helpers\n * need concrete EntityManager / AwilixContainer types.\n */\nexport function asHelperContext(ctx: InboxActionExecutionContext): ExecutionHelperContext {\n return ctx as unknown as ExecutionHelperContext\n}\n\nexport function clearLatestCommandLogEntry(ctx: object) {\n ;(ctx as OperationLogCarrier)[LAST_OPERATION_LOG_ENTRY] = null\n}\n\nexport function readLatestCommandLogEntry(ctx: object): OperationLogEntryLike | null {\n return (ctx as OperationLogCarrier)[LAST_OPERATION_LOG_ENTRY] ?? null\n}\n\ninterface FeatureCheckingRbacService {\n userHasAllFeatures: (\n userId: string,\n features: string[],\n scope: { tenantId: string; organizationId: string },\n ) => Promise<boolean>\n}\n\n// Super-admin status is the immutable `isSuperAdmin` flag derived from the ACL\n// `is_super_admin` column at session resolution. Never fall back to comparing the\n// auth context's `roles` against a literal like 'superadmin' \u2014 role names are\n// tenant-mutable and trivially spoofable, so a role-name fallback would let a tenant\n// admin mint a `superadmin`-named role to bypass every inbox-ops feature gate\n// (#2699). Mirrors the scheduler module's hardened gate.\nfunction hasSuperAdminAccess(auth: ExecutionHelperContext['auth']): boolean {\n if (!auth || typeof auth !== 'object') return false\n return (auth as Record<string, unknown>).isSuperAdmin === true\n}\n\n// ---------------------------------------------------------------------------\n// Error\n// ---------------------------------------------------------------------------\n\nexport class ExecutionError extends Error {\n statusCode: number\n\n constructor(message: string, statusCode = 400) {\n super(message)\n this.statusCode = statusCode\n }\n}\n\n// ---------------------------------------------------------------------------\n// Command execution\n// ---------------------------------------------------------------------------\n\nexport async function executeCommand<TInput, TResult>(\n ctx: ExecutionHelperContext,\n commandId: string,\n input: TInput,\n): Promise<TResult> {\n const commandBus = ctx.container.resolve('commandBus') as CommandBus\n if (!commandBus || typeof commandBus.execute !== 'function') {\n throw new ExecutionError('Command bus is not available', 503)\n }\n\n const auth =\n ctx.auth ??\n ({\n sub: ctx.userId,\n userId: ctx.userId,\n tenantId: ctx.tenantId,\n orgId: ctx.organizationId,\n isSuperAdmin: false,\n } satisfies Exclude<AuthContext, null>)\n\n const commandContext: CommandRuntimeContext = {\n container: ctx.container,\n auth,\n organizationScope: null,\n selectedOrganizationId: ctx.organizationId,\n organizationIds: [ctx.organizationId],\n }\n\n const { result, logEntry } = await commandBus.execute<TInput, TResult>(commandId, {\n input,\n ctx: commandContext,\n })\n\n if (logEntry?.undoToken && logEntry.id && logEntry.commandId) {\n ;(ctx as ExecutionHelperContext & OperationLogCarrier)[LAST_OPERATION_LOG_ENTRY] = logEntry\n }\n\n return result\n}\n\nexport async function userHasFeature(\n ctx: ExecutionHelperContext,\n feature: string,\n): Promise<boolean> {\n if (!feature || feature.trim().length === 0) return false\n if (hasSuperAdminAccess(ctx.auth)) return true\n\n try {\n const rbacService = ctx.container.resolve('rbacService') as FeatureCheckingRbacService\n if (!rbacService || typeof rbacService.userHasAllFeatures !== 'function') {\n return false\n }\n return rbacService.userHasAllFeatures(\n ctx.userId,\n [feature],\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n } catch {\n return false\n }\n}\n\n// ---------------------------------------------------------------------------\n// Entity resolution\n// ---------------------------------------------------------------------------\n\nexport function resolveEntityClass<K extends keyof CrossModuleEntities>(\n ctx: ExecutionHelperContext,\n key: K,\n): CrossModuleEntities[K] | null {\n const fromEntities = ctx.entities?.[key]\n if (fromEntities) return fromEntities\n try { return ctx.container.resolve(key) } catch { return null }\n}\n\n// ---------------------------------------------------------------------------\n// Source metadata\n// ---------------------------------------------------------------------------\n\nexport function buildSourceMetadata(actionId: string, proposalId: string): Record<string, unknown> {\n return {\n source: 'inbox_ops',\n inboxOpsActionId: actionId,\n inboxOpsProposalId: proposalId,\n }\n}\n\n// ---------------------------------------------------------------------------\n// Order resolution\n// ---------------------------------------------------------------------------\n\nexport async function resolveOrderByReference(\n ctx: ExecutionHelperContext,\n orderId?: string,\n orderNumber?: string,\n): Promise<{ id: string; orderNumber: string; currencyCode: string; comments?: string | null }> {\n const SalesOrderClass = resolveEntityClass(ctx, 'SalesOrder')\n if (!SalesOrderClass) {\n throw new ExecutionError('Sales module entities not available', 503)\n }\n\n const where: Record<string, unknown> = {\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n }\n if (orderId) {\n where.id = orderId\n } else if (orderNumber && orderNumber.trim().length > 0) {\n where.orderNumber = orderNumber.trim()\n } else {\n throw new ExecutionError('Order reference is required', 400)\n }\n\n const order = await findOneWithDecryption(\n ctx.em,\n SalesOrderClass,\n where,\n undefined,\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n if (!order) {\n throw new ExecutionError('Referenced order not found', 404)\n }\n return order\n}\n\n// ---------------------------------------------------------------------------\n// Channel resolution\n// ---------------------------------------------------------------------------\n\nexport async function resolveFirstChannelId(ctx: ExecutionHelperContext): Promise<string | null> {\n const SalesChannelClass = resolveEntityClass(ctx, 'SalesChannel')\n if (!SalesChannelClass) return null\n\n try {\n const channel = await findOneWithDecryption(\n ctx.em,\n SalesChannelClass,\n {\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n { orderBy: { name: 'ASC' } },\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n return channel?.id ?? null\n } catch {\n return null\n }\n}\n\nexport async function resolveChannelCurrency(\n ctx: ExecutionHelperContext,\n channelId: string | null,\n): Promise<string | null> {\n const SalesChannelClass = resolveEntityClass(ctx, 'SalesChannel')\n if (!SalesChannelClass) return null\n\n try {\n const where: Record<string, unknown> = {\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n }\n if (channelId) where.id = channelId\n const channel = await findOneWithDecryption(\n ctx.em,\n SalesChannelClass,\n where,\n channelId ? undefined : { orderBy: { name: 'ASC' } },\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n return channel?.currencyCode ?? null\n } catch {\n return null\n }\n}\n\nexport async function resolveEffectiveDocumentKind(\n ctx: ExecutionHelperContext,\n channelId: string,\n): Promise<'order' | 'quote'> {\n const SalesChannelClass = resolveEntityClass(ctx, 'SalesChannel')\n if (!SalesChannelClass) return 'order'\n\n const channel = await findOneWithDecryption(\n ctx.em,\n SalesChannelClass,\n {\n id: channelId,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n undefined,\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n if (!channel) return 'order'\n\n const metadata = channel.metadata as Record<string, unknown> | null\n if (metadata?.quotesRequired === true) {\n return 'quote'\n }\n return 'order'\n}\n\n// ---------------------------------------------------------------------------\n// Shipment status resolution\n// ---------------------------------------------------------------------------\n\nconst SALES_SHIPMENT_STATUS_DICTIONARY_KEY = 'sales.shipment_status'\n\nexport async function resolveShipmentStatusEntryId(\n ctx: ExecutionHelperContext,\n statusLabel: string,\n): Promise<string | null> {\n const DictionaryClass = resolveEntityClass(ctx, 'Dictionary')\n const DictionaryEntryClass = resolveEntityClass(ctx, 'DictionaryEntry')\n if (!DictionaryClass || !DictionaryEntryClass) return null\n\n const encryptionScope = { tenantId: ctx.tenantId, organizationId: ctx.organizationId }\n\n const dictionary = await findOneWithDecryption(\n ctx.em,\n DictionaryClass,\n {\n key: SALES_SHIPMENT_STATUS_DICTIONARY_KEY,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n undefined,\n encryptionScope,\n )\n if (!dictionary) return null\n\n const entries = await findWithDecryption(\n ctx.em,\n DictionaryEntryClass,\n {\n dictionary: dictionary.id,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n },\n undefined,\n encryptionScope,\n )\n if (!entries.length) return null\n\n const normalizedTarget = normalizeDictionaryToken(statusLabel)\n const loweredTarget = statusLabel.trim().toLowerCase()\n\n const match = entries.find((entry) => {\n const label = entry.label.trim().toLowerCase()\n const value = entry.value.trim().toLowerCase()\n return (\n entry.normalizedValue === normalizedTarget ||\n label === loweredTarget ||\n value === loweredTarget\n )\n })\n\n return match?.id ?? null\n}\n\n// ---------------------------------------------------------------------------\n// Customer / contact resolution\n// ---------------------------------------------------------------------------\n\nexport async function resolveCustomerEntityIdByEmail(\n ctx: ExecutionHelperContext,\n email: string,\n): Promise<string | null> {\n const normalized = email.trim().toLowerCase()\n if (!normalized) return null\n\n const CustomerEntityClass = resolveEntityClass(ctx, 'CustomerEntity')\n if (!CustomerEntityClass) return null\n\n const entity = await findOneWithDecryption(\n ctx.em,\n CustomerEntityClass,\n {\n primaryEmail: normalized,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n undefined,\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n if (entity) return entity.id\n\n const candidates = await findWithDecryption(\n ctx.em,\n CustomerEntityClass,\n {\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n { limit: 100, orderBy: { createdAt: 'DESC' } },\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n const match = candidates.find(\n (e) => e.primaryEmail && e.primaryEmail.toLowerCase() === normalized,\n )\n return match?.id ?? null\n}\n\nexport async function resolveContactIdByNameAndType(\n ctx: ExecutionHelperContext,\n contactName: string,\n contactType: string,\n): Promise<string | null> {\n const CustomerEntityClass = resolveEntityClass(ctx, 'CustomerEntity')\n if (!CustomerEntityClass) return null\n\n const normalized = contactName.trim()\n if (!normalized) return null\n\n const entity = await findOneWithDecryption(\n ctx.em,\n CustomerEntityClass,\n {\n displayName: normalized,\n kind: contactType,\n tenantId: ctx.tenantId,\n organizationId: ctx.organizationId,\n deletedAt: null,\n },\n undefined,\n { tenantId: ctx.tenantId, organizationId: ctx.organizationId },\n )\n\n return entity?.id ?? null\n}\n\n// ---------------------------------------------------------------------------\n// Order line items\n// ---------------------------------------------------------------------------\n\nexport interface OrderLineItem {\n id: string\n name?: string | null\n}\n\nexport async function loadOrderLineItems(\n ctx: ExecutionHelperContext,\n orderId: string,\n): Promise<OrderLineItem[]> {\n try {\n const result = await executeCommand<Record<string, unknown>, { lines?: OrderLineItem[] }>(\n ctx,\n 'sales.orders.lines.list',\n { orderId, organizationId: ctx.organizationId, tenantId: ctx.tenantId },\n )\n return result.lines ?? []\n } catch {\n return []\n }\n}\n\nexport function matchLineItemByName(\n orderLines: OrderLineItem[],\n lineItemName: string,\n): string | null {\n const target = lineItemName.trim().toLowerCase()\n if (!target) return null\n\n const exact = orderLines.find((l) => (l.name || '').trim().toLowerCase() === target)\n if (exact) return exact.id\n\n const partial = orderLines.find((l) => {\n const name = (l.name || '').trim().toLowerCase()\n return name.includes(target) || target.includes(name)\n })\n return partial?.id ?? null\n}\n\n// ---------------------------------------------------------------------------\n// Data normalization utilities\n// ---------------------------------------------------------------------------\n\nexport function normalizeAddressSnapshot(\n address: Record<string, unknown>,\n): Record<string, unknown> {\n return {\n addressLine1: address.line1 ?? address.addressLine1 ?? '',\n addressLine2: address.line2 ?? address.addressLine2 ?? null,\n companyName: address.company ?? address.companyName ?? null,\n name: address.contactName ?? address.name ?? null,\n city: address.city ?? null,\n region: address.state ?? address.region ?? null,\n postalCode: address.postalCode ?? null,\n country: address.country ?? null,\n }\n}\n\nexport function parseDateToken(value?: string | null): Date | undefined {\n if (!value) return undefined\n const parsed = new Date(value)\n if (Number.isNaN(parsed.getTime())) return undefined\n return parsed\n}\n\nexport function parseNumberToken(value: string, fieldName: string): number {\n const parsed = Number(value)\n if (!Number.isFinite(parsed)) {\n throw new ExecutionError(`Invalid numeric value for ${fieldName}`, 400)\n }\n return parsed\n}\n\nexport function normalizeDictionaryToken(value: string): string {\n return value.trim().toLowerCase().replace(/[\\s-]+/g, '_')\n}\n\n// ---------------------------------------------------------------------------\n// Product discrepancy resolution (used by catalog inbox action handler)\n// ---------------------------------------------------------------------------\n\nexport async function resolveProductDiscrepanciesInProposal(\n em: EntityManager,\n proposalId: string,\n productTitle: string,\n productId: string,\n scope: { tenantId: string; organizationId: string },\n): Promise<void> {\n const { InboxDiscrepancy, InboxProposalAction } = await import('../data/entities')\n\n const discrepancies = await findWithDecryption(\n em,\n InboxDiscrepancy,\n {\n proposalId,\n type: 'product_not_found',\n resolved: false,\n tenantId: scope.tenantId,\n organizationId: scope.organizationId,\n },\n undefined,\n scope,\n )\n\n const normalizedTitle = productTitle.toLowerCase().trim()\n const matchingDiscrepancies = discrepancies.filter((d) => {\n const foundValue = (d.foundValue || '').toLowerCase().trim()\n return foundValue === normalizedTitle\n })\n\n if (matchingDiscrepancies.length === 0) return\n\n // Phase 1: flush scalar mutations before any queries to avoid UoW tracking loss (SPEC-018)\n for (const discrepancy of matchingDiscrepancies) {\n discrepancy.resolved = true\n }\n await em.flush()\n\n // Phase 2: update line item product IDs (involves findOneWithDecryption queries)\n const actionIds = matchingDiscrepancies\n .map((d) => d.actionId)\n .filter((id): id is string => !!id)\n\n for (const actionId of actionIds) {\n const action = await findOneWithDecryption(\n em,\n InboxProposalAction,\n { id: actionId, deletedAt: null },\n undefined,\n scope,\n )\n if (!action) continue\n\n const payload = action.payload as Record<string, unknown>\n const lineItems = Array.isArray(payload?.lineItems)\n ? (payload.lineItems as Record<string, unknown>[])\n : []\n\n let updated = false\n for (const item of lineItems) {\n if (item.productId) continue\n const itemName = (typeof item.productName === 'string' ? item.productName : '').toLowerCase().trim()\n if (itemName === normalizedTitle) {\n item.productId = productId\n updated = true\n break\n }\n }\n\n if (updated) {\n action.payload = { ...payload, lineItems }\n }\n }\n\n if (actionIds.length > 0) {\n await em.flush()\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,19 +1,44 @@
|
|
|
1
1
|
import { findWithDecryption } from "@open-mercato/shared/lib/encryption/find";
|
|
2
2
|
const DEFAULT_MISMATCH_THRESHOLD = 0.05;
|
|
3
|
+
function readUuid(value) {
|
|
4
|
+
return typeof value === "string" && value.trim() ? value.trim() : void 0;
|
|
5
|
+
}
|
|
6
|
+
function parsePositiveQuantity(value) {
|
|
7
|
+
const parsed = typeof value === "number" ? value : typeof value === "string" ? parseFloat(value) : NaN;
|
|
8
|
+
return Number.isFinite(parsed) && parsed > 0 ? parsed : 1;
|
|
9
|
+
}
|
|
3
10
|
async function validatePrices(em, actions, scope, deps) {
|
|
4
|
-
if (!deps?.catalogProductPriceClass) return [];
|
|
11
|
+
if (!deps?.catalogProductPriceClass || !deps?.catalogPricingService) return [];
|
|
5
12
|
const threshold = parseFloat(process.env.INBOX_OPS_PRICE_MISMATCH_THRESHOLD || String(DEFAULT_MISMATCH_THRESHOLD));
|
|
6
13
|
const discrepancies = [];
|
|
14
|
+
const now = /* @__PURE__ */ new Date();
|
|
7
15
|
for (const action of actions) {
|
|
8
16
|
if (action.actionType !== "create_order" && action.actionType !== "create_quote") {
|
|
9
17
|
continue;
|
|
10
18
|
}
|
|
11
19
|
const lineItems = action.payload.lineItems || [];
|
|
12
20
|
const orderCurrency = typeof action.payload.currencyCode === "string" ? action.payload.currencyCode.trim().toUpperCase() : null;
|
|
21
|
+
const channelId = readUuid(action.payload.channelId) ?? scope.channelId ?? null;
|
|
22
|
+
const customerId = readUuid(action.payload.customerEntityId) ?? scope.customerId ?? null;
|
|
13
23
|
for (const line of lineItems) {
|
|
14
24
|
if (!line.productId || !line.unitPrice) continue;
|
|
15
25
|
try {
|
|
16
|
-
const
|
|
26
|
+
const pricingContext = {
|
|
27
|
+
channelId,
|
|
28
|
+
customerId,
|
|
29
|
+
customerGroupId: null,
|
|
30
|
+
quantity: parsePositiveQuantity(line.quantity),
|
|
31
|
+
date: now
|
|
32
|
+
};
|
|
33
|
+
const catalogResult = await lookupCatalogPrice(
|
|
34
|
+
em,
|
|
35
|
+
line.productId,
|
|
36
|
+
scope,
|
|
37
|
+
deps.catalogProductPriceClass,
|
|
38
|
+
deps.catalogPricingService,
|
|
39
|
+
pricingContext,
|
|
40
|
+
orderCurrency
|
|
41
|
+
);
|
|
17
42
|
if (catalogResult === null) continue;
|
|
18
43
|
if (orderCurrency && catalogResult.currencyCode && orderCurrency !== catalogResult.currencyCode.toUpperCase()) {
|
|
19
44
|
discrepancies.push({
|
|
@@ -47,9 +72,9 @@ async function validatePrices(em, actions, scope, deps) {
|
|
|
47
72
|
}
|
|
48
73
|
return discrepancies;
|
|
49
74
|
}
|
|
50
|
-
async function lookupCatalogPrice(em, productId, scope, entityClass) {
|
|
75
|
+
async function lookupCatalogPrice(em, productId, scope, entityClass, pricingService, pricingContext, orderCurrency) {
|
|
51
76
|
try {
|
|
52
|
-
const
|
|
77
|
+
const rows = await findWithDecryption(
|
|
53
78
|
em,
|
|
54
79
|
entityClass,
|
|
55
80
|
{
|
|
@@ -58,11 +83,14 @@ async function lookupCatalogPrice(em, productId, scope, entityClass) {
|
|
|
58
83
|
organizationId: scope.organizationId,
|
|
59
84
|
tenantId: scope.tenantId
|
|
60
85
|
},
|
|
61
|
-
{
|
|
86
|
+
{ populate: ["priceKind", "offer"] },
|
|
62
87
|
{ tenantId: scope.tenantId, organizationId: scope.organizationId }
|
|
63
88
|
);
|
|
64
|
-
if (!
|
|
65
|
-
const
|
|
89
|
+
if (!rows || rows.length === 0) return null;
|
|
90
|
+
const sameCurrencyRows = orderCurrency ? rows.filter((row) => (row.currencyCode ?? "").toUpperCase() === orderCurrency) : rows;
|
|
91
|
+
const candidateRows = sameCurrencyRows.length > 0 ? sameCurrencyRows : rows;
|
|
92
|
+
const priceRecord = await pricingService.resolvePrice(candidateRows, pricingContext);
|
|
93
|
+
if (!priceRecord) return null;
|
|
66
94
|
const priceValue = priceRecord.unitPriceNet ?? priceRecord.unitPriceGross ?? null;
|
|
67
95
|
if (!priceValue) return null;
|
|
68
96
|
return {
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/modules/inbox_ops/lib/priceValidator.ts"],
|
|
4
|
-
"sourcesContent": ["import type { EntityManager } from '@mikro-orm/postgresql'\nimport type { EntityClass } from '@mikro-orm/core'\nimport { findWithDecryption } from '@open-mercato/shared/lib/encryption/find'\nimport type { InboxDiscrepancyType } from '../data/entities'\n\nexport interface PriceDiscrepancy {\n type: InboxDiscrepancyType\n severity: 'warning' | 'error'\n description: string\n expectedValue?: string\n foundValue?: string\n actionIndex: number\n}\n\ninterface LineItem {\n productName: string\n productId?: string\n unitPrice?: string\n quantity: string\n currencyCode?: string\n}\n\ninterface PriceValidatorScope {\n tenantId: string\n organizationId: string\n channelId?: string\n customerId?: string\n}\n\ninterface CatalogProductPriceLike {\n product?: unknown\n unitPriceNet?: string | null\n unitPriceGross?: string | null\n currencyCode?: string | null\n tenantId?: string\n organizationId?: string\n deletedAt?: Date | null\n createdAt?: Date\n}\n\nconst DEFAULT_MISMATCH_THRESHOLD = 0.05 // 5%\n\nexport async function validatePrices(\n em: EntityManager,\n actions: { actionType: string; payload: Record<string, unknown>; index: number }[],\n scope: PriceValidatorScope,\n deps?:
|
|
5
|
-
"mappings": "AAEA,SAAS,0BAA0B;
|
|
4
|
+
"sourcesContent": ["import type { EntityManager } from '@mikro-orm/postgresql'\nimport type { EntityClass } from '@mikro-orm/core'\nimport { findWithDecryption } from '@open-mercato/shared/lib/encryption/find'\nimport type { InboxDiscrepancyType } from '../data/entities'\n\nexport interface PriceDiscrepancy {\n type: InboxDiscrepancyType\n severity: 'warning' | 'error'\n description: string\n expectedValue?: string\n foundValue?: string\n actionIndex: number\n}\n\ninterface LineItem {\n productName: string\n productId?: string\n unitPrice?: string\n quantity: string\n currencyCode?: string\n}\n\ninterface PriceValidatorScope {\n tenantId: string\n organizationId: string\n channelId?: string\n customerId?: string\n}\n\ninterface CatalogProductPriceLike {\n product?: unknown\n priceKind?: unknown\n offer?: unknown\n unitPriceNet?: string | null\n unitPriceGross?: string | null\n currencyCode?: string | null\n tenantId?: string\n organizationId?: string\n deletedAt?: Date | null\n createdAt?: Date\n}\n\n// Minimal structural view of catalog's `PricingContext`. The validator resolves\n// prices through the catalog pricing engine so channel/customer/quantity/time/kind\n// scoping is honored consistently with the rest of the platform.\nexport interface CatalogPricingContext {\n channelId?: string | null\n customerId?: string | null\n customerGroupId?: string | null\n quantity: number\n date: Date\n}\n\n// Minimal structural view of the `catalogPricingService` DI token. Using the engine\n// (selectBestPrice + resolver pipeline) instead of a bespoke query is mandated by\n// packages/core/src/modules/catalog/AGENTS.md and ensures tenant pricing overrides apply.\nexport interface CatalogPricingServiceLike {\n resolvePrice(\n rows: CatalogProductPriceLike[],\n context: CatalogPricingContext,\n ): Promise<CatalogProductPriceLike | null>\n}\n\ninterface PriceValidatorDeps {\n catalogProductPriceClass: EntityClass<CatalogProductPriceLike>\n catalogPricingService: CatalogPricingServiceLike\n}\n\nconst DEFAULT_MISMATCH_THRESHOLD = 0.05 // 5%\n\nfunction readUuid(value: unknown): string | undefined {\n return typeof value === 'string' && value.trim() ? value.trim() : undefined\n}\n\nfunction parsePositiveQuantity(value: unknown): number {\n const parsed = typeof value === 'number' ? value : typeof value === 'string' ? parseFloat(value) : NaN\n return Number.isFinite(parsed) && parsed > 0 ? parsed : 1\n}\n\nexport async function validatePrices(\n em: EntityManager,\n actions: { actionType: string; payload: Record<string, unknown>; index: number }[],\n scope: PriceValidatorScope,\n deps?: PriceValidatorDeps,\n): Promise<PriceDiscrepancy[]> {\n if (!deps?.catalogProductPriceClass || !deps?.catalogPricingService) return []\n\n const threshold = parseFloat(process.env.INBOX_OPS_PRICE_MISMATCH_THRESHOLD || String(DEFAULT_MISMATCH_THRESHOLD))\n const discrepancies: PriceDiscrepancy[] = []\n const now = new Date()\n\n for (const action of actions) {\n if (action.actionType !== 'create_order' && action.actionType !== 'create_quote') {\n continue\n }\n\n const lineItems = (action.payload.lineItems as LineItem[]) || []\n\n const orderCurrency = typeof action.payload.currencyCode === 'string'\n ? action.payload.currencyCode.trim().toUpperCase()\n : null\n\n // Channel and customer scoping come from the proposal payload (or the caller's\n // scope as a fallback). When unknown, the engine correctly excludes channel/\n // customer-specific rows and falls back to the generally-applicable price.\n const channelId = readUuid(action.payload.channelId) ?? scope.channelId ?? null\n const customerId = readUuid(action.payload.customerEntityId) ?? scope.customerId ?? null\n\n for (const line of lineItems) {\n if (!line.productId || !line.unitPrice) continue\n\n try {\n const pricingContext: CatalogPricingContext = {\n channelId,\n customerId,\n customerGroupId: null,\n quantity: parsePositiveQuantity(line.quantity),\n date: now,\n }\n const catalogResult = await lookupCatalogPrice(\n em,\n line.productId,\n scope,\n deps.catalogProductPriceClass,\n deps.catalogPricingService,\n pricingContext,\n orderCurrency,\n )\n if (catalogResult === null) continue\n\n if (orderCurrency && catalogResult.currencyCode && orderCurrency !== catalogResult.currencyCode.toUpperCase()) {\n discrepancies.push({\n type: 'currency_mismatch',\n severity: 'warning',\n description: `Currency mismatch for \"${line.productName}\": order uses ${orderCurrency} but catalog price is in ${catalogResult.currencyCode.toUpperCase()}`,\n expectedValue: catalogResult.currencyCode.toUpperCase(),\n foundValue: orderCurrency,\n actionIndex: action.index,\n })\n continue\n }\n\n const extractedPrice = parseFloat(line.unitPrice)\n const catPrice = parseFloat(catalogResult.price)\n\n if (isNaN(extractedPrice) || isNaN(catPrice) || catPrice === 0) continue\n\n const priceDiff = Math.abs(extractedPrice - catPrice) / catPrice\n\n if (priceDiff > threshold) {\n const percentDiff = (priceDiff * 100).toFixed(1)\n discrepancies.push({\n type: 'price_mismatch',\n severity: priceDiff > 0.2 ? 'error' : 'warning',\n description: `Price for \"${line.productName}\": email says ${line.unitPrice} but catalog price is ${catalogResult.price} (${percentDiff}% difference)`,\n expectedValue: catalogResult.price,\n foundValue: line.unitPrice,\n actionIndex: action.index,\n })\n }\n } catch {\n // Skip price validation if lookup fails\n }\n }\n }\n\n return discrepancies\n}\n\nasync function lookupCatalogPrice(\n em: EntityManager,\n productId: string,\n scope: PriceValidatorScope,\n entityClass: EntityClass<CatalogProductPriceLike>,\n pricingService: CatalogPricingServiceLike,\n pricingContext: CatalogPricingContext,\n orderCurrency: string | null,\n): Promise<{ price: string; currencyCode: string | null } | null> {\n try {\n // Load all non-deleted price rows for the product (populating the relations the\n // pricing engine scores on), then let the engine pick the applicable one \u2014 never\n // the \"most recently created\" row, which ignores every pricing dimension (#2737).\n const rows = await findWithDecryption(\n em,\n entityClass,\n {\n product: productId,\n deletedAt: null,\n organizationId: scope.organizationId,\n tenantId: scope.tenantId,\n },\n { populate: ['priceKind', 'offer'] },\n { tenantId: scope.tenantId, organizationId: scope.organizationId },\n )\n\n if (!rows || rows.length === 0) return null\n\n // Prefer rows in the order's currency so we compare like-for-like instead of\n // raising a spurious currency mismatch against an unrelated-currency row; fall\n // back to all rows (which surfaces a genuine mismatch when no match exists).\n const sameCurrencyRows = orderCurrency\n ? rows.filter((row) => (row.currencyCode ?? '').toUpperCase() === orderCurrency)\n : rows\n const candidateRows = sameCurrencyRows.length > 0 ? sameCurrencyRows : rows\n\n const priceRecord = await pricingService.resolvePrice(candidateRows, pricingContext)\n if (!priceRecord) return null\n\n const priceValue = priceRecord.unitPriceNet ?? priceRecord.unitPriceGross ?? null\n if (!priceValue) return null\n\n return {\n price: priceValue,\n currencyCode: priceRecord.currencyCode ?? null,\n }\n } catch {\n return null\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|