resend-cli 1.2.1 → 1.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (191) hide show
  1. package/biome.json +1 -1
  2. package/bun.lock +0 -3
  3. package/package.json +2 -3
  4. package/src/cli.ts +11 -1
  5. package/src/commands/auth/login.ts +35 -8
  6. package/src/commands/doctor.ts +33 -115
  7. package/src/commands/teams/remove.ts +5 -2
  8. package/src/commands/teams/switch.ts +3 -0
  9. package/src/lib/config.ts +37 -31
  10. package/src/lib/spinner.ts +17 -10
  11. package/src/lib/update-check.ts +172 -0
  12. package/tests/commands/auth/login.test.ts +37 -0
  13. package/tests/lib/config.test.ts +38 -7
  14. package/tests/lib/update-check.test.ts +169 -0
  15. package/.claude/worktrees/emails-list/.claude/settings.local.json +0 -5
  16. package/.claude/worktrees/emails-list/.github/scripts/pr-title-check.js +0 -34
  17. package/.claude/worktrees/emails-list/.github/workflows/ci.yml +0 -32
  18. package/.claude/worktrees/emails-list/.github/workflows/pr-title-check.yml +0 -13
  19. package/.claude/worktrees/emails-list/.github/workflows/release.yml +0 -93
  20. package/.claude/worktrees/emails-list/CHANGELOG.md +0 -31
  21. package/.claude/worktrees/emails-list/LICENSE +0 -21
  22. package/.claude/worktrees/emails-list/README.md +0 -424
  23. package/.claude/worktrees/emails-list/biome.json +0 -36
  24. package/.claude/worktrees/emails-list/bun.lock +0 -76
  25. package/.claude/worktrees/emails-list/bunfig.toml +0 -2
  26. package/.claude/worktrees/emails-list/install.ps1 +0 -140
  27. package/.claude/worktrees/emails-list/install.sh +0 -301
  28. package/.claude/worktrees/emails-list/package.json +0 -43
  29. package/.claude/worktrees/emails-list/renovate.json +0 -6
  30. package/.claude/worktrees/emails-list/src/cli.ts +0 -74
  31. package/.claude/worktrees/emails-list/src/commands/api-keys/create.ts +0 -114
  32. package/.claude/worktrees/emails-list/src/commands/api-keys/delete.ts +0 -47
  33. package/.claude/worktrees/emails-list/src/commands/api-keys/index.ts +0 -26
  34. package/.claude/worktrees/emails-list/src/commands/api-keys/list.ts +0 -35
  35. package/.claude/worktrees/emails-list/src/commands/api-keys/utils.ts +0 -8
  36. package/.claude/worktrees/emails-list/src/commands/auth/index.ts +0 -20
  37. package/.claude/worktrees/emails-list/src/commands/auth/login.ts +0 -207
  38. package/.claude/worktrees/emails-list/src/commands/auth/logout.ts +0 -105
  39. package/.claude/worktrees/emails-list/src/commands/broadcasts/create.ts +0 -196
  40. package/.claude/worktrees/emails-list/src/commands/broadcasts/delete.ts +0 -46
  41. package/.claude/worktrees/emails-list/src/commands/broadcasts/get.ts +0 -59
  42. package/.claude/worktrees/emails-list/src/commands/broadcasts/index.ts +0 -43
  43. package/.claude/worktrees/emails-list/src/commands/broadcasts/list.ts +0 -60
  44. package/.claude/worktrees/emails-list/src/commands/broadcasts/send.ts +0 -56
  45. package/.claude/worktrees/emails-list/src/commands/broadcasts/update.ts +0 -95
  46. package/.claude/worktrees/emails-list/src/commands/broadcasts/utils.ts +0 -35
  47. package/.claude/worktrees/emails-list/src/commands/contact-properties/create.ts +0 -118
  48. package/.claude/worktrees/emails-list/src/commands/contact-properties/delete.ts +0 -48
  49. package/.claude/worktrees/emails-list/src/commands/contact-properties/get.ts +0 -46
  50. package/.claude/worktrees/emails-list/src/commands/contact-properties/index.ts +0 -48
  51. package/.claude/worktrees/emails-list/src/commands/contact-properties/list.ts +0 -68
  52. package/.claude/worktrees/emails-list/src/commands/contact-properties/update.ts +0 -88
  53. package/.claude/worktrees/emails-list/src/commands/contact-properties/utils.ts +0 -17
  54. package/.claude/worktrees/emails-list/src/commands/contacts/add-segment.ts +0 -78
  55. package/.claude/worktrees/emails-list/src/commands/contacts/create.ts +0 -122
  56. package/.claude/worktrees/emails-list/src/commands/contacts/delete.ts +0 -49
  57. package/.claude/worktrees/emails-list/src/commands/contacts/get.ts +0 -53
  58. package/.claude/worktrees/emails-list/src/commands/contacts/index.ts +0 -58
  59. package/.claude/worktrees/emails-list/src/commands/contacts/list.ts +0 -57
  60. package/.claude/worktrees/emails-list/src/commands/contacts/remove-segment.ts +0 -48
  61. package/.claude/worktrees/emails-list/src/commands/contacts/segments.ts +0 -39
  62. package/.claude/worktrees/emails-list/src/commands/contacts/topics.ts +0 -45
  63. package/.claude/worktrees/emails-list/src/commands/contacts/update-topics.ts +0 -90
  64. package/.claude/worktrees/emails-list/src/commands/contacts/update.ts +0 -77
  65. package/.claude/worktrees/emails-list/src/commands/contacts/utils.ts +0 -119
  66. package/.claude/worktrees/emails-list/src/commands/doctor.ts +0 -298
  67. package/.claude/worktrees/emails-list/src/commands/domains/create.ts +0 -83
  68. package/.claude/worktrees/emails-list/src/commands/domains/delete.ts +0 -42
  69. package/.claude/worktrees/emails-list/src/commands/domains/get.ts +0 -47
  70. package/.claude/worktrees/emails-list/src/commands/domains/index.ts +0 -35
  71. package/.claude/worktrees/emails-list/src/commands/domains/list.ts +0 -53
  72. package/.claude/worktrees/emails-list/src/commands/domains/update.ts +0 -75
  73. package/.claude/worktrees/emails-list/src/commands/domains/utils.ts +0 -44
  74. package/.claude/worktrees/emails-list/src/commands/domains/verify.ts +0 -38
  75. package/.claude/worktrees/emails-list/src/commands/emails/batch.ts +0 -140
  76. package/.claude/worktrees/emails-list/src/commands/emails/index.ts +0 -28
  77. package/.claude/worktrees/emails-list/src/commands/emails/list.ts +0 -73
  78. package/.claude/worktrees/emails-list/src/commands/emails/receiving/attachment.ts +0 -55
  79. package/.claude/worktrees/emails-list/src/commands/emails/receiving/attachments.ts +0 -68
  80. package/.claude/worktrees/emails-list/src/commands/emails/receiving/get.ts +0 -58
  81. package/.claude/worktrees/emails-list/src/commands/emails/receiving/index.ts +0 -28
  82. package/.claude/worktrees/emails-list/src/commands/emails/receiving/list.ts +0 -59
  83. package/.claude/worktrees/emails-list/src/commands/emails/receiving/utils.ts +0 -38
  84. package/.claude/worktrees/emails-list/src/commands/emails/send.ts +0 -189
  85. package/.claude/worktrees/emails-list/src/commands/open.ts +0 -24
  86. package/.claude/worktrees/emails-list/src/commands/segments/create.ts +0 -50
  87. package/.claude/worktrees/emails-list/src/commands/segments/delete.ts +0 -47
  88. package/.claude/worktrees/emails-list/src/commands/segments/get.ts +0 -38
  89. package/.claude/worktrees/emails-list/src/commands/segments/index.ts +0 -36
  90. package/.claude/worktrees/emails-list/src/commands/segments/list.ts +0 -58
  91. package/.claude/worktrees/emails-list/src/commands/segments/utils.ts +0 -7
  92. package/.claude/worktrees/emails-list/src/commands/teams/index.ts +0 -10
  93. package/.claude/worktrees/emails-list/src/commands/teams/list.ts +0 -35
  94. package/.claude/worktrees/emails-list/src/commands/teams/remove.ts +0 -83
  95. package/.claude/worktrees/emails-list/src/commands/teams/switch.ts +0 -73
  96. package/.claude/worktrees/emails-list/src/commands/topics/create.ts +0 -73
  97. package/.claude/worktrees/emails-list/src/commands/topics/delete.ts +0 -47
  98. package/.claude/worktrees/emails-list/src/commands/topics/get.ts +0 -42
  99. package/.claude/worktrees/emails-list/src/commands/topics/index.ts +0 -42
  100. package/.claude/worktrees/emails-list/src/commands/topics/list.ts +0 -34
  101. package/.claude/worktrees/emails-list/src/commands/topics/update.ts +0 -59
  102. package/.claude/worktrees/emails-list/src/commands/topics/utils.ts +0 -16
  103. package/.claude/worktrees/emails-list/src/commands/webhooks/create.ts +0 -128
  104. package/.claude/worktrees/emails-list/src/commands/webhooks/delete.ts +0 -49
  105. package/.claude/worktrees/emails-list/src/commands/webhooks/get.ts +0 -42
  106. package/.claude/worktrees/emails-list/src/commands/webhooks/index.ts +0 -44
  107. package/.claude/worktrees/emails-list/src/commands/webhooks/list.ts +0 -55
  108. package/.claude/worktrees/emails-list/src/commands/webhooks/update.ts +0 -83
  109. package/.claude/worktrees/emails-list/src/commands/webhooks/utils.ts +0 -36
  110. package/.claude/worktrees/emails-list/src/commands/whoami.ts +0 -71
  111. package/.claude/worktrees/emails-list/src/lib/actions.ts +0 -157
  112. package/.claude/worktrees/emails-list/src/lib/client.ts +0 -34
  113. package/.claude/worktrees/emails-list/src/lib/config.ts +0 -211
  114. package/.claude/worktrees/emails-list/src/lib/files.ts +0 -15
  115. package/.claude/worktrees/emails-list/src/lib/help-text.ts +0 -38
  116. package/.claude/worktrees/emails-list/src/lib/output.ts +0 -54
  117. package/.claude/worktrees/emails-list/src/lib/pagination.ts +0 -36
  118. package/.claude/worktrees/emails-list/src/lib/prompts.ts +0 -149
  119. package/.claude/worktrees/emails-list/src/lib/spinner.ts +0 -93
  120. package/.claude/worktrees/emails-list/src/lib/table.ts +0 -57
  121. package/.claude/worktrees/emails-list/src/lib/tty.ts +0 -28
  122. package/.claude/worktrees/emails-list/src/lib/version.ts +0 -4
  123. package/.claude/worktrees/emails-list/tests/commands/api-keys/create.test.ts +0 -195
  124. package/.claude/worktrees/emails-list/tests/commands/api-keys/delete.test.ts +0 -156
  125. package/.claude/worktrees/emails-list/tests/commands/api-keys/list.test.ts +0 -133
  126. package/.claude/worktrees/emails-list/tests/commands/auth/login.test.ts +0 -119
  127. package/.claude/worktrees/emails-list/tests/commands/auth/logout.test.ts +0 -146
  128. package/.claude/worktrees/emails-list/tests/commands/broadcasts/create.test.ts +0 -447
  129. package/.claude/worktrees/emails-list/tests/commands/broadcasts/delete.test.ts +0 -182
  130. package/.claude/worktrees/emails-list/tests/commands/broadcasts/get.test.ts +0 -146
  131. package/.claude/worktrees/emails-list/tests/commands/broadcasts/list.test.ts +0 -196
  132. package/.claude/worktrees/emails-list/tests/commands/broadcasts/send.test.ts +0 -161
  133. package/.claude/worktrees/emails-list/tests/commands/broadcasts/update.test.ts +0 -283
  134. package/.claude/worktrees/emails-list/tests/commands/contact-properties/create.test.ts +0 -250
  135. package/.claude/worktrees/emails-list/tests/commands/contact-properties/delete.test.ts +0 -183
  136. package/.claude/worktrees/emails-list/tests/commands/contact-properties/get.test.ts +0 -144
  137. package/.claude/worktrees/emails-list/tests/commands/contact-properties/list.test.ts +0 -180
  138. package/.claude/worktrees/emails-list/tests/commands/contact-properties/update.test.ts +0 -216
  139. package/.claude/worktrees/emails-list/tests/commands/contacts/add-segment.test.ts +0 -188
  140. package/.claude/worktrees/emails-list/tests/commands/contacts/create.test.ts +0 -270
  141. package/.claude/worktrees/emails-list/tests/commands/contacts/delete.test.ts +0 -192
  142. package/.claude/worktrees/emails-list/tests/commands/contacts/get.test.ts +0 -148
  143. package/.claude/worktrees/emails-list/tests/commands/contacts/list.test.ts +0 -175
  144. package/.claude/worktrees/emails-list/tests/commands/contacts/remove-segment.test.ts +0 -166
  145. package/.claude/worktrees/emails-list/tests/commands/contacts/segments.test.ts +0 -167
  146. package/.claude/worktrees/emails-list/tests/commands/contacts/topics.test.ts +0 -163
  147. package/.claude/worktrees/emails-list/tests/commands/contacts/update-topics.test.ts +0 -247
  148. package/.claude/worktrees/emails-list/tests/commands/contacts/update.test.ts +0 -205
  149. package/.claude/worktrees/emails-list/tests/commands/doctor.test.ts +0 -165
  150. package/.claude/worktrees/emails-list/tests/commands/domains/create.test.ts +0 -192
  151. package/.claude/worktrees/emails-list/tests/commands/domains/delete.test.ts +0 -156
  152. package/.claude/worktrees/emails-list/tests/commands/domains/get.test.ts +0 -137
  153. package/.claude/worktrees/emails-list/tests/commands/domains/list.test.ts +0 -164
  154. package/.claude/worktrees/emails-list/tests/commands/domains/update.test.ts +0 -223
  155. package/.claude/worktrees/emails-list/tests/commands/domains/verify.test.ts +0 -117
  156. package/.claude/worktrees/emails-list/tests/commands/emails/batch.test.ts +0 -313
  157. package/.claude/worktrees/emails-list/tests/commands/emails/list.test.ts +0 -196
  158. package/.claude/worktrees/emails-list/tests/commands/emails/receiving/attachment.test.ts +0 -140
  159. package/.claude/worktrees/emails-list/tests/commands/emails/receiving/attachments.test.ts +0 -168
  160. package/.claude/worktrees/emails-list/tests/commands/emails/receiving/get.test.ts +0 -140
  161. package/.claude/worktrees/emails-list/tests/commands/emails/receiving/list.test.ts +0 -181
  162. package/.claude/worktrees/emails-list/tests/commands/emails/send.test.ts +0 -309
  163. package/.claude/worktrees/emails-list/tests/commands/segments/create.test.ts +0 -163
  164. package/.claude/worktrees/emails-list/tests/commands/segments/delete.test.ts +0 -182
  165. package/.claude/worktrees/emails-list/tests/commands/segments/get.test.ts +0 -137
  166. package/.claude/worktrees/emails-list/tests/commands/segments/list.test.ts +0 -173
  167. package/.claude/worktrees/emails-list/tests/commands/teams/list.test.ts +0 -63
  168. package/.claude/worktrees/emails-list/tests/commands/teams/remove.test.ts +0 -103
  169. package/.claude/worktrees/emails-list/tests/commands/teams/switch.test.ts +0 -96
  170. package/.claude/worktrees/emails-list/tests/commands/topics/create.test.ts +0 -191
  171. package/.claude/worktrees/emails-list/tests/commands/topics/delete.test.ts +0 -156
  172. package/.claude/worktrees/emails-list/tests/commands/topics/get.test.ts +0 -125
  173. package/.claude/worktrees/emails-list/tests/commands/topics/list.test.ts +0 -124
  174. package/.claude/worktrees/emails-list/tests/commands/topics/update.test.ts +0 -177
  175. package/.claude/worktrees/emails-list/tests/commands/webhooks/create.test.ts +0 -224
  176. package/.claude/worktrees/emails-list/tests/commands/webhooks/delete.test.ts +0 -156
  177. package/.claude/worktrees/emails-list/tests/commands/webhooks/get.test.ts +0 -125
  178. package/.claude/worktrees/emails-list/tests/commands/webhooks/list.test.ts +0 -177
  179. package/.claude/worktrees/emails-list/tests/commands/webhooks/update.test.ts +0 -206
  180. package/.claude/worktrees/emails-list/tests/commands/whoami.test.ts +0 -99
  181. package/.claude/worktrees/emails-list/tests/helpers.ts +0 -93
  182. package/.claude/worktrees/emails-list/tests/lib/client.test.ts +0 -71
  183. package/.claude/worktrees/emails-list/tests/lib/config.test.ts +0 -414
  184. package/.claude/worktrees/emails-list/tests/lib/files.test.ts +0 -65
  185. package/.claude/worktrees/emails-list/tests/lib/help-text.test.ts +0 -97
  186. package/.claude/worktrees/emails-list/tests/lib/output.test.ts +0 -127
  187. package/.claude/worktrees/emails-list/tests/lib/prompts.test.ts +0 -178
  188. package/.claude/worktrees/emails-list/tests/lib/spinner.test.ts +0 -146
  189. package/.claude/worktrees/emails-list/tests/lib/table.test.ts +0 -63
  190. package/.claude/worktrees/emails-list/tests/lib/tty.test.ts +0 -85
  191. package/.claude/worktrees/emails-list/tsconfig.json +0 -14
