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,205 @@
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' 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 = { update: mockUpdate };
31
+ },
32
+ }));
33
+
34
+ describe('contacts 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 contact by ID with --unsubscribed', async () => {
59
+ spies = setupOutputSpies();
60
+
61
+ const { updateContactCommand } = await import(
62
+ '../../../src/commands/contacts/update'
63
+ );
64
+ await updateContactCommand.parseAsync(
65
+ ['a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6', '--unsubscribed'],
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('a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6');
72
+ expect(args.unsubscribed).toBe(true);
73
+ });
74
+
75
+ test('updates contact by email with --no-unsubscribed', async () => {
76
+ spies = setupOutputSpies();
77
+
78
+ const { updateContactCommand } = await import(
79
+ '../../../src/commands/contacts/update'
80
+ );
81
+ await updateContactCommand.parseAsync(
82
+ ['jane@example.com', '--no-unsubscribed'],
83
+ { from: 'user' },
84
+ );
85
+
86
+ const args = mockUpdate.mock.calls[0][0] as Record<string, unknown>;
87
+ expect(args.email).toBe('jane@example.com');
88
+ expect(args.unsubscribed).toBe(false);
89
+ });
90
+
91
+ test('parses --properties JSON and passes to SDK', async () => {
92
+ spies = setupOutputSpies();
93
+
94
+ const { updateContactCommand } = await import(
95
+ '../../../src/commands/contacts/update'
96
+ );
97
+ await updateContactCommand.parseAsync(
98
+ [
99
+ 'a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6',
100
+ '--properties',
101
+ '{"plan":"pro"}',
102
+ ],
103
+ { from: 'user' },
104
+ );
105
+
106
+ const args = mockUpdate.mock.calls[0][0] as Record<string, unknown>;
107
+ expect(args.properties).toEqual({ plan: 'pro' });
108
+ });
109
+
110
+ test('does not include unsubscribed when neither flag is passed', async () => {
111
+ spies = setupOutputSpies();
112
+
113
+ const { updateContactCommand } = await import(
114
+ '../../../src/commands/contacts/update'
115
+ );
116
+ await updateContactCommand.parseAsync(
117
+ ['a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6'],
118
+ { from: 'user' },
119
+ );
120
+
121
+ const args = mockUpdate.mock.calls[0][0] as Record<string, unknown>;
122
+ expect(args.unsubscribed).toBeUndefined();
123
+ });
124
+
125
+ test('outputs JSON result when non-interactive', async () => {
126
+ spies = setupOutputSpies();
127
+
128
+ const { updateContactCommand } = await import(
129
+ '../../../src/commands/contacts/update'
130
+ );
131
+ await updateContactCommand.parseAsync(
132
+ ['a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6', '--unsubscribed'],
133
+ { from: 'user' },
134
+ );
135
+
136
+ const output = spies.logSpy.mock.calls[0][0] as string;
137
+ const parsed = JSON.parse(output);
138
+ expect(parsed.id).toBe('a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6');
139
+ expect(parsed.object).toBe('contact');
140
+ });
141
+
142
+ test('errors with invalid_properties when --properties is not valid JSON', async () => {
143
+ setNonInteractive();
144
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
145
+ exitSpy = mockExitThrow();
146
+
147
+ const { updateContactCommand } = await import(
148
+ '../../../src/commands/contacts/update'
149
+ );
150
+ await expectExit1(() =>
151
+ updateContactCommand.parseAsync(
152
+ ['a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6', '--properties', 'bad-json'],
153
+ { from: 'user' },
154
+ ),
155
+ );
156
+
157
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
158
+ expect(output).toContain('invalid_properties');
159
+ });
160
+
161
+ test('errors with auth_error when no API key', async () => {
162
+ setNonInteractive();
163
+ delete process.env.RESEND_API_KEY;
164
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend';
165
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
166
+ exitSpy = mockExitThrow();
167
+
168
+ const { updateContactCommand } = await import(
169
+ '../../../src/commands/contacts/update'
170
+ );
171
+ await expectExit1(() =>
172
+ updateContactCommand.parseAsync(
173
+ ['a1b2c3d4-5e6f-7a8b-9c0d-e1f2a3b4c5d6', '--unsubscribed'],
174
+ {
175
+ from: 'user',
176
+ },
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('Contact 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 { updateContactCommand } = await import(
194
+ '../../../src/commands/contacts/update'
195
+ );
196
+ await expectExit1(() =>
197
+ updateContactCommand.parseAsync(['nonexistent_id', '--unsubscribed'], {
198
+ from: 'user',
199
+ }),
200
+ );
201
+
202
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
203
+ expect(output).toContain('update_error');
204
+ });
205
+ });
@@ -0,0 +1,165 @@
1
+ import {
2
+ afterEach,
3
+ beforeEach,
4
+ describe,
5
+ expect,
6
+ mock,
7
+ spyOn,
8
+ test,
9
+ } from 'bun:test';
10
+ import { Command } from '@commander-js/extra-typings';
11
+ import {
12
+ captureTestEnv,
13
+ expectExit1,
14
+ mockExitThrow,
15
+ setupOutputSpies,
16
+ } from '../helpers';
17
+
18
+ // Mock resend SDK for doctor
19
+ mock.module('resend', () => ({
20
+ Resend: class MockResend {
21
+ constructor(public key: string) {}
22
+ domains = {
23
+ list: mock(async () => ({
24
+ data: { data: [{ name: 'example.com', status: 'verified' }] },
25
+ error: null,
26
+ })),
27
+ };
28
+ },
29
+ }));
30
+
31
+ /**
32
+ * Wraps doctorCommand in a parent program with global --json option,
33
+ * matching the real CLI structure in src/cli.ts.
34
+ */
35
+ async function createDoctorProgram() {
36
+ const { doctorCommand } = await import('../../src/commands/doctor');
37
+ const program = new Command()
38
+ .name('resend')
39
+ .option('--json', 'Force JSON output')
40
+ .addCommand(doctorCommand);
41
+ return program;
42
+ }
43
+
44
+ describe('doctor command', () => {
45
+ const restoreEnv = captureTestEnv();
46
+ let spies: ReturnType<typeof setupOutputSpies> | undefined;
47
+ let exitSpy: ReturnType<typeof spyOn> | undefined;
48
+
49
+ beforeEach(() => {
50
+ process.env.RESEND_API_KEY = 're_test_key_for_doctor';
51
+ });
52
+
53
+ afterEach(() => {
54
+ restoreEnv();
55
+ spies?.restore();
56
+ spies = undefined;
57
+ exitSpy?.mockRestore();
58
+ exitSpy = undefined;
59
+ });
60
+
61
+ test('outputs JSON with --json flag', async () => {
62
+ spies = setupOutputSpies();
63
+
64
+ const program = await createDoctorProgram();
65
+ await program.parseAsync(['doctor', '--json'], { from: 'user' });
66
+
67
+ expect(spies.logSpy).toHaveBeenCalled();
68
+ const output = spies.logSpy.mock.calls[0][0] as string;
69
+ const parsed = JSON.parse(output);
70
+
71
+ expect(parsed).toHaveProperty('ok');
72
+ expect(parsed).toHaveProperty('checks');
73
+ expect(Array.isArray(parsed.checks)).toBe(true);
74
+ });
75
+
76
+ test('JSON output has correct check structure', async () => {
77
+ spies = setupOutputSpies();
78
+
79
+ const program = await createDoctorProgram();
80
+ await program.parseAsync(['doctor', '--json'], { from: 'user' });
81
+
82
+ const output = spies.logSpy.mock.calls[0][0] as string;
83
+ const parsed = JSON.parse(output);
84
+
85
+ for (const check of parsed.checks) {
86
+ expect(check).toHaveProperty('name');
87
+ expect(check).toHaveProperty('status');
88
+ expect(check).toHaveProperty('message');
89
+ expect(['pass', 'warn', 'fail']).toContain(check.status);
90
+ }
91
+ });
92
+
93
+ test('API key check passes when key is set', async () => {
94
+ spies = setupOutputSpies();
95
+
96
+ const program = await createDoctorProgram();
97
+ await program.parseAsync(['doctor', '--json'], { from: 'user' });
98
+
99
+ const output = spies.logSpy.mock.calls[0][0] as string;
100
+ const parsed = JSON.parse(output);
101
+ const keyCheck = parsed.checks.find(
102
+ (c: Record<string, unknown>) => c.name === 'API Key',
103
+ );
104
+
105
+ expect(keyCheck).toBeDefined();
106
+ expect(keyCheck.status).toBe('pass');
107
+ expect(keyCheck.message).toContain('re_');
108
+ expect(keyCheck.message).toContain('env');
109
+ });
110
+
111
+ test('API key check fails when no key is set', async () => {
112
+ delete process.env.RESEND_API_KEY;
113
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend';
114
+
115
+ spies = setupOutputSpies();
116
+ exitSpy = spyOn(process, 'exit').mockImplementation(
117
+ () => undefined as never,
118
+ );
119
+
120
+ const program = await createDoctorProgram();
121
+ await program.parseAsync(['doctor', '--json'], { from: 'user' });
122
+
123
+ const output = spies.logSpy.mock.calls[0][0] as string;
124
+ const parsed = JSON.parse(output);
125
+
126
+ expect(parsed.ok).toBe(false);
127
+ const keyCheck = parsed.checks.find(
128
+ (c: Record<string, unknown>) => c.name === 'API Key',
129
+ );
130
+ expect(keyCheck.status).toBe('fail');
131
+ });
132
+
133
+ test('masks API key in output', async () => {
134
+ process.env.RESEND_API_KEY = 're_abcdefghijklmnop';
135
+ spies = setupOutputSpies();
136
+
137
+ const program = await createDoctorProgram();
138
+ await program.parseAsync(['doctor', '--json'], { from: 'user' });
139
+
140
+ const output = spies.logSpy.mock.calls[0][0] as string;
141
+ const parsed = JSON.parse(output);
142
+ const keyCheck = parsed.checks.find(
143
+ (c: Record<string, unknown>) => c.name === 'API Key',
144
+ );
145
+
146
+ // Should not contain full key
147
+ expect(keyCheck.message).not.toContain('re_abcdefghijklmnop');
148
+ // Should contain masked version (re_...mnop)
149
+ expect(keyCheck.message).toContain('re_');
150
+ expect(keyCheck.message).toContain('...');
151
+ });
152
+
153
+ test('exits with code 1 when checks fail', async () => {
154
+ delete process.env.RESEND_API_KEY;
155
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend';
156
+
157
+ spies = setupOutputSpies();
158
+ exitSpy = mockExitThrow();
159
+
160
+ const program = await createDoctorProgram();
161
+ await expectExit1(() =>
162
+ program.parseAsync(['doctor', '--json'], { from: 'user' }),
163
+ );
164
+ });
165
+ });
@@ -0,0 +1,192 @@
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
+ id: 'test-domain-id',
22
+ name: 'example.com',
23
+ status: 'not_started',
24
+ created_at: '2026-01-01T00:00:00.000Z',
25
+ region: 'us-east-1',
26
+ records: [
27
+ {
28
+ record: 'SPF',
29
+ type: 'MX',
30
+ name: 'send',
31
+ ttl: 'Auto',
32
+ status: 'not_started',
33
+ value: 'feedback-smtp.us-east-1.amazonses.com',
34
+ priority: 10,
35
+ },
36
+ ],
37
+ capabilities: { sending: 'enabled', receiving: 'disabled' },
38
+ },
39
+ error: null,
40
+ }));
41
+
42
+ mock.module('resend', () => ({
43
+ Resend: class MockResend {
44
+ constructor(public key: string) {}
45
+ domains = { create: mockCreate };
46
+ },
47
+ }));
48
+
49
+ describe('domains create command', () => {
50
+ const restoreEnv = captureTestEnv();
51
+ let spies: ReturnType<typeof setupOutputSpies> | undefined;
52
+ let errorSpy: ReturnType<typeof spyOn> | undefined;
53
+ let stderrSpy: ReturnType<typeof spyOn> | undefined;
54
+ let exitSpy: ReturnType<typeof spyOn> | undefined;
55
+
56
+ beforeEach(() => {
57
+ process.env.RESEND_API_KEY = 're_test_key';
58
+ mockCreate.mockClear();
59
+ });
60
+
61
+ afterEach(() => {
62
+ restoreEnv();
63
+ spies?.restore();
64
+ errorSpy?.mockRestore();
65
+ stderrSpy?.mockRestore();
66
+ exitSpy?.mockRestore();
67
+ spies = undefined;
68
+ errorSpy = undefined;
69
+ stderrSpy = undefined;
70
+ exitSpy = undefined;
71
+ });
72
+
73
+ test('creates domain with --name flag', async () => {
74
+ spies = setupOutputSpies();
75
+
76
+ const { createDomainCommand } = await import(
77
+ '../../../src/commands/domains/create'
78
+ );
79
+ await createDomainCommand.parseAsync(['--name', 'example.com'], {
80
+ from: 'user',
81
+ });
82
+
83
+ expect(mockCreate).toHaveBeenCalledTimes(1);
84
+ const args = mockCreate.mock.calls[0][0] as Record<string, unknown>;
85
+ expect(args.name).toBe('example.com');
86
+ });
87
+
88
+ test('passes region and tls flags to SDK', async () => {
89
+ spies = setupOutputSpies();
90
+
91
+ const { createDomainCommand } = await import(
92
+ '../../../src/commands/domains/create'
93
+ );
94
+ await createDomainCommand.parseAsync(
95
+ ['--name', 'example.com', '--region', 'eu-west-1', '--tls', 'enforced'],
96
+ { from: 'user' },
97
+ );
98
+
99
+ const args = mockCreate.mock.calls[0][0] as Record<string, unknown>;
100
+ expect(args.region).toBe('eu-west-1');
101
+ expect(args.tls).toBe('enforced');
102
+ });
103
+
104
+ test('passes receiving capability when --receiving flag is set', async () => {
105
+ spies = setupOutputSpies();
106
+
107
+ const { createDomainCommand } = await import(
108
+ '../../../src/commands/domains/create'
109
+ );
110
+ await createDomainCommand.parseAsync(
111
+ ['--name', 'example.com', '--receiving'],
112
+ { from: 'user' },
113
+ );
114
+
115
+ const args = mockCreate.mock.calls[0][0] as Record<string, unknown>;
116
+ expect(args.capabilities?.receiving).toBe('enabled');
117
+ });
118
+
119
+ test('outputs JSON result when non-interactive', async () => {
120
+ spies = setupOutputSpies();
121
+
122
+ const { createDomainCommand } = await import(
123
+ '../../../src/commands/domains/create'
124
+ );
125
+ await createDomainCommand.parseAsync(['--name', 'example.com'], {
126
+ from: 'user',
127
+ });
128
+
129
+ const output = spies.logSpy.mock.calls[0][0] as string;
130
+ const parsed = JSON.parse(output);
131
+ expect(parsed.id).toBe('test-domain-id');
132
+ expect(parsed.name).toBe('example.com');
133
+ });
134
+
135
+ test('errors with missing_name when --name absent in non-interactive mode', async () => {
136
+ setNonInteractive();
137
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
138
+ exitSpy = mockExitThrow();
139
+
140
+ const { createDomainCommand } = await import(
141
+ '../../../src/commands/domains/create'
142
+ );
143
+ await expectExit1(() =>
144
+ createDomainCommand.parseAsync([], { from: 'user' }),
145
+ );
146
+
147
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
148
+ expect(output).toContain('missing_name');
149
+ });
150
+
151
+ test('errors with auth_error when no API key', async () => {
152
+ setNonInteractive();
153
+ delete process.env.RESEND_API_KEY;
154
+ process.env.XDG_CONFIG_HOME = '/tmp/nonexistent-resend';
155
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
156
+ exitSpy = mockExitThrow();
157
+
158
+ const { createDomainCommand } = await import(
159
+ '../../../src/commands/domains/create'
160
+ );
161
+ await expectExit1(() =>
162
+ createDomainCommand.parseAsync(['--name', 'example.com'], {
163
+ from: 'user',
164
+ }),
165
+ );
166
+
167
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
168
+ expect(output).toContain('auth_error');
169
+ });
170
+
171
+ test('errors with create_error when SDK returns an error', async () => {
172
+ setNonInteractive();
173
+ mockCreate.mockResolvedValueOnce(
174
+ mockSdkError('Domain already exists', 'validation_error'),
175
+ );
176
+ errorSpy = spyOn(console, 'error').mockImplementation(() => {});
177
+ stderrSpy = spyOn(process.stderr, 'write').mockImplementation(() => true);
178
+ exitSpy = mockExitThrow();
179
+
180
+ const { createDomainCommand } = await import(
181
+ '../../../src/commands/domains/create'
182
+ );
183
+ await expectExit1(() =>
184
+ createDomainCommand.parseAsync(['--name', 'example.com'], {
185
+ from: 'user',
186
+ }),
187
+ );
188
+
189
+ const output = errorSpy.mock.calls.map((c) => c[0]).join(' ');
190
+ expect(output).toContain('create_error');
191
+ });
192
+ });