@erinjs/core 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 (389) hide show
  1. package/core/.changeset/README.md +8 -0
  2. package/core/.changeset/community-bootstrap-release.md +17 -0
  3. package/core/.changeset/config.json +11 -0
  4. package/core/.changeset/no-changelog.js +16 -0
  5. package/core/.changeset/pre.json +17 -0
  6. package/core/.editorconfig +13 -0
  7. package/core/.gitattributes +2 -0
  8. package/core/.github/CODE_OF_CONDUCT.md +23 -0
  9. package/core/.github/FUNDING.yml +7 -0
  10. package/core/.github/ISSUE_TEMPLATE/bug_report.md +31 -0
  11. package/core/.github/ISSUE_TEMPLATE/feature_request.md +23 -0
  12. package/core/.github/PULL_REQUEST_TEMPLATE.md +16 -0
  13. package/core/.github/dependabot.yml +16 -0
  14. package/core/.github/workflows/autoapp.yml +16 -0
  15. package/core/.github/workflows/ci.yml +187 -0
  16. package/core/.github/workflows/codeql.yml +30 -0
  17. package/core/.github/workflows/deploy-docs.yml +54 -0
  18. package/core/.github/workflows/publish.yml +43 -0
  19. package/core/.lintstagedrc.json +4 -0
  20. package/core/.nvmrc +1 -0
  21. package/core/.prettierignore +8 -0
  22. package/core/.prettierrc +11 -0
  23. package/core/CONTRIBUTING.md +70 -0
  24. package/core/LICENSE +203 -0
  25. package/core/README.md +61 -0
  26. package/core/SECURITY.md +21 -0
  27. package/core/apps/docs/index.html +28 -0
  28. package/core/apps/docs/middleware.ts +21 -0
  29. package/core/apps/docs/package.json +33 -0
  30. package/core/apps/docs/public/@flux.png +0 -0
  31. package/core/apps/docs/public/docs/latest/guides.json +1420 -0
  32. package/core/apps/docs/public/docs/latest/main.json +14981 -0
  33. package/core/apps/docs/public/docs/latest/rag-index.json +1 -0
  34. package/core/apps/docs/public/docs/v1.0.5/guides.json +226 -0
  35. package/core/apps/docs/public/docs/v1.0.5/main.json +7920 -0
  36. package/core/apps/docs/public/docs/v1.0.6/guides.json +226 -0
  37. package/core/apps/docs/public/docs/v1.0.6/main.json +7920 -0
  38. package/core/apps/docs/public/docs/v1.0.7/guides.json +259 -0
  39. package/core/apps/docs/public/docs/v1.0.7/main.json +8652 -0
  40. package/core/apps/docs/public/docs/v1.0.8/guides.json +313 -0
  41. package/core/apps/docs/public/docs/v1.0.8/main.json +9618 -0
  42. package/core/apps/docs/public/docs/v1.0.9/guides.json +319 -0
  43. package/core/apps/docs/public/docs/v1.0.9/main.json +10694 -0
  44. package/core/apps/docs/public/docs/v1.1.0/guides.json +589 -0
  45. package/core/apps/docs/public/docs/v1.1.0/main.json +12576 -0
  46. package/core/apps/docs/public/docs/v1.1.2/guides.json +650 -0
  47. package/core/apps/docs/public/docs/v1.1.2/main.json +13239 -0
  48. package/core/apps/docs/public/docs/v1.1.3/guides.json +650 -0
  49. package/core/apps/docs/public/docs/v1.1.3/main.json +13239 -0
  50. package/core/apps/docs/public/docs/v1.1.4/guides.json +708 -0
  51. package/core/apps/docs/public/docs/v1.1.4/main.json +13231 -0
  52. package/core/apps/docs/public/docs/v1.1.5/guides.json +1035 -0
  53. package/core/apps/docs/public/docs/v1.1.5/main.json +13838 -0
  54. package/core/apps/docs/public/docs/v1.1.6/guides.json +1041 -0
  55. package/core/apps/docs/public/docs/v1.1.6/main.json +14313 -0
  56. package/core/apps/docs/public/docs/v1.1.8/guides.json +1047 -0
  57. package/core/apps/docs/public/docs/v1.1.8/main.json +14421 -0
  58. package/core/apps/docs/public/docs/v1.1.9/guides.json +1047 -0
  59. package/core/apps/docs/public/docs/v1.1.9/main.json +14421 -0
  60. package/core/apps/docs/public/docs/v1.2.0/guides.json +1212 -0
  61. package/core/apps/docs/public/docs/v1.2.0/main.json +14663 -0
  62. package/core/apps/docs/public/docs/v1.2.1/guides.json +1293 -0
  63. package/core/apps/docs/public/docs/v1.2.1/main.json +14828 -0
  64. package/core/apps/docs/public/docs/v1.2.2/guides.json +1293 -0
  65. package/core/apps/docs/public/docs/v1.2.2/main.json +15025 -0
  66. package/core/apps/docs/public/docs/v1.2.3/guides.json +1420 -0
  67. package/core/apps/docs/public/docs/v1.2.3/main.json +14954 -0
  68. package/core/apps/docs/public/docs/v1.2.4/guides.json +1420 -0
  69. package/core/apps/docs/public/docs/v1.2.4/main.json +14981 -0
  70. package/core/apps/docs/public/docs/versions.json +24 -0
  71. package/core/apps/docs/public/flux.png +0 -0
  72. package/core/apps/docs/public/locales/en.json +50 -0
  73. package/core/apps/docs/public/locales/guides-en.json +512 -0
  74. package/core/apps/docs/public/robots.txt +4 -0
  75. package/core/apps/docs/public/sitemap.xml +33 -0
  76. package/core/apps/docs/src/App.vue +538 -0
  77. package/core/apps/docs/src/components/ApiCategorySection.vue +42 -0
  78. package/core/apps/docs/src/components/ApiDiscordCompat.vue +65 -0
  79. package/core/apps/docs/src/components/ApiEndpointCard.vue +313 -0
  80. package/core/apps/docs/src/components/ApiSchemaBlock.vue +131 -0
  81. package/core/apps/docs/src/components/CodeBlock.vue +177 -0
  82. package/core/apps/docs/src/components/CommunityCallout.vue +90 -0
  83. package/core/apps/docs/src/components/ConstructorSection.vue +82 -0
  84. package/core/apps/docs/src/components/DocDescription.vue +40 -0
  85. package/core/apps/docs/src/components/FluxerLogo.vue +3 -0
  86. package/core/apps/docs/src/components/Footer.vue +106 -0
  87. package/core/apps/docs/src/components/GuideCodeBlock.vue +102 -0
  88. package/core/apps/docs/src/components/GuideDiscordCompat.vue +77 -0
  89. package/core/apps/docs/src/components/GuideDiscordCompatCallout.vue +83 -0
  90. package/core/apps/docs/src/components/GuideTable.vue +77 -0
  91. package/core/apps/docs/src/components/GuideTip.vue +38 -0
  92. package/core/apps/docs/src/components/MethodsSection.vue +195 -0
  93. package/core/apps/docs/src/components/ParamsTable.vue +70 -0
  94. package/core/apps/docs/src/components/PropertiesSection.vue +143 -0
  95. package/core/apps/docs/src/components/SearchBar.vue +76 -0
  96. package/core/apps/docs/src/components/SearchModal.vue +361 -0
  97. package/core/apps/docs/src/components/SidebarNav.vue +225 -0
  98. package/core/apps/docs/src/components/SponsorBanner.vue +153 -0
  99. package/core/apps/docs/src/components/TypeSignature.vue +187 -0
  100. package/core/apps/docs/src/components/VersionPicker.vue +191 -0
  101. package/core/apps/docs/src/composables/useSearchIndex.ts +144 -0
  102. package/core/apps/docs/src/composables/useVersionedPath.ts +20 -0
  103. package/core/apps/docs/src/data/apiEndpoints.ts +1073 -0
  104. package/core/apps/docs/src/data/changelog.ts +717 -0
  105. package/core/apps/docs/src/data/guides.ts +2362 -0
  106. package/core/apps/docs/src/env.d.ts +7 -0
  107. package/core/apps/docs/src/locales/guides-en.json +512 -0
  108. package/core/apps/docs/src/main.ts +27 -0
  109. package/core/apps/docs/src/pages/ApiReferenceLayout.vue +175 -0
  110. package/core/apps/docs/src/pages/ApiReferencePage.vue +128 -0
  111. package/core/apps/docs/src/pages/Changelog.vue +288 -0
  112. package/core/apps/docs/src/pages/ClassPage.vue +319 -0
  113. package/core/apps/docs/src/pages/ClassesList.vue +100 -0
  114. package/core/apps/docs/src/pages/DocsLayout.vue +127 -0
  115. package/core/apps/docs/src/pages/GuidePage.vue +279 -0
  116. package/core/apps/docs/src/pages/GuidesIndex.vue +166 -0
  117. package/core/apps/docs/src/pages/GuidesLayout.vue +245 -0
  118. package/core/apps/docs/src/pages/Home.vue +125 -0
  119. package/core/apps/docs/src/pages/NotFound.vue +57 -0
  120. package/core/apps/docs/src/pages/TypedefPage.vue +230 -0
  121. package/core/apps/docs/src/pages/TypedefsList.vue +168 -0
  122. package/core/apps/docs/src/pages/VersionLayout.vue +15 -0
  123. package/core/apps/docs/src/router.ts +73 -0
  124. package/core/apps/docs/src/stores/docs.ts +54 -0
  125. package/core/apps/docs/src/stores/guides.ts +53 -0
  126. package/core/apps/docs/src/stores/version.ts +67 -0
  127. package/core/apps/docs/src/styles/main.css +278 -0
  128. package/core/apps/docs/src/styles/prism.css +95 -0
  129. package/core/apps/docs/src/types/doc-schema.ts +112 -0
  130. package/core/apps/docs/tsconfig.json +17 -0
  131. package/core/apps/docs/tsconfig.node.json +10 -0
  132. package/core/apps/docs/vite.config.d.ts +2 -0
  133. package/core/apps/docs/vite.config.js +26 -0
  134. package/core/apps/docs/vite.config.ts +28 -0
  135. package/core/apps/docs-vitepress/.vitepress/config.ts +141 -0
  136. package/core/apps/docs-vitepress/api-data/latest/main.json +15035 -0
  137. package/core/apps/docs-vitepress/api-data/v1.2.4/main.json +15035 -0
  138. package/core/apps/docs-vitepress/api-data/versions.json +6 -0
  139. package/core/apps/docs-vitepress/index.md +15 -0
  140. package/core/apps/docs-vitepress/package-lock.json +2924 -0
  141. package/core/apps/docs-vitepress/package.json +20 -0
  142. package/core/apps/docs-vitepress/public/CNAME +1 -0
  143. package/core/apps/docs-vitepress/scripts/generate-api.ts +243 -0
  144. package/core/apps/docs-vitepress/scripts/migrate-guides.ts +129 -0
  145. package/core/apps/docs-vitepress/tsconfig.json +11 -0
  146. package/core/apps/docs-vitepress/v/latest/guides/attachments-by-url.md +57 -0
  147. package/core/apps/docs-vitepress/v/latest/guides/attachments.md +62 -0
  148. package/core/apps/docs-vitepress/v/latest/guides/basic-bot.md +49 -0
  149. package/core/apps/docs-vitepress/v/latest/guides/channels.md +180 -0
  150. package/core/apps/docs-vitepress/v/latest/guides/deprecated-apis.md +58 -0
  151. package/core/apps/docs-vitepress/v/latest/guides/discord-js-compatibility.md +42 -0
  152. package/core/apps/docs-vitepress/v/latest/guides/editing-embeds.md +65 -0
  153. package/core/apps/docs-vitepress/v/latest/guides/embed-media.md +87 -0
  154. package/core/apps/docs-vitepress/v/latest/guides/embeds.md +166 -0
  155. package/core/apps/docs-vitepress/v/latest/guides/emojis.md +77 -0
  156. package/core/apps/docs-vitepress/v/latest/guides/events.md +202 -0
  157. package/core/apps/docs-vitepress/v/latest/guides/gifs.md +47 -0
  158. package/core/apps/docs-vitepress/v/latest/guides/installation.md +10 -0
  159. package/core/apps/docs-vitepress/v/latest/guides/moderation.md +89 -0
  160. package/core/apps/docs-vitepress/v/latest/guides/permissions.md +130 -0
  161. package/core/apps/docs-vitepress/v/latest/guides/prefix-commands.md +41 -0
  162. package/core/apps/docs-vitepress/v/latest/guides/profile-urls.md +58 -0
  163. package/core/apps/docs-vitepress/v/latest/guides/reactions.md +69 -0
  164. package/core/apps/docs-vitepress/v/latest/guides/roles.md +130 -0
  165. package/core/apps/docs-vitepress/v/latest/guides/sending-without-reply.md +172 -0
  166. package/core/apps/docs-vitepress/v/latest/guides/voice.md +109 -0
  167. package/core/apps/docs-vitepress/v/latest/guides/wait-for-guilds.md +37 -0
  168. package/core/apps/docs-vitepress/v/latest/guides/webhook-attachments-embeds.md +73 -0
  169. package/core/apps/docs-vitepress/v/latest/guides/webhooks.md +131 -0
  170. package/core/eslint.config.js +80 -0
  171. package/core/examples/.env.example +22 -0
  172. package/core/examples/README.md +68 -0
  173. package/core/examples/first-steps-bot.js +118 -0
  174. package/core/examples/minimal-bot.js +17 -0
  175. package/core/examples/moderation-bot.js +209 -0
  176. package/core/examples/package.json +14 -0
  177. package/core/examples/ping-bot.js +1146 -0
  178. package/core/examples/reaction-bot.js +70 -0
  179. package/core/examples/reaction-roles-bot.js +140 -0
  180. package/core/examples/webhook-bot.js +239 -0
  181. package/core/flux.png +0 -0
  182. package/core/package.json +78 -0
  183. package/core/packages/builders/package.json +51 -0
  184. package/core/packages/builders/src/index.ts +13 -0
  185. package/core/packages/builders/src/messages/AttachmentBuilder.test.ts +79 -0
  186. package/core/packages/builders/src/messages/AttachmentBuilder.ts +69 -0
  187. package/core/packages/builders/src/messages/EmbedBuilder.test.ts +266 -0
  188. package/core/packages/builders/src/messages/EmbedBuilder.ts +239 -0
  189. package/core/packages/builders/src/messages/MessagePayload.test.ts +118 -0
  190. package/core/packages/builders/src/messages/MessagePayload.ts +122 -0
  191. package/core/packages/builders/tsconfig.json +9 -0
  192. package/core/packages/builders/tsup.config.ts +9 -0
  193. package/core/packages/builders/vitest.config.ts +9 -0
  194. package/core/packages/collection/package.json +47 -0
  195. package/core/packages/collection/src/Collection.test.ts +232 -0
  196. package/core/packages/collection/src/Collection.ts +196 -0
  197. package/core/packages/collection/src/index.ts +1 -0
  198. package/core/packages/collection/tsconfig.json +9 -0
  199. package/core/packages/collection/tsup.config.ts +9 -0
  200. package/core/packages/collection/vitest.config.ts +9 -0
  201. package/core/packages/docgen/package.json +26 -0
  202. package/core/packages/docgen/src/extract.ts +262 -0
  203. package/core/packages/docgen/src/formatType.ts +24 -0
  204. package/core/packages/docgen/src/index.ts +103 -0
  205. package/core/packages/docgen/src/schema.ts +100 -0
  206. package/core/packages/docgen/src/visitor.ts +147 -0
  207. package/core/packages/docgen/tsconfig.json +9 -0
  208. package/core/packages/docgen/tsup.config.ts +9 -0
  209. package/core/packages/fluxer-core/README.md +26 -0
  210. package/core/packages/fluxer-core/package.json +60 -0
  211. package/core/packages/fluxer-core/src/client/ChannelManager.ts +143 -0
  212. package/core/packages/fluxer-core/src/client/Client.gateway.test.ts +84 -0
  213. package/core/packages/fluxer-core/src/client/Client.resolveEmoji.test.ts +45 -0
  214. package/core/packages/fluxer-core/src/client/Client.ts +558 -0
  215. package/core/packages/fluxer-core/src/client/ClientUser.ts +40 -0
  216. package/core/packages/fluxer-core/src/client/EventHandlerRegistry.ts +469 -0
  217. package/core/packages/fluxer-core/src/client/GuildManager.ts +79 -0
  218. package/core/packages/fluxer-core/src/client/GuildMemberManager.ts +91 -0
  219. package/core/packages/fluxer-core/src/client/MessageManager.ts +58 -0
  220. package/core/packages/fluxer-core/src/client/UsersManager.ts +122 -0
  221. package/core/packages/fluxer-core/src/errors/ErrorCodes.test.ts +19 -0
  222. package/core/packages/fluxer-core/src/errors/ErrorCodes.ts +12 -0
  223. package/core/packages/fluxer-core/src/errors/FluxerError.test.ts +32 -0
  224. package/core/packages/fluxer-core/src/errors/FluxerError.ts +15 -0
  225. package/core/packages/fluxer-core/src/index.ts +85 -0
  226. package/core/packages/fluxer-core/src/structures/Base.ts +7 -0
  227. package/core/packages/fluxer-core/src/structures/Channel.ts +508 -0
  228. package/core/packages/fluxer-core/src/structures/Guild.test.ts +189 -0
  229. package/core/packages/fluxer-core/src/structures/Guild.ts +734 -0
  230. package/core/packages/fluxer-core/src/structures/GuildBan.ts +35 -0
  231. package/core/packages/fluxer-core/src/structures/GuildEmoji.ts +57 -0
  232. package/core/packages/fluxer-core/src/structures/GuildMember.test.ts +203 -0
  233. package/core/packages/fluxer-core/src/structures/GuildMember.ts +213 -0
  234. package/core/packages/fluxer-core/src/structures/GuildMemberRoleManager.ts +121 -0
  235. package/core/packages/fluxer-core/src/structures/GuildSticker.ts +56 -0
  236. package/core/packages/fluxer-core/src/structures/Invite.test.ts +103 -0
  237. package/core/packages/fluxer-core/src/structures/Invite.ts +121 -0
  238. package/core/packages/fluxer-core/src/structures/Message.test.ts +109 -0
  239. package/core/packages/fluxer-core/src/structures/Message.ts +397 -0
  240. package/core/packages/fluxer-core/src/structures/MessageReaction.ts +72 -0
  241. package/core/packages/fluxer-core/src/structures/PartialMessage.ts +12 -0
  242. package/core/packages/fluxer-core/src/structures/Role.test.ts +77 -0
  243. package/core/packages/fluxer-core/src/structures/Role.ts +112 -0
  244. package/core/packages/fluxer-core/src/structures/User.test.ts +110 -0
  245. package/core/packages/fluxer-core/src/structures/User.ts +109 -0
  246. package/core/packages/fluxer-core/src/structures/Webhook.test.ts +109 -0
  247. package/core/packages/fluxer-core/src/structures/Webhook.ts +258 -0
  248. package/core/packages/fluxer-core/src/util/Constants.test.ts +16 -0
  249. package/core/packages/fluxer-core/src/util/Constants.ts +7 -0
  250. package/core/packages/fluxer-core/src/util/Events.ts +46 -0
  251. package/core/packages/fluxer-core/src/util/MessageCollector.ts +87 -0
  252. package/core/packages/fluxer-core/src/util/Options.ts +33 -0
  253. package/core/packages/fluxer-core/src/util/ReactionCollector.ts +116 -0
  254. package/core/packages/fluxer-core/src/util/cdn.test.ts +108 -0
  255. package/core/packages/fluxer-core/src/util/cdn.ts +130 -0
  256. package/core/packages/fluxer-core/src/util/guildUtils.ts +33 -0
  257. package/core/packages/fluxer-core/src/util/messageUtils.test.ts +74 -0
  258. package/core/packages/fluxer-core/src/util/messageUtils.ts +119 -0
  259. package/core/packages/fluxer-core/src/util/permissions.test.ts +95 -0
  260. package/core/packages/fluxer-core/src/util/permissions.ts +43 -0
  261. package/core/packages/fluxer-core/tsconfig.json +9 -0
  262. package/core/packages/fluxer-core/tsup.config.ts +9 -0
  263. package/core/packages/fluxer-core/vitest.config.ts +9 -0
  264. package/core/packages/rest/package.json +52 -0
  265. package/core/packages/rest/src/REST.test.ts +64 -0
  266. package/core/packages/rest/src/REST.ts +90 -0
  267. package/core/packages/rest/src/RateLimitManager.test.ts +71 -0
  268. package/core/packages/rest/src/RateLimitManager.ts +60 -0
  269. package/core/packages/rest/src/RequestManager.test.ts +87 -0
  270. package/core/packages/rest/src/RequestManager.ts +172 -0
  271. package/core/packages/rest/src/errors/FluxerAPIError.test.ts +57 -0
  272. package/core/packages/rest/src/errors/FluxerAPIError.ts +21 -0
  273. package/core/packages/rest/src/errors/HTTPError.test.ts +55 -0
  274. package/core/packages/rest/src/errors/HTTPError.ts +25 -0
  275. package/core/packages/rest/src/errors/RateLimitError.test.ts +41 -0
  276. package/core/packages/rest/src/errors/RateLimitError.ts +15 -0
  277. package/core/packages/rest/src/errors/index.ts +3 -0
  278. package/core/packages/rest/src/index.ts +6 -0
  279. package/core/packages/rest/src/utils/constants.test.ts +31 -0
  280. package/core/packages/rest/src/utils/constants.ts +5 -0
  281. package/core/packages/rest/src/utils/files.test.ts +37 -0
  282. package/core/packages/rest/src/utils/files.ts +75 -0
  283. package/core/packages/rest/tsconfig.json +9 -0
  284. package/core/packages/rest/tsup.config.ts +9 -0
  285. package/core/packages/rest/vitest.config.ts +9 -0
  286. package/core/packages/types/package.json +46 -0
  287. package/core/packages/types/src/api/ban.ts +8 -0
  288. package/core/packages/types/src/api/channel.ts +65 -0
  289. package/core/packages/types/src/api/embed.ts +82 -0
  290. package/core/packages/types/src/api/emoji.ts +12 -0
  291. package/core/packages/types/src/api/errors.ts +68 -0
  292. package/core/packages/types/src/api/gateway.ts +14 -0
  293. package/core/packages/types/src/api/guild.ts +123 -0
  294. package/core/packages/types/src/api/index.ts +15 -0
  295. package/core/packages/types/src/api/instance.ts +32 -0
  296. package/core/packages/types/src/api/interaction.ts +26 -0
  297. package/core/packages/types/src/api/invite.ts +28 -0
  298. package/core/packages/types/src/api/message.ts +140 -0
  299. package/core/packages/types/src/api/role.ts +41 -0
  300. package/core/packages/types/src/api/sticker.ts +14 -0
  301. package/core/packages/types/src/api/user.ts +79 -0
  302. package/core/packages/types/src/api/webhook.ts +41 -0
  303. package/core/packages/types/src/common/index.ts +1 -0
  304. package/core/packages/types/src/common/snowflake.test.ts +9 -0
  305. package/core/packages/types/src/common/snowflake.ts +8 -0
  306. package/core/packages/types/src/gateway/events.ts +189 -0
  307. package/core/packages/types/src/gateway/index.ts +3 -0
  308. package/core/packages/types/src/gateway/opcodes.ts +17 -0
  309. package/core/packages/types/src/gateway/payloads.ts +481 -0
  310. package/core/packages/types/src/index.ts +4 -0
  311. package/core/packages/types/src/rest/index.ts +1 -0
  312. package/core/packages/types/src/rest/routes.test.ts +169 -0
  313. package/core/packages/types/src/rest/routes.ts +109 -0
  314. package/core/packages/types/tsconfig.json +9 -0
  315. package/core/packages/types/tsup.config.ts +9 -0
  316. package/core/packages/types/vitest.config.ts +9 -0
  317. package/core/packages/util/package.json +51 -0
  318. package/core/packages/util/src/BitField.test.ts +96 -0
  319. package/core/packages/util/src/BitField.ts +105 -0
  320. package/core/packages/util/src/MessageFlagsBitField.test.ts +42 -0
  321. package/core/packages/util/src/MessageFlagsBitField.ts +20 -0
  322. package/core/packages/util/src/PermissionsBitField.test.ts +79 -0
  323. package/core/packages/util/src/PermissionsBitField.ts +97 -0
  324. package/core/packages/util/src/SnowflakeUtil.test.ts +69 -0
  325. package/core/packages/util/src/SnowflakeUtil.ts +65 -0
  326. package/core/packages/util/src/UserFlagsBitField.test.ts +39 -0
  327. package/core/packages/util/src/UserFlagsBitField.ts +48 -0
  328. package/core/packages/util/src/deprecation.test.ts +44 -0
  329. package/core/packages/util/src/deprecation.ts +28 -0
  330. package/core/packages/util/src/emojiShortcodes.generated.ts +5 -0
  331. package/core/packages/util/src/emojiShortcodes.test.ts +41 -0
  332. package/core/packages/util/src/emojiShortcodes.ts +22 -0
  333. package/core/packages/util/src/formatters.test.ts +65 -0
  334. package/core/packages/util/src/formatters.ts +35 -0
  335. package/core/packages/util/src/index.ts +34 -0
  336. package/core/packages/util/src/resolvers.test.ts +198 -0
  337. package/core/packages/util/src/resolvers.ts +127 -0
  338. package/core/packages/util/src/tenorUtils.test.ts +75 -0
  339. package/core/packages/util/src/tenorUtils.ts +86 -0
  340. package/core/packages/util/tsconfig.json +9 -0
  341. package/core/packages/util/tsup.config.ts +9 -0
  342. package/core/packages/util/vitest.config.ts +9 -0
  343. package/core/packages/voice/README.md +42 -0
  344. package/core/packages/voice/package.json +67 -0
  345. package/core/packages/voice/src/LiveKitRtcConnection.receive.test.ts +24 -0
  346. package/core/packages/voice/src/LiveKitRtcConnection.ts +1767 -0
  347. package/core/packages/voice/src/VoiceConnection.ts +413 -0
  348. package/core/packages/voice/src/VoiceManager.receive.test.ts +61 -0
  349. package/core/packages/voice/src/VoiceManager.test.ts +44 -0
  350. package/core/packages/voice/src/VoiceManager.ts +503 -0
  351. package/core/packages/voice/src/exports.test.ts +38 -0
  352. package/core/packages/voice/src/index.ts +51 -0
  353. package/core/packages/voice/src/livekit.test.ts +48 -0
  354. package/core/packages/voice/src/livekit.ts +33 -0
  355. package/core/packages/voice/src/mp4box.d.ts +32 -0
  356. package/core/packages/voice/src/opusUtils.test.ts +29 -0
  357. package/core/packages/voice/src/opusUtils.ts +86 -0
  358. package/core/packages/voice/src/streamPreviewPlaceholder.test.ts +16 -0
  359. package/core/packages/voice/src/streamPreviewPlaceholder.ts +8 -0
  360. package/core/packages/voice/src/ws.d.ts +1 -0
  361. package/core/packages/voice/tsconfig.json +5 -0
  362. package/core/packages/voice/tsup.config.ts +10 -0
  363. package/core/packages/voice/vitest.config.ts +9 -0
  364. package/core/packages/ws/package.json +52 -0
  365. package/core/packages/ws/src/WebSocketManager.ts +130 -0
  366. package/core/packages/ws/src/WebSocketShard.ts +296 -0
  367. package/core/packages/ws/src/index.ts +12 -0
  368. package/core/packages/ws/src/utils/constants.test.ts +46 -0
  369. package/core/packages/ws/src/utils/constants.ts +22 -0
  370. package/core/packages/ws/src/utils/getWebSocket.ts +55 -0
  371. package/core/packages/ws/src/ws.d.ts +10 -0
  372. package/core/packages/ws/tsconfig.json +9 -0
  373. package/core/packages/ws/tsup.config.ts +9 -0
  374. package/core/pnpm-lock.yaml +7033 -0
  375. package/core/pnpm-workspace.yaml +4 -0
  376. package/core/scripts/generate-ai-rag.ts +240 -0
  377. package/core/scripts/generate-docs.ts +143 -0
  378. package/core/scripts/generate-emoji-shortcodes.ts +58 -0
  379. package/core/scripts/generate-types.ts +6 -0
  380. package/core/scripts/publish-ordered.js +63 -0
  381. package/core/scripts/test-cjs-require.mjs +43 -0
  382. package/core/scripts/test-esm-imports.mjs +42 -0
  383. package/core/scripts/test-package-exports.mjs +98 -0
  384. package/core/scripts/test-smoke.mjs +103 -0
  385. package/core/tsconfig.json +18 -0
  386. package/core/turbo.json +30 -0
  387. package/core/vitest.config.ts +17 -0
  388. package/core/wrangler.jsonc +9 -0
  389. package/package.json +26 -0
