agent-messenger 1.0.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 (394) hide show
  1. package/.claude/commands/release.md +92 -0
  2. package/.claude-plugin/README.md +144 -0
  3. package/.claude-plugin/marketplace.json +37 -0
  4. package/.claude-plugin/plugin.json +17 -0
  5. package/.github/workflows/ci.yml +30 -0
  6. package/CLAUDE.md +106 -0
  7. package/CONTRIBUTING.md +131 -0
  8. package/README.md +140 -0
  9. package/biome.json +34 -0
  10. package/bun.lock +252 -0
  11. package/dist/cli.d.ts +5 -0
  12. package/dist/cli.d.ts.map +1 -0
  13. package/dist/cli.js +21 -0
  14. package/dist/cli.js.map +1 -0
  15. package/dist/commands/auth.d.ts +3 -0
  16. package/dist/commands/auth.d.ts.map +1 -0
  17. package/dist/commands/auth.js +140 -0
  18. package/dist/commands/auth.js.map +1 -0
  19. package/dist/commands/channel.d.ts +3 -0
  20. package/dist/commands/channel.d.ts.map +1 -0
  21. package/dist/commands/channel.js +118 -0
  22. package/dist/commands/channel.js.map +1 -0
  23. package/dist/commands/file.d.ts +3 -0
  24. package/dist/commands/file.d.ts.map +1 -0
  25. package/dist/commands/file.js +113 -0
  26. package/dist/commands/file.js.map +1 -0
  27. package/dist/commands/index.d.ts +9 -0
  28. package/dist/commands/index.d.ts.map +1 -0
  29. package/dist/commands/index.js +9 -0
  30. package/dist/commands/index.js.map +1 -0
  31. package/dist/commands/message.d.ts +3 -0
  32. package/dist/commands/message.d.ts.map +1 -0
  33. package/dist/commands/message.js +214 -0
  34. package/dist/commands/message.js.map +1 -0
  35. package/dist/commands/reaction.d.ts +3 -0
  36. package/dist/commands/reaction.d.ts.map +1 -0
  37. package/dist/commands/reaction.js +100 -0
  38. package/dist/commands/reaction.js.map +1 -0
  39. package/dist/commands/snapshot.d.ts +3 -0
  40. package/dist/commands/snapshot.d.ts.map +1 -0
  41. package/dist/commands/snapshot.js +88 -0
  42. package/dist/commands/snapshot.js.map +1 -0
  43. package/dist/commands/user.d.ts +3 -0
  44. package/dist/commands/user.d.ts.map +1 -0
  45. package/dist/commands/user.js +96 -0
  46. package/dist/commands/user.js.map +1 -0
  47. package/dist/commands/workspace.d.ts +3 -0
  48. package/dist/commands/workspace.d.ts.map +1 -0
  49. package/dist/commands/workspace.js +89 -0
  50. package/dist/commands/workspace.js.map +1 -0
  51. package/dist/lib/credential-manager.d.ts +13 -0
  52. package/dist/lib/credential-manager.d.ts.map +1 -0
  53. package/dist/lib/credential-manager.js +58 -0
  54. package/dist/lib/credential-manager.js.map +1 -0
  55. package/dist/lib/index.d.ts +3 -0
  56. package/dist/lib/index.d.ts.map +1 -0
  57. package/dist/lib/index.js +3 -0
  58. package/dist/lib/index.js.map +1 -0
  59. package/dist/lib/ref-manager.d.ts +26 -0
  60. package/dist/lib/ref-manager.d.ts.map +1 -0
  61. package/dist/lib/ref-manager.js +92 -0
  62. package/dist/lib/ref-manager.js.map +1 -0
  63. package/dist/lib/slack-client.d.ts +37 -0
  64. package/dist/lib/slack-client.d.ts.map +1 -0
  65. package/dist/lib/slack-client.js +379 -0
  66. package/dist/lib/slack-client.js.map +1 -0
  67. package/dist/lib/token-extractor.d.ts +28 -0
  68. package/dist/lib/token-extractor.d.ts.map +1 -0
  69. package/dist/lib/token-extractor.js +401 -0
  70. package/dist/lib/token-extractor.js.map +1 -0
  71. package/dist/package.json +37 -0
  72. package/dist/src/cli.d.ts +5 -0
  73. package/dist/src/cli.d.ts.map +1 -0
  74. package/dist/src/cli.js +22 -0
  75. package/dist/src/cli.js.map +1 -0
  76. package/dist/src/platforms/discord/cli.d.ts +5 -0
  77. package/dist/src/platforms/discord/cli.d.ts.map +1 -0
  78. package/dist/src/platforms/discord/cli.js +22 -0
  79. package/dist/src/platforms/discord/cli.js.map +1 -0
  80. package/dist/src/platforms/discord/client.d.ts +34 -0
  81. package/dist/src/platforms/discord/client.d.ts.map +1 -0
  82. package/dist/src/platforms/discord/client.js +187 -0
  83. package/dist/src/platforms/discord/client.js.map +1 -0
  84. package/dist/src/platforms/discord/client.test.d.ts +2 -0
  85. package/dist/src/platforms/discord/client.test.d.ts.map +1 -0
  86. package/dist/src/platforms/discord/client.test.js +367 -0
  87. package/dist/src/platforms/discord/client.test.js.map +1 -0
  88. package/dist/src/platforms/discord/commands/auth.d.ts +13 -0
  89. package/dist/src/platforms/discord/commands/auth.d.ts.map +1 -0
  90. package/dist/src/platforms/discord/commands/auth.js +155 -0
  91. package/dist/src/platforms/discord/commands/auth.js.map +1 -0
  92. package/dist/src/platforms/discord/commands/auth.test.d.ts +2 -0
  93. package/dist/src/platforms/discord/commands/auth.test.d.ts.map +1 -0
  94. package/dist/src/platforms/discord/commands/auth.test.js +65 -0
  95. package/dist/src/platforms/discord/commands/auth.test.js.map +1 -0
  96. package/dist/src/platforms/discord/commands/channel.d.ts +13 -0
  97. package/dist/src/platforms/discord/commands/channel.d.ts.map +1 -0
  98. package/dist/src/platforms/discord/commands/channel.js +99 -0
  99. package/dist/src/platforms/discord/commands/channel.js.map +1 -0
  100. package/dist/src/platforms/discord/commands/channel.test.d.ts +2 -0
  101. package/dist/src/platforms/discord/commands/channel.test.d.ts.map +1 -0
  102. package/dist/src/platforms/discord/commands/channel.test.js +136 -0
  103. package/dist/src/platforms/discord/commands/channel.test.js.map +1 -0
  104. package/dist/src/platforms/discord/commands/file.d.ts +13 -0
  105. package/dist/src/platforms/discord/commands/file.d.ts.map +1 -0
  106. package/dist/src/platforms/discord/commands/file.js +99 -0
  107. package/dist/src/platforms/discord/commands/file.js.map +1 -0
  108. package/dist/src/platforms/discord/commands/file.test.d.ts +2 -0
  109. package/dist/src/platforms/discord/commands/file.test.d.ts.map +1 -0
  110. package/dist/src/platforms/discord/commands/file.test.js +83 -0
  111. package/dist/src/platforms/discord/commands/file.test.js.map +1 -0
  112. package/dist/src/platforms/discord/commands/guild.d.ts +15 -0
  113. package/dist/src/platforms/discord/commands/guild.d.ts.map +1 -0
  114. package/dist/src/platforms/discord/commands/guild.js +102 -0
  115. package/dist/src/platforms/discord/commands/guild.js.map +1 -0
  116. package/dist/src/platforms/discord/commands/guild.test.d.ts +2 -0
  117. package/dist/src/platforms/discord/commands/guild.test.d.ts.map +1 -0
  118. package/dist/src/platforms/discord/commands/guild.test.js +100 -0
  119. package/dist/src/platforms/discord/commands/guild.test.js.map +1 -0
  120. package/dist/src/platforms/discord/commands/index.d.ts +9 -0
  121. package/dist/src/platforms/discord/commands/index.d.ts.map +1 -0
  122. package/dist/src/platforms/discord/commands/index.js +9 -0
  123. package/dist/src/platforms/discord/commands/index.js.map +1 -0
  124. package/dist/src/platforms/discord/commands/message.d.ts +17 -0
  125. package/dist/src/platforms/discord/commands/message.d.ts.map +1 -0
  126. package/dist/src/platforms/discord/commands/message.js +131 -0
  127. package/dist/src/platforms/discord/commands/message.js.map +1 -0
  128. package/dist/src/platforms/discord/commands/message.test.d.ts +2 -0
  129. package/dist/src/platforms/discord/commands/message.test.d.ts.map +1 -0
  130. package/dist/src/platforms/discord/commands/message.test.js +91 -0
  131. package/dist/src/platforms/discord/commands/message.test.js.map +1 -0
  132. package/dist/src/platforms/discord/commands/reaction.d.ts +12 -0
  133. package/dist/src/platforms/discord/commands/reaction.d.ts.map +1 -0
  134. package/dist/src/platforms/discord/commands/reaction.js +99 -0
  135. package/dist/src/platforms/discord/commands/reaction.js.map +1 -0
  136. package/dist/src/platforms/discord/commands/reaction.test.d.ts +2 -0
  137. package/dist/src/platforms/discord/commands/reaction.test.d.ts.map +1 -0
  138. package/dist/src/platforms/discord/commands/reaction.test.js +115 -0
  139. package/dist/src/platforms/discord/commands/reaction.test.js.map +1 -0
  140. package/dist/src/platforms/discord/commands/snapshot.d.ts +9 -0
  141. package/dist/src/platforms/discord/commands/snapshot.d.ts.map +1 -0
  142. package/dist/src/platforms/discord/commands/snapshot.js +80 -0
  143. package/dist/src/platforms/discord/commands/snapshot.js.map +1 -0
  144. package/dist/src/platforms/discord/commands/snapshot.test.d.ts +2 -0
  145. package/dist/src/platforms/discord/commands/snapshot.test.d.ts.map +1 -0
  146. package/dist/src/platforms/discord/commands/snapshot.test.js +25 -0
  147. package/dist/src/platforms/discord/commands/snapshot.test.js.map +1 -0
  148. package/dist/src/platforms/discord/commands/user.d.ts +3 -0
  149. package/dist/src/platforms/discord/commands/user.d.ts.map +1 -0
  150. package/dist/src/platforms/discord/commands/user.js +94 -0
  151. package/dist/src/platforms/discord/commands/user.js.map +1 -0
  152. package/dist/src/platforms/discord/commands/user.test.d.ts +2 -0
  153. package/dist/src/platforms/discord/commands/user.test.d.ts.map +1 -0
  154. package/dist/src/platforms/discord/commands/user.test.js +103 -0
  155. package/dist/src/platforms/discord/commands/user.test.js.map +1 -0
  156. package/dist/src/platforms/discord/credential-manager.d.ts +33 -0
  157. package/dist/src/platforms/discord/credential-manager.d.ts.map +1 -0
  158. package/dist/src/platforms/discord/credential-manager.js +73 -0
  159. package/dist/src/platforms/discord/credential-manager.js.map +1 -0
  160. package/dist/src/platforms/discord/credential-manager.test.d.ts +2 -0
  161. package/dist/src/platforms/discord/credential-manager.test.d.ts.map +1 -0
  162. package/dist/src/platforms/discord/credential-manager.test.js +136 -0
  163. package/dist/src/platforms/discord/credential-manager.test.js.map +1 -0
  164. package/dist/src/platforms/discord/token-extractor.d.ts +55 -0
  165. package/dist/src/platforms/discord/token-extractor.d.ts.map +1 -0
  166. package/dist/src/platforms/discord/token-extractor.js +462 -0
  167. package/dist/src/platforms/discord/token-extractor.js.map +1 -0
  168. package/dist/src/platforms/discord/token-extractor.test.d.ts +2 -0
  169. package/dist/src/platforms/discord/token-extractor.test.d.ts.map +1 -0
  170. package/dist/src/platforms/discord/token-extractor.test.js +789 -0
  171. package/dist/src/platforms/discord/token-extractor.test.js.map +1 -0
  172. package/dist/src/platforms/discord/types.d.ts +251 -0
  173. package/dist/src/platforms/discord/types.d.ts.map +1 -0
  174. package/dist/src/platforms/discord/types.js +74 -0
  175. package/dist/src/platforms/discord/types.js.map +1 -0
  176. package/dist/src/platforms/discord/types.test.d.ts +2 -0
  177. package/dist/src/platforms/discord/types.test.d.ts.map +1 -0
  178. package/dist/src/platforms/discord/types.test.js +211 -0
  179. package/dist/src/platforms/discord/types.test.js.map +1 -0
  180. package/dist/src/platforms/slack/cli.d.ts +5 -0
  181. package/dist/src/platforms/slack/cli.d.ts.map +1 -0
  182. package/dist/src/platforms/slack/cli.js +22 -0
  183. package/dist/src/platforms/slack/cli.js.map +1 -0
  184. package/dist/src/platforms/slack/client.d.ts +47 -0
  185. package/dist/src/platforms/slack/client.d.ts.map +1 -0
  186. package/dist/src/platforms/slack/client.js +412 -0
  187. package/dist/src/platforms/slack/client.js.map +1 -0
  188. package/dist/src/platforms/slack/commands/auth.d.ts +3 -0
  189. package/dist/src/platforms/slack/commands/auth.d.ts.map +1 -0
  190. package/dist/src/platforms/slack/commands/auth.js +156 -0
  191. package/dist/src/platforms/slack/commands/auth.js.map +1 -0
  192. package/dist/src/platforms/slack/commands/channel.d.ts +3 -0
  193. package/dist/src/platforms/slack/commands/channel.d.ts.map +1 -0
  194. package/dist/src/platforms/slack/commands/channel.js +118 -0
  195. package/dist/src/platforms/slack/commands/channel.js.map +1 -0
  196. package/dist/src/platforms/slack/commands/file.d.ts +3 -0
  197. package/dist/src/platforms/slack/commands/file.d.ts.map +1 -0
  198. package/dist/src/platforms/slack/commands/file.js +113 -0
  199. package/dist/src/platforms/slack/commands/file.js.map +1 -0
  200. package/dist/src/platforms/slack/commands/index.d.ts +9 -0
  201. package/dist/src/platforms/slack/commands/index.d.ts.map +1 -0
  202. package/dist/src/platforms/slack/commands/index.js +9 -0
  203. package/dist/src/platforms/slack/commands/index.js.map +1 -0
  204. package/dist/src/platforms/slack/commands/message.d.ts +3 -0
  205. package/dist/src/platforms/slack/commands/message.d.ts.map +1 -0
  206. package/dist/src/platforms/slack/commands/message.js +263 -0
  207. package/dist/src/platforms/slack/commands/message.js.map +1 -0
  208. package/dist/src/platforms/slack/commands/reaction.d.ts +3 -0
  209. package/dist/src/platforms/slack/commands/reaction.d.ts.map +1 -0
  210. package/dist/src/platforms/slack/commands/reaction.js +100 -0
  211. package/dist/src/platforms/slack/commands/reaction.js.map +1 -0
  212. package/dist/src/platforms/slack/commands/snapshot.d.ts +3 -0
  213. package/dist/src/platforms/slack/commands/snapshot.d.ts.map +1 -0
  214. package/dist/src/platforms/slack/commands/snapshot.js +87 -0
  215. package/dist/src/platforms/slack/commands/snapshot.js.map +1 -0
  216. package/dist/src/platforms/slack/commands/user.d.ts +3 -0
  217. package/dist/src/platforms/slack/commands/user.d.ts.map +1 -0
  218. package/dist/src/platforms/slack/commands/user.js +96 -0
  219. package/dist/src/platforms/slack/commands/user.js.map +1 -0
  220. package/dist/src/platforms/slack/commands/workspace.d.ts +3 -0
  221. package/dist/src/platforms/slack/commands/workspace.d.ts.map +1 -0
  222. package/dist/src/platforms/slack/commands/workspace.js +89 -0
  223. package/dist/src/platforms/slack/commands/workspace.js.map +1 -0
  224. package/dist/src/platforms/slack/credential-manager.d.ts +13 -0
  225. package/dist/src/platforms/slack/credential-manager.d.ts.map +1 -0
  226. package/dist/src/platforms/slack/credential-manager.js +58 -0
  227. package/dist/src/platforms/slack/credential-manager.js.map +1 -0
  228. package/dist/src/platforms/slack/index.d.ts +3 -0
  229. package/dist/src/platforms/slack/index.d.ts.map +1 -0
  230. package/dist/src/platforms/slack/index.js +3 -0
  231. package/dist/src/platforms/slack/index.js.map +1 -0
  232. package/dist/src/platforms/slack/token-extractor.d.ts +28 -0
  233. package/dist/src/platforms/slack/token-extractor.d.ts.map +1 -0
  234. package/dist/src/platforms/slack/token-extractor.js +401 -0
  235. package/dist/src/platforms/slack/token-extractor.js.map +1 -0
  236. package/dist/src/platforms/slack/types.d.ts +369 -0
  237. package/dist/src/platforms/slack/types.d.ts.map +1 -0
  238. package/dist/src/platforms/slack/types.js +92 -0
  239. package/dist/src/platforms/slack/types.js.map +1 -0
  240. package/dist/src/shared/utils/concurrency.d.ts +2 -0
  241. package/dist/src/shared/utils/concurrency.d.ts.map +1 -0
  242. package/dist/src/shared/utils/concurrency.js +14 -0
  243. package/dist/src/shared/utils/concurrency.js.map +1 -0
  244. package/dist/src/shared/utils/concurrency.test.d.ts +2 -0
  245. package/dist/src/shared/utils/concurrency.test.d.ts.map +1 -0
  246. package/dist/src/shared/utils/concurrency.test.js +39 -0
  247. package/dist/src/shared/utils/concurrency.test.js.map +1 -0
  248. package/dist/src/shared/utils/error-handler.d.ts +2 -0
  249. package/dist/src/shared/utils/error-handler.d.ts.map +1 -0
  250. package/dist/src/shared/utils/error-handler.js +5 -0
  251. package/dist/src/shared/utils/error-handler.js.map +1 -0
  252. package/dist/src/shared/utils/output.d.ts +2 -0
  253. package/dist/src/shared/utils/output.d.ts.map +1 -0
  254. package/dist/src/shared/utils/output.js +4 -0
  255. package/dist/src/shared/utils/output.js.map +1 -0
  256. package/dist/tests/cli.test.d.ts +2 -0
  257. package/dist/tests/cli.test.d.ts.map +1 -0
  258. package/dist/tests/cli.test.js +83 -0
  259. package/dist/tests/cli.test.js.map +1 -0
  260. package/dist/tests/commands/.test-slack-data/Local Storage/leveldb/CURRENT +1 -0
  261. package/dist/tests/commands/.test-slack-data/Local Storage/leveldb/LOCK +0 -0
  262. package/dist/tests/commands/.test-slack-data/Local Storage/leveldb/LOG +3 -0
  263. package/dist/tests/commands/.test-slack-data/Local Storage/leveldb/LOG.old +1 -0
  264. package/dist/tests/commands/.test-slack-data/Local Storage/leveldb/MANIFEST-000004 +0 -0
  265. package/dist/tests/commands/auth.test.d.ts +2 -0
  266. package/dist/tests/commands/auth.test.d.ts.map +1 -0
  267. package/dist/tests/commands/auth.test.js +304 -0
  268. package/dist/tests/commands/auth.test.js.map +1 -0
  269. package/dist/tests/commands/channel.test.d.ts +2 -0
  270. package/dist/tests/commands/channel.test.d.ts.map +1 -0
  271. package/dist/tests/commands/channel.test.js +166 -0
  272. package/dist/tests/commands/channel.test.js.map +1 -0
  273. package/dist/tests/commands/file.test.d.ts +2 -0
  274. package/dist/tests/commands/file.test.d.ts.map +1 -0
  275. package/dist/tests/commands/file.test.js +175 -0
  276. package/dist/tests/commands/file.test.js.map +1 -0
  277. package/dist/tests/commands/message.test.d.ts +2 -0
  278. package/dist/tests/commands/message.test.d.ts.map +1 -0
  279. package/dist/tests/commands/message.test.js +293 -0
  280. package/dist/tests/commands/message.test.js.map +1 -0
  281. package/dist/tests/commands/reaction.test.d.ts +2 -0
  282. package/dist/tests/commands/reaction.test.d.ts.map +1 -0
  283. package/dist/tests/commands/reaction.test.js +84 -0
  284. package/dist/tests/commands/reaction.test.js.map +1 -0
  285. package/dist/tests/commands/snapshot.test.d.ts +2 -0
  286. package/dist/tests/commands/snapshot.test.d.ts.map +1 -0
  287. package/dist/tests/commands/snapshot.test.js +280 -0
  288. package/dist/tests/commands/snapshot.test.js.map +1 -0
  289. package/dist/tests/commands/user.test.d.ts +2 -0
  290. package/dist/tests/commands/user.test.d.ts.map +1 -0
  291. package/dist/tests/commands/user.test.js +117 -0
  292. package/dist/tests/commands/user.test.js.map +1 -0
  293. package/dist/tests/commands/workspace.test.d.ts +2 -0
  294. package/dist/tests/commands/workspace.test.d.ts.map +1 -0
  295. package/dist/tests/commands/workspace.test.js +453 -0
  296. package/dist/tests/commands/workspace.test.js.map +1 -0
  297. package/dist/tests/credential-manager.test.d.ts +2 -0
  298. package/dist/tests/credential-manager.test.d.ts.map +1 -0
  299. package/dist/tests/credential-manager.test.js +199 -0
  300. package/dist/tests/credential-manager.test.js.map +1 -0
  301. package/dist/tests/slack-client.test.d.ts +2 -0
  302. package/dist/tests/slack-client.test.d.ts.map +1 -0
  303. package/dist/tests/slack-client.test.js +741 -0
  304. package/dist/tests/slack-client.test.js.map +1 -0
  305. package/dist/tests/types.test.d.ts +2 -0
  306. package/dist/tests/types.test.d.ts.map +1 -0
  307. package/dist/tests/types.test.js +215 -0
  308. package/dist/tests/types.test.js.map +1 -0
  309. package/dist/types/index.d.ts +369 -0
  310. package/dist/types/index.d.ts.map +1 -0
  311. package/dist/types/index.js +92 -0
  312. package/dist/types/index.js.map +1 -0
  313. package/dist/utils/error-handler.d.ts +2 -0
  314. package/dist/utils/error-handler.d.ts.map +1 -0
  315. package/dist/utils/error-handler.js +5 -0
  316. package/dist/utils/error-handler.js.map +1 -0
  317. package/dist/utils/output.d.ts +2 -0
  318. package/dist/utils/output.d.ts.map +1 -0
  319. package/dist/utils/output.js +4 -0
  320. package/dist/utils/output.js.map +1 -0
  321. package/docs/discord.md +182 -0
  322. package/docs/slack.md +160 -0
  323. package/package.json +37 -0
  324. package/skills/agent-discord/SKILL.md +273 -0
  325. package/skills/agent-discord/references/authentication.md +294 -0
  326. package/skills/agent-discord/references/common-patterns.md +455 -0
  327. package/skills/agent-discord/templates/guild-summary.sh +167 -0
  328. package/skills/agent-discord/templates/monitor-channel.sh +180 -0
  329. package/skills/agent-discord/templates/post-message.sh +173 -0
  330. package/skills/agent-slack/SKILL.md +268 -0
  331. package/skills/agent-slack/references/authentication.md +332 -0
  332. package/skills/agent-slack/references/common-patterns.md +527 -0
  333. package/skills/agent-slack/templates/monitor-channel.sh +186 -0
  334. package/skills/agent-slack/templates/post-message.sh +130 -0
  335. package/skills/agent-slack/templates/workspace-summary.sh +149 -0
  336. package/src/cli.ts +29 -0
  337. package/src/platforms/discord/cli.ts +36 -0
  338. package/src/platforms/discord/client.test.ts +456 -0
  339. package/src/platforms/discord/client.ts +281 -0
  340. package/src/platforms/discord/commands/auth.test.ts +72 -0
  341. package/src/platforms/discord/commands/auth.ts +206 -0
  342. package/src/platforms/discord/commands/channel.test.ts +153 -0
  343. package/src/platforms/discord/commands/channel.ts +127 -0
  344. package/src/platforms/discord/commands/file.test.ts +98 -0
  345. package/src/platforms/discord/commands/file.ts +134 -0
  346. package/src/platforms/discord/commands/guild.test.ts +117 -0
  347. package/src/platforms/discord/commands/guild.ts +129 -0
  348. package/src/platforms/discord/commands/index.ts +8 -0
  349. package/src/platforms/discord/commands/message.test.ts +107 -0
  350. package/src/platforms/discord/commands/message.ts +182 -0
  351. package/src/platforms/discord/commands/reaction.test.ts +123 -0
  352. package/src/platforms/discord/commands/reaction.ts +156 -0
  353. package/src/platforms/discord/commands/snapshot.test.ts +29 -0
  354. package/src/platforms/discord/commands/snapshot.ts +104 -0
  355. package/src/platforms/discord/commands/user.test.ts +115 -0
  356. package/src/platforms/discord/commands/user.ts +124 -0
  357. package/src/platforms/discord/credential-manager.test.ts +173 -0
  358. package/src/platforms/discord/credential-manager.ts +95 -0
  359. package/src/platforms/discord/token-extractor.test.ts +918 -0
  360. package/src/platforms/discord/token-extractor.ts +549 -0
  361. package/src/platforms/discord/types.test.ts +245 -0
  362. package/src/platforms/discord/types.ts +158 -0
  363. package/src/platforms/slack/cli.ts +36 -0
  364. package/src/platforms/slack/client.ts +466 -0
  365. package/src/platforms/slack/commands/auth.ts +211 -0
  366. package/src/platforms/slack/commands/channel.ts +158 -0
  367. package/src/platforms/slack/commands/file.ts +153 -0
  368. package/src/platforms/slack/commands/index.ts +8 -0
  369. package/src/platforms/slack/commands/message.ts +369 -0
  370. package/src/platforms/slack/commands/reaction.ts +166 -0
  371. package/src/platforms/slack/commands/snapshot.ts +114 -0
  372. package/src/platforms/slack/commands/user.ts +110 -0
  373. package/src/platforms/slack/commands/workspace.ts +111 -0
  374. package/src/platforms/slack/credential-manager.ts +74 -0
  375. package/src/platforms/slack/index.ts +2 -0
  376. package/src/platforms/slack/token-extractor.ts +496 -0
  377. package/src/platforms/slack/types.ts +193 -0
  378. package/src/shared/utils/concurrency.test.ts +53 -0
  379. package/src/shared/utils/concurrency.ts +20 -0
  380. package/src/shared/utils/error-handler.ts +4 -0
  381. package/src/shared/utils/output.ts +3 -0
  382. package/tests/cli.test.ts +94 -0
  383. package/tests/commands/auth.test.ts +383 -0
  384. package/tests/commands/channel.test.ts +185 -0
  385. package/tests/commands/file.test.ts +204 -0
  386. package/tests/commands/message.test.ts +344 -0
  387. package/tests/commands/reaction.test.ts +101 -0
  388. package/tests/commands/snapshot.test.ts +308 -0
  389. package/tests/commands/user.test.ts +138 -0
  390. package/tests/commands/workspace.test.ts +528 -0
  391. package/tests/credential-manager.test.ts +241 -0
  392. package/tests/slack-client.test.ts +916 -0
  393. package/tests/types.test.ts +241 -0
  394. package/tsconfig.json +36 -0
