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,38 @@
1
+ import { Command } from '@commander-js/extra-typings';
2
+ import { runWrite } from '../../lib/actions';
3
+ import type { GlobalOpts } from '../../lib/client';
4
+ import { buildHelpText } from '../../lib/help-text';
5
+
6
+ export const verifyDomainCommand = new Command('verify')
7
+ .description('Trigger async DNS verification for a domain')
8
+ .argument('<id>', 'Domain ID')
9
+ .addHelpText(
10
+ 'after',
11
+ buildHelpText({
12
+ context: `Verification is async — the domain enters "pending" status while DNS records are checked.
13
+ Poll the status with: resend domains get <id>`,
14
+ output: ' {"object":"domain","id":"<id>"}',
15
+ errorCodes: ['auth_error', 'verify_error'],
16
+ examples: [
17
+ 'resend domains verify 4dd369bc-aa82-4ff3-97de-514ae3000ee0',
18
+ 'resend domains verify 4dd369bc-aa82-4ff3-97de-514ae3000ee0 --json',
19
+ ],
20
+ }),
21
+ )
22
+ .action(async (id, _opts, cmd) => {
23
+ const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
24
+
25
+ await runWrite(
26
+ {
27
+ spinner: {
28
+ loading: 'Verifying domain...',
29
+ success: 'Verification started',
30
+ fail: 'Failed to verify domain',
31
+ },
32
+ sdkCall: (resend) => resend.domains.verify(id),
33
+ errorCode: 'verify_error',
34
+ successMsg: `Domain verification started. Check status with resend domains get ${id}.`,
35
+ },
36
+ globalOpts,
37
+ );
38
+ });
@@ -0,0 +1,140 @@
1
+ import { Command } from '@commander-js/extra-typings';
2
+ import type { CreateBatchOptions } from 'resend';
3
+ import type { GlobalOpts } from '../../lib/client';
4
+ import { requireClient } from '../../lib/client';
5
+ import { readFile } from '../../lib/files';
6
+ import { buildHelpText } from '../../lib/help-text';
7
+ import { outputError, outputResult } from '../../lib/output';
8
+ import { requireText } from '../../lib/prompts';
9
+ import { withSpinner } from '../../lib/spinner';
10
+ import { isInteractive } from '../../lib/tty';
11
+
12
+ export const batchCommand = new Command('batch')
13
+ .description('Send up to 100 emails in a single API request from a JSON file')
14
+ .option(
15
+ '--file <path>',
16
+ 'Path to a JSON file containing an array of email objects (required in non-interactive mode)',
17
+ )
18
+ .option(
19
+ '--idempotency-key <key>',
20
+ 'Deduplicate this batch request using this key',
21
+ )
22
+ .addHelpText(
23
+ 'after',
24
+ buildHelpText({
25
+ context:
26
+ 'Non-interactive: --file\nLimit: 100 emails per request (API hard limit — warned if exceeded)\nUnsupported per-email fields: attachments, scheduled_at\n\nFile format (--file path):\n [\n {"from":"you@domain.com","to":["user@example.com"],"subject":"Hello","text":"Hi"},\n {"from":"you@domain.com","to":["other@example.com"],"subject":"Hello","html":"<b>Hi</b>"}\n ]',
27
+ output: ' [{"id":"<email-id>"},{"id":"<email-id>"}]',
28
+ errorCodes: [
29
+ 'auth_error',
30
+ 'missing_file',
31
+ 'file_read_error',
32
+ 'invalid_json',
33
+ 'invalid_format',
34
+ 'batch_error',
35
+ ],
36
+ examples: [
37
+ 'resend emails batch --file ./emails.json',
38
+ 'resend emails batch --file ./emails.json --json',
39
+ 'resend emails batch --file ./emails.json --idempotency-key my-batch-2026-02-18',
40
+ 'RESEND_API_KEY=re_123 resend emails batch --file ./emails.json --json',
41
+ ],
42
+ }),
43
+ )
44
+ .action(async (opts, cmd) => {
45
+ const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
46
+
47
+ const resend = requireClient(globalOpts);
48
+
49
+ const filePath = await requireText(
50
+ opts.file,
51
+ { message: 'Path to JSON file', placeholder: './emails.json' },
52
+ {
53
+ message:
54
+ 'Missing --file flag. Provide a JSON file with an array of email objects.',
55
+ code: 'missing_file',
56
+ },
57
+ globalOpts,
58
+ );
59
+
60
+ const raw = readFile(filePath, globalOpts);
61
+
62
+ let parsed: unknown;
63
+ try {
64
+ parsed = JSON.parse(raw);
65
+ } catch {
66
+ outputError(
67
+ { message: 'File content is not valid JSON.', code: 'invalid_json' },
68
+ { json: globalOpts.json },
69
+ );
70
+ }
71
+
72
+ if (!Array.isArray(parsed)) {
73
+ outputError(
74
+ {
75
+ message: 'File content must be a JSON array of email objects.',
76
+ code: 'invalid_format',
77
+ },
78
+ { json: globalOpts.json },
79
+ );
80
+ }
81
+
82
+ const emails = parsed as unknown[];
83
+
84
+ if (emails.length > 100) {
85
+ console.warn(
86
+ `Warning: ${emails.length} emails exceeds the 100-email limit. The API may reject this request.`,
87
+ );
88
+ }
89
+
90
+ for (let i = 0; i < emails.length; i++) {
91
+ const email = emails[i] as Record<string, unknown>;
92
+ if ('attachments' in email) {
93
+ outputError(
94
+ {
95
+ message: `Email at index ${i} contains "attachments", which is not supported in batch sends.`,
96
+ code: 'batch_error',
97
+ },
98
+ { json: globalOpts.json },
99
+ );
100
+ }
101
+ if ('scheduled_at' in email) {
102
+ outputError(
103
+ {
104
+ message: `Email at index ${i} contains "scheduled_at", which is not supported in batch sends.`,
105
+ code: 'batch_error',
106
+ },
107
+ { json: globalOpts.json },
108
+ );
109
+ }
110
+ }
111
+
112
+ const batchData = await withSpinner(
113
+ {
114
+ loading: 'Sending batch...',
115
+ success: 'Batch sent',
116
+ fail: 'Failed to send batch',
117
+ },
118
+ () =>
119
+ resend.batch.send(
120
+ emails as CreateBatchOptions,
121
+ opts.idempotencyKey
122
+ ? { idempotencyKey: opts.idempotencyKey }
123
+ : undefined,
124
+ ),
125
+ 'batch_error',
126
+ globalOpts,
127
+ );
128
+
129
+ const emailIds = batchData.data;
130
+ if (!globalOpts.json && isInteractive()) {
131
+ console.log(
132
+ `Sent ${emailIds.length} email${emailIds.length === 1 ? '' : 's'}`,
133
+ );
134
+ for (const email of emailIds) {
135
+ console.log(` ${email.id}`);
136
+ }
137
+ } else {
138
+ outputResult(emailIds, { json: globalOpts.json });
139
+ }
140
+ });
@@ -0,0 +1,24 @@
1
+ import { Command } from '@commander-js/extra-typings';
2
+ import { buildHelpText } from '../../lib/help-text';
3
+ import { batchCommand } from './batch';
4
+ import { receivingCommand } from './receiving/index';
5
+ import { sendCommand } from './send';
6
+
7
+ export const emailsCommand = new Command('emails')
8
+ .description('Send and manage emails')
9
+ .addHelpText(
10
+ 'after',
11
+ buildHelpText({
12
+ examples: [
13
+ 'resend emails send --from you@domain.com --to user@example.com --subject "Hello" --text "Hi"',
14
+ 'resend emails batch --file ./emails.json',
15
+ 'resend emails receiving list',
16
+ 'resend emails receiving get <email-id>',
17
+ 'resend emails receiving attachments <email-id>',
18
+ 'resend emails receiving attachment <email-id> <attachment-id>',
19
+ ],
20
+ }),
21
+ )
22
+ .addCommand(sendCommand)
23
+ .addCommand(batchCommand)
24
+ .addCommand(receivingCommand);
@@ -0,0 +1,55 @@
1
+ import { Command } from '@commander-js/extra-typings';
2
+ import type { GlobalOpts } from '../../../lib/client';
3
+ import { requireClient } from '../../../lib/client';
4
+ import { buildHelpText } from '../../../lib/help-text';
5
+ import { outputResult } from '../../../lib/output';
6
+ import { withSpinner } from '../../../lib/spinner';
7
+ import { isInteractive } from '../../../lib/tty';
8
+
9
+ export const getAttachmentCommand = new Command('attachment')
10
+ .description('Retrieve a single attachment from a received (inbound) email')
11
+ .argument('<emailId>', 'Received email UUID')
12
+ .argument('<attachmentId>', 'Attachment UUID')
13
+ .addHelpText(
14
+ 'after',
15
+ buildHelpText({
16
+ context:
17
+ 'The download_url is a signed URL that expires in ~1 hour. Download the file directly:\n resend emails receiving attachment <emailId> <attachmentId> --json | jq -r .download_url | xargs curl -O',
18
+ output:
19
+ ' {"object":"attachment","id":"<uuid>","filename":"invoice.pdf","size":51200,"content_type":"application/pdf","content_disposition":"attachment","content_id":null,"download_url":"<signed-url>","expires_at":"<iso-date>"}',
20
+ errorCodes: ['auth_error', 'fetch_error'],
21
+ examples: [
22
+ 'resend emails receiving attachment <email-id> <attachment-id>',
23
+ 'resend emails receiving attachment <email-id> <attachment-id> --json',
24
+ ],
25
+ }),
26
+ )
27
+ .action(async (emailId, attachmentId, _opts, cmd) => {
28
+ const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
29
+ const resend = requireClient(globalOpts);
30
+
31
+ const data = await withSpinner(
32
+ {
33
+ loading: 'Fetching attachment...',
34
+ success: 'Attachment fetched',
35
+ fail: 'Failed to fetch attachment',
36
+ },
37
+ () =>
38
+ resend.emails.receiving.attachments.get({ emailId, id: attachmentId }),
39
+ 'fetch_error',
40
+ globalOpts,
41
+ );
42
+
43
+ if (!globalOpts.json && isInteractive()) {
44
+ const d = data;
45
+ console.log(`\n${d.filename ?? '(unnamed)'}`);
46
+ console.log(`ID: ${d.id}`);
47
+ console.log(`Content-Type: ${d.content_type}`);
48
+ console.log(`Size: ${d.size} bytes`);
49
+ console.log(`Disposition: ${d.content_disposition}`);
50
+ console.log(`Download URL: ${d.download_url}`);
51
+ console.log(`Expires: ${d.expires_at}`);
52
+ } else {
53
+ outputResult(data, { json: globalOpts.json });
54
+ }
55
+ });
@@ -0,0 +1,68 @@
1
+ import { Command } from '@commander-js/extra-typings';
2
+ import { runList } from '../../../lib/actions';
3
+ import type { GlobalOpts } from '../../../lib/client';
4
+ import { buildHelpText } from '../../../lib/help-text';
5
+ import {
6
+ buildPaginationOpts,
7
+ parseLimitOpt,
8
+ printPaginationHint,
9
+ } from '../../../lib/pagination';
10
+ import { renderAttachmentsTable } from './utils';
11
+
12
+ export const listAttachmentsCommand = new Command('attachments')
13
+ .description('List attachments on a received (inbound) email')
14
+ .argument('<emailId>', 'Received email UUID')
15
+ .option(
16
+ '--limit <n>',
17
+ 'Maximum number of attachments to return (1-100)',
18
+ '10',
19
+ )
20
+ .option(
21
+ '--after <cursor>',
22
+ 'Return attachments after this cursor (next page)',
23
+ )
24
+ .option(
25
+ '--before <cursor>',
26
+ 'Return attachments before this cursor (previous page)',
27
+ )
28
+ .addHelpText(
29
+ 'after',
30
+ buildHelpText({
31
+ context:
32
+ 'Each attachment has a download_url (signed, expires ~1 hour).\nUse the attachment sub-command to retrieve a single attachment with its download URL:\n resend emails receiving attachment <emailId> <attachmentId>\n\ncontent_disposition: "inline" means the attachment is embedded in the HTML body (e.g. an image).\ncontent_disposition: "attachment" means it is a standalone file download.',
33
+ output:
34
+ ' {"object":"list","has_more":false,"data":[{"id":"<uuid>","filename":"invoice.pdf","size":51200,"content_type":"application/pdf","content_disposition":"attachment","content_id":null,"download_url":"<url>","expires_at":"<iso-date>"}]}',
35
+ errorCodes: ['auth_error', 'invalid_limit', 'list_error'],
36
+ examples: [
37
+ 'resend emails receiving attachments <email-id>',
38
+ 'resend emails receiving attachments <email-id> --json',
39
+ 'resend emails receiving attachments <email-id> --limit 25 --json',
40
+ ],
41
+ }),
42
+ )
43
+ .action(async (emailId, opts, cmd) => {
44
+ const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
45
+
46
+ const limit = parseLimitOpt(opts.limit, globalOpts);
47
+ const paginationOpts = buildPaginationOpts(limit, opts.after, opts.before);
48
+
49
+ await runList(
50
+ {
51
+ spinner: {
52
+ loading: 'Fetching attachments...',
53
+ success: 'Attachments fetched',
54
+ fail: 'Failed to list attachments',
55
+ },
56
+ sdkCall: (resend) =>
57
+ resend.emails.receiving.attachments.list({
58
+ emailId,
59
+ ...paginationOpts,
60
+ }),
61
+ onInteractive: (list) => {
62
+ console.log(renderAttachmentsTable(list.data));
63
+ printPaginationHint(list);
64
+ },
65
+ },
66
+ globalOpts,
67
+ );
68
+ });
@@ -0,0 +1,58 @@
1
+ import { Command } from '@commander-js/extra-typings';
2
+ import { runGet } from '../../../lib/actions';
3
+ import type { GlobalOpts } from '../../../lib/client';
4
+ import { buildHelpText } from '../../../lib/help-text';
5
+
6
+ export const getReceivingCommand = new Command('get')
7
+ .description(
8
+ 'Retrieve a single received (inbound) email with full details including HTML, text, and headers',
9
+ )
10
+ .argument('<id>', 'Received email UUID')
11
+ .addHelpText(
12
+ 'after',
13
+ buildHelpText({
14
+ context:
15
+ 'The raw.download_url field is a signed URL (expires ~1 hour) containing the full RFC 2822\nMIME message. Pipe it to curl to save the original email:\n resend emails receiving get <id> --json | jq -r .raw.download_url | xargs curl > email.eml\n\nAttachments are listed in the attachments array. Use the attachments sub-command to get\ndownload URLs:\n resend emails receiving attachments <id>',
16
+ output:
17
+ ' {"object":"email","id":"<uuid>","to":["inbox@yourdomain.com"],"from":"sender@external.com","subject":"Hello","html":"<p>Hello!</p>","text":"Hello!","headers":{"x-mailer":"..."},"message_id":"<str>","bcc":[],"cc":[],"reply_to":[],"raw":{"download_url":"<url>","expires_at":"<iso-date>"},"attachments":[]}',
18
+ errorCodes: ['auth_error', 'fetch_error'],
19
+ examples: [
20
+ 'resend emails receiving get <email-id>',
21
+ 'resend emails receiving get <email-id> --json',
22
+ ],
23
+ }),
24
+ )
25
+ .action(async (id, _opts, cmd) => {
26
+ const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
27
+ await runGet(
28
+ {
29
+ spinner: {
30
+ loading: 'Fetching received email...',
31
+ success: 'Received email fetched',
32
+ fail: 'Failed to fetch received email',
33
+ },
34
+ sdkCall: (resend) => resend.emails.receiving.get(id),
35
+ onInteractive: (data) => {
36
+ console.log(`\nFrom: ${data.from}`);
37
+ console.log(`To: ${data.to.join(', ')}`);
38
+ console.log(`Subject: ${data.subject}`);
39
+ console.log(`Date: ${data.created_at}`);
40
+ if (data.attachments.length > 0) {
41
+ console.log(`Files: ${data.attachments.length} attachment(s)`);
42
+ }
43
+ if (data.text) {
44
+ const snippet =
45
+ data.text.length > 200
46
+ ? `${data.text.slice(0, 197)}...`
47
+ : data.text;
48
+ console.log(`\n${snippet}`);
49
+ } else if (data.html) {
50
+ console.log(
51
+ '\n(HTML body only — use --json to view or pipe to a browser)',
52
+ );
53
+ }
54
+ },
55
+ },
56
+ globalOpts,
57
+ );
58
+ });
@@ -0,0 +1,28 @@
1
+ import { Command } from '@commander-js/extra-typings';
2
+ import { buildHelpText } from '../../../lib/help-text';
3
+ import { getAttachmentCommand } from './attachment';
4
+ import { listAttachmentsCommand } from './attachments';
5
+ import { getReceivingCommand } from './get';
6
+ import { listReceivingCommand } from './list';
7
+
8
+ export const receivingCommand = new Command('receiving')
9
+ .description(
10
+ 'Manage received (inbound) emails — requires domain receiving to be enabled',
11
+ )
12
+ .addHelpText(
13
+ 'after',
14
+ buildHelpText({
15
+ context:
16
+ 'Receiving must be enabled on the domain first:\n resend domains update <id> --receiving enabled',
17
+ examples: [
18
+ 'resend emails receiving list',
19
+ 'resend emails receiving get <email-id>',
20
+ 'resend emails receiving attachments <email-id>',
21
+ 'resend emails receiving attachment <email-id> <attachment-id>',
22
+ ],
23
+ }),
24
+ )
25
+ .addCommand(listReceivingCommand, { isDefault: true })
26
+ .addCommand(getReceivingCommand)
27
+ .addCommand(listAttachmentsCommand)
28
+ .addCommand(getAttachmentCommand);
@@ -0,0 +1,59 @@
1
+ import { Command } from '@commander-js/extra-typings';
2
+ import { runList } from '../../../lib/actions';
3
+ import type { GlobalOpts } from '../../../lib/client';
4
+ import { buildHelpText } from '../../../lib/help-text';
5
+ import {
6
+ buildPaginationOpts,
7
+ parseLimitOpt,
8
+ printPaginationHint,
9
+ } from '../../../lib/pagination';
10
+ import { renderReceivingEmailsTable } from './utils';
11
+
12
+ export const listReceivingCommand = new Command('list')
13
+ .alias('ls')
14
+ .description(
15
+ 'List received (inbound) emails for domains with receiving enabled',
16
+ )
17
+ .option('--limit <n>', 'Maximum number of emails to return (1-100)', '10')
18
+ .option('--after <cursor>', 'Return emails after this cursor (next page)')
19
+ .option(
20
+ '--before <cursor>',
21
+ 'Return emails before this cursor (previous page)',
22
+ )
23
+ .addHelpText(
24
+ 'after',
25
+ buildHelpText({
26
+ context:
27
+ 'Receiving must be enabled on the domain first:\n resend domains update <id> --receiving enabled\n\nPagination: use --after or --before with a received email ID as the cursor.\nOnly one of --after or --before may be used at a time.\nThe response includes has_more: true when additional pages exist.',
28
+ output:
29
+ ' {"object":"list","has_more":false,"data":[{"id":"<uuid>","to":["inbox@yourdomain.com"],"from":"sender@external.com","subject":"Hello","created_at":"<iso-date>","message_id":"<str>","bcc":null,"cc":null,"reply_to":null,"attachments":[]}]}',
30
+ errorCodes: ['auth_error', 'invalid_limit', 'list_error'],
31
+ examples: [
32
+ 'resend emails receiving list',
33
+ 'resend emails receiving list --limit 25 --json',
34
+ 'resend emails receiving list --after <email-id> --json',
35
+ ],
36
+ }),
37
+ )
38
+ .action(async (opts, cmd) => {
39
+ const globalOpts = cmd.optsWithGlobals() as GlobalOpts;
40
+
41
+ const limit = parseLimitOpt(opts.limit, globalOpts);
42
+ const paginationOpts = buildPaginationOpts(limit, opts.after, opts.before);
43
+
44
+ await runList(
45
+ {
46
+ spinner: {
47
+ loading: 'Fetching received emails...',
48
+ success: 'Received emails fetched',
49
+ fail: 'Failed to list received emails',
50
+ },
51
+ sdkCall: (resend) => resend.emails.receiving.list(paginationOpts),
52
+ onInteractive: (list) => {
53
+ console.log(renderReceivingEmailsTable(list.data));
54
+ printPaginationHint(list);
55
+ },
56
+ },
57
+ globalOpts,
58
+ );
59
+ });
@@ -0,0 +1,38 @@
1
+ import type {
2
+ ListAttachmentsResponseSuccess,
3
+ ListReceivingEmail,
4
+ } from 'resend';
5
+ import { renderTable } from '../../../lib/table';
6
+
7
+ export function renderReceivingEmailsTable(
8
+ emails: ListReceivingEmail[],
9
+ ): string {
10
+ const rows = emails.map((e) => {
11
+ const to = e.to.join(', ');
12
+ const toStr = to.length > 40 ? `${to.slice(0, 37)}...` : to;
13
+ const subject =
14
+ e.subject.length > 50 ? `${e.subject.slice(0, 47)}...` : e.subject;
15
+ return [e.from, toStr, subject, e.created_at, e.id];
16
+ });
17
+ return renderTable(
18
+ ['From', 'To', 'Subject', 'Created At', 'ID'],
19
+ rows,
20
+ '(no received emails)',
21
+ );
22
+ }
23
+
24
+ export function renderAttachmentsTable(
25
+ attachments: ListAttachmentsResponseSuccess['data'],
26
+ ): string {
27
+ const rows = attachments.map((a) => [
28
+ a.filename ?? '(unnamed)',
29
+ a.content_type,
30
+ String(a.size),
31
+ a.id,
32
+ ]);
33
+ return renderTable(
34
+ ['Filename', 'Content-Type', 'Size (bytes)', 'ID'],
35
+ rows,
36
+ '(no attachments)',
37
+ );
38
+ }