resend-cli 1.0.3 → 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 (365) hide show
  1. package/.claude/settings.local.json +5 -0
  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/scripts/pr-title-check.js +34 -0
  180. package/.github/workflows/ci.yml +32 -0
  181. package/.github/workflows/pr-title-check.yml +13 -0
  182. package/.github/workflows/release.yml +93 -0
  183. package/.github/workflows/test-build-windows.yml +44 -0
  184. package/.github/workflows/test-install-windows.yml +48 -0
  185. package/CHANGELOG.md +31 -0
  186. package/LICENSE +21 -21
  187. package/README.md +424 -19
  188. package/biome.json +36 -0
  189. package/bun.lock +76 -0
  190. package/bunfig.toml +2 -0
  191. package/docs/agent-dx-gaps.md +167 -0
  192. package/docs/missing-commands.md +58 -0
  193. package/docs/production-readiness.md +99 -0
  194. package/docs/secure-key-storage.md +174 -0
  195. package/install.ps1 +141 -0
  196. package/install.sh +301 -0
  197. package/package.json +43 -22
  198. package/renovate.json +4 -0
  199. package/src/cli.ts +74 -0
  200. package/src/commands/api-keys/create.ts +114 -0
  201. package/src/commands/api-keys/delete.ts +47 -0
  202. package/src/commands/api-keys/index.ts +26 -0
  203. package/src/commands/api-keys/list.ts +35 -0
  204. package/src/commands/api-keys/utils.ts +8 -0
  205. package/src/commands/auth/index.ts +20 -0
  206. package/src/commands/auth/login.ts +232 -0
  207. package/src/commands/auth/logout.ts +105 -0
  208. package/src/commands/broadcasts/create.ts +196 -0
  209. package/src/commands/broadcasts/delete.ts +46 -0
  210. package/src/commands/broadcasts/get.ts +59 -0
  211. package/src/commands/broadcasts/index.ts +43 -0
  212. package/src/commands/broadcasts/list.ts +60 -0
  213. package/src/commands/broadcasts/send.ts +56 -0
  214. package/src/commands/broadcasts/update.ts +95 -0
  215. package/src/commands/broadcasts/utils.ts +35 -0
  216. package/src/commands/contact-properties/create.ts +118 -0
  217. package/src/commands/contact-properties/delete.ts +48 -0
  218. package/src/commands/contact-properties/get.ts +46 -0
  219. package/src/commands/contact-properties/index.ts +48 -0
  220. package/src/commands/contact-properties/list.ts +68 -0
  221. package/src/commands/contact-properties/update.ts +88 -0
  222. package/src/commands/contact-properties/utils.ts +17 -0
  223. package/src/commands/contacts/add-segment.ts +78 -0
  224. package/src/commands/contacts/create.ts +122 -0
  225. package/src/commands/contacts/delete.ts +49 -0
  226. package/src/commands/contacts/get.ts +53 -0
  227. package/src/commands/contacts/index.ts +58 -0
  228. package/src/commands/contacts/list.ts +57 -0
  229. package/src/commands/contacts/remove-segment.ts +48 -0
  230. package/src/commands/contacts/segments.ts +39 -0
  231. package/src/commands/contacts/topics.ts +45 -0
  232. package/src/commands/contacts/update-topics.ts +90 -0
  233. package/src/commands/contacts/update.ts +77 -0
  234. package/src/commands/contacts/utils.ts +119 -0
  235. package/src/commands/doctor.ts +298 -0
  236. package/src/commands/domains/create.ts +83 -0
  237. package/src/commands/domains/delete.ts +42 -0
  238. package/src/commands/domains/get.ts +47 -0
  239. package/src/commands/domains/index.ts +35 -0
  240. package/src/commands/domains/list.ts +53 -0
  241. package/src/commands/domains/update.ts +75 -0
  242. package/src/commands/domains/utils.ts +44 -0
  243. package/src/commands/domains/verify.ts +38 -0
  244. package/src/commands/emails/batch.ts +140 -0
  245. package/src/commands/emails/index.ts +24 -0
  246. package/src/commands/emails/receiving/attachment.ts +55 -0
  247. package/src/commands/emails/receiving/attachments.ts +68 -0
  248. package/src/commands/emails/receiving/get.ts +58 -0
  249. package/src/commands/emails/receiving/index.ts +28 -0
  250. package/src/commands/emails/receiving/list.ts +59 -0
  251. package/src/commands/emails/receiving/utils.ts +38 -0
  252. package/src/commands/emails/send.ts +189 -0
  253. package/src/commands/open.ts +24 -0
  254. package/src/commands/segments/create.ts +50 -0
  255. package/src/commands/segments/delete.ts +47 -0
  256. package/src/commands/segments/get.ts +38 -0
  257. package/src/commands/segments/index.ts +36 -0
  258. package/src/commands/segments/list.ts +58 -0
  259. package/src/commands/segments/utils.ts +7 -0
  260. package/src/commands/teams/index.ts +10 -0
  261. package/src/commands/teams/list.ts +35 -0
  262. package/src/commands/teams/remove.ts +86 -0
  263. package/src/commands/teams/switch.ts +76 -0
  264. package/src/commands/topics/create.ts +73 -0
  265. package/src/commands/topics/delete.ts +47 -0
  266. package/src/commands/topics/get.ts +42 -0
  267. package/src/commands/topics/index.ts +42 -0
  268. package/src/commands/topics/list.ts +34 -0
  269. package/src/commands/topics/update.ts +59 -0
  270. package/src/commands/topics/utils.ts +16 -0
  271. package/src/commands/webhooks/create.ts +128 -0
  272. package/src/commands/webhooks/delete.ts +49 -0
  273. package/src/commands/webhooks/get.ts +42 -0
  274. package/src/commands/webhooks/index.ts +44 -0
  275. package/src/commands/webhooks/list.ts +55 -0
  276. package/src/commands/webhooks/update.ts +83 -0
  277. package/src/commands/webhooks/utils.ts +36 -0
  278. package/src/commands/whoami.ts +71 -0
  279. package/src/lib/actions.ts +157 -0
  280. package/src/lib/client.ts +34 -0
  281. package/src/lib/config.ts +218 -0
  282. package/src/lib/files.ts +15 -0
  283. package/src/lib/help-text.ts +38 -0
  284. package/src/lib/output.ts +54 -0
  285. package/src/lib/pagination.ts +36 -0
  286. package/src/lib/prompts.ts +149 -0
  287. package/src/lib/spinner.ts +93 -0
  288. package/src/lib/table.ts +57 -0
  289. package/src/lib/tty.ts +28 -0
  290. package/src/lib/version.ts +4 -0
  291. package/tests/commands/api-keys/create.test.ts +195 -0
  292. package/tests/commands/api-keys/delete.test.ts +156 -0
  293. package/tests/commands/api-keys/list.test.ts +133 -0
  294. package/tests/commands/auth/login.test.ts +154 -0
  295. package/tests/commands/auth/logout.test.ts +146 -0
  296. package/tests/commands/broadcasts/create.test.ts +447 -0
  297. package/tests/commands/broadcasts/delete.test.ts +182 -0
  298. package/tests/commands/broadcasts/get.test.ts +146 -0
  299. package/tests/commands/broadcasts/list.test.ts +196 -0
  300. package/tests/commands/broadcasts/send.test.ts +161 -0
  301. package/tests/commands/broadcasts/update.test.ts +283 -0
  302. package/tests/commands/contact-properties/create.test.ts +250 -0
  303. package/tests/commands/contact-properties/delete.test.ts +183 -0
  304. package/tests/commands/contact-properties/get.test.ts +144 -0
  305. package/tests/commands/contact-properties/list.test.ts +180 -0
  306. package/tests/commands/contact-properties/update.test.ts +216 -0
  307. package/tests/commands/contacts/add-segment.test.ts +188 -0
  308. package/tests/commands/contacts/create.test.ts +270 -0
  309. package/tests/commands/contacts/delete.test.ts +192 -0
  310. package/tests/commands/contacts/get.test.ts +148 -0
  311. package/tests/commands/contacts/list.test.ts +175 -0
  312. package/tests/commands/contacts/remove-segment.test.ts +166 -0
  313. package/tests/commands/contacts/segments.test.ts +167 -0
  314. package/tests/commands/contacts/topics.test.ts +163 -0
  315. package/tests/commands/contacts/update-topics.test.ts +247 -0
  316. package/tests/commands/contacts/update.test.ts +205 -0
  317. package/tests/commands/doctor.test.ts +165 -0
  318. package/tests/commands/domains/create.test.ts +192 -0
  319. package/tests/commands/domains/delete.test.ts +156 -0
  320. package/tests/commands/domains/get.test.ts +137 -0
  321. package/tests/commands/domains/list.test.ts +164 -0
  322. package/tests/commands/domains/update.test.ts +223 -0
  323. package/tests/commands/domains/verify.test.ts +117 -0
  324. package/tests/commands/emails/batch.test.ts +313 -0
  325. package/tests/commands/emails/receiving/attachment.test.ts +140 -0
  326. package/tests/commands/emails/receiving/attachments.test.ts +168 -0
  327. package/tests/commands/emails/receiving/get.test.ts +140 -0
  328. package/tests/commands/emails/receiving/list.test.ts +181 -0
  329. package/tests/commands/emails/send.test.ts +309 -0
  330. package/tests/commands/segments/create.test.ts +163 -0
  331. package/tests/commands/segments/delete.test.ts +182 -0
  332. package/tests/commands/segments/get.test.ts +137 -0
  333. package/tests/commands/segments/list.test.ts +173 -0
  334. package/tests/commands/teams/list.test.ts +63 -0
  335. package/tests/commands/teams/remove.test.ts +103 -0
  336. package/tests/commands/teams/switch.test.ts +96 -0
  337. package/tests/commands/topics/create.test.ts +191 -0
  338. package/tests/commands/topics/delete.test.ts +156 -0
  339. package/tests/commands/topics/get.test.ts +125 -0
  340. package/tests/commands/topics/list.test.ts +124 -0
  341. package/tests/commands/topics/update.test.ts +177 -0
  342. package/tests/commands/webhooks/create.test.ts +224 -0
  343. package/tests/commands/webhooks/delete.test.ts +156 -0
  344. package/tests/commands/webhooks/get.test.ts +125 -0
  345. package/tests/commands/webhooks/list.test.ts +177 -0
  346. package/tests/commands/webhooks/update.test.ts +206 -0
  347. package/tests/commands/whoami.test.ts +99 -0
  348. package/tests/helpers.ts +93 -0
  349. package/tests/lib/client.test.ts +71 -0
  350. package/tests/lib/config.test.ts +447 -0
  351. package/tests/lib/files.test.ts +65 -0
  352. package/tests/lib/help-text.test.ts +97 -0
  353. package/tests/lib/output.test.ts +127 -0
  354. package/tests/lib/prompts.test.ts +178 -0
  355. package/tests/lib/spinner.test.ts +146 -0
  356. package/tests/lib/table.test.ts +63 -0
  357. package/tests/lib/tty.test.ts +85 -0
  358. package/tsconfig.json +14 -0
  359. package/src/index.js +0 -72
  360. package/src/routes.js +0 -37
  361. package/src/sections/apikeys.js +0 -99
  362. package/src/sections/audiences.js +0 -84
  363. package/src/sections/contacts.js +0 -177
  364. package/src/sections/domain.js +0 -195
  365. package/src/sections/email.js +0 -132
