resend-cli 1.1.0 → 1.2.0

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 (205) hide show
  1. package/.claude/settings.local.json +1 -10
  2. package/.claude/worktrees/emails-list/.claude/settings.local.json +5 -0
  3. package/.claude/worktrees/emails-list/.github/scripts/pr-title-check.js +34 -0
  4. package/.claude/worktrees/emails-list/.github/workflows/ci.yml +32 -0
  5. package/.claude/worktrees/emails-list/.github/workflows/pr-title-check.yml +13 -0
  6. package/.claude/worktrees/emails-list/.github/workflows/release.yml +93 -0
  7. package/.claude/worktrees/emails-list/CHANGELOG.md +31 -0
  8. package/.claude/worktrees/emails-list/LICENSE +21 -0
  9. package/.claude/worktrees/emails-list/README.md +424 -0
  10. package/.claude/worktrees/emails-list/biome.json +36 -0
  11. package/.claude/worktrees/emails-list/bun.lock +76 -0
  12. package/.claude/worktrees/emails-list/bunfig.toml +2 -0
  13. package/.claude/worktrees/emails-list/install.ps1 +140 -0
  14. package/.claude/worktrees/emails-list/install.sh +301 -0
  15. package/.claude/worktrees/emails-list/package.json +43 -0
  16. package/.claude/worktrees/emails-list/renovate.json +6 -0
  17. package/.claude/worktrees/emails-list/src/cli.ts +74 -0
  18. package/.claude/worktrees/emails-list/src/commands/api-keys/create.ts +114 -0
  19. package/.claude/worktrees/emails-list/src/commands/api-keys/delete.ts +47 -0
  20. package/.claude/worktrees/emails-list/src/commands/api-keys/index.ts +26 -0
  21. package/.claude/worktrees/emails-list/src/commands/api-keys/list.ts +35 -0
  22. package/.claude/worktrees/emails-list/src/commands/api-keys/utils.ts +8 -0
  23. package/.claude/worktrees/emails-list/src/commands/auth/index.ts +20 -0
  24. package/.claude/worktrees/emails-list/src/commands/auth/login.ts +207 -0
  25. package/.claude/worktrees/emails-list/src/commands/auth/logout.ts +105 -0
  26. package/.claude/worktrees/emails-list/src/commands/broadcasts/create.ts +196 -0
  27. package/.claude/worktrees/emails-list/src/commands/broadcasts/delete.ts +46 -0
  28. package/.claude/worktrees/emails-list/src/commands/broadcasts/get.ts +59 -0
  29. package/.claude/worktrees/emails-list/src/commands/broadcasts/index.ts +43 -0
  30. package/.claude/worktrees/emails-list/src/commands/broadcasts/list.ts +60 -0
  31. package/.claude/worktrees/emails-list/src/commands/broadcasts/send.ts +56 -0
  32. package/.claude/worktrees/emails-list/src/commands/broadcasts/update.ts +95 -0
  33. package/.claude/worktrees/emails-list/src/commands/broadcasts/utils.ts +35 -0
  34. package/.claude/worktrees/emails-list/src/commands/contact-properties/create.ts +118 -0
  35. package/.claude/worktrees/emails-list/src/commands/contact-properties/delete.ts +48 -0
  36. package/.claude/worktrees/emails-list/src/commands/contact-properties/get.ts +46 -0
  37. package/.claude/worktrees/emails-list/src/commands/contact-properties/index.ts +48 -0
  38. package/.claude/worktrees/emails-list/src/commands/contact-properties/list.ts +68 -0
  39. package/.claude/worktrees/emails-list/src/commands/contact-properties/update.ts +88 -0
  40. package/.claude/worktrees/emails-list/src/commands/contact-properties/utils.ts +17 -0
  41. package/.claude/worktrees/emails-list/src/commands/contacts/add-segment.ts +78 -0
  42. package/.claude/worktrees/emails-list/src/commands/contacts/create.ts +122 -0
  43. package/.claude/worktrees/emails-list/src/commands/contacts/delete.ts +49 -0
  44. package/.claude/worktrees/emails-list/src/commands/contacts/get.ts +53 -0
  45. package/.claude/worktrees/emails-list/src/commands/contacts/index.ts +58 -0
  46. package/.claude/worktrees/emails-list/src/commands/contacts/list.ts +57 -0
  47. package/.claude/worktrees/emails-list/src/commands/contacts/remove-segment.ts +48 -0
  48. package/.claude/worktrees/emails-list/src/commands/contacts/segments.ts +39 -0
  49. package/.claude/worktrees/emails-list/src/commands/contacts/topics.ts +45 -0
  50. package/.claude/worktrees/emails-list/src/commands/contacts/update-topics.ts +90 -0
  51. package/.claude/worktrees/emails-list/src/commands/contacts/update.ts +77 -0
  52. package/.claude/worktrees/emails-list/src/commands/contacts/utils.ts +119 -0
  53. package/.claude/worktrees/emails-list/src/commands/doctor.ts +298 -0
  54. package/.claude/worktrees/emails-list/src/commands/domains/create.ts +83 -0
  55. package/.claude/worktrees/emails-list/src/commands/domains/delete.ts +42 -0
  56. package/.claude/worktrees/emails-list/src/commands/domains/get.ts +47 -0
  57. package/.claude/worktrees/emails-list/src/commands/domains/index.ts +35 -0
  58. package/.claude/worktrees/emails-list/src/commands/domains/list.ts +53 -0
  59. package/.claude/worktrees/emails-list/src/commands/domains/update.ts +75 -0
  60. package/.claude/worktrees/emails-list/src/commands/domains/utils.ts +44 -0
  61. package/.claude/worktrees/emails-list/src/commands/domains/verify.ts +38 -0
  62. package/.claude/worktrees/emails-list/src/commands/emails/batch.ts +140 -0
  63. package/.claude/worktrees/emails-list/src/commands/emails/index.ts +28 -0
  64. package/.claude/worktrees/emails-list/src/commands/emails/list.ts +73 -0
  65. package/.claude/worktrees/emails-list/src/commands/emails/receiving/attachment.ts +55 -0
  66. package/.claude/worktrees/emails-list/src/commands/emails/receiving/attachments.ts +68 -0
  67. package/.claude/worktrees/emails-list/src/commands/emails/receiving/get.ts +58 -0
  68. package/.claude/worktrees/emails-list/src/commands/emails/receiving/index.ts +28 -0
  69. package/.claude/worktrees/emails-list/src/commands/emails/receiving/list.ts +59 -0
  70. package/.claude/worktrees/emails-list/src/commands/emails/receiving/utils.ts +38 -0
  71. package/.claude/worktrees/emails-list/src/commands/emails/send.ts +189 -0
  72. package/.claude/worktrees/emails-list/src/commands/open.ts +24 -0
  73. package/.claude/worktrees/emails-list/src/commands/segments/create.ts +50 -0
  74. package/.claude/worktrees/emails-list/src/commands/segments/delete.ts +47 -0
  75. package/.claude/worktrees/emails-list/src/commands/segments/get.ts +38 -0
  76. package/.claude/worktrees/emails-list/src/commands/segments/index.ts +36 -0
  77. package/.claude/worktrees/emails-list/src/commands/segments/list.ts +58 -0
  78. package/.claude/worktrees/emails-list/src/commands/segments/utils.ts +7 -0
  79. package/.claude/worktrees/emails-list/src/commands/teams/index.ts +10 -0
  80. package/.claude/worktrees/emails-list/src/commands/teams/list.ts +35 -0
  81. package/.claude/worktrees/emails-list/src/commands/teams/remove.ts +83 -0
  82. package/.claude/worktrees/emails-list/src/commands/teams/switch.ts +73 -0
  83. package/.claude/worktrees/emails-list/src/commands/topics/create.ts +73 -0
  84. package/.claude/worktrees/emails-list/src/commands/topics/delete.ts +47 -0
  85. package/.claude/worktrees/emails-list/src/commands/topics/get.ts +42 -0
  86. package/.claude/worktrees/emails-list/src/commands/topics/index.ts +42 -0
  87. package/.claude/worktrees/emails-list/src/commands/topics/list.ts +34 -0
  88. package/.claude/worktrees/emails-list/src/commands/topics/update.ts +59 -0
  89. package/.claude/worktrees/emails-list/src/commands/topics/utils.ts +16 -0
  90. package/.claude/worktrees/emails-list/src/commands/webhooks/create.ts +128 -0
  91. package/.claude/worktrees/emails-list/src/commands/webhooks/delete.ts +49 -0
  92. package/.claude/worktrees/emails-list/src/commands/webhooks/get.ts +42 -0
  93. package/.claude/worktrees/emails-list/src/commands/webhooks/index.ts +44 -0
  94. package/.claude/worktrees/emails-list/src/commands/webhooks/list.ts +55 -0
  95. package/.claude/worktrees/emails-list/src/commands/webhooks/update.ts +83 -0
  96. package/.claude/worktrees/emails-list/src/commands/webhooks/utils.ts +36 -0
  97. package/.claude/worktrees/emails-list/src/commands/whoami.ts +71 -0
  98. package/.claude/worktrees/emails-list/src/lib/actions.ts +157 -0
  99. package/.claude/worktrees/emails-list/src/lib/client.ts +34 -0
  100. package/.claude/worktrees/emails-list/src/lib/config.ts +211 -0
  101. package/.claude/worktrees/emails-list/src/lib/files.ts +15 -0
  102. package/.claude/worktrees/emails-list/src/lib/help-text.ts +38 -0
  103. package/.claude/worktrees/emails-list/src/lib/output.ts +54 -0
  104. package/.claude/worktrees/emails-list/src/lib/pagination.ts +36 -0
  105. package/.claude/worktrees/emails-list/src/lib/prompts.ts +149 -0
  106. package/.claude/worktrees/emails-list/src/lib/spinner.ts +93 -0
  107. package/.claude/worktrees/emails-list/src/lib/table.ts +57 -0
  108. package/.claude/worktrees/emails-list/src/lib/tty.ts +28 -0
  109. package/.claude/worktrees/emails-list/src/lib/version.ts +4 -0
  110. package/.claude/worktrees/emails-list/tests/commands/api-keys/create.test.ts +195 -0
  111. package/.claude/worktrees/emails-list/tests/commands/api-keys/delete.test.ts +156 -0
  112. package/.claude/worktrees/emails-list/tests/commands/api-keys/list.test.ts +133 -0
  113. package/.claude/worktrees/emails-list/tests/commands/auth/login.test.ts +119 -0
  114. package/.claude/worktrees/emails-list/tests/commands/auth/logout.test.ts +146 -0
  115. package/.claude/worktrees/emails-list/tests/commands/broadcasts/create.test.ts +447 -0
  116. package/.claude/worktrees/emails-list/tests/commands/broadcasts/delete.test.ts +182 -0
  117. package/.claude/worktrees/emails-list/tests/commands/broadcasts/get.test.ts +146 -0
  118. package/.claude/worktrees/emails-list/tests/commands/broadcasts/list.test.ts +196 -0
  119. package/.claude/worktrees/emails-list/tests/commands/broadcasts/send.test.ts +161 -0
  120. package/.claude/worktrees/emails-list/tests/commands/broadcasts/update.test.ts +283 -0
  121. package/.claude/worktrees/emails-list/tests/commands/contact-properties/create.test.ts +250 -0
  122. package/.claude/worktrees/emails-list/tests/commands/contact-properties/delete.test.ts +183 -0
  123. package/.claude/worktrees/emails-list/tests/commands/contact-properties/get.test.ts +144 -0
  124. package/.claude/worktrees/emails-list/tests/commands/contact-properties/list.test.ts +180 -0
  125. package/.claude/worktrees/emails-list/tests/commands/contact-properties/update.test.ts +216 -0
  126. package/.claude/worktrees/emails-list/tests/commands/contacts/add-segment.test.ts +188 -0
  127. package/.claude/worktrees/emails-list/tests/commands/contacts/create.test.ts +270 -0
  128. package/.claude/worktrees/emails-list/tests/commands/contacts/delete.test.ts +192 -0
  129. package/.claude/worktrees/emails-list/tests/commands/contacts/get.test.ts +148 -0
  130. package/.claude/worktrees/emails-list/tests/commands/contacts/list.test.ts +175 -0
  131. package/.claude/worktrees/emails-list/tests/commands/contacts/remove-segment.test.ts +166 -0
  132. package/.claude/worktrees/emails-list/tests/commands/contacts/segments.test.ts +167 -0
  133. package/.claude/worktrees/emails-list/tests/commands/contacts/topics.test.ts +163 -0
  134. package/.claude/worktrees/emails-list/tests/commands/contacts/update-topics.test.ts +247 -0
  135. package/.claude/worktrees/emails-list/tests/commands/contacts/update.test.ts +205 -0
  136. package/.claude/worktrees/emails-list/tests/commands/doctor.test.ts +165 -0
  137. package/.claude/worktrees/emails-list/tests/commands/domains/create.test.ts +192 -0
  138. package/.claude/worktrees/emails-list/tests/commands/domains/delete.test.ts +156 -0
  139. package/.claude/worktrees/emails-list/tests/commands/domains/get.test.ts +137 -0
  140. package/.claude/worktrees/emails-list/tests/commands/domains/list.test.ts +164 -0
  141. package/.claude/worktrees/emails-list/tests/commands/domains/update.test.ts +223 -0
  142. package/.claude/worktrees/emails-list/tests/commands/domains/verify.test.ts +117 -0
  143. package/.claude/worktrees/emails-list/tests/commands/emails/batch.test.ts +313 -0
  144. package/.claude/worktrees/emails-list/tests/commands/emails/list.test.ts +196 -0
  145. package/.claude/worktrees/emails-list/tests/commands/emails/receiving/attachment.test.ts +140 -0
  146. package/.claude/worktrees/emails-list/tests/commands/emails/receiving/attachments.test.ts +168 -0
  147. package/.claude/worktrees/emails-list/tests/commands/emails/receiving/get.test.ts +140 -0
  148. package/.claude/worktrees/emails-list/tests/commands/emails/receiving/list.test.ts +181 -0
  149. package/.claude/worktrees/emails-list/tests/commands/emails/send.test.ts +309 -0
  150. package/.claude/worktrees/emails-list/tests/commands/segments/create.test.ts +163 -0
  151. package/.claude/worktrees/emails-list/tests/commands/segments/delete.test.ts +182 -0
  152. package/.claude/worktrees/emails-list/tests/commands/segments/get.test.ts +137 -0
  153. package/.claude/worktrees/emails-list/tests/commands/segments/list.test.ts +173 -0
  154. package/.claude/worktrees/emails-list/tests/commands/teams/list.test.ts +63 -0
  155. package/.claude/worktrees/emails-list/tests/commands/teams/remove.test.ts +103 -0
  156. package/.claude/worktrees/emails-list/tests/commands/teams/switch.test.ts +96 -0
  157. package/.claude/worktrees/emails-list/tests/commands/topics/create.test.ts +191 -0
  158. package/.claude/worktrees/emails-list/tests/commands/topics/delete.test.ts +156 -0
  159. package/.claude/worktrees/emails-list/tests/commands/topics/get.test.ts +125 -0
  160. package/.claude/worktrees/emails-list/tests/commands/topics/list.test.ts +124 -0
  161. package/.claude/worktrees/emails-list/tests/commands/topics/update.test.ts +177 -0
  162. package/.claude/worktrees/emails-list/tests/commands/webhooks/create.test.ts +224 -0
  163. package/.claude/worktrees/emails-list/tests/commands/webhooks/delete.test.ts +156 -0
  164. package/.claude/worktrees/emails-list/tests/commands/webhooks/get.test.ts +125 -0
  165. package/.claude/worktrees/emails-list/tests/commands/webhooks/list.test.ts +177 -0
  166. package/.claude/worktrees/emails-list/tests/commands/webhooks/update.test.ts +206 -0
  167. package/.claude/worktrees/emails-list/tests/commands/whoami.test.ts +99 -0
  168. package/.claude/worktrees/emails-list/tests/helpers.ts +93 -0
  169. package/.claude/worktrees/emails-list/tests/lib/client.test.ts +71 -0
  170. package/.claude/worktrees/emails-list/tests/lib/config.test.ts +414 -0
  171. package/.claude/worktrees/emails-list/tests/lib/files.test.ts +65 -0
  172. package/.claude/worktrees/emails-list/tests/lib/help-text.test.ts +97 -0
  173. package/.claude/worktrees/emails-list/tests/lib/output.test.ts +127 -0
  174. package/.claude/worktrees/emails-list/tests/lib/prompts.test.ts +178 -0
  175. package/.claude/worktrees/emails-list/tests/lib/spinner.test.ts +146 -0
  176. package/.claude/worktrees/emails-list/tests/lib/table.test.ts +63 -0
  177. package/.claude/worktrees/emails-list/tests/lib/tty.test.ts +85 -0
  178. package/.claude/worktrees/emails-list/tsconfig.json +14 -0
  179. package/.github/workflows/ci.yml +3 -3
  180. package/.github/workflows/pr-title-check.yml +1 -1
  181. package/.github/workflows/release.yml +2 -2
  182. package/.github/workflows/test-build-windows.yml +44 -0
  183. package/.github/workflows/test-install-windows.yml +48 -0
  184. package/README.md +8 -0
  185. package/docs/agent-dx-gaps.md +167 -0
  186. package/docs/missing-commands.md +58 -0
  187. package/docs/production-readiness.md +99 -0
  188. package/docs/secure-key-storage.md +174 -0
  189. package/install.ps1 +1 -0
  190. package/install.sh +11 -4
  191. package/package.json +1 -1
  192. package/renovate.json +4 -0
  193. package/src/cli.ts +9 -0
  194. package/src/commands/auth/login.ts +34 -13
  195. package/src/commands/doctor.ts +3 -3
  196. package/src/commands/open.ts +24 -0
  197. package/src/commands/teams/remove.ts +5 -2
  198. package/src/commands/teams/switch.ts +3 -0
  199. package/src/lib/client.ts +6 -1
  200. package/src/lib/config.ts +37 -30
  201. package/src/lib/help-text.ts +4 -2
  202. package/src/lib/spinner.ts +7 -3
  203. package/tests/commands/auth/login.test.ts +35 -0
  204. package/tests/lib/config.test.ts +40 -7
  205. package/tests/lib/help-text.test.ts +2 -1
