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,216 @@
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: {
21
+ object: 'contact_property' as const,
22
+ id: 'b4a3c2d1-6e5f-8a7b-0c9d-2e1f4a3b6c5d',
23
+ },
24
+ error: null,
25
+ }));
26
+
27
+ mock.module('resend', () => ({
28
+ Resend: class MockResend {
29
+ constructor(public key: string) {}
30
+ contactProperties = { update: mockUpdate };
31
+ },
32
+ }));
33
+
34
+ describe('contact-properties update command', () => {
35
+ const restoreEnv = captureTestEnv();
36
+ let spies: ReturnType<typeof setupOutputSpies> | undefined;
37
+ let errorSpy: ReturnType<typeof spyOn> | undefined;
38
+ let stderrSpy: ReturnType<typeof spyOn> | undefined;
39
+ let exitSpy: ReturnType<typeof spyOn> | undefined;
40
+
41
+ beforeEach(() => {
42
+ process.env.RESEND_API_KEY = 're_test_key';
43
+ mockUpdate.mockClear();
44
+ });
45
+
46
+ afterEach(() => {
47
+ restoreEnv();
48
+ spies?.restore();
49
+ errorSpy?.mockRestore();
50
+ stderrSpy?.mockRestore();
51
+ exitSpy?.mockRestore();
52
+ spies = undefined;
53
+ errorSpy = undefined;
54
+ stderrSpy = undefined;
55
+ exitSpy = undefined;
56
+ });
57
+
58
+ test('updates property fallback value', async () => {
59
+ spies = setupOutputSpies();
60
+
61
+ const { updateContactPropertyCommand } = await import(
62
+ '../../../src/commands/contact-properties/update'
63
+ );
64
+ await updateContactPropertyCommand.parseAsync(
65
+ ['b4a3c2d1-6e5f-8a7b-0c9d-2e1f4a3b6c5d', '--fallback-value', 'Acme Corp'],
66
+ { from: 'user' },
67
+ );
68
+
69
+ expect(mockUpdate).toHaveBeenCalledTimes(1);
70
+ const args = mockUpdate.mock.calls[0][0] as Record<string, unknown>;
71
+ expect(args.id).toBe('b4a3c2d1-6e5f-8a7b-0c9d-2e1f4a3b6c5d');
72
+ expect(args.fallbackValue).toBe('Acme Corp');
73
+ });
74
+
75
+ test('clears fallback value with --clear-fallback-value', async () => {
76
+ spies = setupOutputSpies();
77
+
78
+ const { updateContactPropertyCommand } = await import(
79
+ '../../../src/commands/contact-properties/update'
80
+ );
81
+ await updateContactPropertyCommand.parseAsync(
82
+ ['b4a3c2d1-6e5f-8a7b-0c9d-2e1f4a3b6c5d', '--clear-fallback-value'],
83
+ { from: 'user' },
84
+ );
85
+
86
+ const args = mockUpdate.mock.calls[0][0] as Record<string, unknown>;
87
+ expect(args.id).toBe('b4a3c2d1-6e5f-8a7b-0c9d-2e1f4a3b6c5d');
88
+ expect(args.fallbackValue).toBeNull();
89
+ });
90
+
91
+ test('errors with conflicting_flags when both --fallback-value and --clear-fallback-value are given', async () => {
92
+ setNonInteractive();
93
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
94
+ exitSpy = mockExitThrow();
95
+
96
+ const { updateContactPropertyCommand } = await import(
97
+ '../../../src/commands/contact-properties/update'
98
+ );
99
+ await expectExit1(() =>
100
+ updateContactPropertyCommand.parseAsync(
101
+ [
102
+ 'b4a3c2d1-6e5f-8a7b-0c9d-2e1f4a3b6c5d',
103
+ '--fallback-value',
104
+ 'Acme',
105
+ '--clear-fallback-value',
106
+ ],
107
+ { from: 'user' },
108
+ ),
109
+ );
110
+
111
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
112
+ expect(output).toContain('conflicting_flags');
113
+ });
114
+
115
+ test('outputs JSON result when non-interactive', async () => {
116
+ spies = setupOutputSpies();
117
+
118
+ const { updateContactPropertyCommand } = await import(
119
+ '../../../src/commands/contact-properties/update'
120
+ );
121
+ await updateContactPropertyCommand.parseAsync(
122
+ ['b4a3c2d1-6e5f-8a7b-0c9d-2e1f4a3b6c5d', '--fallback-value', 'Test'],
123
+ { from: 'user' },
124
+ );
125
+
126
+ const output = spies.logSpy.mock.calls[0][0] as string;
127
+ const parsed = JSON.parse(output);
128
+ expect(parsed.object).toBe('contact_property');
129
+ expect(parsed.id).toBe('b4a3c2d1-6e5f-8a7b-0c9d-2e1f4a3b6c5d');
130
+ });
131
+
132
+ test('errors with no_changes when no flags are provided', async () => {
133
+ setNonInteractive();
134
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
135
+ exitSpy = mockExitThrow();
136
+
137
+ const { updateContactPropertyCommand } = await import(
138
+ '../../../src/commands/contact-properties/update'
139
+ );
140
+ await expectExit1(() =>
141
+ updateContactPropertyCommand.parseAsync(
142
+ ['b4a3c2d1-6e5f-8a7b-0c9d-2e1f4a3b6c5d'],
143
+ {
144
+ from: 'user',
145
+ },
146
+ ),
147
+ );
148
+
149
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
150
+ expect(output).toContain('no_changes');
151
+ });
152
+
153
+ test('does not call SDK when no_changes error is raised', async () => {
154
+ setNonInteractive();
155
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
156
+ exitSpy = mockExitThrow();
157
+
158
+ const { updateContactPropertyCommand } = await import(
159
+ '../../../src/commands/contact-properties/update'
160
+ );
161
+ await expectExit1(() =>
162
+ updateContactPropertyCommand.parseAsync(
163
+ ['b4a3c2d1-6e5f-8a7b-0c9d-2e1f4a3b6c5d'],
164
+ {
165
+ from: 'user',
166
+ },
167
+ ),
168
+ );
169
+
170
+ expect(mockUpdate).not.toHaveBeenCalled();
171
+ });
172
+
173
+ test('errors with auth_error when no API key', async () => {
174
+ setNonInteractive();
175
+ delete process.env.RESEND_API_KEY;
176
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend';
177
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
178
+ exitSpy = mockExitThrow();
179
+
180
+ const { updateContactPropertyCommand } = await import(
181
+ '../../../src/commands/contact-properties/update'
182
+ );
183
+ await expectExit1(() =>
184
+ updateContactPropertyCommand.parseAsync(
185
+ ['b4a3c2d1-6e5f-8a7b-0c9d-2e1f4a3b6c5d', '--fallback-value', 'Test'],
186
+ { from: 'user' },
187
+ ),
188
+ );
189
+
190
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
191
+ expect(output).toContain('auth_error');
192
+ });
193
+
194
+ test('errors with update_error when SDK returns an error', async () => {
195
+ setNonInteractive();
196
+ mockUpdate.mockResolvedValueOnce(
197
+ mockSdkError('Property not found', 'not_found'),
198
+ );
199
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
200
+ stderrSpy = spyOn(process.stderr, 'write').mockImplementation(() => true);
201
+ exitSpy = mockExitThrow();
202
+
203
+ const { updateContactPropertyCommand } = await import(
204
+ '../../../src/commands/contact-properties/update'
205
+ );
206
+ await expectExit1(() =>
207
+ updateContactPropertyCommand.parseAsync(
208
+ ['nonexistent_id', '--fallback-value', 'Test'],
209
+ { from: 'user' },
210
+ ),
211
+ );
212
+
213
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
214
+ expect(output).toContain('update_error');
215
+ });
216
+ });
@@ -0,0 +1,188 @@
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 mockAddSegment = mock(async () => ({
20
+ data: { id: '7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d' },
21
+ error: null,
22
+ }));
23
+
24
+ mock.module('resend', () => ({
25
+ Resend: class MockResend {
26
+ constructor(public key: string) {}
27
+ contacts = {
28
+ segments: { add: mockAddSegment },
29
+ };
30
+ },
31
+ }));
32
+
33
+ describe('contacts add-segment command', () => {
34
+ const restoreEnv = captureTestEnv();
35
+ let spies: ReturnType<typeof setupOutputSpies> | undefined;
36
+ let errorSpy: ReturnType<typeof spyOn> | undefined;
37
+ let stderrSpy: ReturnType<typeof spyOn> | undefined;
38
+ let exitSpy: ReturnType<typeof spyOn> | undefined;
39
+
40
+ beforeEach(() => {
41
+ process.env.RESEND_API_KEY = 're_test_key';
42
+ mockAddSegment.mockClear();
43
+ });
44
+
45
+ afterEach(() => {
46
+ restoreEnv();
47
+ spies?.restore();
48
+ errorSpy?.mockRestore();
49
+ stderrSpy?.mockRestore();
50
+ exitSpy?.mockRestore();
51
+ spies = undefined;
52
+ errorSpy = undefined;
53
+ stderrSpy = undefined;
54
+ exitSpy = undefined;
55
+ });
56
+
57
+ test('adds contact to segment by contact ID', async () => {
58
+ spies = setupOutputSpies();
59
+
60
+ const { addContactSegmentCommand } = await import(
61
+ '../../../src/commands/contacts/add-segment'
62
+ );
63
+ await addContactSegmentCommand.parseAsync(
64
+ [
65
+ 'a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6',
66
+ '--segment-id',
67
+ '7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d',
68
+ ],
69
+ { from: 'user' },
70
+ );
71
+
72
+ expect(mockAddSegment).toHaveBeenCalledTimes(1);
73
+ const args = mockAddSegment.mock.calls[0][0] as Record<string, unknown>;
74
+ expect(args.contactId).toBe('a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6');
75
+ expect(args.segmentId).toBe('7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d');
76
+ });
77
+
78
+ test('adds contact to segment by email', async () => {
79
+ spies = setupOutputSpies();
80
+
81
+ const { addContactSegmentCommand } = await import(
82
+ '../../../src/commands/contacts/add-segment'
83
+ );
84
+ await addContactSegmentCommand.parseAsync(
85
+ [
86
+ 'jane@example.com',
87
+ '--segment-id',
88
+ '7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d',
89
+ ],
90
+ { from: 'user' },
91
+ );
92
+
93
+ const args = mockAddSegment.mock.calls[0][0] as Record<string, unknown>;
94
+ expect(args.email).toBe('jane@example.com');
95
+ expect(args.segmentId).toBe('7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d');
96
+ });
97
+
98
+ test('outputs JSON result when non-interactive', async () => {
99
+ spies = setupOutputSpies();
100
+
101
+ const { addContactSegmentCommand } = await import(
102
+ '../../../src/commands/contacts/add-segment'
103
+ );
104
+ await addContactSegmentCommand.parseAsync(
105
+ [
106
+ 'a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6',
107
+ '--segment-id',
108
+ '7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d',
109
+ ],
110
+ { from: 'user' },
111
+ );
112
+
113
+ const output = spies.logSpy.mock.calls[0][0] as string;
114
+ const parsed = JSON.parse(output);
115
+ expect(parsed.id).toBe('7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d');
116
+ });
117
+
118
+ test('errors with missing_segment_id when --segment-id absent in non-interactive mode', async () => {
119
+ setNonInteractive();
120
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
121
+ exitSpy = mockExitThrow();
122
+
123
+ const { addContactSegmentCommand } = await import(
124
+ '../../../src/commands/contacts/add-segment'
125
+ );
126
+ await expectExit1(() =>
127
+ addContactSegmentCommand.parseAsync(
128
+ ['a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6'],
129
+ { from: 'user' },
130
+ ),
131
+ );
132
+
133
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
134
+ expect(output).toContain('missing_segment_id');
135
+ });
136
+
137
+ test('errors with auth_error when no API key', async () => {
138
+ setNonInteractive();
139
+ delete process.env.RESEND_API_KEY;
140
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend';
141
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
142
+ exitSpy = mockExitThrow();
143
+
144
+ const { addContactSegmentCommand } = await import(
145
+ '../../../src/commands/contacts/add-segment'
146
+ );
147
+ await expectExit1(() =>
148
+ addContactSegmentCommand.parseAsync(
149
+ [
150
+ 'a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6',
151
+ '--segment-id',
152
+ '7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d',
153
+ ],
154
+ { from: 'user' },
155
+ ),
156
+ );
157
+
158
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
159
+ expect(output).toContain('auth_error');
160
+ });
161
+
162
+ test('errors with add_segment_error when SDK returns an error', async () => {
163
+ setNonInteractive();
164
+ mockAddSegment.mockResolvedValueOnce(
165
+ mockSdkError('Segment not found', 'not_found'),
166
+ );
167
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
168
+ stderrSpy = spyOn(process.stderr, 'write').mockImplementation(() => true);
169
+ exitSpy = mockExitThrow();
170
+
171
+ const { addContactSegmentCommand } = await import(
172
+ '../../../src/commands/contacts/add-segment'
173
+ );
174
+ await expectExit1(() =>
175
+ addContactSegmentCommand.parseAsync(
176
+ [
177
+ 'a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6',
178
+ '--segment-id',
179
+ '00000000-0000-0000-0000-00000bad0seg',
180
+ ],
181
+ { from: 'user' },
182
+ ),
183
+ );
184
+
185
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
186
+ expect(output).toContain('add_segment_error');
187
+ });
188
+ });
@@ -0,0 +1,270 @@
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 mockCreate = mock(async () => ({
20
+ data: {
21
+ object: 'contact' as const,
22
+ id: 'a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6',
23
+ },
24
+ error: null,
25
+ }));
26
+
27
+ mock.module('resend', () => ({
28
+ Resend: class MockResend {
29
+ constructor(public key: string) {}
30
+ contacts = { create: mockCreate };
31
+ },
32
+ }));
33
+
34
+ describe('contacts create command', () => {
35
+ const restoreEnv = captureTestEnv();
36
+ let spies: ReturnType<typeof setupOutputSpies> | undefined;
37
+ let errorSpy: ReturnType<typeof spyOn> | undefined;
38
+ let stderrSpy: ReturnType<typeof spyOn> | undefined;
39
+ let exitSpy: ReturnType<typeof spyOn> | undefined;
40
+
41
+ beforeEach(() => {
42
+ process.env.RESEND_API_KEY = 're_test_key';
43
+ mockCreate.mockClear();
44
+ });
45
+
46
+ afterEach(() => {
47
+ restoreEnv();
48
+ spies?.restore();
49
+ errorSpy?.mockRestore();
50
+ stderrSpy?.mockRestore();
51
+ exitSpy?.mockRestore();
52
+ spies = undefined;
53
+ errorSpy = undefined;
54
+ stderrSpy = undefined;
55
+ exitSpy = undefined;
56
+ });
57
+
58
+ test('creates contact with --email flag', async () => {
59
+ spies = setupOutputSpies();
60
+
61
+ const { createContactCommand } = await import(
62
+ '../../../src/commands/contacts/create'
63
+ );
64
+ await createContactCommand.parseAsync(['--email', 'jane@example.com'], {
65
+ from: 'user',
66
+ });
67
+
68
+ expect(mockCreate).toHaveBeenCalledTimes(1);
69
+ const args = mockCreate.mock.calls[0][0] as Record<string, unknown>;
70
+ expect(args.email).toBe('jane@example.com');
71
+ });
72
+
73
+ test('outputs JSON id when non-interactive', async () => {
74
+ spies = setupOutputSpies();
75
+
76
+ const { createContactCommand } = await import(
77
+ '../../../src/commands/contacts/create'
78
+ );
79
+ await createContactCommand.parseAsync(['--email', 'jane@example.com'], {
80
+ from: 'user',
81
+ });
82
+
83
+ const output = spies.logSpy.mock.calls[0][0] as string;
84
+ const parsed = JSON.parse(output);
85
+ expect(parsed.id).toBe('a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6');
86
+ expect(parsed.object).toBe('contact');
87
+ });
88
+
89
+ test('passes --first-name and --last-name to SDK', async () => {
90
+ spies = setupOutputSpies();
91
+
92
+ const { createContactCommand } = await import(
93
+ '../../../src/commands/contacts/create'
94
+ );
95
+ await createContactCommand.parseAsync(
96
+ [
97
+ '--email',
98
+ 'jane@example.com',
99
+ '--first-name',
100
+ 'Jane',
101
+ '--last-name',
102
+ 'Smith',
103
+ ],
104
+ { from: 'user' },
105
+ );
106
+
107
+ const args = mockCreate.mock.calls[0][0] as Record<string, unknown>;
108
+ expect(args.firstName).toBe('Jane');
109
+ expect(args.lastName).toBe('Smith');
110
+ });
111
+
112
+ test('passes --unsubscribed flag to SDK', async () => {
113
+ spies = setupOutputSpies();
114
+
115
+ const { createContactCommand } = await import(
116
+ '../../../src/commands/contacts/create'
117
+ );
118
+ await createContactCommand.parseAsync(
119
+ ['--email', 'jane@example.com', '--unsubscribed'],
120
+ { from: 'user' },
121
+ );
122
+
123
+ const args = mockCreate.mock.calls[0][0] as Record<string, unknown>;
124
+ expect(args.unsubscribed).toBe(true);
125
+ });
126
+
127
+ test('parses --properties JSON and passes to SDK', async () => {
128
+ spies = setupOutputSpies();
129
+
130
+ const { createContactCommand } = await import(
131
+ '../../../src/commands/contacts/create'
132
+ );
133
+ await createContactCommand.parseAsync(
134
+ [
135
+ '--email',
136
+ 'jane@example.com',
137
+ '--properties',
138
+ '{"company":"Acme","plan":"pro"}',
139
+ ],
140
+ { from: 'user' },
141
+ );
142
+
143
+ const args = mockCreate.mock.calls[0][0] as Record<string, unknown>;
144
+ expect(args.properties).toEqual({ company: 'Acme', plan: 'pro' });
145
+ });
146
+
147
+ test('passes --segment-id (single) to SDK as segments array', async () => {
148
+ spies = setupOutputSpies();
149
+
150
+ const { createContactCommand } = await import(
151
+ '../../../src/commands/contacts/create'
152
+ );
153
+ await createContactCommand.parseAsync(
154
+ [
155
+ '--email',
156
+ 'jane@example.com',
157
+ '--segment-id',
158
+ '7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d',
159
+ ],
160
+ { from: 'user' },
161
+ );
162
+
163
+ const args = mockCreate.mock.calls[0][0] as Record<string, unknown>;
164
+ expect(args.segments).toEqual([
165
+ { id: '7b1e0a3d-4c5f-4e8a-9b2d-1a3c5e7f9b2d' },
166
+ ]);
167
+ });
168
+
169
+ test('passes multiple --segment-id values to SDK', async () => {
170
+ spies = setupOutputSpies();
171
+
172
+ const { createContactCommand } = await import(
173
+ '../../../src/commands/contacts/create'
174
+ );
175
+ await createContactCommand.parseAsync(
176
+ [
177
+ '--email',
178
+ 'jane@example.com',
179
+ '--segment-id',
180
+ '3f2a1b4c-5d6e-7f8a-9b0c-1d2e3f4a5b6c',
181
+ '--segment-id',
182
+ 'e8d7c6b5-a4f3-2e1d-0c9b-8a7f6e5d4c3b',
183
+ ],
184
+ { from: 'user' },
185
+ );
186
+
187
+ const args = mockCreate.mock.calls[0][0] as Record<string, unknown>;
188
+ expect(args.segments).toEqual([
189
+ { id: '3f2a1b4c-5d6e-7f8a-9b0c-1d2e3f4a5b6c' },
190
+ { id: 'e8d7c6b5-a4f3-2e1d-0c9b-8a7f6e5d4c3b' },
191
+ ]);
192
+ });
193
+
194
+ test('errors with missing_email in non-interactive mode', async () => {
195
+ setNonInteractive();
196
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
197
+ exitSpy = mockExitThrow();
198
+
199
+ const { createContactCommand } = await import(
200
+ '../../../src/commands/contacts/create'
201
+ );
202
+ await expectExit1(() =>
203
+ createContactCommand.parseAsync([], { from: 'user' }),
204
+ );
205
+
206
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
207
+ expect(output).toContain('missing_email');
208
+ });
209
+
210
+ test('errors with invalid_properties when --properties is not valid JSON', async () => {
211
+ setNonInteractive();
212
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
213
+ exitSpy = mockExitThrow();
214
+
215
+ const { createContactCommand } = await import(
216
+ '../../../src/commands/contacts/create'
217
+ );
218
+ await expectExit1(() =>
219
+ createContactCommand.parseAsync(
220
+ ['--email', 'jane@example.com', '--properties', 'not-json'],
221
+ { from: 'user' },
222
+ ),
223
+ );
224
+
225
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
226
+ expect(output).toContain('invalid_properties');
227
+ });
228
+
229
+ test('errors with auth_error when no API key', async () => {
230
+ setNonInteractive();
231
+ delete process.env.RESEND_API_KEY;
232
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend';
233
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
234
+ exitSpy = mockExitThrow();
235
+
236
+ const { createContactCommand } = await import(
237
+ '../../../src/commands/contacts/create'
238
+ );
239
+ await expectExit1(() =>
240
+ createContactCommand.parseAsync(['--email', 'jane@example.com'], {
241
+ from: 'user',
242
+ }),
243
+ );
244
+
245
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
246
+ expect(output).toContain('auth_error');
247
+ });
248
+
249
+ test('errors with create_error when SDK returns an error', async () => {
250
+ setNonInteractive();
251
+ mockCreate.mockResolvedValueOnce(
252
+ mockSdkError('Contact already exists', 'validation_error'),
253
+ );
254
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
255
+ stderrSpy = spyOn(process.stderr, 'write').mockImplementation(() => true);
256
+ exitSpy = mockExitThrow();
257
+
258
+ const { createContactCommand } = await import(
259
+ '../../../src/commands/contacts/create'
260
+ );
261
+ await expectExit1(() =>
262
+ createContactCommand.parseAsync(['--email', 'jane@example.com'], {
263
+ from: 'user',
264
+ }),
265
+ );
266
+
267
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
268
+ expect(output).toContain('create_error');
269
+ });
270
+ });