@@ -0,0 +1,177 @@
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' as const,
22
+ has_more: false,
23
+ data: [
24
+ {
25
+ id: 'wh_abc123',
26
+ endpoint: 'https://app.example.com/hooks/resend',
27
+ events: ['email.sent', 'email.bounced'] as string[],
28
+ status: 'enabled' as const,
29
+ created_at: '2026-01-01T00:00:00.000Z',
30
+ },
31
+ ],
32
+ },
33
+ error: null,
34
+ }));
35
+
36
+ mock.module('resend', () => ({
37
+ Resend: class MockResend {
38
+ constructor(public key: string) {}
39
+ webhooks = { list: mockList };
40
+ },
41
+ }));
42
+
43
+ describe('webhooks list command', () => {
44
+ const restoreEnv = captureTestEnv();
45
+ let spies: ReturnType<typeof setupOutputSpies> | undefined;
46
+ let errorSpy: ReturnType<typeof spyOn> | undefined;
47
+ let stderrSpy: ReturnType<typeof spyOn> | undefined;
48
+ let exitSpy: ReturnType<typeof spyOn> | undefined;
49
+
50
+ beforeEach(() => {
51
+ process.env.RESEND_API_KEY = 're_test_key';
52
+ mockList.mockClear();
53
+ });
54
+
55
+ afterEach(() => {
56
+ restoreEnv();
57
+ spies?.restore();
58
+ errorSpy?.mockRestore();
59
+ stderrSpy?.mockRestore();
60
+ exitSpy?.mockRestore();
61
+ spies = undefined;
62
+ errorSpy = undefined;
63
+ stderrSpy = undefined;
64
+ exitSpy = undefined;
65
+ });
66
+
67
+ test('calls SDK list method with default pagination', async () => {
68
+ spies = setupOutputSpies();
69
+
70
+ const { listWebhooksCommand } = await import(
71
+ '../../../src/commands/webhooks/list'
72
+ );
73
+ await listWebhooksCommand.parseAsync([], { from: 'user' });
74
+
75
+ expect(mockList).toHaveBeenCalledTimes(1);
76
+ const args = mockList.mock.calls[0][0] as Record<string, unknown>;
77
+ expect(args.limit).toBe(10);
78
+ });
79
+
80
+ test('passes --limit to pagination options', async () => {
81
+ spies = setupOutputSpies();
82
+
83
+ const { listWebhooksCommand } = await import(
84
+ '../../../src/commands/webhooks/list'
85
+ );
86
+ await listWebhooksCommand.parseAsync(['--limit', '5'], { from: 'user' });
87
+
88
+ const args = mockList.mock.calls[0][0] as Record<string, unknown>;
89
+ expect(args.limit).toBe(5);
90
+ });
91
+
92
+ test('passes --after cursor to pagination options', async () => {
93
+ spies = setupOutputSpies();
94
+
95
+ const { listWebhooksCommand } = await import(
96
+ '../../../src/commands/webhooks/list'
97
+ );
98
+ await listWebhooksCommand.parseAsync(['--after', 'wh_cursor123'], {
99
+ from: 'user',
100
+ });
101
+
102
+ const args = mockList.mock.calls[0][0] as Record<string, unknown>;
103
+ expect(args.after).toBe('wh_cursor123');
104
+ });
105
+
106
+ test('outputs JSON list with webhook data when non-interactive', async () => {
107
+ spies = setupOutputSpies();
108
+
109
+ const { listWebhooksCommand } = await import(
110
+ '../../../src/commands/webhooks/list'
111
+ );
112
+ await listWebhooksCommand.parseAsync([], { from: 'user' });
113
+
114
+ const output = spies.logSpy.mock.calls[0][0] as string;
115
+ const parsed = JSON.parse(output);
116
+ expect(Array.isArray(parsed.data)).toBe(true);
117
+ expect(parsed.data[0].id).toBe('wh_abc123');
118
+ expect(parsed.data[0].endpoint).toBe(
119
+ 'https://app.example.com/hooks/resend',
120
+ );
121
+ expect(parsed.has_more).toBe(false);
122
+ });
123
+
124
+ test('errors with invalid_limit for out-of-range limit', async () => {
125
+ setNonInteractive();
126
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
127
+ exitSpy = mockExitThrow();
128
+
129
+ const { listWebhooksCommand } = await import(
130
+ '../../../src/commands/webhooks/list'
131
+ );
132
+ await expectExit1(() =>
133
+ listWebhooksCommand.parseAsync(['--limit', '200'], { from: 'user' }),
134
+ );
135
+
136
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
137
+ expect(output).toContain('invalid_limit');
138
+ });
139
+
140
+ test('errors with auth_error when no API key', async () => {
141
+ setNonInteractive();
142
+ delete process.env.RESEND_API_KEY;
143
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend';
144
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
145
+ exitSpy = mockExitThrow();
146
+
147
+ const { listWebhooksCommand } = await import(
148
+ '../../../src/commands/webhooks/list'
149
+ );
150
+ await expectExit1(() =>
151
+ listWebhooksCommand.parseAsync([], { from: 'user' }),
152
+ );
153
+
154
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
155
+ expect(output).toContain('auth_error');
156
+ });
157
+
158
+ test('errors with list_error when SDK returns an error', async () => {
159
+ setNonInteractive();
160
+ mockList.mockResolvedValueOnce(
161
+ mockSdkError('Server error', 'server_error'),
162
+ );
163
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
164
+ stderrSpy = spyOn(process.stderr, 'write').mockImplementation(() => true);
165
+ exitSpy = mockExitThrow();
166
+
167
+ const { listWebhooksCommand } = await import(
168
+ '../../../src/commands/webhooks/list'
169
+ );
170
+ await expectExit1(() =>
171
+ listWebhooksCommand.parseAsync([], { from: 'user' }),
172
+ );
173
+
174
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
175
+ expect(output).toContain('list_error');
176
+ });
177
+ });
@@ -0,0 +1,206 @@
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: 'webhook' as const, id: 'wh_abc123' },
21
+ error: null,
22
+ }));
23
+
24
+ mock.module('resend', () => ({
25
+ Resend: class MockResend {
26
+ constructor(public key: string) {}
27
+ webhooks = { update: mockUpdate };
28
+ },
29
+ }));
30
+
31
+ describe('webhooks 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('updates endpoint with --endpoint flag', async () => {
56
+ spies = setupOutputSpies();
57
+
58
+ const { updateWebhookCommand } = await import(
59
+ '../../../src/commands/webhooks/update'
60
+ );
61
+ await updateWebhookCommand.parseAsync(
62
+ ['wh_abc123', '--endpoint', 'https://new-app.example.com/hooks'],
63
+ { from: 'user' },
64
+ );
65
+
66
+ expect(mockUpdate).toHaveBeenCalledTimes(1);
67
+ expect(mockUpdate.mock.calls[0][0]).toBe('wh_abc123');
68
+ const payload = mockUpdate.mock.calls[0][1] as Record<string, unknown>;
69
+ expect(payload.endpoint).toBe('https://new-app.example.com/hooks');
70
+ });
71
+
72
+ test('updates events with --events flag', async () => {
73
+ spies = setupOutputSpies();
74
+
75
+ const { updateWebhookCommand } = await import(
76
+ '../../../src/commands/webhooks/update'
77
+ );
78
+ await updateWebhookCommand.parseAsync(
79
+ ['wh_abc123', '--events', 'email.sent', 'email.bounced'],
80
+ { from: 'user' },
81
+ );
82
+
83
+ const payload = mockUpdate.mock.calls[0][1] as Record<string, unknown>;
84
+ expect(payload.events).toEqual(['email.sent', 'email.bounced']);
85
+ });
86
+
87
+ test('expands "all" shorthand in --events to 17 events', async () => {
88
+ spies = setupOutputSpies();
89
+
90
+ const { updateWebhookCommand } = await import(
91
+ '../../../src/commands/webhooks/update'
92
+ );
93
+ await updateWebhookCommand.parseAsync(['wh_abc123', '--events', 'all'], {
94
+ from: 'user',
95
+ });
96
+
97
+ const payload = mockUpdate.mock.calls[0][1] as Record<string, unknown>;
98
+ expect(payload.events).toHaveLength(17);
99
+ });
100
+
101
+ test('updates status with --status flag', async () => {
102
+ spies = setupOutputSpies();
103
+
104
+ const { updateWebhookCommand } = await import(
105
+ '../../../src/commands/webhooks/update'
106
+ );
107
+ await updateWebhookCommand.parseAsync(
108
+ ['wh_abc123', '--status', 'disabled'],
109
+ { from: 'user' },
110
+ );
111
+
112
+ const payload = mockUpdate.mock.calls[0][1] as Record<string, unknown>;
113
+ expect(payload.status).toBe('disabled');
114
+ });
115
+
116
+ test('outputs JSON result when non-interactive', async () => {
117
+ spies = setupOutputSpies();
118
+
119
+ const { updateWebhookCommand } = await import(
120
+ '../../../src/commands/webhooks/update'
121
+ );
122
+ await updateWebhookCommand.parseAsync(
123
+ ['wh_abc123', '--status', 'enabled'],
124
+ { from: 'user' },
125
+ );
126
+
127
+ const output = spies.logSpy.mock.calls[0][0] as string;
128
+ const parsed = JSON.parse(output);
129
+ expect(parsed.object).toBe('webhook');
130
+ expect(parsed.id).toBe('wh_abc123');
131
+ });
132
+
133
+ test('errors with no_changes when no flags are provided', async () => {
134
+ setNonInteractive();
135
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
136
+ exitSpy = mockExitThrow();
137
+
138
+ const { updateWebhookCommand } = await import(
139
+ '../../../src/commands/webhooks/update'
140
+ );
141
+ await expectExit1(() =>
142
+ updateWebhookCommand.parseAsync(['wh_abc123'], { from: 'user' }),
143
+ );
144
+
145
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
146
+ expect(output).toContain('no_changes');
147
+ });
148
+
149
+ test('does not call SDK when no_changes error is raised', async () => {
150
+ setNonInteractive();
151
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
152
+ exitSpy = mockExitThrow();
153
+
154
+ const { updateWebhookCommand } = await import(
155
+ '../../../src/commands/webhooks/update'
156
+ );
157
+ await expectExit1(() =>
158
+ updateWebhookCommand.parseAsync(['wh_abc123'], { from: 'user' }),
159
+ );
160
+
161
+ expect(mockUpdate).not.toHaveBeenCalled();
162
+ });
163
+
164
+ test('errors with auth_error when no API key', async () => {
165
+ setNonInteractive();
166
+ delete process.env.RESEND_API_KEY;
167
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend';
168
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
169
+ exitSpy = mockExitThrow();
170
+
171
+ const { updateWebhookCommand } = await import(
172
+ '../../../src/commands/webhooks/update'
173
+ );
174
+ await expectExit1(() =>
175
+ updateWebhookCommand.parseAsync(['wh_abc123', '--status', 'enabled'], {
176
+ from: 'user',
177
+ }),
178
+ );
179
+
180
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
181
+ expect(output).toContain('auth_error');
182
+ });
183
+
184
+ test('errors with update_error when SDK returns an error', async () => {
185
+ setNonInteractive();
186
+ mockUpdate.mockResolvedValueOnce(
187
+ mockSdkError('Webhook not found', 'not_found'),
188
+ );
189
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
190
+ stderrSpy = spyOn(process.stderr, 'write').mockImplementation(() => true);
191
+ exitSpy = mockExitThrow();
192
+
193
+ const { updateWebhookCommand } = await import(
194
+ '../../../src/commands/webhooks/update'
195
+ );
196
+ await expectExit1(() =>
197
+ updateWebhookCommand.parseAsync(
198
+ ['wh_nonexistent', '--status', 'disabled'],
199
+ { from: 'user' },
200
+ ),
201
+ );
202
+
203
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
204
+ expect(output).toContain('update_error');
205
+ });
206
+ });
@@ -0,0 +1,99 @@
1
+ import {
2
+ afterEach,
3
+ beforeEach,
4
+ describe,
5
+ expect,
6
+ type spyOn,
7
+ test,
8
+ } from 'bun:test';
9
+ import { mkdirSync, rmSync, writeFileSync } from 'node:fs';
10
+ import { tmpdir } from 'node:os';
11
+ import { join } from 'node:path';
12
+ import {
13
+ captureTestEnv,
14
+ expectExit1,
15
+ mockExitThrow,
16
+ setupOutputSpies,
17
+ } from '../helpers';
18
+
19
+ describe('whoami command', () => {
20
+ const restoreEnv = captureTestEnv();
21
+ let spies: ReturnType<typeof setupOutputSpies> | undefined;
22
+ let errorSpy: ReturnType<typeof spyOn> | undefined;
23
+ let exitSpy: ReturnType<typeof spyOn> | undefined;
24
+ let tmpDir: string;
25
+
26
+ beforeEach(() => {
27
+ tmpDir = join(
28
+ tmpdir(),
29
+ `resend-test-${Date.now()}-${Math.random().toString(36).slice(2)}`,
30
+ );
31
+ mkdirSync(tmpDir, { recursive: true });
32
+ process.env.XDG_CONFIG_HOME = tmpDir;
33
+ delete process.env.RESEND_API_KEY;
34
+ delete process.env.RESEND_TEAM;
35
+ });
36
+
37
+ afterEach(() => {
38
+ restoreEnv();
39
+ spies?.restore();
40
+ spies = undefined;
41
+ errorSpy?.mockRestore();
42
+ errorSpy = undefined;
43
+ exitSpy?.mockRestore();
44
+ exitSpy = undefined;
45
+ rmSync(tmpDir, { recursive: true, force: true });
46
+ });
47
+
48
+ test('exits 1 when not authenticated (non-interactive)', async () => {
49
+ spies = setupOutputSpies();
50
+ exitSpy = mockExitThrow();
51
+
52
+ const { whoamiCommand } = await import('../../src/commands/whoami');
53
+ await expectExit1(() => whoamiCommand.parseAsync([], { from: 'user' }));
54
+
55
+ const output = spies.logSpy.mock.calls[0]?.[0] as string | undefined;
56
+ if (output) {
57
+ const parsed = JSON.parse(output);
58
+ expect(parsed.authenticated).toBe(false);
59
+ }
60
+ });
61
+
62
+ test('shows authenticated JSON when key exists in config', async () => {
63
+ const configDir = join(tmpDir, 'resend');
64
+ mkdirSync(configDir, { recursive: true });
65
+ writeFileSync(
66
+ join(configDir, 'credentials.json'),
67
+ JSON.stringify({
68
+ active_team: 'production',
69
+ teams: { production: { api_key: 're_test_key_abcd' } },
70
+ }),
71
+ );
72
+
73
+ spies = setupOutputSpies();
74
+
75
+ const { whoamiCommand } = await import('../../src/commands/whoami');
76
+ await whoamiCommand.parseAsync([], { from: 'user' });
77
+
78
+ const output = spies.logSpy.mock.calls[0][0] as string;
79
+ const parsed = JSON.parse(output);
80
+ expect(parsed.authenticated).toBe(true);
81
+ expect(parsed.team).toBe('production');
82
+ expect(parsed.api_key).toBe('re_...abcd');
83
+ expect(parsed.source).toBe('config');
84
+ });
85
+
86
+ test('shows env source when RESEND_API_KEY is set', async () => {
87
+ process.env.RESEND_API_KEY = 're_env_key_5678';
88
+
89
+ spies = setupOutputSpies();
90
+
91
+ const { whoamiCommand } = await import('../../src/commands/whoami');
92
+ await whoamiCommand.parseAsync([], { from: 'user' });
93
+
94
+ const output = spies.logSpy.mock.calls[0][0] as string;
95
+ const parsed = JSON.parse(output);
96
+ expect(parsed.authenticated).toBe(true);
97
+ expect(parsed.source).toBe('env');
98
+ });
99
+ });
@@ -0,0 +1,93 @@
1
+ import { expect, spyOn } from 'bun:test';
2
+
3
+ export class ExitError extends Error {
4
+ constructor(public code: number) {
5
+ super(`process.exit(${code})`);
6
+ }
7
+ }
8
+
9
+ export function setNonInteractive(): void {
10
+ Object.defineProperty(process.stdin, 'isTTY', {
11
+ value: undefined,
12
+ writable: true,
13
+ });
14
+ Object.defineProperty(process.stdout, 'isTTY', {
15
+ value: undefined,
16
+ writable: true,
17
+ });
18
+ }
19
+
20
+ export function mockExitThrow(): ReturnType<typeof spyOn> {
21
+ return spyOn(process, 'exit').mockImplementation((code?: number) => {
22
+ throw new ExitError(code ?? 0);
23
+ });
24
+ }
25
+
26
+ /**
27
+ * Captures current env and TTY state and returns a function that restores it.
28
+ * Call once at the top of a describe block (not inside beforeEach) so state
29
+ * is captured before any test runs.
30
+ */
31
+ export function captureTestEnv(): () => void {
32
+ const originalEnv = { ...process.env };
33
+ const originalStdinIsTTY = process.stdin.isTTY;
34
+ const originalStdoutIsTTY = process.stdout.isTTY;
35
+ return () => {
36
+ process.env = { ...originalEnv };
37
+ Object.defineProperty(process.stdin, 'isTTY', {
38
+ value: originalStdinIsTTY,
39
+ writable: true,
40
+ });
41
+ Object.defineProperty(process.stdout, 'isTTY', {
42
+ value: originalStdoutIsTTY,
43
+ writable: true,
44
+ });
45
+ };
46
+ }
47
+
48
+ /**
49
+ * Sets non-interactive mode and mocks console.log + process.stderr.write.
50
+ * Returns the spies and a restore() function. Use in happy-path tests.
51
+ */
52
+ export function setupOutputSpies() {
53
+ setNonInteractive();
54
+ const logSpy = spyOn(console, 'log').mockImplementation(() => {});
55
+ const stderrSpy = spyOn(process.stderr, 'write').mockImplementation(
56
+ () => true,
57
+ );
58
+ return {
59
+ logSpy,
60
+ stderrSpy,
61
+ restore() {
62
+ logSpy.mockRestore();
63
+ stderrSpy.mockRestore();
64
+ },
65
+ };
66
+ }
67
+
68
+ /**
69
+ * Asserts that fn throws ExitError with code 1.
70
+ * Eliminates the expect(true).toBe(false) anti-pattern in error-path tests.
71
+ */
72
+ /**
73
+ * Returns a properly-typed SDK error response without needing `as any`.
74
+ */
75
+ export function mockSdkError(message: string, name = 'error') {
76
+ return { data: null, error: { message, name }, headers: null };
77
+ }
78
+
79
+ export async function expectExit1(fn: () => Promise<unknown>): Promise<void> {
80
+ let threw = false;
81
+ try {
82
+ await fn();
83
+ } catch (err) {
84
+ threw = true;
85
+ expect(err).toBeInstanceOf(ExitError);
86
+ expect((err as ExitError).code).toBe(1);
87
+ }
88
+ if (!threw) {
89
+ throw new Error(
90
+ 'Expected command to exit with code 1 but it completed successfully',
91
+ );
92
+ }
93
+ }
@@ -0,0 +1,71 @@
1
+ import { afterEach, describe, expect, test } from 'bun:test';
2
+ import { mkdirSync, rmSync } from 'node:fs';
3
+ import { tmpdir } from 'node:os';
4
+ import { join } from 'node:path';
5
+ import { Resend } from 'resend';
6
+ import { captureTestEnv } from '../helpers';
7
+
8
+ describe('createClient', () => {
9
+ const restoreEnv = captureTestEnv();
10
+
11
+ afterEach(() => {
12
+ restoreEnv();
13
+ });
14
+
15
+ test('returns Resend instance when flag value provided', () => {
16
+ const { createClient } = require('../../src/lib/client');
17
+ const client = createClient('re_test_key');
18
+ expect(client).toBeInstanceOf(Resend);
19
+ });
20
+
21
+ test('returns Resend instance when env var set', () => {
22
+ process.env.RESEND_API_KEY = 're_env_key';
23
+ const { createClient } = require('../../src/lib/client');
24
+ const client = createClient();
25
+ expect(client).toBeInstanceOf(Resend);
26
+ });
27
+
28
+ test('throws when no API key available', () => {
29
+ delete process.env.RESEND_API_KEY;
30
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend-test';
31
+ const { createClient } = require('../../src/lib/client');
32
+ expect(() => createClient()).toThrow('No API key found');
33
+ });
34
+
35
+ test('flag value takes priority over env var', () => {
36
+ process.env.RESEND_API_KEY = 're_env_key';
37
+ const { createClient } = require('../../src/lib/client');
38
+ const client = createClient('re_flag_key');
39
+ expect(client).toBeInstanceOf(Resend);
40
+ });
41
+
42
+ test('team name is threaded through to resolveApiKey', () => {
43
+ delete process.env.RESEND_API_KEY;
44
+ const tmpDir = join(
45
+ tmpdir(),
46
+ `resend-test-${Date.now()}-${Math.random().toString(36).slice(2)}`,
47
+ );
48
+ mkdirSync(tmpDir, { recursive: true });
49
+ process.env.XDG_CONFIG_HOME = tmpDir;
50
+
51
+ const configDir = join(tmpDir, 'resend');
52
+ mkdirSync(configDir, { recursive: true });
53
+ Bun.write(
54
+ join(configDir, 'credentials.json'),
55
+ JSON.stringify({
56
+ active_team: 'default',
57
+ teams: {
58
+ default: { api_key: 're_default_key' },
59
+ staging: { api_key: 're_staging_key' },
60
+ },
61
+ }),
62
+ );
63
+
64
+ const { createClient } = require('../../src/lib/client');
65
+ // Should not throw — resolves staging team's key
66
+ const client = createClient(undefined, 'staging');
67
+ expect(client).toBeInstanceOf(Resend);
68
+
69
+ rmSync(tmpDir, { recursive: true, force: true });
70
+ });
71
+ });