@@ -0,0 +1,156 @@
1
+ import {
2
+ afterEach,
3
+ beforeEach,
4
+ describe,
5
+ expect,
6
+ mock,
7
+ spyOn,
8
+ test,
9
+ } from 'bun:test';
10
+ import {
11
+ captureTestEnv,
12
+ expectExit1,
13
+ mockExitThrow,
14
+ mockSdkError,
15
+ setNonInteractive,
16
+ setupOutputSpies,
17
+ } from '../../helpers';
18
+
19
+ const mockRemove = mock(async () => ({
20
+ data: { object: 'domain', id: 'test-domain-id', deleted: true },
21
+ error: null,
22
+ }));
23
+
24
+ mock.module('resend', () => ({
25
+ Resend: class MockResend {
26
+ constructor(public key: string) {}
27
+ domains = { remove: mockRemove };
28
+ },
29
+ }));
30
+
31
+ describe('domains delete command', () => {
32
+ const restoreEnv = captureTestEnv();
33
+ let spies: ReturnType<typeof setupOutputSpies> | undefined;
34
+ let errorSpy: ReturnType<typeof spyOn> | undefined;
35
+ let stderrSpy: ReturnType<typeof spyOn> | undefined;
36
+ let exitSpy: ReturnType<typeof spyOn> | undefined;
37
+
38
+ beforeEach(() => {
39
+ process.env.RESEND_API_KEY = 're_test_key';
40
+ mockRemove.mockClear();
41
+ });
42
+
43
+ afterEach(() => {
44
+ restoreEnv();
45
+ spies?.restore();
46
+ errorSpy?.mockRestore();
47
+ stderrSpy?.mockRestore();
48
+ exitSpy?.mockRestore();
49
+ spies = undefined;
50
+ errorSpy = undefined;
51
+ stderrSpy = undefined;
52
+ exitSpy = undefined;
53
+ });
54
+
55
+ test('deletes domain with --yes flag', async () => {
56
+ spies = setupOutputSpies();
57
+
58
+ const { deleteDomainCommand } = await import(
59
+ '../../../src/commands/domains/delete'
60
+ );
61
+ await deleteDomainCommand.parseAsync(['test-domain-id', '--yes'], {
62
+ from: 'user',
63
+ });
64
+
65
+ expect(mockRemove).toHaveBeenCalledWith('test-domain-id');
66
+ });
67
+
68
+ test('outputs deleted domain JSON on success', async () => {
69
+ spies = setupOutputSpies();
70
+
71
+ const { deleteDomainCommand } = await import(
72
+ '../../../src/commands/domains/delete'
73
+ );
74
+ await deleteDomainCommand.parseAsync(['test-domain-id', '--yes'], {
75
+ from: 'user',
76
+ });
77
+
78
+ const output = spies.logSpy.mock.calls[0][0] as string;
79
+ const parsed = JSON.parse(output);
80
+ expect(parsed.deleted).toBe(true);
81
+ expect(parsed.id).toBe('test-domain-id');
82
+ });
83
+
84
+ test('errors with confirmation_required when --yes absent in non-interactive mode', async () => {
85
+ setNonInteractive();
86
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
87
+ exitSpy = mockExitThrow();
88
+
89
+ const { deleteDomainCommand } = await import(
90
+ '../../../src/commands/domains/delete'
91
+ );
92
+ await expectExit1(() =>
93
+ deleteDomainCommand.parseAsync(['test-domain-id'], { from: 'user' }),
94
+ );
95
+
96
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
97
+ expect(output).toContain('confirmation_required');
98
+ });
99
+
100
+ test('does not call SDK when confirmation is required but not given', async () => {
101
+ setNonInteractive();
102
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
103
+ exitSpy = mockExitThrow();
104
+
105
+ const { deleteDomainCommand } = await import(
106
+ '../../../src/commands/domains/delete'
107
+ );
108
+ await expectExit1(() =>
109
+ deleteDomainCommand.parseAsync(['test-domain-id'], { from: 'user' }),
110
+ );
111
+
112
+ expect(mockRemove).not.toHaveBeenCalled();
113
+ });
114
+
115
+ test('errors with auth_error when no API key', async () => {
116
+ setNonInteractive();
117
+ delete process.env.RESEND_API_KEY;
118
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend';
119
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
120
+ exitSpy = mockExitThrow();
121
+
122
+ const { deleteDomainCommand } = await import(
123
+ '../../../src/commands/domains/delete'
124
+ );
125
+ await expectExit1(() =>
126
+ deleteDomainCommand.parseAsync(['test-domain-id', '--yes'], {
127
+ from: 'user',
128
+ }),
129
+ );
130
+
131
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
132
+ expect(output).toContain('auth_error');
133
+ });
134
+
135
+ test('errors with delete_error when SDK returns an error', async () => {
136
+ setNonInteractive();
137
+ mockRemove.mockResolvedValueOnce(
138
+ mockSdkError('Domain not found', 'not_found'),
139
+ );
140
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
141
+ stderrSpy = spyOn(process.stderr, 'write').mockImplementation(() => true);
142
+ exitSpy = mockExitThrow();
143
+
144
+ const { deleteDomainCommand } = await import(
145
+ '../../../src/commands/domains/delete'
146
+ );
147
+ await expectExit1(() =>
148
+ deleteDomainCommand.parseAsync(['test-domain-id', '--yes'], {
149
+ from: 'user',
150
+ }),
151
+ );
152
+
153
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
154
+ expect(output).toContain('delete_error');
155
+ });
156
+ });
@@ -0,0 +1,137 @@
1
+ import {
2
+ afterEach,
3
+ beforeEach,
4
+ describe,
5
+ expect,
6
+ mock,
7
+ spyOn,
8
+ test,
9
+ } from 'bun:test';
10
+ import {
11
+ captureTestEnv,
12
+ expectExit1,
13
+ mockExitThrow,
14
+ mockSdkError,
15
+ setNonInteractive,
16
+ setupOutputSpies,
17
+ } from '../../helpers';
18
+
19
+ const mockGet = mock(async () => ({
20
+ data: {
21
+ object: 'domain',
22
+ id: 'test-domain-id',
23
+ name: 'example.com',
24
+ status: 'verified',
25
+ created_at: '2026-01-01T00:00:00.000Z',
26
+ region: 'us-east-1',
27
+ records: [
28
+ {
29
+ record: 'SPF',
30
+ type: 'MX',
31
+ name: 'send',
32
+ ttl: 'Auto',
33
+ status: 'verified',
34
+ value: 'feedback-smtp.us-east-1.amazonses.com',
35
+ priority: 10,
36
+ },
37
+ ],
38
+ capabilities: { sending: 'enabled', receiving: 'disabled' },
39
+ },
40
+ error: null,
41
+ }));
42
+
43
+ mock.module('resend', () => ({
44
+ Resend: class MockResend {
45
+ constructor(public key: string) {}
46
+ domains = { get: mockGet };
47
+ },
48
+ }));
49
+
50
+ describe('domains get command', () => {
51
+ const restoreEnv = captureTestEnv();
52
+ let spies: ReturnType<typeof setupOutputSpies> | undefined;
53
+ let errorSpy: ReturnType<typeof spyOn> | undefined;
54
+ let stderrSpy: ReturnType<typeof spyOn> | undefined;
55
+ let exitSpy: ReturnType<typeof spyOn> | undefined;
56
+
57
+ beforeEach(() => {
58
+ process.env.RESEND_API_KEY = 're_test_key';
59
+ mockGet.mockClear();
60
+ });
61
+
62
+ afterEach(() => {
63
+ restoreEnv();
64
+ spies?.restore();
65
+ errorSpy?.mockRestore();
66
+ stderrSpy?.mockRestore();
67
+ exitSpy?.mockRestore();
68
+ spies = undefined;
69
+ errorSpy = undefined;
70
+ stderrSpy = undefined;
71
+ exitSpy = undefined;
72
+ });
73
+
74
+ test('calls SDK get with correct id', async () => {
75
+ spies = setupOutputSpies();
76
+
77
+ const { getDomainCommand } = await import(
78
+ '../../../src/commands/domains/get'
79
+ );
80
+ await getDomainCommand.parseAsync(['test-domain-id'], { from: 'user' });
81
+
82
+ expect(mockGet).toHaveBeenCalledWith('test-domain-id');
83
+ });
84
+
85
+ test('outputs full domain JSON in non-interactive mode', async () => {
86
+ spies = setupOutputSpies();
87
+
88
+ const { getDomainCommand } = await import(
89
+ '../../../src/commands/domains/get'
90
+ );
91
+ await getDomainCommand.parseAsync(['test-domain-id'], { from: 'user' });
92
+
93
+ const output = spies.logSpy.mock.calls[0][0] as string;
94
+ const parsed = JSON.parse(output);
95
+ expect(parsed.id).toBe('test-domain-id');
96
+ expect(parsed.status).toBe('verified');
97
+ expect(parsed.records).toHaveLength(1);
98
+ });
99
+
100
+ test('errors with auth_error when no API key', async () => {
101
+ setNonInteractive();
102
+ delete process.env.RESEND_API_KEY;
103
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend';
104
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
105
+ exitSpy = mockExitThrow();
106
+
107
+ const { getDomainCommand } = await import(
108
+ '../../../src/commands/domains/get'
109
+ );
110
+ await expectExit1(() =>
111
+ getDomainCommand.parseAsync(['test-domain-id'], { from: 'user' }),
112
+ );
113
+
114
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
115
+ expect(output).toContain('auth_error');
116
+ });
117
+
118
+ test('errors with fetch_error when SDK returns an error', async () => {
119
+ setNonInteractive();
120
+ mockGet.mockResolvedValueOnce(
121
+ mockSdkError('Domain not found', 'not_found'),
122
+ );
123
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
124
+ stderrSpy = spyOn(process.stderr, 'write').mockImplementation(() => true);
125
+ exitSpy = mockExitThrow();
126
+
127
+ const { getDomainCommand } = await import(
128
+ '../../../src/commands/domains/get'
129
+ );
130
+ await expectExit1(() =>
131
+ getDomainCommand.parseAsync(['test-domain-id'], { from: 'user' }),
132
+ );
133
+
134
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
135
+ expect(output).toContain('fetch_error');
136
+ });
137
+ });
@@ -0,0 +1,164 @@
1
+ import {
2
+ afterEach,
3
+ beforeEach,
4
+ describe,
5
+ expect,
6
+ mock,
7
+ spyOn,
8
+ test,
9
+ } from 'bun:test';
10
+ import {
11
+ captureTestEnv,
12
+ expectExit1,
13
+ mockExitThrow,
14
+ mockSdkError,
15
+ setNonInteractive,
16
+ setupOutputSpies,
17
+ } from '../../helpers';
18
+
19
+ const mockList = mock(async () => ({
20
+ data: {
21
+ object: 'list',
22
+ data: [
23
+ {
24
+ id: 'domain-1',
25
+ name: 'example.com',
26
+ status: 'verified',
27
+ region: 'us-east-1',
28
+ created_at: '2026-01-01T00:00:00.000Z',
29
+ capabilities: { sending: 'enabled', receiving: 'disabled' },
30
+ },
31
+ {
32
+ id: 'domain-2',
33
+ name: 'test.com',
34
+ status: 'pending',
35
+ region: 'eu-west-1',
36
+ created_at: '2026-01-02T00:00:00.000Z',
37
+ capabilities: { sending: 'enabled', receiving: 'disabled' },
38
+ },
39
+ ],
40
+ has_more: false,
41
+ },
42
+ error: null,
43
+ }));
44
+
45
+ mock.module('resend', () => ({
46
+ Resend: class MockResend {
47
+ constructor(public key: string) {}
48
+ domains = { list: mockList };
49
+ },
50
+ }));
51
+
52
+ describe('domains list command', () => {
53
+ const restoreEnv = captureTestEnv();
54
+ let spies: ReturnType<typeof setupOutputSpies> | undefined;
55
+ let errorSpy: ReturnType<typeof spyOn> | undefined;
56
+ let stderrSpy: ReturnType<typeof spyOn> | undefined;
57
+ let exitSpy: ReturnType<typeof spyOn> | undefined;
58
+
59
+ beforeEach(() => {
60
+ process.env.RESEND_API_KEY = 're_test_key';
61
+ mockList.mockClear();
62
+ });
63
+
64
+ afterEach(() => {
65
+ restoreEnv();
66
+ spies?.restore();
67
+ errorSpy?.mockRestore();
68
+ stderrSpy?.mockRestore();
69
+ exitSpy?.mockRestore();
70
+ spies = undefined;
71
+ errorSpy = undefined;
72
+ stderrSpy = undefined;
73
+ exitSpy = undefined;
74
+ });
75
+
76
+ test('calls SDK list and outputs domains as JSON', async () => {
77
+ spies = setupOutputSpies();
78
+
79
+ const { listDomainsCommand } = await import(
80
+ '../../../src/commands/domains/list'
81
+ );
82
+ await listDomainsCommand.parseAsync([], { from: 'user' });
83
+
84
+ expect(mockList).toHaveBeenCalledTimes(1);
85
+ const output = spies.logSpy.mock.calls[0][0] as string;
86
+ const parsed = JSON.parse(output);
87
+ expect(parsed.object).toBe('list');
88
+ expect(parsed.data).toHaveLength(2);
89
+ });
90
+
91
+ test('passes limit to SDK', async () => {
92
+ spies = setupOutputSpies();
93
+
94
+ const { listDomainsCommand } = await import(
95
+ '../../../src/commands/domains/list'
96
+ );
97
+ await listDomainsCommand.parseAsync(['--limit', '25'], { from: 'user' });
98
+
99
+ const callArgs = mockList.mock.calls[0][0] as Record<string, unknown>;
100
+ expect(callArgs.limit).toBe(25);
101
+ });
102
+
103
+ test('passes after cursor to SDK', async () => {
104
+ spies = setupOutputSpies();
105
+
106
+ const { listDomainsCommand } = await import(
107
+ '../../../src/commands/domains/list'
108
+ );
109
+ await listDomainsCommand.parseAsync(['--after', 'some-cursor'], {
110
+ from: 'user',
111
+ });
112
+
113
+ const callArgs = mockList.mock.calls[0][0] as Record<string, unknown>;
114
+ expect(callArgs.after).toBe('some-cursor');
115
+ });
116
+
117
+ test('uses default limit of 10 when not specified', async () => {
118
+ spies = setupOutputSpies();
119
+
120
+ const { listDomainsCommand } = await import(
121
+ '../../../src/commands/domains/list'
122
+ );
123
+ await listDomainsCommand.parseAsync([], { from: 'user' });
124
+
125
+ const callArgs = mockList.mock.calls[0][0] as Record<string, unknown>;
126
+ expect(callArgs.limit).toBe(10);
127
+ });
128
+
129
+ test('errors with auth_error when no API key', async () => {
130
+ setNonInteractive();
131
+ delete process.env.RESEND_API_KEY;
132
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend';
133
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
134
+ exitSpy = mockExitThrow();
135
+
136
+ const { listDomainsCommand } = await import(
137
+ '../../../src/commands/domains/list'
138
+ );
139
+ await expectExit1(() =>
140
+ listDomainsCommand.parseAsync([], { from: 'user' }),
141
+ );
142
+
143
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
144
+ expect(output).toContain('auth_error');
145
+ });
146
+
147
+ test('errors with list_error when SDK returns an error', async () => {
148
+ setNonInteractive();
149
+ mockList.mockResolvedValueOnce(mockSdkError('Unauthorized', 'auth_error'));
150
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
151
+ stderrSpy = spyOn(process.stderr, 'write').mockImplementation(() => true);
152
+ exitSpy = mockExitThrow();
153
+
154
+ const { listDomainsCommand } = await import(
155
+ '../../../src/commands/domains/list'
156
+ );
157
+ await expectExit1(() =>
158
+ listDomainsCommand.parseAsync([], { from: 'user' }),
159
+ );
160
+
161
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
162
+ expect(output).toContain('list_error');
163
+ });
164
+ });
@@ -0,0 +1,223 @@
1
+ import {
2
+ afterEach,
3
+ beforeEach,
4
+ describe,
5
+ expect,
6
+ mock,
7
+ spyOn,
8
+ test,
9
+ } from 'bun:test';
10
+ import {
11
+ captureTestEnv,
12
+ expectExit1,
13
+ mockExitThrow,
14
+ mockSdkError,
15
+ setNonInteractive,
16
+ setupOutputSpies,
17
+ } from '../../helpers';
18
+
19
+ const mockUpdate = mock(async () => ({
20
+ data: { object: 'domain', id: 'test-domain-id' },
21
+ error: null,
22
+ }));
23
+
24
+ mock.module('resend', () => ({
25
+ Resend: class MockResend {
26
+ constructor(public key: string) {}
27
+ domains = { update: mockUpdate };
28
+ },
29
+ }));
30
+
31
+ describe('domains update command', () => {
32
+ const restoreEnv = captureTestEnv();
33
+ let spies: ReturnType<typeof setupOutputSpies> | undefined;
34
+ let errorSpy: ReturnType<typeof spyOn> | undefined;
35
+ let stderrSpy: ReturnType<typeof spyOn> | undefined;
36
+ let exitSpy: ReturnType<typeof spyOn> | undefined;
37
+
38
+ beforeEach(() => {
39
+ process.env.RESEND_API_KEY = 're_test_key';
40
+ mockUpdate.mockClear();
41
+ });
42
+
43
+ afterEach(() => {
44
+ restoreEnv();
45
+ spies?.restore();
46
+ errorSpy?.mockRestore();
47
+ stderrSpy?.mockRestore();
48
+ exitSpy?.mockRestore();
49
+ spies = undefined;
50
+ errorSpy = undefined;
51
+ stderrSpy = undefined;
52
+ exitSpy = undefined;
53
+ });
54
+
55
+ test('calls SDK update with correct id', async () => {
56
+ spies = setupOutputSpies();
57
+
58
+ const { updateDomainCommand } = await import(
59
+ '../../../src/commands/domains/update'
60
+ );
61
+ await updateDomainCommand.parseAsync(
62
+ ['test-domain-id', '--tls', 'enforced'],
63
+ { from: 'user' },
64
+ );
65
+
66
+ expect(mockUpdate).toHaveBeenCalledTimes(1);
67
+ const args = mockUpdate.mock.calls[0][0] as Record<string, unknown>;
68
+ expect(args.id).toBe('test-domain-id');
69
+ expect(args.tls).toBe('enforced');
70
+ });
71
+
72
+ test('passes openTracking=true when --open-tracking is set', async () => {
73
+ spies = setupOutputSpies();
74
+
75
+ const { updateDomainCommand } = await import(
76
+ '../../../src/commands/domains/update'
77
+ );
78
+ await updateDomainCommand.parseAsync(
79
+ ['test-domain-id', '--open-tracking'],
80
+ { from: 'user' },
81
+ );
82
+
83
+ const args = mockUpdate.mock.calls[0][0] as Record<string, unknown>;
84
+ expect(args.openTracking).toBe(true);
85
+ });
86
+
87
+ test('passes openTracking=false when --no-open-tracking is set', async () => {
88
+ spies = setupOutputSpies();
89
+
90
+ const { updateDomainCommand } = await import(
91
+ '../../../src/commands/domains/update'
92
+ );
93
+ await updateDomainCommand.parseAsync(
94
+ ['test-domain-id', '--no-open-tracking'],
95
+ { from: 'user' },
96
+ );
97
+
98
+ const args = mockUpdate.mock.calls[0][0] as Record<string, unknown>;
99
+ expect(args.openTracking).toBe(false);
100
+ });
101
+
102
+ test('does not include tracking keys in payload when no tracking flags are passed', async () => {
103
+ spies = setupOutputSpies();
104
+
105
+ const { updateDomainCommand } = await import(
106
+ '../../../src/commands/domains/update'
107
+ );
108
+ await updateDomainCommand.parseAsync(
109
+ ['test-domain-id', '--tls', 'enforced'],
110
+ { from: 'user' },
111
+ );
112
+
113
+ const args = mockUpdate.mock.calls[0][0] as Record<string, unknown>;
114
+ expect(args.openTracking).toBeUndefined();
115
+ expect(args.clickTracking).toBeUndefined();
116
+ });
117
+
118
+ test('passes clickTracking=true when --click-tracking is set', async () => {
119
+ spies = setupOutputSpies();
120
+
121
+ const { updateDomainCommand } = await import(
122
+ '../../../src/commands/domains/update'
123
+ );
124
+ await updateDomainCommand.parseAsync(
125
+ ['test-domain-id', '--click-tracking'],
126
+ { from: 'user' },
127
+ );
128
+
129
+ const args = mockUpdate.mock.calls[0][0] as Record<string, unknown>;
130
+ expect(args.clickTracking).toBe(true);
131
+ });
132
+
133
+ test('passes clickTracking=false when --no-click-tracking is set', async () => {
134
+ spies = setupOutputSpies();
135
+
136
+ const { updateDomainCommand } = await import(
137
+ '../../../src/commands/domains/update'
138
+ );
139
+ await updateDomainCommand.parseAsync(
140
+ ['test-domain-id', '--no-click-tracking'],
141
+ { from: 'user' },
142
+ );
143
+
144
+ const args = mockUpdate.mock.calls[0][0] as Record<string, unknown>;
145
+ expect(args.clickTracking).toBe(false);
146
+ });
147
+
148
+ test('errors with no_changes when no update flags are provided', async () => {
149
+ setNonInteractive();
150
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
151
+ exitSpy = mockExitThrow();
152
+
153
+ const { updateDomainCommand } = await import(
154
+ '../../../src/commands/domains/update'
155
+ );
156
+ await expectExit1(() =>
157
+ updateDomainCommand.parseAsync(['test-domain-id'], { from: 'user' }),
158
+ );
159
+
160
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
161
+ expect(output).toContain('no_changes');
162
+ expect(mockUpdate).not.toHaveBeenCalled();
163
+ });
164
+
165
+ test('outputs domain JSON on success', async () => {
166
+ spies = setupOutputSpies();
167
+
168
+ const { updateDomainCommand } = await import(
169
+ '../../../src/commands/domains/update'
170
+ );
171
+ await updateDomainCommand.parseAsync(
172
+ ['test-domain-id', '--tls', 'opportunistic'],
173
+ { from: 'user' },
174
+ );
175
+
176
+ const output = spies.logSpy.mock.calls[0][0] as string;
177
+ const parsed = JSON.parse(output);
178
+ expect(parsed.object).toBe('domain');
179
+ expect(parsed.id).toBe('test-domain-id');
180
+ });
181
+
182
+ test('errors with auth_error when no API key', async () => {
183
+ setNonInteractive();
184
+ delete process.env.RESEND_API_KEY;
185
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend';
186
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
187
+ exitSpy = mockExitThrow();
188
+
189
+ const { updateDomainCommand } = await import(
190
+ '../../../src/commands/domains/update'
191
+ );
192
+ await expectExit1(() =>
193
+ updateDomainCommand.parseAsync(['test-domain-id', '--tls', 'enforced'], {
194
+ from: 'user',
195
+ }),
196
+ );
197
+
198
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
199
+ expect(output).toContain('auth_error');
200
+ });
201
+
202
+ test('errors with update_error when SDK returns an error', async () => {
203
+ setNonInteractive();
204
+ mockUpdate.mockResolvedValueOnce(
205
+ mockSdkError('Domain not found', 'not_found'),
206
+ );
207
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
208
+ stderrSpy = spyOn(process.stderr, 'write').mockImplementation(() => true);
209
+ exitSpy = mockExitThrow();
210
+
211
+ const { updateDomainCommand } = await import(
212
+ '../../../src/commands/domains/update'
213
+ );
214
+ await expectExit1(() =>
215
+ updateDomainCommand.parseAsync(['test-domain-id', '--tls', 'enforced'], {
216
+ from: 'user',
217
+ }),
218
+ );
219
+
220
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
221
+ expect(output).toContain('update_error');
222
+ });
223
+ });