@@ -1,83 +0,0 @@
1
- import { Command, Option } from '@commander-js/extra-typings';
2
- import { runCreate } from '../../lib/actions';
3
- import type { GlobalOpts } from '../../lib/client';
4
- import { buildHelpText } from '../../lib/help-text';
5
- import { requireText } from '../../lib/prompts';
6
- import { renderDnsRecordsTable } from './utils';
7
-
8
- export const createDomainCommand = new Command('create')
9
- .description('Create a new domain and receive DNS records to configure')
10
- .option('--name <domain>', 'Domain name (e.g. example.com)')
11
- .addOption(
12
- new Option('--region <region>', 'Sending region').choices([
13
- 'us-east-1',
14
- 'eu-west-1',
15
- 'sa-east-1',
16
- 'ap-northeast-1',
17
- ] as const),
18
- )
19
- .addOption(
20
- new Option('--tls <mode>', 'TLS mode (default: opportunistic)').choices([
21
- 'opportunistic',
22
- 'enforced',
23
- ] as const),
24
- )
25
- .option('--sending', 'Enable sending capability (default: enabled)')
26
- .option('--receiving', 'Enable receiving capability (default: disabled)')
27
- .addHelpText(
28
- 'after',
29
- buildHelpText({
30
- context:
31
- 'Non-interactive: --name is required (no prompts when stdin/stdout is not a TTY)',
32
- output:
33
- ' Full domain object with DNS records array to configure in your DNS provider.',
34
- errorCodes: ['auth_error', 'missing_name', 'create_error'],
35
- examples: [
36
- 'resend domains create --name example.com',
37
- 'resend domains create --name example.com --region eu-west-1 --tls enforced',
38
- 'resend domains create --name example.com --receiving --json',
39
- 'resend domains create --name example.com --sending --receiving --json',
40
- ],
41
- }),
42
- )
43
- .action(async (opts, cmd) => {
44
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
45
-
46
- const name = await requireText(
47
- opts.name,
48
- { message: 'Domain name', placeholder: 'example.com' },
49
- { message: 'Missing --name flag.', code: 'missing_name' },
50
- globalOpts,
51
- );
52
-
53
- await runCreate(
54
- {
55
- spinner: {
56
- loading: 'Creating domain...',
57
- success: 'Domain created',
58
- fail: 'Failed to create domain',
59
- },
60
- sdkCall: (resend) =>
61
- resend.domains.create({
62
- name,
63
- ...(opts.region && { region: opts.region }),
64
- ...(opts.tls && { tls: opts.tls }),
65
- ...((opts.sending || opts.receiving) && {
66
- capabilities: {
67
- ...(opts.sending && { sending: 'enabled' as const }),
68
- ...(opts.receiving && { receiving: 'enabled' as const }),
69
- },
70
- }),
71
- }),
72
- onInteractive: (d) => {
73
- console.log(`\nDomain created: ${d.name} (id: ${d.id})`);
74
- console.log('\nDNS Records to configure:');
75
- console.log(renderDnsRecordsTable(d.records, d.name));
76
- console.log(
77
- `\nRun \`resend domains verify ${d.id}\` after configuring DNS.`,
78
- );
79
- },
80
- },
81
- globalOpts,
82
- );
83
- });
@@ -1,42 +0,0 @@
1
- import { Command } from '@commander-js/extra-typings';
2
- import { runDelete } from '../../lib/actions';
3
- import type { GlobalOpts } from '../../lib/client';
4
- import { buildHelpText } from '../../lib/help-text';
5
-
6
- export const deleteDomainCommand = new Command('delete')
7
- .alias('rm')
8
- .description('Delete a domain')
9
- .argument('<id>', 'Domain ID')
10
- .option('--yes', 'Skip confirmation prompt')
11
- .addHelpText(
12
- 'after',
13
- buildHelpText({
14
- context:
15
- 'Non-interactive: --yes is required to confirm deletion when stdin/stdout is not a TTY.',
16
- output: ' {"object":"domain","id":"<id>","deleted":true}',
17
- errorCodes: ['auth_error', 'confirmation_required', 'delete_error'],
18
- examples: [
19
- 'resend domains delete 4dd369bc-aa82-4ff3-97de-514ae3000ee0 --yes',
20
- 'resend domains delete 4dd369bc-aa82-4ff3-97de-514ae3000ee0 --yes --json',
21
- ],
22
- }),
23
- )
24
- .action(async (id, opts, cmd) => {
25
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
26
- await runDelete(
27
- id,
28
- !!opts.yes,
29
- {
30
- confirmMessage: `Delete domain ${id}?\nThis cannot be undone.`,
31
- spinner: {
32
- loading: 'Deleting domain...',
33
- success: 'Domain deleted',
34
- fail: 'Failed to delete domain',
35
- },
36
- object: 'domain',
37
- successMsg: 'Domain deleted.',
38
- sdkCall: (resend) => resend.domains.remove(id),
39
- },
40
- globalOpts,
41
- );
42
- });
@@ -1,47 +0,0 @@
1
- import { Command } from '@commander-js/extra-typings';
2
- import { runGet } from '../../lib/actions';
3
- import type { GlobalOpts } from '../../lib/client';
4
- import { buildHelpText } from '../../lib/help-text';
5
- import { renderDnsRecordsTable, statusIndicator } from './utils';
6
-
7
- export const getDomainCommand = new Command('get')
8
- .description(
9
- 'Retrieve a domain with its DNS records and current verification status',
10
- )
11
- .argument('<id>', 'Domain ID')
12
- .addHelpText(
13
- 'after',
14
- buildHelpText({
15
- output:
16
- ' Full domain object including records array and current status.\n\nDomain status values: not_started | pending | verified | failed | temporary_failure',
17
- errorCodes: ['auth_error', 'fetch_error'],
18
- examples: [
19
- 'resend domains get 4dd369bc-aa82-4ff3-97de-514ae3000ee0',
20
- 'resend domains get 4dd369bc-aa82-4ff3-97de-514ae3000ee0 --json',
21
- ],
22
- }),
23
- )
24
- .action(async (id, _opts, cmd) => {
25
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
26
- await runGet(
27
- {
28
- spinner: {
29
- loading: 'Fetching domain...',
30
- success: 'Domain fetched',
31
- fail: 'Failed to fetch domain',
32
- },
33
- sdkCall: (resend) => resend.domains.get(id),
34
- onInteractive: (d) => {
35
- console.log(`\n${d.name} — ${statusIndicator(d.status)}`);
36
- console.log(`ID: ${d.id}`);
37
- console.log(`Region: ${d.region}`);
38
- console.log(`Created: ${d.created_at}`);
39
- if (d.records.length > 0) {
40
- console.log('\nDNS Records:');
41
- console.log(renderDnsRecordsTable(d.records, d.name));
42
- }
43
- },
44
- },
45
- globalOpts,
46
- );
47
- });
@@ -1,35 +0,0 @@
1
- import { Command } from '@commander-js/extra-typings';
2
- import { buildHelpText } from '../../lib/help-text';
3
- import { createDomainCommand } from './create';
4
- import { deleteDomainCommand } from './delete';
5
- import { getDomainCommand } from './get';
6
- import { listDomainsCommand } from './list';
7
- import { updateDomainCommand } from './update';
8
- import { verifyDomainCommand } from './verify';
9
-
10
- export const domainsCommand = new Command('domains')
11
- .description('Manage verified sending and receiving domains')
12
- .addHelpText(
13
- 'after',
14
- buildHelpText({
15
- context: `Domain lifecycle:
16
- 1. resend domains create --name example.com (get DNS records)
17
- 2. Configure DNS records at your DNS provider
18
- 3. resend domains verify <id> (trigger verification)
19
- 4. resend domains get <id> (poll until "verified")`,
20
- examples: [
21
- 'resend domains list',
22
- 'resend domains create --name example.com --region us-east-1',
23
- 'resend domains verify 4dd369bc-aa82-4ff3-97de-514ae3000ee0',
24
- 'resend domains get 4dd369bc-aa82-4ff3-97de-514ae3000ee0',
25
- 'resend domains update <id> --tls enforced --open-tracking',
26
- 'resend domains delete <id> --yes',
27
- ],
28
- }),
29
- )
30
- .addCommand(createDomainCommand)
31
- .addCommand(verifyDomainCommand)
32
- .addCommand(getDomainCommand)
33
- .addCommand(listDomainsCommand, { isDefault: true })
34
- .addCommand(updateDomainCommand)
35
- .addCommand(deleteDomainCommand);
@@ -1,53 +0,0 @@
1
- import { Command } from '@commander-js/extra-typings';
2
- import { runList } from '../../lib/actions';
3
- import type { GlobalOpts } from '../../lib/client';
4
- import { buildHelpText } from '../../lib/help-text';
5
- import {
6
- buildPaginationOpts,
7
- parseLimitOpt,
8
- printPaginationHint,
9
- } from '../../lib/pagination';
10
- import { renderDomainsTable } from './utils';
11
-
12
- export const listDomainsCommand = new Command('list')
13
- .alias('ls')
14
- .description('List all domains')
15
- .option('--limit <n>', 'Maximum number of domains to return (1-100)', '10')
16
- .option('--after <cursor>', 'Return domains after this cursor (next page)')
17
- .option(
18
- '--before <cursor>',
19
- 'Return domains before this cursor (previous page)',
20
- )
21
- .addHelpText(
22
- 'after',
23
- buildHelpText({
24
- output:
25
- ' {"object":"list","data":[...],"has_more":true}\n The list response does not include DNS records — use "resend domains get <id>" for that.',
26
- errorCodes: ['auth_error', 'invalid_limit', 'list_error'],
27
- examples: [
28
- 'resend domains list',
29
- 'resend domains list --limit 25 --json',
30
- 'resend domains list --after <cursor> --json',
31
- ],
32
- }),
33
- )
34
- .action(async (opts, cmd) => {
35
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
36
- const limit = parseLimitOpt(opts.limit, globalOpts);
37
- const paginationOpts = buildPaginationOpts(limit, opts.after, opts.before);
38
- await runList(
39
- {
40
- spinner: {
41
- loading: 'Fetching domains...',
42
- success: 'Domains fetched',
43
- fail: 'Failed to list domains',
44
- },
45
- sdkCall: (resend) => resend.domains.list(paginationOpts),
46
- onInteractive: (list) => {
47
- console.log(renderDomainsTable(list.data));
48
- printPaginationHint(list);
49
- },
50
- },
51
- globalOpts,
52
- );
53
- });
@@ -1,75 +0,0 @@
1
- import { Command, Option } from '@commander-js/extra-typings';
2
- import type { UpdateDomainsOptions } from 'resend';
3
- import { runWrite } from '../../lib/actions';
4
- import type { GlobalOpts } from '../../lib/client';
5
- import { buildHelpText } from '../../lib/help-text';
6
- import { outputError } from '../../lib/output';
7
-
8
- export const updateDomainCommand = new Command('update')
9
- .description(
10
- 'Update domain settings: TLS mode, open tracking, and click tracking',
11
- )
12
- .argument('<id>', 'Domain ID')
13
- .addOption(
14
- new Option('--tls <mode>', 'TLS mode').choices([
15
- 'opportunistic',
16
- 'enforced',
17
- ] as const),
18
- )
19
- .option('--open-tracking', 'Enable open tracking')
20
- .option('--no-open-tracking', 'Disable open tracking')
21
- .option('--click-tracking', 'Enable click tracking')
22
- .option('--no-click-tracking', 'Disable click tracking')
23
- .addHelpText(
24
- 'after',
25
- buildHelpText({
26
- output: ' {"object":"domain","id":"<id>"}',
27
- errorCodes: ['auth_error', 'no_changes', 'update_error'],
28
- examples: [
29
- 'resend domains update <id> --tls enforced',
30
- 'resend domains update <id> --open-tracking --click-tracking',
31
- 'resend domains update <id> --no-open-tracking --json',
32
- ],
33
- }),
34
- )
35
- .action(async (id, opts, cmd) => {
36
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
37
-
38
- const { tls, openTracking, clickTracking } = opts;
39
-
40
- if (!tls && openTracking === undefined && clickTracking === undefined) {
41
- outputError(
42
- {
43
- message:
44
- 'Provide at least one option to update: --tls, --open-tracking, or --click-tracking.',
45
- code: 'no_changes',
46
- },
47
- { json: globalOpts.json },
48
- );
49
- }
50
-
51
- const payload: UpdateDomainsOptions = { id };
52
- if (tls) {
53
- payload.tls = tls;
54
- }
55
- if (openTracking !== undefined) {
56
- payload.openTracking = openTracking;
57
- }
58
- if (clickTracking !== undefined) {
59
- payload.clickTracking = clickTracking;
60
- }
61
-
62
- await runWrite(
63
- {
64
- spinner: {
65
- loading: 'Updating domain...',
66
- success: 'Domain updated',
67
- fail: 'Failed to update domain',
68
- },
69
- sdkCall: (resend) => resend.domains.update(payload),
70
- errorCode: 'update_error',
71
- successMsg: `Domain updated: ${id}`,
72
- },
73
- globalOpts,
74
- );
75
- });
@@ -1,44 +0,0 @@
1
- import type { DomainRecords } from 'resend';
2
- import { renderTable } from '../../lib/table';
3
-
4
- export function renderDnsRecordsTable(
5
- records: DomainRecords[],
6
- domainName: string,
7
- ): string {
8
- const rows = records.map((r) => {
9
- const displayName = r.name
10
- ? r.name.includes('.')
11
- ? r.name
12
- : `${r.name}.${domainName}`
13
- : domainName;
14
- return [r.type, displayName, r.ttl, r.value];
15
- });
16
- return renderTable(
17
- ['Type', 'Name', 'TTL', 'Value'],
18
- rows,
19
- '(no DNS records)',
20
- );
21
- }
22
-
23
- export function renderDomainsTable(
24
- domains: Array<{ id: string; name: string; status: string; region: string }>,
25
- ): string {
26
- const rows = domains.map((d) => [d.name, d.status, d.region, d.id]);
27
- return renderTable(['Name', 'Status', 'Region', 'ID'], rows, '(no domains)');
28
- }
29
-
30
- export function statusIndicator(status: string): string {
31
- switch (status) {
32
- case 'verified':
33
- return '✓ Verified';
34
- case 'pending':
35
- return '⏳ Pending';
36
- case 'not_started':
37
- return '○ Not started';
38
- case 'failed':
39
- case 'temporary_failure':
40
- return '✗ Failed';
41
- default:
42
- return status;
43
- }
44
- }
@@ -1,38 +0,0 @@
1
- import { Command } from '@commander-js/extra-typings';
2
- import { runWrite } from '../../lib/actions';
3
- import type { GlobalOpts } from '../../lib/client';
4
- import { buildHelpText } from '../../lib/help-text';
5
-
6
- export const verifyDomainCommand = new Command('verify')
7
- .description('Trigger async DNS verification for a domain')
8
- .argument('<id>', 'Domain ID')
9
- .addHelpText(
10
- 'after',
11
- buildHelpText({
12
- context: `Verification is async — the domain enters "pending" status while DNS records are checked.
13
- Poll the status with: resend domains get <id>`,
14
- output: ' {"object":"domain","id":"<id>"}',
15
- errorCodes: ['auth_error', 'verify_error'],
16
- examples: [
17
- 'resend domains verify 4dd369bc-aa82-4ff3-97de-514ae3000ee0',
18
- 'resend domains verify 4dd369bc-aa82-4ff3-97de-514ae3000ee0 --json',
19
- ],
20
- }),
21
- )
22
- .action(async (id, _opts, cmd) => {
23
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
24
-
25
- await runWrite(
26
- {
27
- spinner: {
28
- loading: 'Verifying domain...',
29
- success: 'Verification started',
30
- fail: 'Failed to verify domain',
31
- },
32
- sdkCall: (resend) => resend.domains.verify(id),
33
- errorCode: 'verify_error',
34
- successMsg: `Domain verification started. Check status with resend domains get ${id}.`,
35
- },
36
- globalOpts,
37
- );
38
- });
@@ -1,140 +0,0 @@
1
- import { Command } from '@commander-js/extra-typings';
2
- import type { CreateBatchOptions } from 'resend';
3
- import type { GlobalOpts } from '../../lib/client';
4
- import { requireClient } from '../../lib/client';
5
- import { readFile } from '../../lib/files';
6
- import { buildHelpText } from '../../lib/help-text';
7
- import { outputError, outputResult } from '../../lib/output';
8
- import { requireText } from '../../lib/prompts';
9
- import { withSpinner } from '../../lib/spinner';
10
- import { isInteractive } from '../../lib/tty';
11
-
12
- export const batchCommand = new Command('batch')
13
- .description('Send up to 100 emails in a single API request from a JSON file')
14
- .option(
15
- '--file <path>',
16
- 'Path to a JSON file containing an array of email objects (required in non-interactive mode)',
17
- )
18
- .option(
19
- '--idempotency-key <key>',
20
- 'Deduplicate this batch request using this key',
21
- )
22
- .addHelpText(
23
- 'after',
24
- buildHelpText({
25
- context:
26
- 'Non-interactive: --file\nLimit: 100 emails per request (API hard limit — warned if exceeded)\nUnsupported per-email fields: attachments, scheduled_at\n\nFile format (--file path):\n [\n {"from":"you@domain.com","to":["user@example.com"],"subject":"Hello","text":"Hi"},\n {"from":"you@domain.com","to":["other@example.com"],"subject":"Hello","html":"<b>Hi</b>"}\n ]',
27
- output: ' [{"id":"<email-id>"},{"id":"<email-id>"}]',
28
- errorCodes: [
29
- 'auth_error',
30
- 'missing_file',
31
- 'file_read_error',
32
- 'invalid_json',
33
- 'invalid_format',
34
- 'batch_error',
35
- ],
36
- examples: [
37
- 'resend emails batch --file ./emails.json',
38
- 'resend emails batch --file ./emails.json --json',
39
- 'resend emails batch --file ./emails.json --idempotency-key my-batch-2026-02-18',
40
- 'RESEND_API_KEY=re_123 resend emails batch --file ./emails.json --json',
41
- ],
42
- }),
43
- )
44
- .action(async (opts, cmd) => {
45
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
46
-
47
- const resend = requireClient(globalOpts);
48
-
49
- const filePath = await requireText(
50
- opts.file,
51
- { message: 'Path to JSON file', placeholder: './emails.json' },
52
- {
53
- message:
54
- 'Missing --file flag. Provide a JSON file with an array of email objects.',
55
- code: 'missing_file',
56
- },
57
- globalOpts,
58
- );
59
-
60
- const raw = readFile(filePath, globalOpts);
61
-
62
- let parsed: unknown;
63
- try {
64
- parsed = JSON.parse(raw);
65
- } catch {
66
- outputError(
67
- { message: 'File content is not valid JSON.', code: 'invalid_json' },
68
- { json: globalOpts.json },
69
- );
70
- }
71
-
72
- if (!Array.isArray(parsed)) {
73
- outputError(
74
- {
75
- message: 'File content must be a JSON array of email objects.',
76
- code: 'invalid_format',
77
- },
78
- { json: globalOpts.json },
79
- );
80
- }
81
-
82
- const emails = parsed as unknown[];
83
-
84
- if (emails.length > 100) {
85
- console.warn(
86
- `Warning: ${emails.length} emails exceeds the 100-email limit. The API may reject this request.`,
87
- );
88
- }
89
-
90
- for (let i = 0; i < emails.length; i++) {
91
- const email = emails[i] as Record<string, unknown>;
92
- if ('attachments' in email) {
93
- outputError(
94
- {
95
- message: `Email at index ${i} contains "attachments", which is not supported in batch sends.`,
96
- code: 'batch_error',
97
- },
98
- { json: globalOpts.json },
99
- );
100
- }
101
- if ('scheduled_at' in email) {
102
- outputError(
103
- {
104
- message: `Email at index ${i} contains "scheduled_at", which is not supported in batch sends.`,
105
- code: 'batch_error',
106
- },
107
- { json: globalOpts.json },
108
- );
109
- }
110
- }
111
-
112
- const batchData = await withSpinner(
113
- {
114
- loading: 'Sending batch...',
115
- success: 'Batch sent',
116
- fail: 'Failed to send batch',
117
- },
118
- () =>
119
- resend.batch.send(
120
- emails as CreateBatchOptions,
121
- opts.idempotencyKey
122
- ? { idempotencyKey: opts.idempotencyKey }
123
- : undefined,
124
- ),
125
- 'batch_error',
126
- globalOpts,
127
- );
128
-
129
- const emailIds = batchData.data;
130
- if (!globalOpts.json && isInteractive()) {
131
- console.log(
132
- `Sent ${emailIds.length} email${emailIds.length === 1 ? '' : 's'}`,
133
- );
134
- for (const email of emailIds) {
135
- console.log(` ${email.id}`);
136
- }
137
- } else {
138
- outputResult(emailIds, { json: globalOpts.json });
139
- }
140
- });
@@ -1,28 +0,0 @@
1
- import { Command } from '@commander-js/extra-typings';
2
- import { buildHelpText } from '../../lib/help-text';
3
- import { batchCommand } from './batch';
4
- import { listCommand } from './list';
5
- import { receivingCommand } from './receiving/index';
6
- import { sendCommand } from './send';
7
-
8
- export const emailsCommand = new Command('emails')
9
- .description('Send and manage emails')
10
- .addHelpText(
11
- 'after',
12
- buildHelpText({
13
- examples: [
14
- 'resend emails send --from you@domain.com --to user@example.com --subject "Hello" --text "Hi"',
15
- 'resend emails list',
16
- 'resend emails list --limit 25 --json',
17
- 'resend emails batch --file ./emails.json',
18
- 'resend emails receiving list',
19
- 'resend emails receiving get <email-id>',
20
- 'resend emails receiving attachments <email-id>',
21
- 'resend emails receiving attachment <email-id> <attachment-id>',
22
- ],
23
- }),
24
- )
25
- .addCommand(sendCommand)
26
- .addCommand(listCommand)
27
- .addCommand(batchCommand)
28
- .addCommand(receivingCommand);
@@ -1,73 +0,0 @@
1
- import { Command } from '@commander-js/extra-typings';
2
- import type { ListEmail } from 'resend';
3
- import { runList } from '../../lib/actions';
4
- import type { GlobalOpts } from '../../lib/client';
5
- import { buildHelpText } from '../../lib/help-text';
6
- import {
7
- buildPaginationOpts,
8
- parseLimitOpt,
9
- printPaginationHint,
10
- } from '../../lib/pagination';
11
- import { renderTable } from '../../lib/table';
12
-
13
- function renderEmailsTable(emails: ListEmail[]): string {
14
- const rows = emails.map((e) => {
15
- const to = e.to.join(', ');
16
- const toStr = to.length > 40 ? `${to.slice(0, 37)}...` : to;
17
- const subject =
18
- e.subject.length > 50 ? `${e.subject.slice(0, 47)}...` : e.subject;
19
- return [e.from, toStr, subject, e.last_event, e.created_at, e.id];
20
- });
21
- return renderTable(
22
- ['From', 'To', 'Subject', 'Last Event', 'Created At', 'ID'],
23
- rows,
24
- '(no emails)',
25
- );
26
- }
27
-
28
- export const listCommand = new Command('list')
29
- .alias('ls')
30
- .description('List sent emails')
31
- .option('--limit <n>', 'Maximum number of emails to return (1-100)', '10')
32
- .option('--after <cursor>', 'Return emails after this cursor (next page)')
33
- .option(
34
- '--before <cursor>',
35
- 'Return emails before this cursor (previous page)',
36
- )
37
- .addHelpText(
38
- 'after',
39
- buildHelpText({
40
- context:
41
- 'Pagination: use --after or --before with an email ID as the cursor.\nOnly one of --after or --before may be used at a time.\nThe response includes has_more: true when additional pages exist.',
42
- output:
43
- ' {"object":"list","has_more":false,"data":[{"id":"<uuid>","from":"you@domain.com","to":["user@example.com"],"subject":"Hello","created_at":"<iso-date>","last_event":"delivered","bcc":null,"cc":null,"reply_to":null}]}',
44
- errorCodes: ['auth_error', 'invalid_limit', 'list_error'],
45
- examples: [
46
- 'resend emails list',
47
- 'resend emails list --limit 25 --json',
48
- 'resend emails list --after <email-id> --json',
49
- ],
50
- }),
51
- )
52
- .action(async (opts, cmd) => {
53
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
54
-
55
- const limit = parseLimitOpt(opts.limit, globalOpts);
56
- const paginationOpts = buildPaginationOpts(limit, opts.after, opts.before);
57
-
58
- await runList(
59
- {
60
- spinner: {
61
- loading: 'Fetching emails...',
62
- success: 'Emails fetched',
63
- fail: 'Failed to list emails',
64
- },
65
- sdkCall: (resend) => resend.emails.list(paginationOpts),
66
- onInteractive: (list) => {
67
- console.log(renderEmailsTable(list.data));
68
- printPaginationHint(list);
69
- },
70
- },
71
- globalOpts,
72
- );
73
- });