resend-cli 1.2.0 → 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 (193) hide show
  1. package/.github/workflows/release.yml +35 -8
  2. package/README.md +12 -1
  3. package/biome.json +1 -1
  4. package/bun.lock +0 -3
  5. package/package.json +3 -4
  6. package/src/cli.ts +23 -5
  7. package/src/commands/auth/login.ts +10 -8
  8. package/src/commands/doctor.ts +30 -112
  9. package/src/lib/client.ts +3 -0
  10. package/src/lib/config.ts +2 -3
  11. package/src/lib/spinner.ts +17 -10
  12. package/src/lib/update-check.ts +172 -0
  13. package/tests/commands/auth/login.test.ts +3 -1
  14. package/tests/lib/config.test.ts +4 -6
  15. package/tests/lib/update-check.test.ts +169 -0
  16. package/.claude/worktrees/emails-list/.claude/settings.local.json +0 -5
  17. package/.claude/worktrees/emails-list/.github/scripts/pr-title-check.js +0 -34
  18. package/.claude/worktrees/emails-list/.github/workflows/ci.yml +0 -32
  19. package/.claude/worktrees/emails-list/.github/workflows/pr-title-check.yml +0 -13
  20. package/.claude/worktrees/emails-list/.github/workflows/release.yml +0 -93
  21. package/.claude/worktrees/emails-list/CHANGELOG.md +0 -31
  22. package/.claude/worktrees/emails-list/LICENSE +0 -21
  23. package/.claude/worktrees/emails-list/README.md +0 -424
  24. package/.claude/worktrees/emails-list/biome.json +0 -36
  25. package/.claude/worktrees/emails-list/bun.lock +0 -76
  26. package/.claude/worktrees/emails-list/bunfig.toml +0 -2
  27. package/.claude/worktrees/emails-list/install.ps1 +0 -140
  28. package/.claude/worktrees/emails-list/install.sh +0 -301
  29. package/.claude/worktrees/emails-list/package.json +0 -43
  30. package/.claude/worktrees/emails-list/renovate.json +0 -6
  31. package/.claude/worktrees/emails-list/src/cli.ts +0 -74
  32. package/.claude/worktrees/emails-list/src/commands/api-keys/create.ts +0 -114
  33. package/.claude/worktrees/emails-list/src/commands/api-keys/delete.ts +0 -47
  34. package/.claude/worktrees/emails-list/src/commands/api-keys/index.ts +0 -26
  35. package/.claude/worktrees/emails-list/src/commands/api-keys/list.ts +0 -35
  36. package/.claude/worktrees/emails-list/src/commands/api-keys/utils.ts +0 -8
  37. package/.claude/worktrees/emails-list/src/commands/auth/index.ts +0 -20
  38. package/.claude/worktrees/emails-list/src/commands/auth/login.ts +0 -207
  39. package/.claude/worktrees/emails-list/src/commands/auth/logout.ts +0 -105
  40. package/.claude/worktrees/emails-list/src/commands/broadcasts/create.ts +0 -196
  41. package/.claude/worktrees/emails-list/src/commands/broadcasts/delete.ts +0 -46
  42. package/.claude/worktrees/emails-list/src/commands/broadcasts/get.ts +0 -59
  43. package/.claude/worktrees/emails-list/src/commands/broadcasts/index.ts +0 -43
  44. package/.claude/worktrees/emails-list/src/commands/broadcasts/list.ts +0 -60
  45. package/.claude/worktrees/emails-list/src/commands/broadcasts/send.ts +0 -56
  46. package/.claude/worktrees/emails-list/src/commands/broadcasts/update.ts +0 -95
  47. package/.claude/worktrees/emails-list/src/commands/broadcasts/utils.ts +0 -35
  48. package/.claude/worktrees/emails-list/src/commands/contact-properties/create.ts +0 -118
  49. package/.claude/worktrees/emails-list/src/commands/contact-properties/delete.ts +0 -48
  50. package/.claude/worktrees/emails-list/src/commands/contact-properties/get.ts +0 -46
  51. package/.claude/worktrees/emails-list/src/commands/contact-properties/index.ts +0 -48
  52. package/.claude/worktrees/emails-list/src/commands/contact-properties/list.ts +0 -68
  53. package/.claude/worktrees/emails-list/src/commands/contact-properties/update.ts +0 -88
  54. package/.claude/worktrees/emails-list/src/commands/contact-properties/utils.ts +0 -17
  55. package/.claude/worktrees/emails-list/src/commands/contacts/add-segment.ts +0 -78
  56. package/.claude/worktrees/emails-list/src/commands/contacts/create.ts +0 -122
  57. package/.claude/worktrees/emails-list/src/commands/contacts/delete.ts +0 -49
  58. package/.claude/worktrees/emails-list/src/commands/contacts/get.ts +0 -53
  59. package/.claude/worktrees/emails-list/src/commands/contacts/index.ts +0 -58
  60. package/.claude/worktrees/emails-list/src/commands/contacts/list.ts +0 -57
  61. package/.claude/worktrees/emails-list/src/commands/contacts/remove-segment.ts +0 -48
  62. package/.claude/worktrees/emails-list/src/commands/contacts/segments.ts +0 -39
  63. package/.claude/worktrees/emails-list/src/commands/contacts/topics.ts +0 -45
  64. package/.claude/worktrees/emails-list/src/commands/contacts/update-topics.ts +0 -90
  65. package/.claude/worktrees/emails-list/src/commands/contacts/update.ts +0 -77
  66. package/.claude/worktrees/emails-list/src/commands/contacts/utils.ts +0 -119
  67. package/.claude/worktrees/emails-list/src/commands/doctor.ts +0 -298
  68. package/.claude/worktrees/emails-list/src/commands/domains/create.ts +0 -83
  69. package/.claude/worktrees/emails-list/src/commands/domains/delete.ts +0 -42
  70. package/.claude/worktrees/emails-list/src/commands/domains/get.ts +0 -47
  71. package/.claude/worktrees/emails-list/src/commands/domains/index.ts +0 -35
  72. package/.claude/worktrees/emails-list/src/commands/domains/list.ts +0 -53
  73. package/.claude/worktrees/emails-list/src/commands/domains/update.ts +0 -75
  74. package/.claude/worktrees/emails-list/src/commands/domains/utils.ts +0 -44
  75. package/.claude/worktrees/emails-list/src/commands/domains/verify.ts +0 -38
  76. package/.claude/worktrees/emails-list/src/commands/emails/batch.ts +0 -140
  77. package/.claude/worktrees/emails-list/src/commands/emails/index.ts +0 -28
  78. package/.claude/worktrees/emails-list/src/commands/emails/list.ts +0 -73
  79. package/.claude/worktrees/emails-list/src/commands/emails/receiving/attachment.ts +0 -55
  80. package/.claude/worktrees/emails-list/src/commands/emails/receiving/attachments.ts +0 -68
  81. package/.claude/worktrees/emails-list/src/commands/emails/receiving/get.ts +0 -58
  82. package/.claude/worktrees/emails-list/src/commands/emails/receiving/index.ts +0 -28
  83. package/.claude/worktrees/emails-list/src/commands/emails/receiving/list.ts +0 -59
  84. package/.claude/worktrees/emails-list/src/commands/emails/receiving/utils.ts +0 -38
  85. package/.claude/worktrees/emails-list/src/commands/emails/send.ts +0 -189
  86. package/.claude/worktrees/emails-list/src/commands/open.ts +0 -24
  87. package/.claude/worktrees/emails-list/src/commands/segments/create.ts +0 -50
  88. package/.claude/worktrees/emails-list/src/commands/segments/delete.ts +0 -47
  89. package/.claude/worktrees/emails-list/src/commands/segments/get.ts +0 -38
  90. package/.claude/worktrees/emails-list/src/commands/segments/index.ts +0 -36
  91. package/.claude/worktrees/emails-list/src/commands/segments/list.ts +0 -58
  92. package/.claude/worktrees/emails-list/src/commands/segments/utils.ts +0 -7
  93. package/.claude/worktrees/emails-list/src/commands/teams/index.ts +0 -10
  94. package/.claude/worktrees/emails-list/src/commands/teams/list.ts +0 -35
  95. package/.claude/worktrees/emails-list/src/commands/teams/remove.ts +0 -83
  96. package/.claude/worktrees/emails-list/src/commands/teams/switch.ts +0 -73
  97. package/.claude/worktrees/emails-list/src/commands/topics/create.ts +0 -73
  98. package/.claude/worktrees/emails-list/src/commands/topics/delete.ts +0 -47
  99. package/.claude/worktrees/emails-list/src/commands/topics/get.ts +0 -42
  100. package/.claude/worktrees/emails-list/src/commands/topics/index.ts +0 -42
  101. package/.claude/worktrees/emails-list/src/commands/topics/list.ts +0 -34
  102. package/.claude/worktrees/emails-list/src/commands/topics/update.ts +0 -59
  103. package/.claude/worktrees/emails-list/src/commands/topics/utils.ts +0 -16
  104. package/.claude/worktrees/emails-list/src/commands/webhooks/create.ts +0 -128
  105. package/.claude/worktrees/emails-list/src/commands/webhooks/delete.ts +0 -49
  106. package/.claude/worktrees/emails-list/src/commands/webhooks/get.ts +0 -42
  107. package/.claude/worktrees/emails-list/src/commands/webhooks/index.ts +0 -44
  108. package/.claude/worktrees/emails-list/src/commands/webhooks/list.ts +0 -55
  109. package/.claude/worktrees/emails-list/src/commands/webhooks/update.ts +0 -83
  110. package/.claude/worktrees/emails-list/src/commands/webhooks/utils.ts +0 -36
  111. package/.claude/worktrees/emails-list/src/commands/whoami.ts +0 -71
  112. package/.claude/worktrees/emails-list/src/lib/actions.ts +0 -157
  113. package/.claude/worktrees/emails-list/src/lib/client.ts +0 -34
  114. package/.claude/worktrees/emails-list/src/lib/config.ts +0 -211
  115. package/.claude/worktrees/emails-list/src/lib/files.ts +0 -15
  116. package/.claude/worktrees/emails-list/src/lib/help-text.ts +0 -38
  117. package/.claude/worktrees/emails-list/src/lib/output.ts +0 -54
  118. package/.claude/worktrees/emails-list/src/lib/pagination.ts +0 -36
  119. package/.claude/worktrees/emails-list/src/lib/prompts.ts +0 -149
  120. package/.claude/worktrees/emails-list/src/lib/spinner.ts +0 -93
  121. package/.claude/worktrees/emails-list/src/lib/table.ts +0 -57
  122. package/.claude/worktrees/emails-list/src/lib/tty.ts +0 -28
  123. package/.claude/worktrees/emails-list/src/lib/version.ts +0 -4
  124. package/.claude/worktrees/emails-list/tests/commands/api-keys/create.test.ts +0 -195
  125. package/.claude/worktrees/emails-list/tests/commands/api-keys/delete.test.ts +0 -156
  126. package/.claude/worktrees/emails-list/tests/commands/api-keys/list.test.ts +0 -133
  127. package/.claude/worktrees/emails-list/tests/commands/auth/login.test.ts +0 -119
  128. package/.claude/worktrees/emails-list/tests/commands/auth/logout.test.ts +0 -146
  129. package/.claude/worktrees/emails-list/tests/commands/broadcasts/create.test.ts +0 -447
  130. package/.claude/worktrees/emails-list/tests/commands/broadcasts/delete.test.ts +0 -182
  131. package/.claude/worktrees/emails-list/tests/commands/broadcasts/get.test.ts +0 -146
  132. package/.claude/worktrees/emails-list/tests/commands/broadcasts/list.test.ts +0 -196
  133. package/.claude/worktrees/emails-list/tests/commands/broadcasts/send.test.ts +0 -161
  134. package/.claude/worktrees/emails-list/tests/commands/broadcasts/update.test.ts +0 -283
  135. package/.claude/worktrees/emails-list/tests/commands/contact-properties/create.test.ts +0 -250
  136. package/.claude/worktrees/emails-list/tests/commands/contact-properties/delete.test.ts +0 -183
  137. package/.claude/worktrees/emails-list/tests/commands/contact-properties/get.test.ts +0 -144
  138. package/.claude/worktrees/emails-list/tests/commands/contact-properties/list.test.ts +0 -180
  139. package/.claude/worktrees/emails-list/tests/commands/contact-properties/update.test.ts +0 -216
  140. package/.claude/worktrees/emails-list/tests/commands/contacts/add-segment.test.ts +0 -188
  141. package/.claude/worktrees/emails-list/tests/commands/contacts/create.test.ts +0 -270
  142. package/.claude/worktrees/emails-list/tests/commands/contacts/delete.test.ts +0 -192
  143. package/.claude/worktrees/emails-list/tests/commands/contacts/get.test.ts +0 -148
  144. package/.claude/worktrees/emails-list/tests/commands/contacts/list.test.ts +0 -175
  145. package/.claude/worktrees/emails-list/tests/commands/contacts/remove-segment.test.ts +0 -166
  146. package/.claude/worktrees/emails-list/tests/commands/contacts/segments.test.ts +0 -167
  147. package/.claude/worktrees/emails-list/tests/commands/contacts/topics.test.ts +0 -163
  148. package/.claude/worktrees/emails-list/tests/commands/contacts/update-topics.test.ts +0 -247
  149. package/.claude/worktrees/emails-list/tests/commands/contacts/update.test.ts +0 -205
  150. package/.claude/worktrees/emails-list/tests/commands/doctor.test.ts +0 -165
  151. package/.claude/worktrees/emails-list/tests/commands/domains/create.test.ts +0 -192
  152. package/.claude/worktrees/emails-list/tests/commands/domains/delete.test.ts +0 -156
  153. package/.claude/worktrees/emails-list/tests/commands/domains/get.test.ts +0 -137
  154. package/.claude/worktrees/emails-list/tests/commands/domains/list.test.ts +0 -164
  155. package/.claude/worktrees/emails-list/tests/commands/domains/update.test.ts +0 -223
  156. package/.claude/worktrees/emails-list/tests/commands/domains/verify.test.ts +0 -117
  157. package/.claude/worktrees/emails-list/tests/commands/emails/batch.test.ts +0 -313
  158. package/.claude/worktrees/emails-list/tests/commands/emails/list.test.ts +0 -196
  159. package/.claude/worktrees/emails-list/tests/commands/emails/receiving/attachment.test.ts +0 -140
  160. package/.claude/worktrees/emails-list/tests/commands/emails/receiving/attachments.test.ts +0 -168
  161. package/.claude/worktrees/emails-list/tests/commands/emails/receiving/get.test.ts +0 -140
  162. package/.claude/worktrees/emails-list/tests/commands/emails/receiving/list.test.ts +0 -181
  163. package/.claude/worktrees/emails-list/tests/commands/emails/send.test.ts +0 -309
  164. package/.claude/worktrees/emails-list/tests/commands/segments/create.test.ts +0 -163
  165. package/.claude/worktrees/emails-list/tests/commands/segments/delete.test.ts +0 -182
  166. package/.claude/worktrees/emails-list/tests/commands/segments/get.test.ts +0 -137
  167. package/.claude/worktrees/emails-list/tests/commands/segments/list.test.ts +0 -173
  168. package/.claude/worktrees/emails-list/tests/commands/teams/list.test.ts +0 -63
  169. package/.claude/worktrees/emails-list/tests/commands/teams/remove.test.ts +0 -103
  170. package/.claude/worktrees/emails-list/tests/commands/teams/switch.test.ts +0 -96
  171. package/.claude/worktrees/emails-list/tests/commands/topics/create.test.ts +0 -191
  172. package/.claude/worktrees/emails-list/tests/commands/topics/delete.test.ts +0 -156
  173. package/.claude/worktrees/emails-list/tests/commands/topics/get.test.ts +0 -125
  174. package/.claude/worktrees/emails-list/tests/commands/topics/list.test.ts +0 -124
  175. package/.claude/worktrees/emails-list/tests/commands/topics/update.test.ts +0 -177
  176. package/.claude/worktrees/emails-list/tests/commands/webhooks/create.test.ts +0 -224
  177. package/.claude/worktrees/emails-list/tests/commands/webhooks/delete.test.ts +0 -156
  178. package/.claude/worktrees/emails-list/tests/commands/webhooks/get.test.ts +0 -125
  179. package/.claude/worktrees/emails-list/tests/commands/webhooks/list.test.ts +0 -177
  180. package/.claude/worktrees/emails-list/tests/commands/webhooks/update.test.ts +0 -206
  181. package/.claude/worktrees/emails-list/tests/commands/whoami.test.ts +0 -99
  182. package/.claude/worktrees/emails-list/tests/helpers.ts +0 -93
  183. package/.claude/worktrees/emails-list/tests/lib/client.test.ts +0 -71
  184. package/.claude/worktrees/emails-list/tests/lib/config.test.ts +0 -414
  185. package/.claude/worktrees/emails-list/tests/lib/files.test.ts +0 -65
  186. package/.claude/worktrees/emails-list/tests/lib/help-text.test.ts +0 -97
  187. package/.claude/worktrees/emails-list/tests/lib/output.test.ts +0 -127
  188. package/.claude/worktrees/emails-list/tests/lib/prompts.test.ts +0 -178
  189. package/.claude/worktrees/emails-list/tests/lib/spinner.test.ts +0 -146
  190. package/.claude/worktrees/emails-list/tests/lib/table.test.ts +0 -63
  191. package/.claude/worktrees/emails-list/tests/lib/tty.test.ts +0 -85
  192. package/.claude/worktrees/emails-list/tsconfig.json +0 -14
  193. package/.github/workflows/test-build-windows.yml +0 -44
