@elevasis/core 0.23.0 → 0.24.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.d.ts +4343 -2690
- package/dist/index.js +1101 -156
- package/dist/knowledge/index.d.ts +574 -210
- package/dist/knowledge/index.js +104 -1
- package/dist/organization-model/index.d.ts +4343 -2690
- package/dist/organization-model/index.js +1101 -156
- package/dist/test-utils/index.d.ts +483 -109
- package/dist/test-utils/index.js +904 -144
- package/package.json +3 -3
- package/src/README.md +14 -14
- package/src/__tests__/publish.test.ts +24 -24
- package/src/__tests__/template-core-compatibility.test.ts +9 -12
- package/src/_gen/__tests__/__snapshots__/contracts.md.snap +2137 -2093
- package/src/_gen/__tests__/scaffold-contracts.test.ts +30 -30
- package/src/auth/multi-tenancy/credentials/__tests__/encryption.test.ts +217 -217
- package/src/auth/multi-tenancy/credentials/server/encryption.ts +69 -69
- package/src/auth/multi-tenancy/credentials/server/kek-loader.ts +37 -37
- package/src/auth/multi-tenancy/index.ts +26 -26
- package/src/auth/multi-tenancy/invitations/api-schemas.ts +104 -104
- package/src/auth/multi-tenancy/memberships/api-schemas.ts +143 -143
- package/src/auth/multi-tenancy/memberships/index.ts +26 -26
- package/src/auth/multi-tenancy/memberships/membership.ts +130 -130
- package/src/auth/multi-tenancy/organizations/__tests__/api-schemas.test.ts +194 -194
- package/src/auth/multi-tenancy/organizations/api-schemas.ts +136 -136
- package/src/auth/multi-tenancy/permissions.test.ts +42 -42
- package/src/auth/multi-tenancy/permissions.ts +123 -123
- package/src/auth/multi-tenancy/role-management/api-schemas.ts +78 -78
- package/src/auth/multi-tenancy/role-management/index.ts +16 -16
- package/src/auth/multi-tenancy/theme-presets.ts +45 -45
- package/src/auth/multi-tenancy/types.ts +57 -57
- package/src/auth/multi-tenancy/users/api-schemas.ts +165 -165
- package/src/business/README.md +2 -2
- package/src/business/acquisition/activity-events.test.ts +250 -250
- package/src/business/acquisition/activity-events.ts +93 -93
- package/src/business/acquisition/api-schemas.test.ts +1883 -1843
- package/src/business/acquisition/api-schemas.ts +1492 -1497
- package/src/business/acquisition/build-templates.test.ts +240 -240
- package/src/business/acquisition/build-templates.ts +98 -98
- package/src/business/acquisition/crm-next-action.test.ts +262 -262
- package/src/business/acquisition/crm-next-action.ts +220 -220
- package/src/business/acquisition/crm-priority.test.ts +216 -216
- package/src/business/acquisition/crm-priority.ts +349 -349
- package/src/business/acquisition/crm-state-actions.test.ts +153 -153
- package/src/business/acquisition/deal-ownership.test.ts +351 -351
- package/src/business/acquisition/deal-ownership.ts +120 -120
- package/src/business/acquisition/derive-actions.test.ts +129 -104
- package/src/business/acquisition/derive-actions.ts +74 -84
- package/src/business/acquisition/index.ts +171 -170
- package/src/business/acquisition/ontology-validation.ts +309 -0
- package/src/business/acquisition/stateful.ts +30 -30
- package/src/business/acquisition/types.ts +396 -396
- package/src/business/clients/api-schemas.test.ts +115 -115
- package/src/business/clients/api-schemas.ts +158 -158
- package/src/business/clients/index.ts +1 -1
- package/src/business/crm/api-schemas.ts +40 -40
- package/src/business/crm/index.ts +1 -1
- package/src/business/deals/api-schemas.ts +87 -87
- package/src/business/deals/index.ts +1 -1
- package/src/business/index.ts +5 -5
- package/src/business/projects/types.ts +144 -144
- package/src/commands/queue/types/task.ts +15 -15
- package/src/execution/core/runner-types.ts +61 -61
- package/src/execution/core/sse-executions.ts +7 -7
- package/src/execution/engine/__tests__/fixtures/test-agents.ts +10 -10
- package/src/execution/engine/agent/core/__tests__/agent.test.ts +16 -16
- package/src/execution/engine/agent/core/__tests__/error-passthrough.test.ts +4 -4
- package/src/execution/engine/agent/core/types.ts +25 -25
- package/src/execution/engine/agent/index.ts +6 -6
- package/src/execution/engine/agent/reasoning/__tests__/request-builder.test.ts +24 -24
- package/src/execution/engine/index.ts +443 -443
- package/src/execution/engine/tools/integration/server/adapters/apify/__tests__/apify-run-actor.integration.test.ts +298 -298
- package/src/execution/engine/tools/integration/server/adapters/apify/apify-adapter.test.ts +55 -55
- package/src/execution/engine/tools/integration/server/adapters/apify/apify-adapter.ts +107 -107
- package/src/execution/engine/tools/integration/server/adapters/apollo/apollo-adapter.test.ts +48 -48
- package/src/execution/engine/tools/integration/server/adapters/apollo/apollo-adapter.ts +99 -99
- package/src/execution/engine/tools/integration/server/adapters/apollo/index.ts +1 -1
- package/src/execution/engine/tools/integration/server/adapters/attio/__tests__/attio-crud.integration.test.ts +363 -363
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/get-record/index.test.ts +162 -162
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/list-records/index.test.ts +316 -316
- package/src/execution/engine/tools/integration/server/adapters/clickup/clickup-adapter.test.ts +18 -18
- package/src/execution/engine/tools/integration/server/adapters/clickup/clickup-adapter.ts +194 -194
- package/src/execution/engine/tools/integration/server/adapters/clickup/index.ts +7 -7
- package/src/execution/engine/tools/integration/server/adapters/gmail/gmail-adapter.ts +204 -204
- package/src/execution/engine/tools/integration/server/adapters/gmail/gmail-tools.ts +105 -105
- package/src/execution/engine/tools/integration/server/adapters/google-calendar/google-calendar-adapter.ts +428 -428
- package/src/execution/engine/tools/integration/server/adapters/google-calendar/index.ts +2 -2
- package/src/execution/engine/tools/integration/server/adapters/google-sheets/__tests__/google-sheets.integration.test.ts +261 -261
- package/src/execution/engine/tools/integration/server/adapters/instantly/instantly-tools.ts +1474 -1474
- package/src/execution/engine/tools/integration/server/adapters/millionverifier/millionverifier-tools.ts +103 -103
- package/src/execution/engine/tools/integration/server/adapters/resend/fetch/send-email/index.test.ts +88 -88
- package/src/execution/engine/tools/integration/server/adapters/resend/fetch/send-email/index.ts +141 -141
- package/src/execution/engine/tools/integration/server/adapters/resend/fetch/utils/types.ts +76 -76
- package/src/execution/engine/tools/integration/server/adapters/signature-api/signature-api-tools.ts +182 -182
- package/src/execution/engine/tools/integration/server/adapters/stripe/stripe-tools.ts +310 -310
- package/src/execution/engine/tools/integration/service.test.ts +239 -239
- package/src/execution/engine/tools/integration/service.ts +172 -172
- package/src/execution/engine/tools/integration/tool.ts +255 -255
- package/src/execution/engine/tools/lead-service-types.ts +1005 -1005
- package/src/execution/engine/tools/messages.ts +43 -43
- package/src/execution/engine/tools/platform/acquisition/company-tools.ts +7 -7
- package/src/execution/engine/tools/platform/acquisition/contact-tools.ts +6 -6
- package/src/execution/engine/tools/platform/acquisition/list-tools.ts +6 -6
- package/src/execution/engine/tools/platform/acquisition/types.ts +280 -280
- package/src/execution/engine/tools/platform/email/types.ts +97 -97
- package/src/execution/engine/tools/registry.ts +704 -704
- package/src/execution/engine/tools/tool-maps.ts +831 -831
- package/src/execution/engine/tools/types.ts +234 -234
- package/src/execution/engine/workflow/types.ts +195 -197
- package/src/execution/external/__tests__/api-schemas.test.ts +127 -127
- package/src/execution/external/api-schemas.ts +40 -40
- package/src/execution/external/index.ts +1 -1
- package/src/index.ts +18 -18
- package/src/integrations/credentials/__tests__/api-schemas.test.ts +420 -420
- package/src/integrations/credentials/api-schemas.ts +146 -146
- package/src/integrations/credentials/schemas.ts +200 -200
- package/src/integrations/oauth/__tests__/provider-registry.test.ts +7 -7
- package/src/integrations/oauth/provider-registry.ts +74 -74
- package/src/integrations/oauth/server/credentials.ts +43 -43
- package/src/integrations/webhook-endpoints/__tests__/api-schemas.test.ts +327 -327
- package/src/integrations/webhook-endpoints/api-schemas.ts +103 -103
- package/src/integrations/webhook-endpoints/types.ts +58 -58
- package/src/knowledge/README.md +32 -32
- package/src/knowledge/__tests__/queries.test.ts +626 -535
- package/src/knowledge/format.ts +99 -99
- package/src/knowledge/index.ts +5 -5
- package/src/knowledge/published.ts +5 -5
- package/src/knowledge/queries.ts +269 -218
- package/src/operations/activities/api-schemas.ts +80 -80
- package/src/operations/activities/types.ts +64 -64
- package/src/organization-model/README.md +149 -149
- package/src/organization-model/__tests__/content-kinds-registry.test.ts +210 -210
- package/src/organization-model/__tests__/defaults.test.ts +168 -168
- package/src/organization-model/__tests__/domains/actions.test.ts +78 -56
- package/src/organization-model/__tests__/domains/customers.test.ts +299 -299
- package/src/organization-model/__tests__/domains/entities.test.ts +56 -56
- package/src/organization-model/__tests__/domains/goals.test.ts +493 -493
- package/src/organization-model/__tests__/domains/identity.test.ts +280 -280
- package/src/organization-model/__tests__/domains/navigation.test.ts +268 -268
- package/src/organization-model/__tests__/domains/offerings.test.ts +414 -414
- package/src/organization-model/__tests__/domains/policies.test.ts +323 -323
- package/src/organization-model/__tests__/domains/resource-mappings.test.ts +293 -293
- package/src/organization-model/__tests__/domains/resources.test.ts +387 -277
- package/src/organization-model/__tests__/domains/roles.test.ts +463 -463
- package/src/organization-model/__tests__/domains/statuses.test.ts +246 -246
- package/src/organization-model/__tests__/domains/systems.test.ts +209 -209
- package/src/organization-model/__tests__/domains/topology.test.ts +188 -0
- package/src/organization-model/__tests__/flatten-additive-merge.test.ts +362 -361
- package/src/organization-model/__tests__/foundation.test.ts +77 -77
- package/src/organization-model/__tests__/get-resources-for-system.test.ts +144 -144
- package/src/organization-model/__tests__/graph.test.ts +1312 -862
- package/src/organization-model/__tests__/icons.test.ts +10 -1
- package/src/organization-model/__tests__/knowledge.test.ts +251 -15
- package/src/organization-model/__tests__/lookup-helpers.test.ts +438 -438
- package/src/organization-model/__tests__/migration-helpers.test.ts +591 -591
- package/src/organization-model/__tests__/prospecting-ssot.test.ts +103 -103
- package/src/organization-model/__tests__/recursive-system-schema.test.ts +535 -506
- package/src/organization-model/__tests__/resolve.test.ts +274 -164
- package/src/organization-model/__tests__/schema.test.ts +844 -301
- package/src/organization-model/__tests__/surface-projection.test.ts +284 -284
- package/src/organization-model/catalogs/lead-gen.ts +144 -144
- package/src/organization-model/content-kinds/config.ts +36 -36
- package/src/organization-model/content-kinds/index.ts +76 -72
- package/src/organization-model/content-kinds/pipeline.ts +68 -68
- package/src/organization-model/content-kinds/registry.ts +44 -44
- package/src/organization-model/content-kinds/status.ts +71 -71
- package/src/organization-model/content-kinds/template.ts +83 -83
- package/src/organization-model/content-kinds/types.ts +117 -117
- package/src/organization-model/contracts.ts +27 -27
- package/src/organization-model/defaults.ts +42 -50
- package/src/organization-model/domains/actions.ts +333 -239
- package/src/organization-model/domains/customers.ts +78 -78
- package/src/organization-model/domains/entities.ts +144 -144
- package/src/organization-model/domains/goals.ts +83 -83
- package/src/organization-model/domains/knowledge.ts +117 -101
- package/src/organization-model/domains/navigation.ts +139 -139
- package/src/organization-model/domains/offerings.ts +71 -71
- package/src/organization-model/domains/policies.ts +102 -102
- package/src/organization-model/domains/projects.ts +14 -14
- package/src/organization-model/domains/prospecting.ts +395 -395
- package/src/organization-model/domains/resources.ts +202 -124
- package/src/organization-model/domains/roles.ts +96 -96
- package/src/organization-model/domains/sales.test.ts +218 -218
- package/src/organization-model/domains/sales.ts +380 -380
- package/src/organization-model/domains/shared.ts +63 -63
- package/src/organization-model/domains/statuses.ts +339 -339
- package/src/organization-model/domains/systems.ts +217 -172
- package/src/organization-model/domains/topology.ts +261 -0
- package/src/organization-model/foundation.ts +75 -75
- package/src/organization-model/graph/build.ts +1043 -867
- package/src/organization-model/graph/index.ts +4 -4
- package/src/organization-model/graph/link.ts +10 -10
- package/src/organization-model/graph/schema.ts +75 -68
- package/src/organization-model/graph/types.ts +71 -64
- package/src/organization-model/helpers.ts +289 -241
- package/src/organization-model/icons.ts +78 -66
- package/src/organization-model/index.ts +128 -125
- package/src/organization-model/migration-helpers.ts +247 -244
- package/src/organization-model/ontology.ts +658 -0
- package/src/organization-model/organization-graph.mdx +110 -90
- package/src/organization-model/organization-model.mdx +225 -213
- package/src/organization-model/published.ts +299 -222
- package/src/organization-model/resolve.ts +146 -91
- package/src/organization-model/schema.ts +818 -659
- package/src/organization-model/surface-projection.ts +212 -212
- package/src/organization-model/types.ts +179 -155
- package/src/platform/api/types.ts +38 -38
- package/src/platform/constants/versions.ts +3 -3
- package/src/platform/index.ts +23 -23
- package/src/platform/registry/__tests__/command-view.test.ts +10 -10
- package/src/platform/registry/__tests__/resource-link.test.ts +35 -35
- package/src/platform/registry/__tests__/resource-registry.integration.test.ts +20 -20
- package/src/platform/registry/__tests__/resource-registry.nested-systems.test.ts +245 -245
- package/src/platform/registry/__tests__/resource-registry.test.ts +2053 -2053
- package/src/platform/registry/__tests__/validation.test.ts +1444 -1259
- package/src/platform/registry/command-view.ts +10 -10
- package/src/platform/registry/index.ts +103 -103
- package/src/platform/registry/resource-link.ts +32 -32
- package/src/platform/registry/resource-registry.ts +886 -886
- package/src/platform/registry/serialization.ts +295 -295
- package/src/platform/registry/serialized-types.ts +166 -166
- package/src/platform/registry/stats-types.ts +68 -68
- package/src/platform/registry/types.ts +425 -425
- package/src/platform/registry/validation.ts +876 -684
- package/src/platform/utils/__tests__/validation.test.ts +1084 -1084
- package/src/platform/utils/validation.ts +425 -425
- package/src/projects/api-schemas.test.ts +39 -39
- package/src/projects/api-schemas.ts +291 -291
- package/src/reference/_generated/contracts.md +2136 -2093
- package/src/reference/glossary.md +76 -76
- package/src/scaffold-registry/__tests__/index.test.ts +206 -206
- package/src/scaffold-registry/__tests__/schema.test.ts +166 -166
- package/src/scaffold-registry/index.ts +392 -392
- package/src/scaffold-registry/schema.ts +243 -243
- package/src/server.ts +289 -289
- package/src/supabase/database.types.ts +3 -0
- package/src/test-utils/README.md +37 -37
- package/src/test-utils/entities.ts +108 -108
- package/src/test-utils/fixtures/memberships.ts +82 -82
- package/src/test-utils/index.ts +12 -12
- package/src/test-utils/organization-model.ts +65 -65
- package/src/test-utils/published.ts +6 -6
- package/src/test-utils/rls/RLSTestContext.ts +588 -588
- package/src/test-utils/test-utils.test.ts +44 -44
|
@@ -1,103 +1,103 @@
|
|
|
1
|
-
import { z } from 'zod'
|
|
2
|
-
import type { Tool } from '../../../../types'
|
|
3
|
-
import { createIntegrationTool } from '../../../tool'
|
|
4
|
-
import { EmailSchema } from '../../../../../../../platform/utils/validation'
|
|
5
|
-
|
|
6
|
-
/**
|
|
7
|
-
* Create a tool that verifies email addresses via MillionVerifier
|
|
8
|
-
*
|
|
9
|
-
* @param credentialName - Name of the MillionVerifier credential to use
|
|
10
|
-
* @returns Tool instance
|
|
11
|
-
*
|
|
12
|
-
* @example
|
|
13
|
-
* ```typescript
|
|
14
|
-
* const tool = createMillionVerifierVerifyEmailTool('elevasis-millionverifier')
|
|
15
|
-
* const result = await tool.execute({
|
|
16
|
-
* email: 'john@example.com'
|
|
17
|
-
* }, context)
|
|
18
|
-
* // result.result: 'ok' | 'catch_all' | 'unknown' | 'error' | 'disposable' | 'invalid'
|
|
19
|
-
* // result.quality: 'good' | 'bad' | 'risky' | ''
|
|
20
|
-
* ```
|
|
21
|
-
*/
|
|
22
|
-
export function createMillionVerifierVerifyEmailTool(credentialName: string): Tool {
|
|
23
|
-
return createIntegrationTool({
|
|
24
|
-
name: 'millionverifier_verify_email',
|
|
25
|
-
description: `Verify email address deliverability using MillionVerifier.
|
|
26
|
-
|
|
27
|
-
Use this tool to validate if an email address is real and can receive emails before
|
|
28
|
-
adding it to a campaign. MillionVerifier is highly accurate and fast (160 req/s).
|
|
29
|
-
|
|
30
|
-
Result codes:
|
|
31
|
-
- ok (1): Email is valid and deliverable
|
|
32
|
-
- catch_all (2): Domain accepts all email (cannot confirm individual mailbox) - FREE
|
|
33
|
-
- unknown (3): Result could not be determined - FREE
|
|
34
|
-
- error (4): Verification error
|
|
35
|
-
- disposable (5): Disposable/temporary email address
|
|
36
|
-
- invalid (6): Email address is invalid
|
|
37
|
-
|
|
38
|
-
Quality signals:
|
|
39
|
-
- good: Safe to send
|
|
40
|
-
- risky: May bounce or flag spam filter
|
|
41
|
-
- bad: Do not send
|
|
42
|
-
|
|
43
|
-
Credits are only deducted for ok, invalid, and disposable results.
|
|
44
|
-
|
|
45
|
-
Returns full verification details including free/role detection and suggested correction.`,
|
|
46
|
-
inputSchema: z.object({
|
|
47
|
-
email: EmailSchema.describe('Email address to verify')
|
|
48
|
-
}),
|
|
49
|
-
outputSchema: z.object({
|
|
50
|
-
email: z.string().describe('Verified email address'),
|
|
51
|
-
quality: z.enum(['', 'good', 'bad', 'risky']).describe('Quality signal for sending decision'),
|
|
52
|
-
result: z.enum(['ok', 'catch_all', 'unknown', 'error', 'disposable', 'invalid']).describe('Verification result'),
|
|
53
|
-
resultCode: z
|
|
54
|
-
.number()
|
|
55
|
-
.describe('Numeric result code (1=ok, 2=catch_all, 3=unknown, 4=error, 5=disposable, 6=invalid)'),
|
|
56
|
-
subresult: z.string().describe('Detailed sub-result (e.g., ok, no_mailbox, dns_no_mx)'),
|
|
57
|
-
free: z.boolean().describe('Whether this is a free email provider (Gmail, Yahoo, etc.)'),
|
|
58
|
-
role: z.boolean().describe('Whether this is a role-based address (info@, support@, etc.)'),
|
|
59
|
-
didYouMean: z.string().describe('Suggested correction if email looks like a typo (empty if none)'),
|
|
60
|
-
credits: z.number().describe('Remaining credits after this verification'),
|
|
61
|
-
executionTime: z.number().describe('Time taken to verify in milliseconds'),
|
|
62
|
-
error: z.string().describe('Error message when result is error (empty otherwise)'),
|
|
63
|
-
liveMode: z.boolean().describe('Whether the verification was performed in live mode')
|
|
64
|
-
}),
|
|
65
|
-
integration: 'millionverifier' as const,
|
|
66
|
-
method: 'verifyEmail' as const,
|
|
67
|
-
credentialName
|
|
68
|
-
})
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
/**
|
|
72
|
-
* Create a tool that checks remaining MillionVerifier credits
|
|
73
|
-
*
|
|
74
|
-
* @param credentialName - Name of the MillionVerifier credential to use
|
|
75
|
-
* @returns Tool instance
|
|
76
|
-
*
|
|
77
|
-
* @example
|
|
78
|
-
* ```typescript
|
|
79
|
-
* const tool = createMillionVerifierCheckCreditsTool('elevasis-millionverifier')
|
|
80
|
-
* const result = await tool.execute({}, context)
|
|
81
|
-
* // result.credits: 5000
|
|
82
|
-
* ```
|
|
83
|
-
*/
|
|
84
|
-
export function createMillionVerifierCheckCreditsTool(credentialName: string): Tool {
|
|
85
|
-
return createIntegrationTool({
|
|
86
|
-
name: 'millionverifier_check_credits',
|
|
87
|
-
description: `Check remaining MillionVerifier verification credits.
|
|
88
|
-
|
|
89
|
-
Use this tool to monitor credit balance before running large verification batches.
|
|
90
|
-
|
|
91
|
-
Returns current credits, bulk credits, renewing credits, and plan tier.`,
|
|
92
|
-
inputSchema: z.object({}),
|
|
93
|
-
outputSchema: z.object({
|
|
94
|
-
credits: z.number().describe('Remaining single-verification credits'),
|
|
95
|
-
bulkCredits: z.number().describe('Remaining bulk verification credits'),
|
|
96
|
-
renewingCredits: z.number().describe('Credits that will renew on next billing cycle'),
|
|
97
|
-
plan: z.number().describe('Current plan tier')
|
|
98
|
-
}),
|
|
99
|
-
integration: 'millionverifier' as const,
|
|
100
|
-
method: 'checkCredits' as const,
|
|
101
|
-
credentialName
|
|
102
|
-
})
|
|
103
|
-
}
|
|
1
|
+
import { z } from 'zod'
|
|
2
|
+
import type { Tool } from '../../../../types'
|
|
3
|
+
import { createIntegrationTool } from '../../../tool'
|
|
4
|
+
import { EmailSchema } from '../../../../../../../platform/utils/validation'
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Create a tool that verifies email addresses via MillionVerifier
|
|
8
|
+
*
|
|
9
|
+
* @param credentialName - Name of the MillionVerifier credential to use
|
|
10
|
+
* @returns Tool instance
|
|
11
|
+
*
|
|
12
|
+
* @example
|
|
13
|
+
* ```typescript
|
|
14
|
+
* const tool = createMillionVerifierVerifyEmailTool('elevasis-millionverifier')
|
|
15
|
+
* const result = await tool.execute({
|
|
16
|
+
* email: 'john@example.com'
|
|
17
|
+
* }, context)
|
|
18
|
+
* // result.result: 'ok' | 'catch_all' | 'unknown' | 'error' | 'disposable' | 'invalid'
|
|
19
|
+
* // result.quality: 'good' | 'bad' | 'risky' | ''
|
|
20
|
+
* ```
|
|
21
|
+
*/
|
|
22
|
+
export function createMillionVerifierVerifyEmailTool(credentialName: string): Tool {
|
|
23
|
+
return createIntegrationTool({
|
|
24
|
+
name: 'millionverifier_verify_email',
|
|
25
|
+
description: `Verify email address deliverability using MillionVerifier.
|
|
26
|
+
|
|
27
|
+
Use this tool to validate if an email address is real and can receive emails before
|
|
28
|
+
adding it to a campaign. MillionVerifier is highly accurate and fast (160 req/s).
|
|
29
|
+
|
|
30
|
+
Result codes:
|
|
31
|
+
- ok (1): Email is valid and deliverable
|
|
32
|
+
- catch_all (2): Domain accepts all email (cannot confirm individual mailbox) - FREE
|
|
33
|
+
- unknown (3): Result could not be determined - FREE
|
|
34
|
+
- error (4): Verification error
|
|
35
|
+
- disposable (5): Disposable/temporary email address
|
|
36
|
+
- invalid (6): Email address is invalid
|
|
37
|
+
|
|
38
|
+
Quality signals:
|
|
39
|
+
- good: Safe to send
|
|
40
|
+
- risky: May bounce or flag spam filter
|
|
41
|
+
- bad: Do not send
|
|
42
|
+
|
|
43
|
+
Credits are only deducted for ok, invalid, and disposable results.
|
|
44
|
+
|
|
45
|
+
Returns full verification details including free/role detection and suggested correction.`,
|
|
46
|
+
inputSchema: z.object({
|
|
47
|
+
email: EmailSchema.describe('Email address to verify')
|
|
48
|
+
}),
|
|
49
|
+
outputSchema: z.object({
|
|
50
|
+
email: z.string().describe('Verified email address'),
|
|
51
|
+
quality: z.enum(['', 'good', 'bad', 'risky']).describe('Quality signal for sending decision'),
|
|
52
|
+
result: z.enum(['ok', 'catch_all', 'unknown', 'error', 'disposable', 'invalid']).describe('Verification result'),
|
|
53
|
+
resultCode: z
|
|
54
|
+
.number()
|
|
55
|
+
.describe('Numeric result code (1=ok, 2=catch_all, 3=unknown, 4=error, 5=disposable, 6=invalid)'),
|
|
56
|
+
subresult: z.string().describe('Detailed sub-result (e.g., ok, no_mailbox, dns_no_mx)'),
|
|
57
|
+
free: z.boolean().describe('Whether this is a free email provider (Gmail, Yahoo, etc.)'),
|
|
58
|
+
role: z.boolean().describe('Whether this is a role-based address (info@, support@, etc.)'),
|
|
59
|
+
didYouMean: z.string().describe('Suggested correction if email looks like a typo (empty if none)'),
|
|
60
|
+
credits: z.number().describe('Remaining credits after this verification'),
|
|
61
|
+
executionTime: z.number().describe('Time taken to verify in milliseconds'),
|
|
62
|
+
error: z.string().describe('Error message when result is error (empty otherwise)'),
|
|
63
|
+
liveMode: z.boolean().describe('Whether the verification was performed in live mode')
|
|
64
|
+
}),
|
|
65
|
+
integration: 'millionverifier' as const,
|
|
66
|
+
method: 'verifyEmail' as const,
|
|
67
|
+
credentialName
|
|
68
|
+
})
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
/**
|
|
72
|
+
* Create a tool that checks remaining MillionVerifier credits
|
|
73
|
+
*
|
|
74
|
+
* @param credentialName - Name of the MillionVerifier credential to use
|
|
75
|
+
* @returns Tool instance
|
|
76
|
+
*
|
|
77
|
+
* @example
|
|
78
|
+
* ```typescript
|
|
79
|
+
* const tool = createMillionVerifierCheckCreditsTool('elevasis-millionverifier')
|
|
80
|
+
* const result = await tool.execute({}, context)
|
|
81
|
+
* // result.credits: 5000
|
|
82
|
+
* ```
|
|
83
|
+
*/
|
|
84
|
+
export function createMillionVerifierCheckCreditsTool(credentialName: string): Tool {
|
|
85
|
+
return createIntegrationTool({
|
|
86
|
+
name: 'millionverifier_check_credits',
|
|
87
|
+
description: `Check remaining MillionVerifier verification credits.
|
|
88
|
+
|
|
89
|
+
Use this tool to monitor credit balance before running large verification batches.
|
|
90
|
+
|
|
91
|
+
Returns current credits, bulk credits, renewing credits, and plan tier.`,
|
|
92
|
+
inputSchema: z.object({}),
|
|
93
|
+
outputSchema: z.object({
|
|
94
|
+
credits: z.number().describe('Remaining single-verification credits'),
|
|
95
|
+
bulkCredits: z.number().describe('Remaining bulk verification credits'),
|
|
96
|
+
renewingCredits: z.number().describe('Credits that will renew on next billing cycle'),
|
|
97
|
+
plan: z.number().describe('Current plan tier')
|
|
98
|
+
}),
|
|
99
|
+
integration: 'millionverifier' as const,
|
|
100
|
+
method: 'checkCredits' as const,
|
|
101
|
+
credentialName
|
|
102
|
+
})
|
|
103
|
+
}
|
package/src/execution/engine/tools/integration/server/adapters/resend/fetch/send-email/index.test.ts
CHANGED
|
@@ -1,88 +1,88 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Resend sendEmail — idempotency-key regression tests
|
|
3
|
-
*
|
|
4
|
-
* Guards the Idempotency-Key header plumbing added as part of the
|
|
5
|
-
* cache-as-contract-leak fix. Platform-internal retries (TypeError /
|
|
6
|
-
* ReferenceError reclassified as platform_internal) must be safe for write
|
|
7
|
-
* tools — Resend dedupes server-side when the same Idempotency-Key arrives.
|
|
8
|
-
*/
|
|
9
|
-
|
|
10
|
-
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest'
|
|
11
|
-
import { sendEmail } from './index'
|
|
12
|
-
import type { ResendCredentials, SendEmailParams } from '../utils/types'
|
|
13
|
-
|
|
14
|
-
const credentials: ResendCredentials = { apiKey: 're_test_123' }
|
|
15
|
-
const baseParams: SendEmailParams = {
|
|
16
|
-
from: 'Test <test@elevasis.io>',
|
|
17
|
-
to: 'recipient@example.com',
|
|
18
|
-
subject: 'Hello',
|
|
19
|
-
html: '<p>Hi</p>'
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
function mockResendOk(): ReturnType<typeof vi.fn> {
|
|
23
|
-
const fetchMock = vi.fn(async () =>
|
|
24
|
-
Promise.resolve(
|
|
25
|
-
new Response(JSON.stringify({ id: 'email_abc' }), {
|
|
26
|
-
status: 200,
|
|
27
|
-
headers: { 'Content-Type': 'application/json' }
|
|
28
|
-
})
|
|
29
|
-
)
|
|
30
|
-
)
|
|
31
|
-
;(globalThis as unknown as { fetch: typeof fetch }).fetch = fetchMock as unknown as typeof fetch
|
|
32
|
-
return fetchMock
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
describe('Resend sendEmail — idempotency key', () => {
|
|
36
|
-
let originalFetch: typeof fetch
|
|
37
|
-
|
|
38
|
-
beforeEach(() => {
|
|
39
|
-
originalFetch = globalThis.fetch
|
|
40
|
-
})
|
|
41
|
-
|
|
42
|
-
afterEach(() => {
|
|
43
|
-
globalThis.fetch = originalFetch
|
|
44
|
-
})
|
|
45
|
-
|
|
46
|
-
it('omits the Idempotency-Key header when idempotencyKey is not provided', async () => {
|
|
47
|
-
const fetchMock = mockResendOk()
|
|
48
|
-
|
|
49
|
-
await sendEmail(credentials, baseParams)
|
|
50
|
-
|
|
51
|
-
expect(fetchMock).toHaveBeenCalledOnce()
|
|
52
|
-
const init = fetchMock.mock.calls[0][1] as RequestInit
|
|
53
|
-
const headers = init.headers as Record<string, string>
|
|
54
|
-
expect(headers['Idempotency-Key']).toBeUndefined()
|
|
55
|
-
// Sanity check: required headers are still present
|
|
56
|
-
expect(headers.Authorization).toBe('Bearer re_test_123')
|
|
57
|
-
expect(headers['Content-Type']).toBe('application/json')
|
|
58
|
-
})
|
|
59
|
-
|
|
60
|
-
it('passes idempotencyKey through as the Idempotency-Key header', async () => {
|
|
61
|
-
const fetchMock = mockResendOk()
|
|
62
|
-
|
|
63
|
-
await sendEmail(credentials, { ...baseParams, idempotencyKey: 'wf-exec-001-step-send-approval' })
|
|
64
|
-
|
|
65
|
-
const init = fetchMock.mock.calls[0][1] as RequestInit
|
|
66
|
-
const headers = init.headers as Record<string, string>
|
|
67
|
-
expect(headers['Idempotency-Key']).toBe('wf-exec-001-step-send-approval')
|
|
68
|
-
})
|
|
69
|
-
|
|
70
|
-
it('sends the same Idempotency-Key on caller-driven retries (regression: platform_internal retries must dedupe)', async () => {
|
|
71
|
-
const fetchMock = mockResendOk()
|
|
72
|
-
|
|
73
|
-
const params: SendEmailParams = { ...baseParams, idempotencyKey: 'stable-key-xyz' }
|
|
74
|
-
|
|
75
|
-
// Simulate two calls with identical params (e.g. caller retries after a
|
|
76
|
-
// platform_internal failure). Both must carry the same Idempotency-Key
|
|
77
|
-
// so Resend collapses them into a single send.
|
|
78
|
-
await sendEmail(credentials, params)
|
|
79
|
-
await sendEmail(credentials, params)
|
|
80
|
-
|
|
81
|
-
expect(fetchMock).toHaveBeenCalledTimes(2)
|
|
82
|
-
|
|
83
|
-
const firstHeaders = (fetchMock.mock.calls[0][1] as RequestInit).headers as Record<string, string>
|
|
84
|
-
const secondHeaders = (fetchMock.mock.calls[1][1] as RequestInit).headers as Record<string, string>
|
|
85
|
-
expect(firstHeaders['Idempotency-Key']).toBe('stable-key-xyz')
|
|
86
|
-
expect(secondHeaders['Idempotency-Key']).toBe('stable-key-xyz')
|
|
87
|
-
})
|
|
88
|
-
})
|
|
1
|
+
/**
|
|
2
|
+
* Resend sendEmail — idempotency-key regression tests
|
|
3
|
+
*
|
|
4
|
+
* Guards the Idempotency-Key header plumbing added as part of the
|
|
5
|
+
* cache-as-contract-leak fix. Platform-internal retries (TypeError /
|
|
6
|
+
* ReferenceError reclassified as platform_internal) must be safe for write
|
|
7
|
+
* tools — Resend dedupes server-side when the same Idempotency-Key arrives.
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest'
|
|
11
|
+
import { sendEmail } from './index'
|
|
12
|
+
import type { ResendCredentials, SendEmailParams } from '../utils/types'
|
|
13
|
+
|
|
14
|
+
const credentials: ResendCredentials = { apiKey: 're_test_123' }
|
|
15
|
+
const baseParams: SendEmailParams = {
|
|
16
|
+
from: 'Test <test@elevasis.io>',
|
|
17
|
+
to: 'recipient@example.com',
|
|
18
|
+
subject: 'Hello',
|
|
19
|
+
html: '<p>Hi</p>'
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
function mockResendOk(): ReturnType<typeof vi.fn> {
|
|
23
|
+
const fetchMock = vi.fn(async () =>
|
|
24
|
+
Promise.resolve(
|
|
25
|
+
new Response(JSON.stringify({ id: 'email_abc' }), {
|
|
26
|
+
status: 200,
|
|
27
|
+
headers: { 'Content-Type': 'application/json' }
|
|
28
|
+
})
|
|
29
|
+
)
|
|
30
|
+
)
|
|
31
|
+
;(globalThis as unknown as { fetch: typeof fetch }).fetch = fetchMock as unknown as typeof fetch
|
|
32
|
+
return fetchMock
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
describe('Resend sendEmail — idempotency key', () => {
|
|
36
|
+
let originalFetch: typeof fetch
|
|
37
|
+
|
|
38
|
+
beforeEach(() => {
|
|
39
|
+
originalFetch = globalThis.fetch
|
|
40
|
+
})
|
|
41
|
+
|
|
42
|
+
afterEach(() => {
|
|
43
|
+
globalThis.fetch = originalFetch
|
|
44
|
+
})
|
|
45
|
+
|
|
46
|
+
it('omits the Idempotency-Key header when idempotencyKey is not provided', async () => {
|
|
47
|
+
const fetchMock = mockResendOk()
|
|
48
|
+
|
|
49
|
+
await sendEmail(credentials, baseParams)
|
|
50
|
+
|
|
51
|
+
expect(fetchMock).toHaveBeenCalledOnce()
|
|
52
|
+
const init = fetchMock.mock.calls[0][1] as RequestInit
|
|
53
|
+
const headers = init.headers as Record<string, string>
|
|
54
|
+
expect(headers['Idempotency-Key']).toBeUndefined()
|
|
55
|
+
// Sanity check: required headers are still present
|
|
56
|
+
expect(headers.Authorization).toBe('Bearer re_test_123')
|
|
57
|
+
expect(headers['Content-Type']).toBe('application/json')
|
|
58
|
+
})
|
|
59
|
+
|
|
60
|
+
it('passes idempotencyKey through as the Idempotency-Key header', async () => {
|
|
61
|
+
const fetchMock = mockResendOk()
|
|
62
|
+
|
|
63
|
+
await sendEmail(credentials, { ...baseParams, idempotencyKey: 'wf-exec-001-step-send-approval' })
|
|
64
|
+
|
|
65
|
+
const init = fetchMock.mock.calls[0][1] as RequestInit
|
|
66
|
+
const headers = init.headers as Record<string, string>
|
|
67
|
+
expect(headers['Idempotency-Key']).toBe('wf-exec-001-step-send-approval')
|
|
68
|
+
})
|
|
69
|
+
|
|
70
|
+
it('sends the same Idempotency-Key on caller-driven retries (regression: platform_internal retries must dedupe)', async () => {
|
|
71
|
+
const fetchMock = mockResendOk()
|
|
72
|
+
|
|
73
|
+
const params: SendEmailParams = { ...baseParams, idempotencyKey: 'stable-key-xyz' }
|
|
74
|
+
|
|
75
|
+
// Simulate two calls with identical params (e.g. caller retries after a
|
|
76
|
+
// platform_internal failure). Both must carry the same Idempotency-Key
|
|
77
|
+
// so Resend collapses them into a single send.
|
|
78
|
+
await sendEmail(credentials, params)
|
|
79
|
+
await sendEmail(credentials, params)
|
|
80
|
+
|
|
81
|
+
expect(fetchMock).toHaveBeenCalledTimes(2)
|
|
82
|
+
|
|
83
|
+
const firstHeaders = (fetchMock.mock.calls[0][1] as RequestInit).headers as Record<string, string>
|
|
84
|
+
const secondHeaders = (fetchMock.mock.calls[1][1] as RequestInit).headers as Record<string, string>
|
|
85
|
+
expect(firstHeaders['Idempotency-Key']).toBe('stable-key-xyz')
|
|
86
|
+
expect(secondHeaders['Idempotency-Key']).toBe('stable-key-xyz')
|
|
87
|
+
})
|
|
88
|
+
})
|