@@ -0,0 +1,508 @@
1
+ import { Client } from '../client/Client.js';
2
+ import { MessageManager } from '../client/MessageManager.js';
3
+ import { MessageCollector } from '../util/MessageCollector.js';
4
+ import { MessageCollectorOptions } from '../util/MessageCollector.js';
5
+ import { Base } from './Base.js';
6
+ import { buildSendBody, resolveMessageFiles } from '../util/messageUtils.js';
7
+ import { MessageSendOptions } from '../util/messageUtils.js';
8
+ import {
9
+ APIChannel,
10
+ APIChannelPartial,
11
+ APIChannelOverwrite,
12
+ APIUser,
13
+ APIMessage,
14
+ APIWebhook,
15
+ APIInvite,
16
+ } from '@erinjs/types';
17
+ import { ChannelType, Routes } from '@erinjs/types';
18
+ import { PermissionFlags } from '@erinjs/util';
19
+ import { emitDeprecationWarning } from '@erinjs/util';
20
+ import { User } from './User.js';
21
+ import { Webhook } from './Webhook.js';
22
+ import { Message } from './Message';
23
+ import { Invite } from './Invite';
24
+
25
+ /** Base class for all channel types. */
26
+ export abstract class Channel extends Base {
27
+ /** Whether this channel has a send method (TextChannel, DMChannel). */
28
+ isTextBased(): this is TextChannel | DMChannel {
29
+ return 'send' in this;
30
+ }
31
+
32
+ /** Whether this channel is a DM or Group DM. */
33
+ isDM(): this is DMChannel {
34
+ return this.type === ChannelType.DM || this.type === ChannelType.GroupDM;
35
+ }
36
+
37
+ /** Whether this channel is voice-based (VoiceChannel). */
38
+ isVoice(): this is VoiceChannel {
39
+ return 'bitrate' in this;
40
+ }
41
+
42
+ isLink(): this is LinkChannel {
43
+ return 'url' in this;
44
+ }
45
+
46
+ /** Create a DM channel from API data (type DM or GroupDM). */
47
+ static createDM(client: Client, data: APIChannelPartial): DMChannel {
48
+ return new DMChannel(client, data);
49
+ }
50
+ readonly client: Client;
51
+ readonly id: string;
52
+ type: ChannelType;
53
+ /** Channel name. Guild channels and Group DMs have names; 1:1 DMs are typically null. */
54
+ name: string | null;
55
+ /** Channel icon hash (Group DMs). Null if none. */
56
+ icon: string | null;
57
+ /** ISO timestamp when the last message was pinned. Null if never pinned. */
58
+ lastPinTimestamp: string | null;
59
+
60
+ /** @param data - API channel from GET /channels/{id} or GET /guilds/{id}/channels */
61
+ constructor(client: Client, data: APIChannelPartial) {
62
+ super();
63
+ this.client = client;
64
+ this.id = data.id;
65
+ this.type = data.type;
66
+ this.name = data.name ?? null;
67
+ this.icon = data.icon ?? null;
68
+ this.lastPinTimestamp = (data as APIChannel).last_pin_timestamp ?? null;
69
+ }
70
+
71
+ /**
72
+ * Create the appropriate channel subclass from API data.
73
+ * @param client - The client instance
74
+ * @param data - Channel data from the API
75
+ */
76
+ static from(
77
+ client: Client,
78
+ data: APIChannel | APIChannelPartial,
79
+ ): GuildChannel | TextChannel | null {
80
+ const type = data.type ?? 0;
81
+ if (type === ChannelType.GuildText) return new TextChannel(client, data as APIChannel);
82
+ if (type === ChannelType.GuildCategory) return new CategoryChannel(client, data as APIChannel);
83
+ if (type === ChannelType.GuildVoice) return new VoiceChannel(client, data as APIChannel);
84
+ if (type === ChannelType.GuildLink || type === ChannelType.GuildLinkExtended)
85
+ return new LinkChannel(client, data as APIChannel);
86
+ return new GuildChannel(client, data as APIChannel);
87
+ }
88
+
89
+ /**
90
+ * Create a channel from API data, including DM and GroupDM.
91
+ * Used by ChannelManager.fetch() for GET /channels/{id}.
92
+ */
93
+ static fromOrCreate(
94
+ client: Client,
95
+ data: APIChannel | APIChannelPartial,
96
+ ): TextChannel | DMChannel | GuildChannel | null {
97
+ const type = data.type ?? 0;
98
+ if (type === ChannelType.DM || type === ChannelType.GroupDM)
99
+ return Channel.createDM(client, data);
100
+ return Channel.from(client, data);
101
+ }
102
+
103
+ /**
104
+ * Bulk delete messages. Requires Manage Messages permission.
105
+ * @param messageIds - Array of message IDs to delete (2–100)
106
+ */
107
+ async bulkDeleteMessages(messageIds: string[]): Promise<void> {
108
+ await this.client.rest.post(Routes.channelBulkDelete(this.id), {
109
+ body: { message_ids: messageIds },
110
+ auth: true,
111
+ });
112
+ }
113
+
114
+ /**
115
+ * Send a typing indicator to the channel. Lasts ~10 seconds.
116
+ */
117
+ async sendTyping(): Promise<void> {
118
+ await this.client.rest.post(Routes.channelTyping(this.id), { auth: true });
119
+ }
120
+
121
+ /**
122
+ * Whether the bot can send messages in this channel.
123
+ * For DMs: always true (when the channel exists).
124
+ * For guild channels: checks ViewChannel and SendMessages permissions via guild.members.me.
125
+ */
126
+ canSendMessage(): boolean {
127
+ if (this.isDM()) return true;
128
+ return false;
129
+ }
130
+ }
131
+
132
+ export class GuildChannel extends Channel {
133
+ readonly guildId: string;
134
+ name: string | null;
135
+ position?: number;
136
+ parentId: string | null;
137
+ /** Permission overwrites for roles and members. */
138
+ permissionOverwrites: APIChannelOverwrite[];
139
+
140
+ constructor(client: Client, data: APIChannel) {
141
+ super(client, data);
142
+ this.guildId = data.guild_id ?? '';
143
+ this.name = data.name ?? null;
144
+ this.position = data.position;
145
+ this.parentId = data.parent_id ?? null;
146
+ this.permissionOverwrites = data.permission_overwrites ?? [];
147
+ }
148
+
149
+ /**
150
+ * Create a webhook in this channel.
151
+ * @param options - Webhook name and optional avatar URL
152
+ * @returns The webhook with token (required for send()). Requires Manage Webhooks permission.
153
+ */
154
+ async createWebhook(options: { name: string; avatar?: string | null }): Promise<Webhook> {
155
+ const data = await this.client.rest.post(Routes.channelWebhooks(this.id), {
156
+ body: options,
157
+ auth: true,
158
+ });
159
+ return new Webhook(this.client, data as APIWebhook);
160
+ }
161
+
162
+ /**
163
+ * Fetch all webhooks in this channel.
164
+ * @returns Webhooks (includes token when listing from channel; can send via send())
165
+ */
166
+ async fetchWebhooks(): Promise<Webhook[]> {
167
+ const data = await this.client.rest.get(Routes.channelWebhooks(this.id));
168
+ const list = Array.isArray(data) ? data : Object.values(data ?? {});
169
+ return list.map((w) => new Webhook(this.client, w));
170
+ }
171
+
172
+ /**
173
+ * Create an invite for this channel.
174
+ * @param options - max_uses (0–100), max_age (0–604800 seconds), unique, temporary
175
+ * Requires Create Instant Invite permission.
176
+ */
177
+ async createInvite(options?: {
178
+ max_uses?: number;
179
+ max_age?: number;
180
+ unique?: boolean;
181
+ temporary?: boolean;
182
+ }): Promise<Invite> {
183
+ const body: Record<string, unknown> = {};
184
+ if (options?.max_uses != null) body.max_uses = options.max_uses;
185
+ if (options?.max_age != null) body.max_age = options.max_age;
186
+ if (options?.unique != null) body.unique = options.unique;
187
+ if (options?.temporary != null) body.temporary = options.temporary;
188
+ const data = await this.client.rest.post(Routes.channelInvites(this.id), {
189
+ body: Object.keys(body).length ? body : undefined,
190
+ auth: true,
191
+ });
192
+ return new Invite(this.client, data as APIInvite);
193
+ }
194
+
195
+ /**
196
+ * Fetch invites for this channel.
197
+ * Requires Manage Channel permission.
198
+ */
199
+ async fetchInvites(): Promise<Invite[]> {
200
+ const data = await this.client.rest.get(Routes.channelInvites(this.id));
201
+ const list = Array.isArray(data) ? data : Object.values(data ?? {});
202
+ return list.map((i) => new Invite(this.client, i as APIInvite));
203
+ }
204
+
205
+ /**
206
+ * Set or update a permission overwrite. PUT /channels/{id}/permissions/{overwriteId}.
207
+ * @param overwriteId - Role or member ID
208
+ * @param options - type (0=role, 1=member), allow, deny (permission bitfields)
209
+ */
210
+ async editPermission(
211
+ overwriteId: string,
212
+ options: { type: 0 | 1; allow?: string; deny?: string },
213
+ ): Promise<void> {
214
+ await this.client.rest.put(Routes.channelPermission(this.id, overwriteId), {
215
+ body: options,
216
+ auth: true,
217
+ });
218
+ const idx = this.permissionOverwrites.findIndex((o) => o.id === overwriteId);
219
+ const entry = {
220
+ id: overwriteId,
221
+ type: options.type,
222
+ allow: options.allow ?? '0',
223
+ deny: options.deny ?? '0',
224
+ };
225
+ if (idx >= 0) this.permissionOverwrites[idx] = entry;
226
+ else this.permissionOverwrites.push(entry);
227
+ }
228
+
229
+ /**
230
+ * Whether the bot can send messages in this channel.
231
+ * Checks ViewChannel and SendMessages via guild.members.me permissions.
232
+ * Returns false if guild or bot member not cached.
233
+ */
234
+ override canSendMessage(): boolean {
235
+ const guild = this.client.guilds.get(this.guildId);
236
+ if (!guild) return false;
237
+ const me = guild.members.me;
238
+ if (!me) return false;
239
+ const perms = me.permissionsIn(this);
240
+ return perms.has(PermissionFlags.ViewChannel) && perms.has(PermissionFlags.SendMessages);
241
+ }
242
+
243
+ /**
244
+ * Send a message to this guild channel.
245
+ * Works for text and announcement channels. Voice/category/link channels will fail at the API.
246
+ */
247
+ async send(options: MessageSendOptions): Promise<Message> {
248
+ const opts = typeof options === 'string' ? { content: options } : options;
249
+ const body = buildSendBody(options);
250
+ const files = opts.files?.length ? await resolveMessageFiles(opts.files) : undefined;
251
+ const postOptions = files?.length ? { body, files } : { body };
252
+ const data = await this.client.rest.post(Routes.channelMessages(this.id), postOptions);
253
+ this.client._addMessageToCache(this.id, data as APIMessage);
254
+ return new Message(this.client, data as APIMessage);
255
+ }
256
+
257
+ /**
258
+ * Remove a permission overwrite. DELETE /channels/{id}/permissions/{overwriteId}.
259
+ */
260
+ async deletePermission(overwriteId: string): Promise<void> {
261
+ await this.client.rest.delete(Routes.channelPermission(this.id, overwriteId), { auth: true });
262
+ const idx = this.permissionOverwrites.findIndex((o) => o.id === overwriteId);
263
+ if (idx >= 0) this.permissionOverwrites.splice(idx, 1);
264
+ }
265
+
266
+ /**
267
+ * Edit this channel. PATCH /channels/{id}.
268
+ * Requires Manage Channel permission.
269
+ */
270
+ async edit(options: {
271
+ name?: string | null;
272
+ topic?: string | null;
273
+ parent_id?: string | null;
274
+ bitrate?: number | null;
275
+ user_limit?: number | null;
276
+ nsfw?: boolean;
277
+ rate_limit_per_user?: number;
278
+ rtc_region?: string | null;
279
+ permission_overwrites?: Array<{ id: string; type: number; allow?: string; deny?: string }>;
280
+ }): Promise<this> {
281
+ const data = await this.client.rest.patch<APIChannel>(Routes.channel(this.id), {
282
+ body: options,
283
+ auth: true,
284
+ });
285
+ this.name = data.name ?? this.name;
286
+ this.parentId = data.parent_id ?? this.parentId;
287
+ this.permissionOverwrites = data.permission_overwrites ?? this.permissionOverwrites;
288
+ const self = this as Record<string, unknown>;
289
+ if ('topic' in self && 'topic' in data) self.topic = data.topic ?? null;
290
+ if ('nsfw' in self && 'nsfw' in data) self.nsfw = data.nsfw ?? false;
291
+ if ('rate_limit_per_user' in data) self.rateLimitPerUser = data.rate_limit_per_user ?? 0;
292
+ if ('bitrate' in self && 'bitrate' in data) self.bitrate = data.bitrate ?? null;
293
+ if ('user_limit' in data) self.userLimit = data.user_limit ?? null;
294
+ if ('rtc_region' in data) self.rtcRegion = data.rtc_region ?? null;
295
+ return this;
296
+ }
297
+
298
+ /**
299
+ * Delete this channel. Requires Manage Channel permission.
300
+ * @param options - silent: if true, does not send a system message (default false)
301
+ */
302
+ async delete(options?: { silent?: boolean }): Promise<void> {
303
+ const url = Routes.channel(this.id) + (options?.silent ? '?silent=true' : '');
304
+ await this.client.rest.delete(url, { auth: true });
305
+ this.client.channels.delete(this.id);
306
+ const guild = this.client.guilds.get(this.guildId);
307
+ if (guild) guild.channels.delete(this.id);
308
+ }
309
+ }
310
+
311
+ export class TextChannel extends GuildChannel {
312
+ topic?: string | null;
313
+ nsfw?: boolean;
314
+ rateLimitPerUser?: number;
315
+ lastMessageId?: string | null;
316
+
317
+ constructor(client: Client, data: APIChannel) {
318
+ super(client, data);
319
+ this.topic = data.topic ?? null;
320
+ this.nsfw = data.nsfw ?? false;
321
+ this.rateLimitPerUser = data.rate_limit_per_user ?? 0;
322
+ this.lastMessageId = data.last_message_id ?? null;
323
+ }
324
+
325
+ /**
326
+ * Send a message to this channel.
327
+ * @param options - Text content or object with content, embeds, and/or files
328
+ */
329
+ async send(options: MessageSendOptions): Promise<Message> {
330
+ const opts = typeof options === 'string' ? { content: options } : options;
331
+ const body = buildSendBody(options);
332
+ const files = opts.files?.length ? await resolveMessageFiles(opts.files) : undefined;
333
+ const postOptions = files?.length ? { body, files } : { body };
334
+ const data = await this.client.rest.post(Routes.channelMessages(this.id), postOptions);
335
+ this.client._addMessageToCache(this.id, data as APIMessage);
336
+ return new Message(this.client, data as APIMessage);
337
+ }
338
+
339
+ /** Message manager for this channel. Use channel.messages.fetch(messageId). */
340
+ get messages(): MessageManager {
341
+ return new MessageManager(this.client, this.id);
342
+ }
343
+
344
+ /**
345
+ * Create a message collector for this channel.
346
+ * Collects messages matching the filter until time expires or max is reached.
347
+ * @param options - Filter, time (ms), and max count
348
+ * @example
349
+ * const collector = channel.createMessageCollector({ filter: m => m.author.id === userId, time: 10000 });
350
+ * collector.on('collect', m => console.log(m.content));
351
+ * collector.on('end', (collected, reason) => { ... });
352
+ */
353
+ createMessageCollector(options?: MessageCollectorOptions): MessageCollector {
354
+ return new MessageCollector(this.client, this.id, options);
355
+ }
356
+
357
+ /**
358
+ * Fetch pinned messages in this channel.
359
+ * @returns Pinned messages
360
+ */
361
+ async fetchPinnedMessages(): Promise<Message[]> {
362
+ type PinnedItem = APIMessage | { message?: APIMessage };
363
+ const data = (await this.client.rest.get(Routes.channelPins(this.id))) as
364
+ | { items?: PinnedItem[] }
365
+ | APIMessage[];
366
+ const list: PinnedItem[] = Array.isArray(data) ? data : (data?.items ?? []);
367
+ return list.map((item) => {
368
+ const msg = typeof item === 'object' && item && 'message' in item ? item.message : item;
369
+ return new Message(this.client, msg as APIMessage);
370
+ });
371
+ }
372
+
373
+ /**
374
+ * Fetch a message by ID from this channel.
375
+ * @param messageId - Snowflake of the message
376
+ * @returns The message, or null if not found
377
+ * @deprecated Use channel.messages.fetch(messageId) instead.
378
+ */
379
+ async fetchMessage(messageId: string): Promise<Message> {
380
+ emitDeprecationWarning(
381
+ 'Channel.fetchMessage()',
382
+ 'Use channel.messages.fetch(messageId) instead.',
383
+ );
384
+ return this.client.channels.fetchMessage(this.id, messageId);
385
+ }
386
+ }
387
+
388
+ export class CategoryChannel extends GuildChannel {}
389
+ export class VoiceChannel extends GuildChannel {
390
+ bitrate?: number | null;
391
+ userLimit?: number | null;
392
+ rtcRegion?: string | null;
393
+
394
+ constructor(client: Client, data: APIChannel) {
395
+ super(client, data);
396
+ this.bitrate = data.bitrate ?? null;
397
+ this.userLimit = data.user_limit ?? null;
398
+ this.rtcRegion = data.rtc_region ?? null;
399
+ }
400
+ }
401
+
402
+ export class LinkChannel extends GuildChannel {
403
+ url: string | null;
404
+ constructor(client: Client, data: APIChannel) {
405
+ super(client, data);
406
+ this.url = data.url ?? null;
407
+ }
408
+ }
409
+
410
+ /** DM channel (direct message between bot and a user). */
411
+ export class DMChannel extends Channel {
412
+ lastMessageId?: string | null;
413
+ /** Group DM creator ID. Null for 1:1 DMs. */
414
+ ownerId: string | null;
415
+ /** Group DM recipients as User objects. Empty for 1:1 DMs. */
416
+ recipients: User[];
417
+ /** Group DM member display names (userId -> nickname). */
418
+ nicks: Record<string, string>;
419
+
420
+ constructor(client: Client, data: APIChannelPartial & Partial<APIChannel>) {
421
+ super(client, data);
422
+ this.lastMessageId = (data as APIChannel).last_message_id ?? null;
423
+ this.ownerId = (data as APIChannel).owner_id ?? null;
424
+ this.recipients = ((data as APIChannel).recipients ?? []).map((u: APIUser) =>
425
+ client.getOrCreateUser(u),
426
+ );
427
+ this.nicks = (data as APIChannel).nicks ?? {};
428
+ }
429
+
430
+ /**
431
+ * Send a message to this DM channel.
432
+ * @param options - Text content or object with content, embeds, and/or files
433
+ */
434
+ async send(options: MessageSendOptions): Promise<Message> {
435
+ const opts = typeof options === 'string' ? { content: options } : options;
436
+ const body = buildSendBody(options);
437
+ const files = opts.files?.length ? await resolveMessageFiles(opts.files) : undefined;
438
+ const postOptions = files?.length ? { body, files } : { body };
439
+ const data = await this.client.rest.post(Routes.channelMessages(this.id), postOptions);
440
+ this.client._addMessageToCache(this.id, data as APIMessage);
441
+ return new Message(this.client, data as APIMessage);
442
+ }
443
+
444
+ /** Message manager for this channel. Use channel.messages.fetch(messageId). */
445
+ get messages(): MessageManager {
446
+ return new MessageManager(this.client, this.id);
447
+ }
448
+
449
+ /**
450
+ * Create a message collector for this DM channel.
451
+ * @param options - Filter, time (ms), and max count
452
+ */
453
+ createMessageCollector(options?: MessageCollectorOptions): MessageCollector {
454
+ return new MessageCollector(this.client, this.id, options);
455
+ }
456
+
457
+ /**
458
+ * Fetch pinned messages in this DM channel.
459
+ * @returns Pinned messages
460
+ */
461
+ async fetchPinnedMessages(): Promise<Message[]> {
462
+ type PinnedItem = APIMessage | { message?: APIMessage };
463
+ const data = (await this.client.rest.get(Routes.channelPins(this.id))) as
464
+ | { items?: PinnedItem[] }
465
+ | APIMessage[];
466
+ const list: PinnedItem[] = Array.isArray(data) ? data : (data?.items ?? []);
467
+ return list.map((item) => {
468
+ const msg = typeof item === 'object' && item && 'message' in item ? item.message : item;
469
+ return new Message(this.client, msg as APIMessage);
470
+ });
471
+ }
472
+
473
+ /**
474
+ * Fetch a message by ID from this DM channel.
475
+ * @param messageId - Snowflake of the message
476
+ * @returns The message, or null if not found
477
+ * @deprecated Use channel.messages.fetch(messageId) instead.
478
+ */
479
+ async fetchMessage(messageId: string): Promise<Message> {
480
+ emitDeprecationWarning(
481
+ 'Channel.fetchMessage()',
482
+ 'Use channel.messages.fetch(messageId) instead.',
483
+ );
484
+ return this.client.channels.fetchMessage(this.id, messageId);
485
+ }
486
+
487
+ /**
488
+ * Add a recipient to this Group DM. Requires Group DM (type GroupDM).
489
+ * PUT /channels/{id}/recipients/{userId}.
490
+ */
491
+ async addRecipient(userId: string): Promise<void> {
492
+ await this.client.rest.put(Routes.channelRecipient(this.id, userId), { auth: true });
493
+ const user = this.client.users.get(userId) ?? (await this.client.users.fetch(userId));
494
+ if (user) this.recipients.push(user);
495
+ }
496
+
497
+ /**
498
+ * Remove a recipient from this Group DM. Requires Group DM (type GroupDM).
499
+ * DELETE /channels/{id}/recipients/{userId}.
500
+ * @param options - silent: if true, does not send a system message (default false)
501
+ */
502
+ async removeRecipient(userId: string, options?: { silent?: boolean }): Promise<void> {
503
+ const url = Routes.channelRecipient(this.id, userId) + (options?.silent ? '?silent=true' : '');
504
+ await this.client.rest.delete(url, { auth: true });
505
+ const idx = this.recipients.findIndex((u) => u.id === userId);
506
+ if (idx >= 0) this.recipients.splice(idx, 1);
507
+ }
508
+ }
@@ -0,0 +1,189 @@
1
+ import { describe, it, expect, vi } from 'vitest';
2
+ import { Guild, Client } from '../';
3
+ import { Routes } from '@erinjs/types';
4
+
5
+ function createMockClient() {
6
+ return {} as Client;
7
+ }
8
+
9
+ function createRestBackedGuild(restGetResponse: unknown) {
10
+ const get = vi.fn().mockResolvedValue(restGetResponse);
11
+ const client = {
12
+ rest: { get },
13
+ getOrCreateUser: (data: { id: string; username?: string; discriminator?: string }) => ({
14
+ id: data.id,
15
+ username: data.username ?? 'user',
16
+ discriminator: data.discriminator ?? '0',
17
+ }),
18
+ guilds: new Map(),
19
+ channels: new Map(),
20
+ } as unknown as Client;
21
+
22
+ const guild = new Guild(client, {
23
+ id: 'guild1',
24
+ name: 'Test Guild',
25
+ icon: null,
26
+ banner: null,
27
+ splash: null,
28
+ owner_id: 'owner1',
29
+ features: [],
30
+ afk_timeout: 0,
31
+ nsfw_level: 0,
32
+ verification_level: 0,
33
+ mfa_level: 0,
34
+ explicit_content_filter: 0,
35
+ default_message_notifications: 0,
36
+ });
37
+
38
+ return { guild, get };
39
+ }
40
+
41
+ function createGuild(
42
+ overrides: {
43
+ id?: string;
44
+ icon?: string | null;
45
+ banner?: string | null;
46
+ splash?: string | null;
47
+ } = {},
48
+ ) {
49
+ return new Guild(createMockClient(), {
50
+ id: overrides.id ?? 'guild1',
51
+ name: 'Test Guild',
52
+ icon: overrides.icon ?? null,
53
+ banner: overrides.banner ?? null,
54
+ splash: overrides.splash ?? null,
55
+ owner_id: 'owner1',
56
+ features: [],
57
+ afk_timeout: 0,
58
+ nsfw_level: 0,
59
+ verification_level: 0,
60
+ mfa_level: 0,
61
+ explicit_content_filter: 0,
62
+ default_message_notifications: 0,
63
+ });
64
+ }
65
+
66
+ describe('Guild', () => {
67
+ describe('iconURL()', () => {
68
+ it('returns null when icon is null', () => {
69
+ const guild = createGuild({ icon: null });
70
+ expect(guild.iconURL()).toBeNull();
71
+ });
72
+
73
+ it('builds icon URL when icon is set', () => {
74
+ const guild = createGuild({ icon: 'iconhash123' });
75
+ const url = guild.iconURL();
76
+ expect(url).toContain('fluxerusercontent.com/icons/guild1/iconhash123.png');
77
+ });
78
+
79
+ it('appends size when provided', () => {
80
+ const guild = createGuild({ icon: 'hash' });
81
+ const url = guild.iconURL({ size: 512 });
82
+ expect(url).toContain('?size=512');
83
+ });
84
+ });
85
+
86
+ describe('bannerURL()', () => {
87
+ it('returns null when banner is null', () => {
88
+ const guild = createGuild({ banner: null });
89
+ expect(guild.bannerURL()).toBeNull();
90
+ });
91
+
92
+ it('builds banner URL when banner is set', () => {
93
+ const guild = createGuild({ banner: 'bannerhash' });
94
+ const url = guild.bannerURL();
95
+ expect(url).toContain('fluxerusercontent.com/banners/guild1/bannerhash.png');
96
+ });
97
+ });
98
+
99
+ describe('splashURL()', () => {
100
+ it('returns null when splash is null', () => {
101
+ const guild = createGuild({ splash: null });
102
+ expect(guild.splashURL()).toBeNull();
103
+ });
104
+
105
+ it('builds splash URL when splash is set', () => {
106
+ const guild = createGuild({ splash: 'splashhash' });
107
+ const url = guild.splashURL();
108
+ expect(url).toContain('fluxerusercontent.com/splashes/guild1/splashhash.png');
109
+ });
110
+ });
111
+
112
+ describe('constructor', () => {
113
+ it('parses guild id and name', () => {
114
+ const guild = createGuild({ id: 'custom123' });
115
+ expect(guild.id).toBe('custom123');
116
+ expect(guild.name).toBe('Test Guild');
117
+ });
118
+ });
119
+
120
+ describe('invite helper methods', () => {
121
+ it('fetchInvites() fetches and maps guild invites', async () => {
122
+ const { guild, get } = createRestBackedGuild([
123
+ {
124
+ code: 'abc123',
125
+ type: 0,
126
+ guild: { id: 'guild1', name: 'Test Guild' },
127
+ channel: { id: 'channel1', type: 0, name: 'general' },
128
+ },
129
+ ]);
130
+
131
+ const invites = await guild.fetchInvites();
132
+
133
+ expect(get).toHaveBeenCalledWith(Routes.guildInvites('guild1'));
134
+ expect(invites).toHaveLength(1);
135
+ expect(invites[0]?.code).toBe('abc123');
136
+ });
137
+
138
+ it('fetchInvite() supports code-or-url helper flow', async () => {
139
+ const { guild, get } = createRestBackedGuild({
140
+ code: 'xyz789',
141
+ type: 0,
142
+ guild: { id: 'guild1', name: 'Test Guild' },
143
+ channel: { id: 'channel1', type: 0, name: 'general' },
144
+ });
145
+
146
+ const invite = await guild.fetchInvite('https://fluxer.gg/xyz789');
147
+
148
+ expect(get).toHaveBeenCalledWith(Routes.invite('xyz789'));
149
+ expect(invite.code).toBe('xyz789');
150
+ });
151
+ });
152
+
153
+ describe('guild sticker helper methods', () => {
154
+ it('fetchStickers() fetches and caches guild stickers', async () => {
155
+ const { guild, get } = createRestBackedGuild([
156
+ {
157
+ id: 'sticker1',
158
+ name: 'sample_sticker_a',
159
+ description: 'Sample sticker A',
160
+ tags: ['sample', 'a'],
161
+ animated: false,
162
+ },
163
+ ]);
164
+
165
+ const stickers = await guild.fetchStickers();
166
+
167
+ expect(get).toHaveBeenCalledWith(Routes.guildStickers('guild1'));
168
+ expect(stickers).toHaveLength(1);
169
+ expect(stickers[0]?.id).toBe('sticker1');
170
+ expect(guild.stickers.get('sticker1')?.name).toBe('sample_sticker_a');
171
+ });
172
+
173
+ it('fetchSticker() fetches and caches a single guild sticker', async () => {
174
+ const { guild, get } = createRestBackedGuild({
175
+ id: 'sticker42',
176
+ name: 'sample_sticker_b',
177
+ description: 'Sample sticker B',
178
+ tags: ['sample', 'b'],
179
+ animated: false,
180
+ });
181
+
182
+ const sticker = await guild.fetchSticker('sticker42');
183
+
184
+ expect(get).toHaveBeenCalledWith(Routes.guildSticker('guild1', 'sticker42'));
185
+ expect(sticker.id).toBe('sticker42');
186
+ expect(guild.stickers.get('sticker42')?.name).toBe('sample_sticker_b');
187
+ });
188
+ });
189
+ });