@@ -1,77 +0,0 @@
1
- import { Command } from '@commander-js/extra-typings';
2
- import type { UpdateContactOptions } from 'resend';
3
- import { runWrite } from '../../lib/actions';
4
- import type { GlobalOpts } from '../../lib/client';
5
- import { buildHelpText } from '../../lib/help-text';
6
- import { contactIdentifier, parsePropertiesJson } from './utils';
7
-
8
- export const updateContactCommand = new Command('update')
9
- .description("Update a contact's subscription status or custom properties")
10
- .argument(
11
- '<id>',
12
- 'Contact UUID or email address — both are accepted by the API',
13
- )
14
- .option(
15
- '--unsubscribed',
16
- 'Globally unsubscribe the contact from all broadcasts',
17
- )
18
- .option(
19
- '--no-unsubscribed',
20
- 'Re-subscribe the contact (clears the global unsubscribe flag)',
21
- )
22
- .option(
23
- '--properties <json>',
24
- 'JSON object of properties to merge (e.g. \'{"company":"Acme"}\'); set a key to null to clear it',
25
- )
26
- .addHelpText(
27
- 'after',
28
- buildHelpText({
29
- context: `The <id> argument accepts either a UUID or an email address.
30
-
31
- Subscription toggle:
32
- --unsubscribed Sets unsubscribed: true — contact will not receive any broadcasts.
33
- --no-unsubscribed Sets unsubscribed: false — re-enables broadcast delivery.
34
- Omitting both flags leaves the subscription status unchanged.
35
-
36
- Properties: --properties merges the given JSON object with existing properties.
37
- Set a key to null to clear it: '{"company":null}'.`,
38
- output: ` {"object":"contact","id":"<id>"}`,
39
- errorCodes: ['auth_error', 'invalid_properties', 'update_error'],
40
- examples: [
41
- 'resend contacts update 479e3145-dd38-4932-8c0c-e58b548c9e76 --unsubscribed',
42
- 'resend contacts update user@example.com --no-unsubscribed',
43
- `resend contacts update 479e3145-dd38-4932-8c0c-e58b548c9e76 --properties '{"plan":"pro"}'`,
44
- 'resend contacts update user@example.com --unsubscribed --json',
45
- ],
46
- }),
47
- )
48
- .action(async (id, opts, cmd) => {
49
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
50
-
51
- const properties = parsePropertiesJson(opts.properties, globalOpts);
52
-
53
- // contactIdentifier resolves UUID vs email. The spread of a discriminated
54
- // union requires an explicit cast because TypeScript cannot narrow it
55
- // through a spread at the call site.
56
- const payload = {
57
- ...contactIdentifier(id),
58
- ...(opts.unsubscribed !== undefined && {
59
- unsubscribed: opts.unsubscribed,
60
- }),
61
- ...(properties && { properties }),
62
- } as UpdateContactOptions;
63
-
64
- await runWrite(
65
- {
66
- spinner: {
67
- loading: 'Updating contact...',
68
- success: 'Contact updated',
69
- fail: 'Failed to update contact',
70
- },
71
- sdkCall: (resend) => resend.contacts.update(payload),
72
- errorCode: 'update_error',
73
- successMsg: `Contact updated: ${id}`,
74
- },
75
- globalOpts,
76
- );
77
- });
@@ -1,119 +0,0 @@
1
- import type { ContactSegmentsBaseOptions, ContactTopic } from 'resend';
2
- import type { GlobalOpts } from '../../lib/client';
3
- import { outputError } from '../../lib/output';
4
- import { renderTable } from '../../lib/table';
5
-
6
- // ─── Table renderers ─────────────────────────────────────────────────────────
7
-
8
- export function renderContactsTable(
9
- contacts: Array<{
10
- id: string;
11
- email: string;
12
- first_name: string | null;
13
- last_name: string | null;
14
- unsubscribed: boolean;
15
- }>,
16
- ): string {
17
- const rows = contacts.map((c) => [
18
- c.email,
19
- c.first_name ?? '',
20
- c.last_name ?? '',
21
- c.unsubscribed ? 'yes' : 'no',
22
- c.id,
23
- ]);
24
- return renderTable(
25
- ['Email', 'First Name', 'Last Name', 'Unsubscribed', 'ID'],
26
- rows,
27
- '(no contacts)',
28
- );
29
- }
30
-
31
- export function renderContactTopicsTable(topics: ContactTopic[]): string {
32
- const rows = topics.map((t) => [
33
- t.name,
34
- t.subscription,
35
- t.id,
36
- t.description ?? '',
37
- ]);
38
- return renderTable(
39
- ['Name', 'Subscription', 'ID', 'Description'],
40
- rows,
41
- '(no topic subscriptions)',
42
- );
43
- }
44
-
45
- // ─── Contact identifier helpers ───────────────────────────────────────────────
46
- //
47
- // The Resend SDK uses two different discriminated-union shapes depending on
48
- // the API surface:
49
- //
50
- // • contactIdentifier — produces { id } | { email } for endpoints that
51
- // accept SelectingField (update, get, remove) or { id?, email? }
52
- // (topics list/update).
53
- //
54
- // • segmentContactIdentifier — produces { contactId } | { email } for the
55
- // contacts.segments.* endpoints, which use ContactSegmentsBaseOptions.
56
- //
57
- // Centralising the `str.includes('@')` check here prevents it from drifting
58
- // across six separate command files.
59
-
60
- export function contactIdentifier(
61
- id: string,
62
- ): { id: string } | { email: string } {
63
- return id.includes('@') ? { email: id } : { id };
64
- }
65
-
66
- export function segmentContactIdentifier(
67
- id: string,
68
- ): ContactSegmentsBaseOptions {
69
- return id.includes('@') ? { email: id } : { contactId: id };
70
- }
71
-
72
- // ─── JSON flag helpers ────────────────────────────────────────────────────────
73
-
74
- export function parseTopicsJson(
75
- raw: string,
76
- globalOpts: GlobalOpts,
77
- ): Array<{ id: string; subscription: 'opt_in' | 'opt_out' }> {
78
- let parsed: unknown;
79
- try {
80
- parsed = JSON.parse(raw);
81
- } catch {
82
- outputError(
83
- {
84
- message:
85
- 'Invalid --topics JSON. Expected an array of {id, subscription} objects.',
86
- code: 'invalid_topics',
87
- },
88
- { json: globalOpts.json },
89
- );
90
- }
91
- if (!Array.isArray(parsed)) {
92
- outputError(
93
- {
94
- message:
95
- 'Invalid --topics JSON. Expected an array of {id, subscription} objects.',
96
- code: 'invalid_topics',
97
- },
98
- { json: globalOpts.json },
99
- );
100
- }
101
- return parsed as Array<{ id: string; subscription: 'opt_in' | 'opt_out' }>;
102
- }
103
-
104
- export function parsePropertiesJson(
105
- raw: string | undefined,
106
- globalOpts: GlobalOpts,
107
- ): Record<string, string | number | null> | undefined {
108
- if (!raw) {
109
- return undefined;
110
- }
111
- try {
112
- return JSON.parse(raw) as Record<string, string | number | null>;
113
- } catch {
114
- outputError(
115
- { message: 'Invalid --properties JSON.', code: 'invalid_properties' },
116
- { json: globalOpts.json },
117
- );
118
- }
119
- }
@@ -1,298 +0,0 @@
1
- import { existsSync } from 'node:fs';
2
- import { homedir } from 'node:os';
3
- import { join } from 'node:path';
4
- import { Command } from '@commander-js/extra-typings';
5
- import { Resend } from 'resend';
6
- import type { GlobalOpts } from '../lib/client';
7
- import { maskKey, resolveApiKey } from '../lib/config';
8
- import { buildHelpText } from '../lib/help-text';
9
- import { errorMessage, outputResult } from '../lib/output';
10
- import { createSpinner } from '../lib/spinner';
11
- import { isInteractive } from '../lib/tty';
12
- import { PACKAGE_NAME, VERSION } from '../lib/version';
13
-
14
- type CheckStatus = 'pass' | 'warn' | 'fail';
15
-
16
- type CheckResult = {
17
- name: string;
18
- status: CheckStatus;
19
- message: string;
20
- detail?: string;
21
- };
22
-
23
- const statusIcons: Record<CheckStatus, string> = {
24
- pass: '\x1B[32m✔\x1B[0m',
25
- warn: '\x1B[33m!\x1B[0m',
26
- fail: '\x1B[31m✗\x1B[0m',
27
- };
28
-
29
- async function checkCliVersion(): Promise<CheckResult> {
30
- try {
31
- const encodedName = encodeURIComponent(PACKAGE_NAME);
32
- const res = await fetch(
33
- `https://registry.npmjs.org/${encodedName}/latest`,
34
- {
35
- signal: AbortSignal.timeout(5000),
36
- },
37
- );
38
- if (!res.ok) {
39
- return {
40
- name: 'CLI Version',
41
- status: 'warn',
42
- message: `v${VERSION} (could not check for updates)`,
43
- };
44
- }
45
- const data = (await res.json()) as { version?: string };
46
- const latest = data.version ?? 'unknown';
47
- if (latest === VERSION) {
48
- return {
49
- name: 'CLI Version',
50
- status: 'pass',
51
- message: `v${VERSION} (latest)`,
52
- };
53
- }
54
- return {
55
- name: 'CLI Version',
56
- status: 'warn',
57
- message: `v${VERSION} (latest: v${latest})`,
58
- detail: 'Update available',
59
- };
60
- } catch {
61
- return {
62
- name: 'CLI Version',
63
- status: 'warn',
64
- message: `v${VERSION} (could not check for updates)`,
65
- };
66
- }
67
- }
68
-
69
- function checkApiKeyPresence(): CheckResult {
70
- const resolved = resolveApiKey();
71
- if (!resolved) {
72
- return {
73
- name: 'API Key',
74
- status: 'fail',
75
- message: 'No API key found',
76
- detail: 'Run: resend login',
77
- };
78
- }
79
- const teamInfo = resolved.team ? `, team: ${resolved.team}` : '';
80
- return {
81
- name: 'API Key',
82
- status: 'pass',
83
- message: `${maskKey(resolved.key)} (source: ${resolved.source}${teamInfo})`,
84
- };
85
- }
86
-
87
- async function checkApiValidationAndDomains(): Promise<CheckResult> {
88
- const resolved = resolveApiKey();
89
- if (!resolved) {
90
- return {
91
- name: 'API Validation',
92
- status: 'fail',
93
- message: 'Skipped — no API key',
94
- };
95
- }
96
-
97
- try {
98
- const resend = new Resend(resolved.key);
99
- const { data, error } = await resend.domains.list();
100
-
101
- if (error) {
102
- return {
103
- name: 'API Validation',
104
- status: 'fail',
105
- message: `API key invalid: ${error.message}`,
106
- };
107
- }
108
-
109
- const domains = data?.data ?? [];
110
- const verified = domains.filter((d) => d.status === 'verified');
111
- const pending = domains.filter((d) => d.status !== 'verified');
112
-
113
- if (domains.length === 0) {
114
- return {
115
- name: 'Domains',
116
- status: 'warn',
117
- message: 'No domains configured',
118
- detail: 'Add a domain at https://resend.com/domains',
119
- };
120
- }
121
-
122
- if (verified.length === 0) {
123
- return {
124
- name: 'Domains',
125
- status: 'warn',
126
- message: `${pending.length} domain(s) pending verification`,
127
- detail: domains.map((d) => `${d.name} (${d.status})`).join(', '),
128
- };
129
- }
130
-
131
- return {
132
- name: 'Domains',
133
- status: 'pass',
134
- message: `${verified.length} verified, ${pending.length} pending`,
135
- detail: domains.map((d) => `${d.name} (${d.status})`).join(', '),
136
- };
137
- } catch (err) {
138
- return {
139
- name: 'API Validation',
140
- status: 'fail',
141
- message: errorMessage(err, 'Failed to validate API key'),
142
- };
143
- }
144
- }
145
-
146
- function checkAgentDetection(): CheckResult {
147
- const home = homedir();
148
- const agents: { name: string; found: boolean }[] = [];
149
-
150
- // OpenClaw
151
- agents.push({
152
- name: 'OpenClaw',
153
- found: existsSync(join(home, 'clawd', 'skills')),
154
- });
155
-
156
- // Cursor
157
- agents.push({ name: 'Cursor', found: existsSync(join(home, '.cursor')) });
158
-
159
- // Claude Desktop
160
- const claudeConfigPaths =
161
- process.platform === 'darwin'
162
- ? [
163
- join(
164
- home,
165
- 'Library',
166
- 'Application Support',
167
- 'Claude',
168
- 'claude_desktop_config.json',
169
- ),
170
- ]
171
- : process.platform === 'win32'
172
- ? [
173
- join(
174
- process.env.APPDATA ?? '',
175
- 'Claude',
176
- 'claude_desktop_config.json',
177
- ),
178
- ]
179
- : [join(home, '.config', 'Claude', 'claude_desktop_config.json')];
180
- agents.push({
181
- name: 'Claude Desktop',
182
- found: claudeConfigPaths.some(existsSync),
183
- });
184
-
185
- // VS Code MCP
186
- agents.push({
187
- name: 'VS Code',
188
- found: existsSync(join(process.cwd(), '.vscode', 'mcp.json')),
189
- });
190
-
191
- const detected = agents.filter((a) => a.found);
192
-
193
- if (detected.length === 0) {
194
- return {
195
- name: 'AI Agents',
196
- status: 'pass',
197
- message: 'No AI agents detected',
198
- };
199
- }
200
-
201
- return {
202
- name: 'AI Agents',
203
- status: 'pass',
204
- message: `Detected: ${detected.map((a) => a.name).join(', ')}`,
205
- detail: 'Future: run `resend setup <agent>` to configure integration',
206
- };
207
- }
208
-
209
- export const doctorCommand = new Command('doctor')
210
- .description(
211
- 'Check CLI version, API key, domain status, and AI agent detection',
212
- )
213
- .addHelpText(
214
- 'after',
215
- buildHelpText({
216
- setup: true,
217
- context: `Checks performed:
218
- CLI Version Is the installed version up to date?
219
- API Key Is a key present (--api-key, RESEND_API_KEY, or credentials file)?
220
- API Validation Is the key valid and accepted by the Resend API?
221
- AI Agents Detected: Claude Desktop, Cursor, VS Code MCP, OpenClaw`,
222
- output: ` {\n "ok": true,\n "checks": [\n {"name":"CLI Version","status":"pass","message":"v0.1.0 (latest)"},\n {"name":"API Key","status":"pass","message":"re_...abcd (source: env)"},\n {"name":"Domains","status":"pass","message":"1 verified, 0 pending"},\n {"name":"AI Agents","status":"pass","message":"Detected: Claude Desktop"}\n ]\n }\n status values: "pass" | "warn" | "fail"\n Exit code 1 if any check has status "fail"`,
223
- examples: ['resend doctor', 'resend doctor --json'],
224
- }),
225
- )
226
- .action(async (_opts, cmd) => {
227
- const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
228
- const checks: CheckResult[] = [];
229
- const interactive = isInteractive() && !globalOpts.json;
230
-
231
- if (interactive) {
232
- console.log('\n Resend Doctor\n');
233
- }
234
-
235
- // Check 1: CLI Version
236
- let spinner = interactive
237
- ? createSpinner('Checking CLI version...', 'orbit')
238
- : null;
239
- const versionCheck = await checkCliVersion();
240
- checks.push(versionCheck);
241
- if (versionCheck.status === 'warn') {
242
- spinner?.warn(versionCheck.message);
243
- } else {
244
- spinner?.stop(versionCheck.message);
245
- }
246
-
247
- // Check 2: API Key
248
- spinner = interactive ? createSpinner('Checking API key...', 'scan') : null;
249
- const keyCheck = checkApiKeyPresence();
250
- checks.push(keyCheck);
251
- if (keyCheck.status === 'fail') {
252
- spinner?.fail(keyCheck.message);
253
- } else {
254
- spinner?.stop(keyCheck.message);
255
- }
256
-
257
- // Check 3: API Validation + Domains
258
- spinner = interactive
259
- ? createSpinner('Validating API key & domains...', 'scan')
260
- : null;
261
- const domainCheck = await checkApiValidationAndDomains();
262
- checks.push(domainCheck);
263
- if (domainCheck.status === 'fail') {
264
- spinner?.fail(domainCheck.message);
265
- } else if (domainCheck.status === 'warn') {
266
- spinner?.warn(domainCheck.message);
267
- } else {
268
- spinner?.stop(domainCheck.message);
269
- }
270
-
271
- // Check 4: Agent Detection
272
- spinner = interactive
273
- ? createSpinner('Detecting AI agents...', 'scan')
274
- : null;
275
- const agentCheck = checkAgentDetection();
276
- checks.push(agentCheck);
277
- spinner?.stop(agentCheck.message);
278
-
279
- const hasFails = checks.some((c) => c.status === 'fail');
280
-
281
- if (!globalOpts.json && isInteractive()) {
282
- console.log('');
283
- for (const check of checks) {
284
- const icon = statusIcons[check.status];
285
- console.log(` ${icon} ${check.name}: ${check.message}`);
286
- if (check.detail) {
287
- console.log(` ${check.detail}`);
288
- }
289
- }
290
- console.log('');
291
- } else {
292
- outputResult({ ok: !hasFails, checks }, { json: globalOpts.json });
293
- }
294
-
295
- if (hasFails) {
296
- process.exit(1);
297
- }
298
- });
@@ -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
- });