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,47 +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 deleteApiKeyCommand = new Command('delete')
7
- .alias('rm')
8
- .description(
9
- 'Delete an API key — any services using it will immediately lose access',
10
- )
11
- .argument('<id>', 'API key ID')
12
- .option('--yes', 'Skip confirmation prompt')
13
- .addHelpText(
14
- 'after',
15
- buildHelpText({
16
- context: `Non-interactive: --yes is required to confirm deletion when stdin/stdout is not a TTY.
17
-
18
- Warning: Deleting a key is immediate and irreversible. Any service using this key
19
- will stop authenticating instantly. The current key (used to call this command)
20
- can delete itself — the API does not prevent self-deletion.`,
21
- output: ` {"object":"api-key","id":"<id>","deleted":true}`,
22
- errorCodes: ['auth_error', 'confirmation_required', 'delete_error'],
23
- examples: [
24
- 'resend api-keys delete dacf4072-aa82-4ff3-97de-514ae3000ee0 --yes',
25
- 'resend api-keys delete dacf4072-aa82-4ff3-97de-514ae3000ee0 --yes --json',
26
- ],
27
- }),
28
- )
29
- .action(async (id, opts, cmd) => {
30
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
31
- await runDelete(
32
- id,
33
- !!opts.yes,
34
- {
35
- confirmMessage: `Delete API key ${id}?\nAny services using this key will stop working.`,
36
- spinner: {
37
- loading: 'Deleting API key...',
38
- success: 'API key deleted',
39
- fail: 'Failed to delete API key',
40
- },
41
- object: 'api-key',
42
- successMsg: 'API key deleted.',
43
- sdkCall: (resend) => resend.apiKeys.remove(id),
44
- },
45
- globalOpts,
46
- );
47
- });
@@ -1,26 +0,0 @@
1
- import { Command } from '@commander-js/extra-typings';
2
- import { buildHelpText } from '../../lib/help-text';
3
- import { createApiKeyCommand } from './create';
4
- import { deleteApiKeyCommand } from './delete';
5
- import { listApiKeysCommand } from './list';
6
-
7
- export const apiKeysCommand = new Command('api-keys')
8
- .description('Manage API keys for authentication')
9
- .addHelpText(
10
- 'after',
11
- buildHelpText({
12
- context: `Security notes:
13
- - Tokens are only shown at creation time and cannot be retrieved again.
14
- - Use sending_access keys with --domain-id for per-domain CI tokens.
15
- - Deleting a key is immediate — any service using it loses access instantly.`,
16
- examples: [
17
- 'resend api-keys list',
18
- 'resend api-keys create --name "Production"',
19
- 'resend api-keys create --name "CI Token" --permission sending_access --domain-id <domain-id>',
20
- 'resend api-keys delete <id> --yes',
21
- ],
22
- }),
23
- )
24
- .addCommand(createApiKeyCommand)
25
- .addCommand(listApiKeysCommand, { isDefault: true })
26
- .addCommand(deleteApiKeyCommand);
@@ -1,35 +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 { renderApiKeysTable } from './utils';
6
-
7
- export const listApiKeysCommand = new Command('list')
8
- .alias('ls')
9
- .description(
10
- 'List all API keys (IDs and names — tokens are never returned by this endpoint)',
11
- )
12
- .addHelpText(
13
- 'after',
14
- buildHelpText({
15
- output: ` {"object":"list","data":[{"id":"<id>","name":"<name>","created_at":"<date>"}]}
16
- Tokens are never included in list responses.`,
17
- errorCodes: ['auth_error', 'list_error'],
18
- examples: ['resend api-keys list', 'resend api-keys list --json'],
19
- }),
20
- )
21
- .action(async (_opts, cmd) => {
22
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
23
- await runList(
24
- {
25
- spinner: {
26
- loading: 'Fetching API keys...',
27
- success: 'API keys fetched',
28
- fail: 'Failed to list API keys',
29
- },
30
- sdkCall: (resend) => resend.apiKeys.list(),
31
- onInteractive: (list) => console.log(renderApiKeysTable(list.data)),
32
- },
33
- globalOpts,
34
- );
35
- });
@@ -1,8 +0,0 @@
1
- import { renderTable } from '../../lib/table';
2
-
3
- export function renderApiKeysTable(
4
- keys: Array<{ id: string; name: string; created_at: string }>,
5
- ): string {
6
- const rows = keys.map((k) => [k.name, k.id, k.created_at]);
7
- return renderTable(['Name', 'ID', 'Created'], rows, '(no API keys)');
8
- }
@@ -1,20 +0,0 @@
1
- import { Command } from '@commander-js/extra-typings';
2
- import { buildHelpText } from '../../lib/help-text';
3
- import { loginCommand } from './login';
4
- import { logoutCommand } from './logout';
5
-
6
- export const authCommand = new Command('auth')
7
- .description('Manage authentication')
8
- .addHelpText(
9
- 'after',
10
- buildHelpText({
11
- setup: true,
12
- examples: [
13
- 'resend login',
14
- 'resend login --key re_123456789',
15
- 'resend logout',
16
- ],
17
- }),
18
- )
19
- .addCommand(loginCommand)
20
- .addCommand(logoutCommand);
@@ -1,207 +0,0 @@
1
- import { execFile } from 'node:child_process';
2
- import * as p from '@clack/prompts';
3
- import { Command } from '@commander-js/extra-typings';
4
- import { Resend } from 'resend';
5
- import type { GlobalOpts } from '../../lib/client';
6
- import { listTeams, resolveApiKey, storeApiKey } from '../../lib/config';
7
- import { buildHelpText } from '../../lib/help-text';
8
- import { errorMessage, outputError, outputResult } from '../../lib/output';
9
- import { cancelAndExit } from '../../lib/prompts';
10
- import { createSpinner } from '../../lib/spinner';
11
- import { isInteractive } from '../../lib/tty';
12
-
13
- const RESEND_API_KEYS_URL = 'https://resend.com/api-keys?new=true';
14
-
15
- function openInBrowser(url: string): Promise<boolean> {
16
- return new Promise((resolve) => {
17
- // `start` on Windows is a shell built-in, not an executable.
18
- // Must invoke via `cmd.exe /c start <url>`.
19
- const cmd =
20
- process.platform === 'win32'
21
- ? 'cmd.exe'
22
- : process.platform === 'darwin'
23
- ? 'open'
24
- : 'xdg-open';
25
- const args =
26
- process.platform === 'win32' ? ['/c', 'start', '""', url] : [url];
27
- execFile(cmd, args, { timeout: 5000 }, (err) => resolve(!err));
28
- });
29
- }
30
-
31
- export const loginCommand = new Command('login')
32
- .description('Save a Resend API key to the local credentials file')
33
- .option('--key <key>', 'API key to store (required in non-interactive mode)')
34
- .addHelpText(
35
- 'after',
36
- buildHelpText({
37
- setup: true,
38
- context:
39
- 'Non-interactive: --key is required (no prompts will appear when stdin/stdout is not a TTY).',
40
- output: ` {"success":true,"config_path":"<path>"}`,
41
- errorCodes: ['missing_key', 'invalid_key_format', 'validation_failed'],
42
- examples: [
43
- 'resend login --key re_123456789',
44
- 'resend login (interactive — prompts and opens browser)',
45
- 'RESEND_API_KEY=re_123 resend emails send ... (skip login; use env var directly)',
46
- ],
47
- }),
48
- )
49
- .action(async (opts, cmd) => {
50
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
51
- let apiKey = opts.key;
52
-
53
- if (!apiKey) {
54
- if (!isInteractive()) {
55
- outputError(
56
- {
57
- message:
58
- 'Missing --key flag. Provide your API key in non-interactive mode.',
59
- code: 'missing_key',
60
- },
61
- { json: globalOpts.json },
62
- );
63
- }
64
-
65
- p.intro('Resend Authentication');
66
-
67
- const existing = resolveApiKey();
68
- if (existing) {
69
- p.log.info(
70
- `Existing API key found (source: ${existing.source}). Enter a new key to replace it.`,
71
- );
72
- }
73
-
74
- const method = await p.select({
75
- message: 'How would you like to get your API key?',
76
- options: [
77
- {
78
- value: 'browser' as const,
79
- label: 'Open resend.com/api-keys in browser',
80
- },
81
- { value: 'manual' as const, label: 'Enter API key manually' },
82
- ],
83
- });
84
-
85
- if (p.isCancel(method)) {
86
- cancelAndExit('Login cancelled.');
87
- }
88
-
89
- if (method === 'browser') {
90
- const opened = await openInBrowser(RESEND_API_KEYS_URL);
91
- if (opened) {
92
- p.log.info(`Opened ${RESEND_API_KEYS_URL}`);
93
- } else {
94
- p.log.warn(
95
- `Could not open browser. Visit ${RESEND_API_KEYS_URL} manually.`,
96
- );
97
- }
98
- }
99
-
100
- const result = await p.password({
101
- message: 'Enter your Resend API key:',
102
- validate: (value) => {
103
- if (!value) {
104
- return 'API key is required';
105
- }
106
- if (!value.startsWith('re_')) {
107
- return 'API key must start with re_';
108
- }
109
- return undefined;
110
- },
111
- });
112
-
113
- if (p.isCancel(result)) {
114
- cancelAndExit('Login cancelled.');
115
- }
116
-
117
- apiKey = result;
118
- }
119
-
120
- if (!apiKey.startsWith('re_')) {
121
- outputError(
122
- {
123
- message: 'Invalid API key format. Key must start with re_',
124
- code: 'invalid_key_format',
125
- },
126
- { json: globalOpts.json },
127
- );
128
- }
129
-
130
- const spinner = createSpinner(
131
- 'Validating API key...',
132
- 'braille',
133
- globalOpts.quiet,
134
- );
135
-
136
- try {
137
- const resend = new Resend(apiKey);
138
- await resend.domains.list();
139
- spinner.stop('API key is valid');
140
- } catch (err) {
141
- spinner.fail('API key validation failed');
142
- outputError(
143
- {
144
- message: errorMessage(err, 'Failed to validate API key'),
145
- code: 'validation_failed',
146
- },
147
- { json: globalOpts.json },
148
- );
149
- }
150
-
151
- let teamName = globalOpts.team;
152
-
153
- if (!teamName && isInteractive()) {
154
- const existingTeams = listTeams();
155
- if (existingTeams.length > 0) {
156
- const options = [
157
- ...existingTeams.map((t) => ({
158
- value: t.name,
159
- label: `${t.name} (overwrite)`,
160
- })),
161
- { value: '__new__' as const, label: '+ Create new team' },
162
- ];
163
-
164
- const choice = await p.select({
165
- message: 'Save API key to which team?',
166
- options,
167
- });
168
-
169
- if (p.isCancel(choice)) {
170
- cancelAndExit('Login cancelled.');
171
- }
172
-
173
- if (choice === '__new__') {
174
- const newName = await p.text({
175
- message: 'Enter a name for the new team:',
176
- validate: (v) =>
177
- !v || v.length === 0 ? 'Team name is required' : undefined,
178
- });
179
- if (p.isCancel(newName)) {
180
- cancelAndExit('Login cancelled.');
181
- }
182
- teamName = newName;
183
- } else {
184
- teamName = choice;
185
- }
186
- } else {
187
- teamName = 'default';
188
- }
189
- }
190
-
191
- const configPath = storeApiKey(apiKey, teamName);
192
- const teamLabel = teamName || 'default';
193
-
194
- if (globalOpts.json) {
195
- outputResult(
196
- { success: true, config_path: configPath, team: teamLabel },
197
- { json: true },
198
- );
199
- } else {
200
- const msg = `API key stored for team '${teamLabel}' at ${configPath}`;
201
- if (isInteractive()) {
202
- p.outro(msg);
203
- } else {
204
- console.log(msg);
205
- }
206
- }
207
- });
@@ -1,105 +0,0 @@
1
- import { existsSync } from 'node:fs';
2
- import { join } from 'node:path';
3
- import * as p from '@clack/prompts';
4
- import { Command } from '@commander-js/extra-typings';
5
- import type { GlobalOpts } from '../../lib/client';
6
- import {
7
- getConfigDir,
8
- removeAllApiKeys,
9
- removeApiKey,
10
- resolveTeamName,
11
- } from '../../lib/config';
12
- import { buildHelpText } from '../../lib/help-text';
13
- import { errorMessage, outputError, outputResult } from '../../lib/output';
14
- import { cancelAndExit } from '../../lib/prompts';
15
- import { isInteractive } from '../../lib/tty';
16
-
17
- export const logoutCommand = new Command('logout')
18
- .description(
19
- 'Remove the saved Resend API key from the local credentials file',
20
- )
21
- .addHelpText(
22
- 'after',
23
- buildHelpText({
24
- setup: true,
25
- context: `Removes the saved API key from ~/.config/resend/credentials.json.
26
- (Linux: $XDG_CONFIG_HOME/resend/credentials.json)
27
- (Windows: %APPDATA%\\resend\\credentials.json)
28
-
29
- When --team is specified, only that team's entry is removed.
30
- When no team is specified, all teams are removed.
31
-
32
- If no credentials file exists, exits cleanly with no error.`,
33
- output: ` {"success":true,"config_path":"<path>"}`,
34
- errorCodes: ['remove_failed'],
35
- examples: [
36
- 'resend logout',
37
- 'resend logout --team staging',
38
- 'resend logout --json',
39
- ],
40
- }),
41
- )
42
- .action(async (_opts, cmd) => {
43
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
44
-
45
- const configPath = join(getConfigDir(), 'credentials.json');
46
-
47
- if (!existsSync(configPath)) {
48
- if (!globalOpts.json && isInteractive()) {
49
- console.log('No saved credentials found. Nothing to remove.');
50
- } else {
51
- outputResult(
52
- { success: true, already_logged_out: true },
53
- { json: globalOpts.json },
54
- );
55
- }
56
- return;
57
- }
58
-
59
- const logoutAll = !globalOpts.team;
60
- const teamLabel = globalOpts.team || resolveTeamName();
61
-
62
- if (!globalOpts.json && isInteractive()) {
63
- const message = logoutAll
64
- ? `Remove all saved API keys at ${configPath}?`
65
- : `Remove saved API key for team '${teamLabel}'?`;
66
-
67
- const confirmed = await p.confirm({ message });
68
-
69
- if (p.isCancel(confirmed) || !confirmed) {
70
- cancelAndExit('Logout cancelled.');
71
- }
72
- }
73
-
74
- try {
75
- if (logoutAll) {
76
- removeAllApiKeys();
77
- } else {
78
- removeApiKey(teamLabel);
79
- }
80
- } catch (err) {
81
- outputError(
82
- {
83
- message: errorMessage(err, 'Failed to remove credentials'),
84
- code: 'remove_failed',
85
- },
86
- { json: globalOpts.json },
87
- );
88
- }
89
-
90
- if (!globalOpts.json && isInteractive()) {
91
- const msg = logoutAll
92
- ? 'Logged out. All API keys removed.'
93
- : `Logged out. API key removed for team '${teamLabel}'.`;
94
- p.outro(msg);
95
- } else {
96
- outputResult(
97
- {
98
- success: true,
99
- config_path: configPath,
100
- team: logoutAll ? 'all' : teamLabel,
101
- },
102
- { json: globalOpts.json },
103
- );
104
- }
105
- });
@@ -1,196 +0,0 @@
1
- import * as p from '@clack/prompts';
2
- import { Command } from '@commander-js/extra-typings';
3
- import type { CreateBroadcastOptions } from 'resend';
4
- import { runCreate } from '../../lib/actions';
5
- import type { GlobalOpts } from '../../lib/client';
6
- import { readFile } from '../../lib/files';
7
- import { buildHelpText } from '../../lib/help-text';
8
- import { outputError } from '../../lib/output';
9
- import { cancelAndExit } from '../../lib/prompts';
10
- import { isInteractive } from '../../lib/tty';
11
-
12
- export const createBroadcastCommand = new Command('create')
13
- .description('Create a broadcast draft (or send immediately with --send)')
14
- .option('--from <address>', 'Sender address — required')
15
- .option('--subject <subject>', 'Email subject — required')
16
- .option('--segment-id <id>', 'Target segment ID — required')
17
- .option(
18
- '--html <html>',
19
- 'HTML body (supports {{{FIRST_NAME|fallback}}} triple-brace variable interpolation)',
20
- )
21
- .option(
22
- '--html-file <path>',
23
- 'Path to an HTML file for the body (supports {{{FIRST_NAME|fallback}}} variable interpolation)',
24
- )
25
- .option('--text <text>', 'Plain-text body')
26
- .option('--name <name>', 'Internal label for the broadcast (optional)')
27
- .option('--reply-to <address>', 'Reply-to address (optional)')
28
- .option(
29
- '--preview-text <text>',
30
- 'Preview text shown in inbox below the subject line (optional)',
31
- )
32
- .option(
33
- '--topic-id <id>',
34
- 'Associate with a topic for subscription filtering (optional)',
35
- )
36
- .option('--send', 'Send immediately on create instead of saving as draft')
37
- .option(
38
- '--scheduled-at <datetime>',
39
- 'Schedule delivery — ISO 8601 or natural language e.g. "in 1 hour", "tomorrow at 9am ET" (only valid with --send)',
40
- )
41
- .addHelpText(
42
- 'after',
43
- buildHelpText({
44
- context: `Non-interactive: --from, --subject, and --segment-id are required.
45
- Body: provide exactly one of --html, --html-file, or --text.
46
-
47
- Variable interpolation:
48
- HTML bodies support triple-brace syntax for contact properties.
49
- Example: {{{FIRST_NAME|Friend}}} — uses FIRST_NAME or falls back to "Friend".
50
-
51
- Scheduling:
52
- Use --scheduled-at with --send to schedule delivery.
53
- Accepts ISO 8601 (e.g. 2026-08-05T11:52:01Z) or natural language (e.g. "in 1 hour").
54
- --scheduled-at without --send is ignored.`,
55
- output: ` {"id":"<broadcast-id>"}`,
56
- errorCodes: [
57
- 'auth_error',
58
- 'missing_from',
59
- 'missing_subject',
60
- 'missing_segment',
61
- 'missing_body',
62
- 'file_read_error',
63
- 'create_error',
64
- ],
65
- examples: [
66
- 'resend broadcasts create --from hello@domain.com --subject "Weekly Update" --segment-id 7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d --html "<p>Hello {{{FIRST_NAME|there}}}</p>"',
67
- 'resend broadcasts create --from hello@domain.com --subject "Launch" --segment-id 7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d --html-file ./email.html --send',
68
- 'resend broadcasts create --from hello@domain.com --subject "Launch" --segment-id 7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d --text "Hello!" --send --scheduled-at "tomorrow at 9am ET"',
69
- 'resend broadcasts create --from hello@domain.com --subject "News" --segment-id 7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d --html "<p>Hi</p>" --json',
70
- ],
71
- }),
72
- )
73
- .action(async (opts, cmd) => {
74
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
75
-
76
- let from = opts.from;
77
- let subject = opts.subject;
78
- let segmentId = opts.segmentId;
79
-
80
- if (!from) {
81
- if (!isInteractive()) {
82
- outputError(
83
- { message: 'Missing --from flag.', code: 'missing_from' },
84
- { json: globalOpts.json },
85
- );
86
- }
87
- const result = await p.text({
88
- message: 'From address',
89
- placeholder: 'hello@domain.com',
90
- validate: (v) => (!v ? 'Required' : undefined),
91
- });
92
- if (p.isCancel(result)) {
93
- cancelAndExit('Cancelled.');
94
- }
95
- from = result;
96
- }
97
-
98
- if (!subject) {
99
- if (!isInteractive()) {
100
- outputError(
101
- { message: 'Missing --subject flag.', code: 'missing_subject' },
102
- { json: globalOpts.json },
103
- );
104
- }
105
- const result = await p.text({
106
- message: 'Subject',
107
- placeholder: 'Weekly Newsletter',
108
- validate: (v) => (!v ? 'Required' : undefined),
109
- });
110
- if (p.isCancel(result)) {
111
- cancelAndExit('Cancelled.');
112
- }
113
- subject = result;
114
- }
115
-
116
- if (!segmentId) {
117
- if (!isInteractive()) {
118
- outputError(
119
- { message: 'Missing --segment-id flag.', code: 'missing_segment' },
120
- { json: globalOpts.json },
121
- );
122
- }
123
- const result = await p.text({
124
- message: 'Segment ID',
125
- placeholder: '7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d',
126
- validate: (v) => (!v ? 'Required' : undefined),
127
- });
128
- if (p.isCancel(result)) {
129
- cancelAndExit('Cancelled.');
130
- }
131
- segmentId = result;
132
- }
133
-
134
- let html = opts.html;
135
- let text = opts.text;
136
-
137
- if (opts.htmlFile) {
138
- html = readFile(opts.htmlFile, globalOpts);
139
- }
140
-
141
- if (!html && !text) {
142
- if (!isInteractive()) {
143
- outputError(
144
- {
145
- message: 'Missing body. Provide --html, --html-file, or --text.',
146
- code: 'missing_body',
147
- },
148
- { json: globalOpts.json },
149
- );
150
- }
151
- const result = await p.text({
152
- message: 'Body (plain text)',
153
- placeholder: 'Hello {{{FIRST_NAME|there}}}!',
154
- validate: (v) => (!v ? 'Required' : undefined),
155
- });
156
- if (p.isCancel(result)) {
157
- cancelAndExit('Cancelled.');
158
- }
159
- text = result;
160
- }
161
-
162
- await runCreate(
163
- {
164
- spinner: {
165
- loading: 'Creating broadcast...',
166
- success: opts.send ? 'Broadcast sent' : 'Broadcast created',
167
- fail: 'Failed to create broadcast',
168
- },
169
- sdkCall: (resend) =>
170
- resend.broadcasts.create({
171
- from,
172
- subject,
173
- segmentId,
174
- ...(html && { html }),
175
- ...(text && { text }),
176
- ...(opts.name && { name: opts.name }),
177
- ...(opts.replyTo && { replyTo: opts.replyTo }),
178
- ...(opts.previewText && { previewText: opts.previewText }),
179
- ...(opts.topicId && { topicId: opts.topicId }),
180
- ...(opts.send && { send: true as const }),
181
- ...(opts.send &&
182
- opts.scheduledAt && { scheduledAt: opts.scheduledAt }),
183
- } as CreateBroadcastOptions),
184
- onInteractive: (d) => {
185
- if (opts.send) {
186
- console.log(`\nBroadcast sent: ${d.id}`);
187
- } else {
188
- console.log(`\nBroadcast created: ${d.id}`);
189
- console.log('Status: draft');
190
- console.log(`\nSend it with: resend broadcasts send ${d.id}`);
191
- }
192
- },
193
- },
194
- globalOpts,
195
- );
196
- });