@@ -0,0 +1,456 @@
1
+ import { afterEach, beforeEach, describe, expect, test } from 'bun:test'
2
+ import { DiscordClient, DiscordError } from './client'
3
+
4
+ describe('DiscordClient', () => {
5
+ const originalFetch = globalThis.fetch
6
+ let fetchCalls: Array<{ url: string; options?: RequestInit }> = []
7
+ let fetchResponses: Response[] = []
8
+ let fetchIndex = 0
9
+
10
+ beforeEach(() => {
11
+ fetchCalls = []
12
+ fetchResponses = []
13
+ fetchIndex = 0
14
+ ;(globalThis as any).fetch = async (
15
+ url: string | URL | Request,
16
+ options?: RequestInit
17
+ ): Promise<Response> => {
18
+ fetchCalls.push({ url: url.toString(), options })
19
+ const response = fetchResponses[fetchIndex]
20
+ fetchIndex++
21
+ if (!response) {
22
+ throw new Error('No mock response configured')
23
+ }
24
+ return response
25
+ }
26
+ })
27
+
28
+ afterEach(() => {
29
+ globalThis.fetch = originalFetch
30
+ })
31
+
32
+ afterEach(() => {
33
+ globalThis.fetch = originalFetch
34
+ })
35
+
36
+ const mockResponse = (body: unknown, status = 200, headers: Record<string, string> = {}) => {
37
+ const defaultHeaders: Record<string, string> = {
38
+ 'Content-Type': 'application/json',
39
+ 'X-RateLimit-Remaining': '10',
40
+ 'X-RateLimit-Reset': String(Date.now() / 1000 + 60),
41
+ 'X-RateLimit-Bucket': 'test-bucket',
42
+ ...headers,
43
+ }
44
+ fetchResponses.push(
45
+ new Response(body === null ? null : JSON.stringify(body), {
46
+ status,
47
+ headers: defaultHeaders,
48
+ })
49
+ )
50
+ }
51
+
52
+ describe('constructor', () => {
53
+ test('requires token', () => {
54
+ expect(() => new DiscordClient('')).toThrow(DiscordError)
55
+ expect(() => new DiscordClient('')).toThrow('Token is required')
56
+ })
57
+
58
+ test('accepts valid token', () => {
59
+ const client = new DiscordClient('test-token')
60
+ expect(client).toBeInstanceOf(DiscordClient)
61
+ })
62
+ })
63
+
64
+ describe('testAuth', () => {
65
+ test('returns current user info', async () => {
66
+ mockResponse({
67
+ id: '123456789',
68
+ username: 'testuser',
69
+ global_name: 'Test User',
70
+ avatar: 'abc123',
71
+ })
72
+
73
+ const client = new DiscordClient('test-token')
74
+ const user = await client.testAuth()
75
+
76
+ expect(user.id).toBe('123456789')
77
+ expect(user.username).toBe('testuser')
78
+ expect(fetchCalls.length).toBe(1)
79
+ expect(fetchCalls[0].url).toBe('https://discord.com/api/v10/users/@me')
80
+ expect(fetchCalls[0].options?.headers).toMatchObject({
81
+ Authorization: 'test-token',
82
+ })
83
+ })
84
+
85
+ test('throws DiscordError on API error', async () => {
86
+ mockResponse({ message: 'Unauthorized', code: 401 }, 401)
87
+
88
+ const client = new DiscordClient('bad-token')
89
+ await expect(client.testAuth()).rejects.toThrow(DiscordError)
90
+ })
91
+ })
92
+
93
+ describe('listGuilds', () => {
94
+ test('returns list of guilds', async () => {
95
+ mockResponse([
96
+ { id: '111', name: 'Guild One' },
97
+ { id: '222', name: 'Guild Two' },
98
+ ])
99
+
100
+ const client = new DiscordClient('test-token')
101
+ const guilds = await client.listGuilds()
102
+
103
+ expect(guilds).toHaveLength(2)
104
+ expect(guilds[0].name).toBe('Guild One')
105
+ expect(fetchCalls[0].url).toBe('https://discord.com/api/v10/users/@me/guilds')
106
+ })
107
+ })
108
+
109
+ describe('getGuild', () => {
110
+ test('returns guild info', async () => {
111
+ mockResponse({ id: '111', name: 'Test Guild' })
112
+
113
+ const client = new DiscordClient('test-token')
114
+ const guild = await client.getGuild('111')
115
+
116
+ expect(guild.id).toBe('111')
117
+ expect(guild.name).toBe('Test Guild')
118
+ expect(fetchCalls[0].url).toBe('https://discord.com/api/v10/guilds/111')
119
+ })
120
+ })
121
+
122
+ describe('listChannels', () => {
123
+ test('returns list of channels for guild', async () => {
124
+ mockResponse([
125
+ { id: 'ch1', guild_id: '111', name: 'general', type: 0 },
126
+ { id: 'ch2', guild_id: '111', name: 'random', type: 0 },
127
+ ])
128
+
129
+ const client = new DiscordClient('test-token')
130
+ const channels = await client.listChannels('111')
131
+
132
+ expect(channels).toHaveLength(2)
133
+ expect(channels[0].name).toBe('general')
134
+ expect(fetchCalls[0].url).toBe('https://discord.com/api/v10/guilds/111/channels')
135
+ })
136
+ })
137
+
138
+ describe('getChannel', () => {
139
+ test('returns channel info', async () => {
140
+ mockResponse({ id: 'ch1', guild_id: '111', name: 'general', type: 0 })
141
+
142
+ const client = new DiscordClient('test-token')
143
+ const channel = await client.getChannel('ch1')
144
+
145
+ expect(channel.id).toBe('ch1')
146
+ expect(channel.name).toBe('general')
147
+ expect(fetchCalls[0].url).toBe('https://discord.com/api/v10/channels/ch1')
148
+ })
149
+ })
150
+
151
+ describe('sendMessage', () => {
152
+ test('sends message to channel', async () => {
153
+ mockResponse({
154
+ id: 'msg1',
155
+ channel_id: 'ch1',
156
+ author: { id: '123', username: 'bot' },
157
+ content: 'Hello world',
158
+ timestamp: '2024-01-01T00:00:00.000Z',
159
+ })
160
+
161
+ const client = new DiscordClient('test-token')
162
+ const message = await client.sendMessage('ch1', 'Hello world')
163
+
164
+ expect(message.content).toBe('Hello world')
165
+ expect(fetchCalls[0].url).toBe('https://discord.com/api/v10/channels/ch1/messages')
166
+ expect(fetchCalls[0].options?.method).toBe('POST')
167
+ expect(fetchCalls[0].options?.body).toBe(JSON.stringify({ content: 'Hello world' }))
168
+ })
169
+ })
170
+
171
+ describe('getMessages', () => {
172
+ test('returns messages from channel', async () => {
173
+ mockResponse([
174
+ {
175
+ id: 'msg1',
176
+ channel_id: 'ch1',
177
+ author: { id: '123', username: 'user1' },
178
+ content: 'Message 1',
179
+ timestamp: '2024-01-01T00:00:00.000Z',
180
+ },
181
+ ])
182
+
183
+ const client = new DiscordClient('test-token')
184
+ const messages = await client.getMessages('ch1', 50)
185
+
186
+ expect(messages).toHaveLength(1)
187
+ expect(messages[0].content).toBe('Message 1')
188
+ expect(fetchCalls[0].url).toBe('https://discord.com/api/v10/channels/ch1/messages?limit=50')
189
+ })
190
+
191
+ test('uses default limit of 50', async () => {
192
+ mockResponse([])
193
+
194
+ const client = new DiscordClient('test-token')
195
+ await client.getMessages('ch1')
196
+
197
+ expect(fetchCalls[0].url).toBe('https://discord.com/api/v10/channels/ch1/messages?limit=50')
198
+ })
199
+ })
200
+
201
+ describe('getMessage', () => {
202
+ test('returns single message', async () => {
203
+ mockResponse({
204
+ id: 'msg1',
205
+ channel_id: 'ch1',
206
+ author: { id: '123', username: 'user1' },
207
+ content: 'Message 1',
208
+ timestamp: '2024-01-01T00:00:00.000Z',
209
+ })
210
+
211
+ const client = new DiscordClient('test-token')
212
+ const message = await client.getMessage('ch1', 'msg1')
213
+
214
+ expect(message.id).toBe('msg1')
215
+ expect(fetchCalls[0].url).toBe('https://discord.com/api/v10/channels/ch1/messages/msg1')
216
+ })
217
+ })
218
+
219
+ describe('deleteMessage', () => {
220
+ test('deletes message', async () => {
221
+ mockResponse(null, 204)
222
+
223
+ const client = new DiscordClient('test-token')
224
+ await client.deleteMessage('ch1', 'msg1')
225
+
226
+ expect(fetchCalls[0].url).toBe('https://discord.com/api/v10/channels/ch1/messages/msg1')
227
+ expect(fetchCalls[0].options?.method).toBe('DELETE')
228
+ })
229
+ })
230
+
231
+ describe('addReaction', () => {
232
+ test('adds reaction to message', async () => {
233
+ mockResponse(null, 204)
234
+
235
+ const client = new DiscordClient('test-token')
236
+ await client.addReaction('ch1', 'msg1', '👍')
237
+
238
+ expect(fetchCalls[0].url).toBe(
239
+ 'https://discord.com/api/v10/channels/ch1/messages/msg1/reactions/%F0%9F%91%8D/@me'
240
+ )
241
+ expect(fetchCalls[0].options?.method).toBe('PUT')
242
+ })
243
+ })
244
+
245
+ describe('removeReaction', () => {
246
+ test('removes reaction from message', async () => {
247
+ mockResponse(null, 204)
248
+
249
+ const client = new DiscordClient('test-token')
250
+ await client.removeReaction('ch1', 'msg1', '👍')
251
+
252
+ expect(fetchCalls[0].url).toBe(
253
+ 'https://discord.com/api/v10/channels/ch1/messages/msg1/reactions/%F0%9F%91%8D/@me'
254
+ )
255
+ expect(fetchCalls[0].options?.method).toBe('DELETE')
256
+ })
257
+ })
258
+
259
+ describe('listUsers', () => {
260
+ test('returns list of guild members', async () => {
261
+ mockResponse([
262
+ { user: { id: 'u1', username: 'user1' } },
263
+ { user: { id: 'u2', username: 'user2' } },
264
+ ])
265
+
266
+ const client = new DiscordClient('test-token')
267
+ const users = await client.listUsers('111')
268
+
269
+ expect(users).toHaveLength(2)
270
+ expect(users[0].username).toBe('user1')
271
+ expect(fetchCalls[0].url).toBe('https://discord.com/api/v10/guilds/111/members?limit=1000')
272
+ })
273
+ })
274
+
275
+ describe('getUser', () => {
276
+ test('returns user info', async () => {
277
+ mockResponse({ id: 'u1', username: 'testuser' })
278
+
279
+ const client = new DiscordClient('test-token')
280
+ const user = await client.getUser('u1')
281
+
282
+ expect(user.id).toBe('u1')
283
+ expect(user.username).toBe('testuser')
284
+ expect(fetchCalls[0].url).toBe('https://discord.com/api/v10/users/u1')
285
+ })
286
+ })
287
+
288
+ describe('uploadFile', () => {
289
+ test('uploads file to channel', async () => {
290
+ const tempFile = '/tmp/test-upload.txt'
291
+ await Bun.write(tempFile, 'test content')
292
+
293
+ mockResponse({
294
+ id: 'msg1',
295
+ channel_id: 'ch1',
296
+ author: { id: '123', username: 'bot' },
297
+ content: '',
298
+ timestamp: '2024-01-01T00:00:00.000Z',
299
+ attachments: [
300
+ {
301
+ id: 'att1',
302
+ filename: 'test-upload.txt',
303
+ size: 12,
304
+ url: 'https://cdn.discord.com/attachments/ch1/att1/test-upload.txt',
305
+ },
306
+ ],
307
+ })
308
+
309
+ const client = new DiscordClient('test-token')
310
+ const file = await client.uploadFile('ch1', tempFile)
311
+
312
+ expect(file.filename).toBe('test-upload.txt')
313
+ expect(fetchCalls[0].url).toBe('https://discord.com/api/v10/channels/ch1/messages')
314
+ expect(fetchCalls[0].options?.method).toBe('POST')
315
+ })
316
+ })
317
+
318
+ describe('listFiles', () => {
319
+ test('returns files from recent messages', async () => {
320
+ mockResponse([
321
+ {
322
+ id: 'msg1',
323
+ channel_id: 'ch1',
324
+ author: { id: '123', username: 'user1' },
325
+ content: '',
326
+ timestamp: '2024-01-01T00:00:00.000Z',
327
+ attachments: [
328
+ { id: 'att1', filename: 'file1.txt', size: 100, url: 'https://example.com/file1.txt' },
329
+ ],
330
+ },
331
+ {
332
+ id: 'msg2',
333
+ channel_id: 'ch1',
334
+ author: { id: '456', username: 'user2' },
335
+ content: 'No attachments',
336
+ timestamp: '2024-01-01T00:00:01.000Z',
337
+ attachments: [],
338
+ },
339
+ ])
340
+
341
+ const client = new DiscordClient('test-token')
342
+ const files = await client.listFiles('ch1')
343
+
344
+ expect(files).toHaveLength(1)
345
+ expect(files[0].filename).toBe('file1.txt')
346
+ })
347
+ })
348
+
349
+ describe('rate limiting', () => {
350
+ test('waits when bucket is exhausted before making request', async () => {
351
+ mockResponse({ id: '1', username: 'user1' }, 200, {
352
+ 'X-RateLimit-Remaining': '0',
353
+ 'X-RateLimit-Reset': String(Date.now() / 1000 + 0.1),
354
+ 'X-RateLimit-Bucket': 'user-bucket',
355
+ })
356
+ mockResponse({ id: '2', username: 'user2' }, 200, {
357
+ 'X-RateLimit-Remaining': '10',
358
+ 'X-RateLimit-Reset': String(Date.now() / 1000 + 60),
359
+ 'X-RateLimit-Bucket': 'user-bucket',
360
+ })
361
+
362
+ const client = new DiscordClient('test-token')
363
+ await client.testAuth()
364
+
365
+ const startTime = Date.now()
366
+ await client.testAuth()
367
+ const elapsed = Date.now() - startTime
368
+
369
+ expect(elapsed).toBeGreaterThanOrEqual(50)
370
+ expect(fetchCalls.length).toBe(2)
371
+ })
372
+
373
+ test('retries on 429 with Retry-After header', async () => {
374
+ mockResponse({ message: 'Rate limited', retry_after: 0.1 }, 429, { 'Retry-After': '0.1' })
375
+ mockResponse({ id: '123', username: 'user' })
376
+
377
+ const client = new DiscordClient('test-token')
378
+ const user = await client.testAuth()
379
+
380
+ expect(user.id).toBe('123')
381
+ expect(fetchCalls.length).toBe(2)
382
+ })
383
+
384
+ test('handles global rate limit', async () => {
385
+ mockResponse({ message: 'Global rate limited', global: true }, 429, {
386
+ 'Retry-After': '0.1',
387
+ 'X-RateLimit-Global': 'true',
388
+ })
389
+ mockResponse({ id: '123', username: 'user' })
390
+
391
+ const client = new DiscordClient('test-token')
392
+ const user = await client.testAuth()
393
+
394
+ expect(user.id).toBe('123')
395
+ expect(fetchCalls.length).toBe(2)
396
+ })
397
+
398
+ test('throws after max retries exceeded', async () => {
399
+ for (let i = 0; i <= 3; i++) {
400
+ mockResponse({ message: 'Rate limited' }, 429, { 'Retry-After': '0.01' })
401
+ }
402
+
403
+ const client = new DiscordClient('test-token')
404
+ await expect(client.testAuth()).rejects.toThrow(DiscordError)
405
+ expect(fetchCalls.length).toBeLessThanOrEqual(4)
406
+ })
407
+ })
408
+
409
+ describe('retry logic', () => {
410
+ test('retries on 500 server error', async () => {
411
+ mockResponse({ message: 'Internal Server Error' }, 500)
412
+ mockResponse({ id: '123', username: 'user' })
413
+
414
+ const client = new DiscordClient('test-token')
415
+ const user = await client.testAuth()
416
+
417
+ expect(user.id).toBe('123')
418
+ expect(fetchCalls.length).toBe(2)
419
+ })
420
+
421
+ test('does not retry on 4xx client errors (except 429)', async () => {
422
+ mockResponse({ message: 'Not Found' }, 404)
423
+
424
+ const client = new DiscordClient('test-token')
425
+ await expect(client.testAuth()).rejects.toThrow(DiscordError)
426
+ expect(fetchCalls.length).toBe(1)
427
+ })
428
+
429
+ test('exponential backoff increases delay', async () => {
430
+ mockResponse({ message: 'Error' }, 500)
431
+ mockResponse({ message: 'Error' }, 500)
432
+ mockResponse({ id: '123', username: 'user' })
433
+
434
+ const client = new DiscordClient('test-token')
435
+ const startTime = Date.now()
436
+ await client.testAuth()
437
+ const elapsed = Date.now() - startTime
438
+
439
+ expect(elapsed).toBeGreaterThanOrEqual(150)
440
+ expect(fetchCalls.length).toBe(3)
441
+ })
442
+ })
443
+
444
+ describe('bucket key normalization', () => {
445
+ test('normalizes channel IDs in routes', async () => {
446
+ mockResponse([])
447
+ mockResponse([])
448
+
449
+ const client = new DiscordClient('test-token')
450
+ await client.getMessages('123456789')
451
+ await client.getMessages('987654321')
452
+
453
+ expect(fetchCalls.length).toBe(2)
454
+ })
455
+ })
456
+ })