@erinjs/core 1.0.0 → 1.2.4

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/{core/LICENSE → LICENSE} +203 -203
  2. package/README.md +46 -0
  3. package/dist/index.d.mts +2084 -0
  4. package/dist/index.d.ts +2084 -0
  5. package/dist/index.js +3867 -0
  6. package/dist/index.mjs +3809 -0
  7. package/package.json +50 -15
  8. package/core/.changeset/README.md +0 -8
  9. package/core/.changeset/community-bootstrap-release.md +0 -17
  10. package/core/.changeset/config.json +0 -11
  11. package/core/.changeset/no-changelog.js +0 -16
  12. package/core/.changeset/pre.json +0 -17
  13. package/core/.editorconfig +0 -13
  14. package/core/.gitattributes +0 -2
  15. package/core/.github/CODE_OF_CONDUCT.md +0 -23
  16. package/core/.github/FUNDING.yml +0 -7
  17. package/core/.github/ISSUE_TEMPLATE/bug_report.md +0 -31
  18. package/core/.github/ISSUE_TEMPLATE/feature_request.md +0 -23
  19. package/core/.github/PULL_REQUEST_TEMPLATE.md +0 -16
  20. package/core/.github/dependabot.yml +0 -16
  21. package/core/.github/workflows/autoapp.yml +0 -16
  22. package/core/.github/workflows/ci.yml +0 -187
  23. package/core/.github/workflows/codeql.yml +0 -30
  24. package/core/.github/workflows/deploy-docs.yml +0 -54
  25. package/core/.github/workflows/publish.yml +0 -43
  26. package/core/.lintstagedrc.json +0 -4
  27. package/core/.nvmrc +0 -1
  28. package/core/.prettierignore +0 -8
  29. package/core/.prettierrc +0 -11
  30. package/core/CONTRIBUTING.md +0 -70
  31. package/core/README.md +0 -61
  32. package/core/SECURITY.md +0 -21
  33. package/core/apps/docs/index.html +0 -28
  34. package/core/apps/docs/middleware.ts +0 -21
  35. package/core/apps/docs/package.json +0 -33
  36. package/core/apps/docs/public/@flux.png +0 -0
  37. package/core/apps/docs/public/docs/latest/guides.json +0 -1420
  38. package/core/apps/docs/public/docs/latest/main.json +0 -14981
  39. package/core/apps/docs/public/docs/latest/rag-index.json +0 -1
  40. package/core/apps/docs/public/docs/v1.0.5/guides.json +0 -226
  41. package/core/apps/docs/public/docs/v1.0.5/main.json +0 -7920
  42. package/core/apps/docs/public/docs/v1.0.6/guides.json +0 -226
  43. package/core/apps/docs/public/docs/v1.0.6/main.json +0 -7920
  44. package/core/apps/docs/public/docs/v1.0.7/guides.json +0 -259
  45. package/core/apps/docs/public/docs/v1.0.7/main.json +0 -8652
  46. package/core/apps/docs/public/docs/v1.0.8/guides.json +0 -313
  47. package/core/apps/docs/public/docs/v1.0.8/main.json +0 -9618
  48. package/core/apps/docs/public/docs/v1.0.9/guides.json +0 -319
  49. package/core/apps/docs/public/docs/v1.0.9/main.json +0 -10694
  50. package/core/apps/docs/public/docs/v1.1.0/guides.json +0 -589
  51. package/core/apps/docs/public/docs/v1.1.0/main.json +0 -12576
  52. package/core/apps/docs/public/docs/v1.1.2/guides.json +0 -650
  53. package/core/apps/docs/public/docs/v1.1.2/main.json +0 -13239
  54. package/core/apps/docs/public/docs/v1.1.3/guides.json +0 -650
  55. package/core/apps/docs/public/docs/v1.1.3/main.json +0 -13239
  56. package/core/apps/docs/public/docs/v1.1.4/guides.json +0 -708
  57. package/core/apps/docs/public/docs/v1.1.4/main.json +0 -13231
  58. package/core/apps/docs/public/docs/v1.1.5/guides.json +0 -1035
  59. package/core/apps/docs/public/docs/v1.1.5/main.json +0 -13838
  60. package/core/apps/docs/public/docs/v1.1.6/guides.json +0 -1041
  61. package/core/apps/docs/public/docs/v1.1.6/main.json +0 -14313
  62. package/core/apps/docs/public/docs/v1.1.8/guides.json +0 -1047
  63. package/core/apps/docs/public/docs/v1.1.8/main.json +0 -14421
  64. package/core/apps/docs/public/docs/v1.1.9/guides.json +0 -1047
  65. package/core/apps/docs/public/docs/v1.1.9/main.json +0 -14421
  66. package/core/apps/docs/public/docs/v1.2.0/guides.json +0 -1212
  67. package/core/apps/docs/public/docs/v1.2.0/main.json +0 -14663
  68. package/core/apps/docs/public/docs/v1.2.1/guides.json +0 -1293
  69. package/core/apps/docs/public/docs/v1.2.1/main.json +0 -14828
  70. package/core/apps/docs/public/docs/v1.2.2/guides.json +0 -1293
  71. package/core/apps/docs/public/docs/v1.2.2/main.json +0 -15025
  72. package/core/apps/docs/public/docs/v1.2.3/guides.json +0 -1420
  73. package/core/apps/docs/public/docs/v1.2.3/main.json +0 -14954
  74. package/core/apps/docs/public/docs/v1.2.4/guides.json +0 -1420
  75. package/core/apps/docs/public/docs/v1.2.4/main.json +0 -14981
  76. package/core/apps/docs/public/docs/versions.json +0 -24
  77. package/core/apps/docs/public/flux.png +0 -0
  78. package/core/apps/docs/public/locales/en.json +0 -50
  79. package/core/apps/docs/public/locales/guides-en.json +0 -512
  80. package/core/apps/docs/public/robots.txt +0 -4
  81. package/core/apps/docs/public/sitemap.xml +0 -33
  82. package/core/apps/docs/src/App.vue +0 -538
  83. package/core/apps/docs/src/components/ApiCategorySection.vue +0 -42
  84. package/core/apps/docs/src/components/ApiDiscordCompat.vue +0 -65
  85. package/core/apps/docs/src/components/ApiEndpointCard.vue +0 -313
  86. package/core/apps/docs/src/components/ApiSchemaBlock.vue +0 -131
  87. package/core/apps/docs/src/components/CodeBlock.vue +0 -177
  88. package/core/apps/docs/src/components/CommunityCallout.vue +0 -90
  89. package/core/apps/docs/src/components/ConstructorSection.vue +0 -82
  90. package/core/apps/docs/src/components/DocDescription.vue +0 -40
  91. package/core/apps/docs/src/components/FluxerLogo.vue +0 -3
  92. package/core/apps/docs/src/components/Footer.vue +0 -106
  93. package/core/apps/docs/src/components/GuideCodeBlock.vue +0 -102
  94. package/core/apps/docs/src/components/GuideDiscordCompat.vue +0 -77
  95. package/core/apps/docs/src/components/GuideDiscordCompatCallout.vue +0 -83
  96. package/core/apps/docs/src/components/GuideTable.vue +0 -77
  97. package/core/apps/docs/src/components/GuideTip.vue +0 -38
  98. package/core/apps/docs/src/components/MethodsSection.vue +0 -195
  99. package/core/apps/docs/src/components/ParamsTable.vue +0 -70
  100. package/core/apps/docs/src/components/PropertiesSection.vue +0 -143
  101. package/core/apps/docs/src/components/SearchBar.vue +0 -76
  102. package/core/apps/docs/src/components/SearchModal.vue +0 -361
  103. package/core/apps/docs/src/components/SidebarNav.vue +0 -225
  104. package/core/apps/docs/src/components/SponsorBanner.vue +0 -153
  105. package/core/apps/docs/src/components/TypeSignature.vue +0 -187
  106. package/core/apps/docs/src/components/VersionPicker.vue +0 -191
  107. package/core/apps/docs/src/composables/useSearchIndex.ts +0 -144
  108. package/core/apps/docs/src/composables/useVersionedPath.ts +0 -20
  109. package/core/apps/docs/src/data/apiEndpoints.ts +0 -1073
  110. package/core/apps/docs/src/data/changelog.ts +0 -717
  111. package/core/apps/docs/src/data/guides.ts +0 -2362
  112. package/core/apps/docs/src/env.d.ts +0 -7
  113. package/core/apps/docs/src/locales/guides-en.json +0 -512
  114. package/core/apps/docs/src/main.ts +0 -27
  115. package/core/apps/docs/src/pages/ApiReferenceLayout.vue +0 -175
  116. package/core/apps/docs/src/pages/ApiReferencePage.vue +0 -128
  117. package/core/apps/docs/src/pages/Changelog.vue +0 -288
  118. package/core/apps/docs/src/pages/ClassPage.vue +0 -319
  119. package/core/apps/docs/src/pages/ClassesList.vue +0 -100
  120. package/core/apps/docs/src/pages/DocsLayout.vue +0 -127
  121. package/core/apps/docs/src/pages/GuidePage.vue +0 -279
  122. package/core/apps/docs/src/pages/GuidesIndex.vue +0 -166
  123. package/core/apps/docs/src/pages/GuidesLayout.vue +0 -245
  124. package/core/apps/docs/src/pages/Home.vue +0 -125
  125. package/core/apps/docs/src/pages/NotFound.vue +0 -57
  126. package/core/apps/docs/src/pages/TypedefPage.vue +0 -230
  127. package/core/apps/docs/src/pages/TypedefsList.vue +0 -168
  128. package/core/apps/docs/src/pages/VersionLayout.vue +0 -15
  129. package/core/apps/docs/src/router.ts +0 -73
  130. package/core/apps/docs/src/stores/docs.ts +0 -54
  131. package/core/apps/docs/src/stores/guides.ts +0 -53
  132. package/core/apps/docs/src/stores/version.ts +0 -67
  133. package/core/apps/docs/src/styles/main.css +0 -278
  134. package/core/apps/docs/src/styles/prism.css +0 -95
  135. package/core/apps/docs/src/types/doc-schema.ts +0 -112
  136. package/core/apps/docs/tsconfig.json +0 -17
  137. package/core/apps/docs/tsconfig.node.json +0 -10
  138. package/core/apps/docs/vite.config.d.ts +0 -2
  139. package/core/apps/docs/vite.config.js +0 -26
  140. package/core/apps/docs/vite.config.ts +0 -28
  141. package/core/apps/docs-vitepress/.vitepress/config.ts +0 -141
  142. package/core/apps/docs-vitepress/api-data/latest/main.json +0 -15035
  143. package/core/apps/docs-vitepress/api-data/v1.2.4/main.json +0 -15035
  144. package/core/apps/docs-vitepress/api-data/versions.json +0 -6
  145. package/core/apps/docs-vitepress/index.md +0 -15
  146. package/core/apps/docs-vitepress/package-lock.json +0 -2924
  147. package/core/apps/docs-vitepress/package.json +0 -20
  148. package/core/apps/docs-vitepress/public/CNAME +0 -1
  149. package/core/apps/docs-vitepress/scripts/generate-api.ts +0 -243
  150. package/core/apps/docs-vitepress/scripts/migrate-guides.ts +0 -129
  151. package/core/apps/docs-vitepress/tsconfig.json +0 -11
  152. package/core/apps/docs-vitepress/v/latest/guides/attachments-by-url.md +0 -57
  153. package/core/apps/docs-vitepress/v/latest/guides/attachments.md +0 -62
  154. package/core/apps/docs-vitepress/v/latest/guides/basic-bot.md +0 -49
  155. package/core/apps/docs-vitepress/v/latest/guides/channels.md +0 -180
  156. package/core/apps/docs-vitepress/v/latest/guides/deprecated-apis.md +0 -58
  157. package/core/apps/docs-vitepress/v/latest/guides/discord-js-compatibility.md +0 -42
  158. package/core/apps/docs-vitepress/v/latest/guides/editing-embeds.md +0 -65
  159. package/core/apps/docs-vitepress/v/latest/guides/embed-media.md +0 -87
  160. package/core/apps/docs-vitepress/v/latest/guides/embeds.md +0 -166
  161. package/core/apps/docs-vitepress/v/latest/guides/emojis.md +0 -77
  162. package/core/apps/docs-vitepress/v/latest/guides/events.md +0 -202
  163. package/core/apps/docs-vitepress/v/latest/guides/gifs.md +0 -47
  164. package/core/apps/docs-vitepress/v/latest/guides/installation.md +0 -10
  165. package/core/apps/docs-vitepress/v/latest/guides/moderation.md +0 -89
  166. package/core/apps/docs-vitepress/v/latest/guides/permissions.md +0 -130
  167. package/core/apps/docs-vitepress/v/latest/guides/prefix-commands.md +0 -41
  168. package/core/apps/docs-vitepress/v/latest/guides/profile-urls.md +0 -58
  169. package/core/apps/docs-vitepress/v/latest/guides/reactions.md +0 -69
  170. package/core/apps/docs-vitepress/v/latest/guides/roles.md +0 -130
  171. package/core/apps/docs-vitepress/v/latest/guides/sending-without-reply.md +0 -172
  172. package/core/apps/docs-vitepress/v/latest/guides/voice.md +0 -109
  173. package/core/apps/docs-vitepress/v/latest/guides/wait-for-guilds.md +0 -37
  174. package/core/apps/docs-vitepress/v/latest/guides/webhook-attachments-embeds.md +0 -73
  175. package/core/apps/docs-vitepress/v/latest/guides/webhooks.md +0 -131
  176. package/core/eslint.config.js +0 -80
  177. package/core/examples/.env.example +0 -22
  178. package/core/examples/README.md +0 -68
  179. package/core/examples/first-steps-bot.js +0 -118
  180. package/core/examples/minimal-bot.js +0 -17
  181. package/core/examples/moderation-bot.js +0 -209
  182. package/core/examples/package.json +0 -14
  183. package/core/examples/ping-bot.js +0 -1146
  184. package/core/examples/reaction-bot.js +0 -70
  185. package/core/examples/reaction-roles-bot.js +0 -140
  186. package/core/examples/webhook-bot.js +0 -239
  187. package/core/flux.png +0 -0
  188. package/core/package.json +0 -78
  189. package/core/packages/builders/package.json +0 -51
  190. package/core/packages/builders/src/index.ts +0 -13
  191. package/core/packages/builders/src/messages/AttachmentBuilder.test.ts +0 -79
  192. package/core/packages/builders/src/messages/AttachmentBuilder.ts +0 -69
  193. package/core/packages/builders/src/messages/EmbedBuilder.test.ts +0 -266
  194. package/core/packages/builders/src/messages/EmbedBuilder.ts +0 -239
  195. package/core/packages/builders/src/messages/MessagePayload.test.ts +0 -118
  196. package/core/packages/builders/src/messages/MessagePayload.ts +0 -122
  197. package/core/packages/builders/tsconfig.json +0 -9
  198. package/core/packages/builders/tsup.config.ts +0 -9
  199. package/core/packages/builders/vitest.config.ts +0 -9
  200. package/core/packages/collection/package.json +0 -47
  201. package/core/packages/collection/src/Collection.test.ts +0 -232
  202. package/core/packages/collection/src/Collection.ts +0 -196
  203. package/core/packages/collection/src/index.ts +0 -1
  204. package/core/packages/collection/tsconfig.json +0 -9
  205. package/core/packages/collection/tsup.config.ts +0 -9
  206. package/core/packages/collection/vitest.config.ts +0 -9
  207. package/core/packages/docgen/package.json +0 -26
  208. package/core/packages/docgen/src/extract.ts +0 -262
  209. package/core/packages/docgen/src/formatType.ts +0 -24
  210. package/core/packages/docgen/src/index.ts +0 -103
  211. package/core/packages/docgen/src/schema.ts +0 -100
  212. package/core/packages/docgen/src/visitor.ts +0 -147
  213. package/core/packages/docgen/tsconfig.json +0 -9
  214. package/core/packages/docgen/tsup.config.ts +0 -9
  215. package/core/packages/fluxer-core/README.md +0 -26
  216. package/core/packages/fluxer-core/package.json +0 -60
  217. package/core/packages/fluxer-core/src/client/ChannelManager.ts +0 -143
  218. package/core/packages/fluxer-core/src/client/Client.gateway.test.ts +0 -84
  219. package/core/packages/fluxer-core/src/client/Client.resolveEmoji.test.ts +0 -45
  220. package/core/packages/fluxer-core/src/client/Client.ts +0 -558
  221. package/core/packages/fluxer-core/src/client/ClientUser.ts +0 -40
  222. package/core/packages/fluxer-core/src/client/EventHandlerRegistry.ts +0 -469
  223. package/core/packages/fluxer-core/src/client/GuildManager.ts +0 -79
  224. package/core/packages/fluxer-core/src/client/GuildMemberManager.ts +0 -91
  225. package/core/packages/fluxer-core/src/client/MessageManager.ts +0 -58
  226. package/core/packages/fluxer-core/src/client/UsersManager.ts +0 -122
  227. package/core/packages/fluxer-core/src/errors/ErrorCodes.test.ts +0 -19
  228. package/core/packages/fluxer-core/src/errors/ErrorCodes.ts +0 -12
  229. package/core/packages/fluxer-core/src/errors/FluxerError.test.ts +0 -32
  230. package/core/packages/fluxer-core/src/errors/FluxerError.ts +0 -15
  231. package/core/packages/fluxer-core/src/index.ts +0 -85
  232. package/core/packages/fluxer-core/src/structures/Base.ts +0 -7
  233. package/core/packages/fluxer-core/src/structures/Channel.ts +0 -508
  234. package/core/packages/fluxer-core/src/structures/Guild.test.ts +0 -189
  235. package/core/packages/fluxer-core/src/structures/Guild.ts +0 -734
  236. package/core/packages/fluxer-core/src/structures/GuildBan.ts +0 -35
  237. package/core/packages/fluxer-core/src/structures/GuildEmoji.ts +0 -57
  238. package/core/packages/fluxer-core/src/structures/GuildMember.test.ts +0 -203
  239. package/core/packages/fluxer-core/src/structures/GuildMember.ts +0 -213
  240. package/core/packages/fluxer-core/src/structures/GuildMemberRoleManager.ts +0 -121
  241. package/core/packages/fluxer-core/src/structures/GuildSticker.ts +0 -56
  242. package/core/packages/fluxer-core/src/structures/Invite.test.ts +0 -103
  243. package/core/packages/fluxer-core/src/structures/Invite.ts +0 -121
  244. package/core/packages/fluxer-core/src/structures/Message.test.ts +0 -109
  245. package/core/packages/fluxer-core/src/structures/Message.ts +0 -397
  246. package/core/packages/fluxer-core/src/structures/MessageReaction.ts +0 -72
  247. package/core/packages/fluxer-core/src/structures/PartialMessage.ts +0 -12
  248. package/core/packages/fluxer-core/src/structures/Role.test.ts +0 -77
  249. package/core/packages/fluxer-core/src/structures/Role.ts +0 -112
  250. package/core/packages/fluxer-core/src/structures/User.test.ts +0 -110
  251. package/core/packages/fluxer-core/src/structures/User.ts +0 -109
  252. package/core/packages/fluxer-core/src/structures/Webhook.test.ts +0 -109
  253. package/core/packages/fluxer-core/src/structures/Webhook.ts +0 -258
  254. package/core/packages/fluxer-core/src/util/Constants.test.ts +0 -16
  255. package/core/packages/fluxer-core/src/util/Constants.ts +0 -7
  256. package/core/packages/fluxer-core/src/util/Events.ts +0 -46
  257. package/core/packages/fluxer-core/src/util/MessageCollector.ts +0 -87
  258. package/core/packages/fluxer-core/src/util/Options.ts +0 -33
  259. package/core/packages/fluxer-core/src/util/ReactionCollector.ts +0 -116
  260. package/core/packages/fluxer-core/src/util/cdn.test.ts +0 -108
  261. package/core/packages/fluxer-core/src/util/cdn.ts +0 -130
  262. package/core/packages/fluxer-core/src/util/guildUtils.ts +0 -33
  263. package/core/packages/fluxer-core/src/util/messageUtils.test.ts +0 -74
  264. package/core/packages/fluxer-core/src/util/messageUtils.ts +0 -119
  265. package/core/packages/fluxer-core/src/util/permissions.test.ts +0 -95
  266. package/core/packages/fluxer-core/src/util/permissions.ts +0 -43
  267. package/core/packages/fluxer-core/tsconfig.json +0 -9
  268. package/core/packages/fluxer-core/tsup.config.ts +0 -9
  269. package/core/packages/fluxer-core/vitest.config.ts +0 -9
  270. package/core/packages/rest/package.json +0 -52
  271. package/core/packages/rest/src/REST.test.ts +0 -64
  272. package/core/packages/rest/src/REST.ts +0 -90
  273. package/core/packages/rest/src/RateLimitManager.test.ts +0 -71
  274. package/core/packages/rest/src/RateLimitManager.ts +0 -60
  275. package/core/packages/rest/src/RequestManager.test.ts +0 -87
  276. package/core/packages/rest/src/RequestManager.ts +0 -172
  277. package/core/packages/rest/src/errors/FluxerAPIError.test.ts +0 -57
  278. package/core/packages/rest/src/errors/FluxerAPIError.ts +0 -21
  279. package/core/packages/rest/src/errors/HTTPError.test.ts +0 -55
  280. package/core/packages/rest/src/errors/HTTPError.ts +0 -25
  281. package/core/packages/rest/src/errors/RateLimitError.test.ts +0 -41
  282. package/core/packages/rest/src/errors/RateLimitError.ts +0 -15
  283. package/core/packages/rest/src/errors/index.ts +0 -3
  284. package/core/packages/rest/src/index.ts +0 -6
  285. package/core/packages/rest/src/utils/constants.test.ts +0 -31
  286. package/core/packages/rest/src/utils/constants.ts +0 -5
  287. package/core/packages/rest/src/utils/files.test.ts +0 -37
  288. package/core/packages/rest/src/utils/files.ts +0 -75
  289. package/core/packages/rest/tsconfig.json +0 -9
  290. package/core/packages/rest/tsup.config.ts +0 -9
  291. package/core/packages/rest/vitest.config.ts +0 -9
  292. package/core/packages/types/package.json +0 -46
  293. package/core/packages/types/src/api/ban.ts +0 -8
  294. package/core/packages/types/src/api/channel.ts +0 -65
  295. package/core/packages/types/src/api/embed.ts +0 -82
  296. package/core/packages/types/src/api/emoji.ts +0 -12
  297. package/core/packages/types/src/api/errors.ts +0 -68
  298. package/core/packages/types/src/api/gateway.ts +0 -14
  299. package/core/packages/types/src/api/guild.ts +0 -123
  300. package/core/packages/types/src/api/index.ts +0 -15
  301. package/core/packages/types/src/api/instance.ts +0 -32
  302. package/core/packages/types/src/api/interaction.ts +0 -26
  303. package/core/packages/types/src/api/invite.ts +0 -28
  304. package/core/packages/types/src/api/message.ts +0 -140
  305. package/core/packages/types/src/api/role.ts +0 -41
  306. package/core/packages/types/src/api/sticker.ts +0 -14
  307. package/core/packages/types/src/api/user.ts +0 -79
  308. package/core/packages/types/src/api/webhook.ts +0 -41
  309. package/core/packages/types/src/common/index.ts +0 -1
  310. package/core/packages/types/src/common/snowflake.test.ts +0 -9
  311. package/core/packages/types/src/common/snowflake.ts +0 -8
  312. package/core/packages/types/src/gateway/events.ts +0 -189
  313. package/core/packages/types/src/gateway/index.ts +0 -3
  314. package/core/packages/types/src/gateway/opcodes.ts +0 -17
  315. package/core/packages/types/src/gateway/payloads.ts +0 -481
  316. package/core/packages/types/src/index.ts +0 -4
  317. package/core/packages/types/src/rest/index.ts +0 -1
  318. package/core/packages/types/src/rest/routes.test.ts +0 -169
  319. package/core/packages/types/src/rest/routes.ts +0 -109
  320. package/core/packages/types/tsconfig.json +0 -9
  321. package/core/packages/types/tsup.config.ts +0 -9
  322. package/core/packages/types/vitest.config.ts +0 -9
  323. package/core/packages/util/package.json +0 -51
  324. package/core/packages/util/src/BitField.test.ts +0 -96
  325. package/core/packages/util/src/BitField.ts +0 -105
  326. package/core/packages/util/src/MessageFlagsBitField.test.ts +0 -42
  327. package/core/packages/util/src/MessageFlagsBitField.ts +0 -20
  328. package/core/packages/util/src/PermissionsBitField.test.ts +0 -79
  329. package/core/packages/util/src/PermissionsBitField.ts +0 -97
  330. package/core/packages/util/src/SnowflakeUtil.test.ts +0 -69
  331. package/core/packages/util/src/SnowflakeUtil.ts +0 -65
  332. package/core/packages/util/src/UserFlagsBitField.test.ts +0 -39
  333. package/core/packages/util/src/UserFlagsBitField.ts +0 -48
  334. package/core/packages/util/src/deprecation.test.ts +0 -44
  335. package/core/packages/util/src/deprecation.ts +0 -28
  336. package/core/packages/util/src/emojiShortcodes.generated.ts +0 -5
  337. package/core/packages/util/src/emojiShortcodes.test.ts +0 -41
  338. package/core/packages/util/src/emojiShortcodes.ts +0 -22
  339. package/core/packages/util/src/formatters.test.ts +0 -65
  340. package/core/packages/util/src/formatters.ts +0 -35
  341. package/core/packages/util/src/index.ts +0 -34
  342. package/core/packages/util/src/resolvers.test.ts +0 -198
  343. package/core/packages/util/src/resolvers.ts +0 -127
  344. package/core/packages/util/src/tenorUtils.test.ts +0 -75
  345. package/core/packages/util/src/tenorUtils.ts +0 -86
  346. package/core/packages/util/tsconfig.json +0 -9
  347. package/core/packages/util/tsup.config.ts +0 -9
  348. package/core/packages/util/vitest.config.ts +0 -9
  349. package/core/packages/voice/README.md +0 -42
  350. package/core/packages/voice/package.json +0 -67
  351. package/core/packages/voice/src/LiveKitRtcConnection.receive.test.ts +0 -24
  352. package/core/packages/voice/src/LiveKitRtcConnection.ts +0 -1767
  353. package/core/packages/voice/src/VoiceConnection.ts +0 -413
  354. package/core/packages/voice/src/VoiceManager.receive.test.ts +0 -61
  355. package/core/packages/voice/src/VoiceManager.test.ts +0 -44
  356. package/core/packages/voice/src/VoiceManager.ts +0 -503
  357. package/core/packages/voice/src/exports.test.ts +0 -38
  358. package/core/packages/voice/src/index.ts +0 -51
  359. package/core/packages/voice/src/livekit.test.ts +0 -48
  360. package/core/packages/voice/src/livekit.ts +0 -33
  361. package/core/packages/voice/src/mp4box.d.ts +0 -32
  362. package/core/packages/voice/src/opusUtils.test.ts +0 -29
  363. package/core/packages/voice/src/opusUtils.ts +0 -86
  364. package/core/packages/voice/src/streamPreviewPlaceholder.test.ts +0 -16
  365. package/core/packages/voice/src/streamPreviewPlaceholder.ts +0 -8
  366. package/core/packages/voice/src/ws.d.ts +0 -1
  367. package/core/packages/voice/tsconfig.json +0 -5
  368. package/core/packages/voice/tsup.config.ts +0 -10
  369. package/core/packages/voice/vitest.config.ts +0 -9
  370. package/core/packages/ws/package.json +0 -52
  371. package/core/packages/ws/src/WebSocketManager.ts +0 -130
  372. package/core/packages/ws/src/WebSocketShard.ts +0 -296
  373. package/core/packages/ws/src/index.ts +0 -12
  374. package/core/packages/ws/src/utils/constants.test.ts +0 -46
  375. package/core/packages/ws/src/utils/constants.ts +0 -22
  376. package/core/packages/ws/src/utils/getWebSocket.ts +0 -55
  377. package/core/packages/ws/src/ws.d.ts +0 -10
  378. package/core/packages/ws/tsconfig.json +0 -9
  379. package/core/packages/ws/tsup.config.ts +0 -9
  380. package/core/pnpm-lock.yaml +0 -7033
  381. package/core/pnpm-workspace.yaml +0 -4
  382. package/core/scripts/generate-ai-rag.ts +0 -240
  383. package/core/scripts/generate-docs.ts +0 -143
  384. package/core/scripts/generate-emoji-shortcodes.ts +0 -58
  385. package/core/scripts/generate-types.ts +0 -6
  386. package/core/scripts/publish-ordered.js +0 -63
  387. package/core/scripts/test-cjs-require.mjs +0 -43
  388. package/core/scripts/test-esm-imports.mjs +0 -42
  389. package/core/scripts/test-package-exports.mjs +0 -98
  390. package/core/scripts/test-smoke.mjs +0 -103
  391. package/core/tsconfig.json +0 -18
  392. package/core/turbo.json +0 -30
  393. package/core/vitest.config.ts +0 -17
  394. package/core/wrangler.jsonc +0 -9
@@ -1,119 +0,0 @@
1
- import { APIEmbed } from '@erinjs/types';
2
- import { EmbedBuilder } from '@erinjs/builders';
3
-
4
- /** Resolved file data (after URL fetch). Used internally by REST layer. */
5
- export interface ResolvedMessageFile {
6
- name: string;
7
- data: Blob | ArrayBuffer | Uint8Array | Buffer;
8
- filename?: string;
9
- }
10
-
11
- /** File data for message attachment uploads. Use `data` for buffers or `url` to fetch from a URL. */
12
- export type MessageFileData =
13
- | {
14
- name: string;
15
- data: Blob | ArrayBuffer | Uint8Array | Buffer;
16
- filename?: string;
17
- }
18
- | {
19
- name: string;
20
- url: string;
21
- filename?: string;
22
- };
23
-
24
- const FILE_FETCH_TIMEOUT_MS = 30_000;
25
-
26
- /** Resolve files: fetch URLs to buffers, pass through data as-is. */
27
- export async function resolveMessageFiles(
28
- files: MessageFileData[],
29
- ): Promise<ResolvedMessageFile[]> {
30
- const result: ResolvedMessageFile[] = [];
31
- for (let i = 0; i < files.length; i++) {
32
- const f = files[i]!;
33
- const filename = f.filename ?? f.name;
34
- if ('url' in f && f.url) {
35
- if (!URL.canParse(f.url)) {
36
- throw new Error(`Invalid file URL at index ${i}: ${f.url}`);
37
- }
38
- const res = await fetch(f.url, {
39
- signal: AbortSignal.timeout(FILE_FETCH_TIMEOUT_MS),
40
- });
41
- if (!res.ok) {
42
- throw new Error(`Failed to fetch file from ${f.url}: ${res.status} ${res.statusText}`);
43
- }
44
- const data = await res.arrayBuffer();
45
- result.push({ name: f.name, data, filename });
46
- } else if ('data' in f && f.data != null) {
47
- result.push({ name: f.name, data: f.data, filename });
48
- } else {
49
- throw new Error(`File at index ${i} must have either "data" or "url"`);
50
- }
51
- }
52
- return result;
53
- }
54
-
55
- /** Attachment metadata for file uploads (id matches FormData index). */
56
- export interface MessageAttachmentMeta {
57
- id: number;
58
- filename: string;
59
- title?: string | null;
60
- description?: string | null;
61
- /** MessageAttachmentFlags: IS_SPOILER (8), CONTAINS_EXPLICIT_MEDIA (16), IS_ANIMATED (32) */
62
- flags?: number;
63
- }
64
-
65
- /** Options for sending a message (content, embeds, files). Used by Message.send, Channel.send, ChannelManager.send.
66
- * EmbedBuilder instances are auto-converted to API format—no need to call .toJSON().
67
- */
68
- export type MessageSendOptions = {
69
- content?: string;
70
- /** EmbedBuilder instances are auto-converted; raw APIEmbed also supported. */
71
- embeds?: (APIEmbed | EmbedBuilder)[];
72
- /** File attachments. When present, request uses multipart/form-data. */
73
- files?: MessageFileData[];
74
- /** Attachment metadata for files (id = index). Use when files are provided. */
75
- attachments?: MessageAttachmentMeta[];
76
- /** Message flags (e.g. MessageFlags.SuppressNotifications for reply without ping). */
77
- flags?: number;
78
- };
79
-
80
- /** API-ready body from MessageSendOptions or text content (serializes EmbedBuilder, includes attachments when files present). */
81
- export interface SendBodyResult {
82
- content?: string;
83
- embeds?: APIEmbed[];
84
- attachments?: Array<{
85
- id: number;
86
- filename: string;
87
- title?: string | null;
88
- description?: string | null;
89
- flags?: number;
90
- }>;
91
- /** Message flags (e.g. SuppressNotifications for reply without ping). */
92
- flags?: number;
93
- }
94
-
95
- /** Build API-ready body from MessageSendOptions or text content (serializes EmbedBuilder to APIEmbed). */
96
- export function buildSendBody(options: string | MessageSendOptions): SendBodyResult {
97
- const body = typeof options === 'string' ? { content: options } : options;
98
- const result: SendBodyResult = {};
99
- if (body.content !== undefined) result.content = body.content;
100
- if (body.embeds?.length) {
101
- result.embeds = body.embeds.map((e) => (e instanceof EmbedBuilder ? e.toJSON() : e));
102
- }
103
- if (body.files?.length && body.attachments) {
104
- result.attachments = body.attachments.map((a) => ({
105
- id: a.id,
106
- filename: a.filename,
107
- ...(a.title != null && { title: a.title }),
108
- ...(a.description != null && { description: a.description }),
109
- ...(a.flags != null && { flags: a.flags }),
110
- }));
111
- } else if (body.files?.length) {
112
- result.attachments = body.files.map((f, i) => ({
113
- id: i,
114
- filename: f.filename ?? f.name,
115
- }));
116
- }
117
- if (body.flags !== undefined) result.flags = body.flags;
118
- return result;
119
- }
@@ -1,95 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { computePermissions, hasPermission } from './permissions.js';
3
- import { OverwriteType } from '@erinjs/types';
4
- import { PermissionFlags } from '@erinjs/util';
5
-
6
- describe('hasPermission', () => {
7
- it('returns true when Administrator is set (grants all permissions)', () => {
8
- expect(hasPermission(PermissionFlags.Administrator, PermissionFlags.Administrator)).toBe(true);
9
- expect(hasPermission(PermissionFlags.Administrator, PermissionFlags.SendMessages)).toBe(true);
10
- });
11
-
12
- it('returns true when specific permission is set', () => {
13
- expect(hasPermission(PermissionFlags.SendMessages, PermissionFlags.SendMessages)).toBe(true);
14
- });
15
-
16
- it('returns false when permission not set', () => {
17
- expect(hasPermission(0n, PermissionFlags.SendMessages)).toBe(false);
18
- expect(hasPermission(0n, PermissionFlags.Administrator)).toBe(false);
19
- });
20
-
21
- it('returns false when Administrator not set but other permission is', () => {
22
- expect(hasPermission(PermissionFlags.BanMembers, PermissionFlags.Administrator)).toBe(false);
23
- });
24
- });
25
-
26
- describe('computePermissions', () => {
27
- it('returns base permissions when no overwrites', () => {
28
- const base = PermissionFlags.SendMessages | PermissionFlags.ViewChannel;
29
- const perms = computePermissions(base, [], [], 'user1', false);
30
- expect(hasPermission(perms, PermissionFlags.SendMessages)).toBe(true);
31
- expect(hasPermission(perms, PermissionFlags.ViewChannel)).toBe(true);
32
- expect(hasPermission(perms, PermissionFlags.BanMembers)).toBe(false);
33
- });
34
-
35
- it('returns full permissions when owner (including high-bit Fluxer perms)', () => {
36
- const perms = computePermissions(0n, [], [], 'user1', true);
37
- expect(hasPermission(perms, PermissionFlags.SendMessages)).toBe(true);
38
- expect(hasPermission(perms, PermissionFlags.BanMembers)).toBe(true);
39
- expect(hasPermission(perms, BigInt(PermissionFlags.ManageRoles))).toBe(true);
40
- expect(hasPermission(perms, BigInt(PermissionFlags.PinMessages))).toBe(true);
41
- expect(hasPermission(perms, BigInt(PermissionFlags.BypassSlowmode))).toBe(true);
42
- });
43
-
44
- it('applies role deny overwrite', () => {
45
- const base = PermissionFlags.SendMessages;
46
- const overwrites = [{ id: 'role1', type: OverwriteType.Role, allow: '0', deny: '2048' }];
47
- const perms = computePermissions(base, overwrites, ['role1'], 'user1', false);
48
- expect(hasPermission(perms, PermissionFlags.SendMessages)).toBe(false);
49
- });
50
-
51
- it('applies role allow overwrite', () => {
52
- const base = 0n;
53
- const overwrites = [{ id: 'role1', type: OverwriteType.Role, allow: '2048', deny: '0' }];
54
- const perms = computePermissions(base, overwrites, ['role1'], 'user1', false);
55
- expect(hasPermission(perms, PermissionFlags.SendMessages)).toBe(true);
56
- });
57
-
58
- it('applies member overwrite', () => {
59
- const base = 0n;
60
- const overwrites = [{ id: 'user1', type: OverwriteType.Member, allow: '2048', deny: '0' }];
61
- const perms = computePermissions(base, overwrites, [], 'user1', false);
62
- expect(hasPermission(perms, PermissionFlags.SendMessages)).toBe(true);
63
- });
64
-
65
- it('member overwrite can deny base permission', () => {
66
- const base = PermissionFlags.SendMessages;
67
- const overwrites = [{ id: 'user1', type: OverwriteType.Member, allow: '0', deny: '2048' }];
68
- const perms = computePermissions(base, overwrites, [], 'user1', false);
69
- expect(hasPermission(perms, PermissionFlags.SendMessages)).toBe(false);
70
- });
71
-
72
- it('ignores overwrites that do not apply to member', () => {
73
- const base = PermissionFlags.SendMessages;
74
- const overwrites = [{ id: 'role99', type: OverwriteType.Role, allow: '0', deny: '2048' }];
75
- const perms = computePermissions(base, overwrites, ['role1'], 'user1', false);
76
- expect(hasPermission(perms, PermissionFlags.SendMessages)).toBe(true);
77
- });
78
-
79
- it('applies both role and member overwrites (member overwrite wins for same permission)', () => {
80
- const base = 0n;
81
- const overwrites = [
82
- { id: 'role1', type: OverwriteType.Role, allow: '2048', deny: '0' },
83
- { id: 'user1', type: OverwriteType.Member, allow: '0', deny: '2048' },
84
- ];
85
- const perms = computePermissions(base, overwrites, ['role1'], 'user1', false);
86
- // Member deny overwrites role allow
87
- expect(hasPermission(perms, PermissionFlags.SendMessages)).toBe(false);
88
- });
89
-
90
- it('empty overwrites returns base unchanged', () => {
91
- const base = 2048n;
92
- const perms = computePermissions(base, [], [], 'user1', false);
93
- expect(perms).toBe(base);
94
- });
95
- });
@@ -1,43 +0,0 @@
1
- import { OverwriteType, type APIChannelOverwrite } from '@erinjs/types';
2
- import { ALL_PERMISSIONS_BIGINT, PermissionFlags } from '@erinjs/util';
3
-
4
- /**
5
- * Compute the effective permission bitfield for a member in a channel.
6
- * Applies role permissions and channel overwrites.
7
- * Guild owner always receives all permissions (matches Fluxer API behavior).
8
- * @param basePermissions - Combined permissions from all member roles (guild base)
9
- * @param overwrites - Channel permission overwrites
10
- * @param memberRoles - Role IDs the member has
11
- * @param memberId - Member user ID (for member-specific overwrites)
12
- * @param isOwner - Whether the member is the guild owner (gets all permissions)
13
- * @returns Effective permission bitfield as bigint
14
- */
15
- export function computePermissions(
16
- basePermissions: bigint,
17
- overwrites: APIChannelOverwrite[],
18
- memberRoles: string[],
19
- memberId: string,
20
- isOwner: boolean,
21
- ): bigint {
22
- if (isOwner) return ALL_PERMISSIONS_BIGINT;
23
- let perms = basePermissions;
24
- for (const overwrite of overwrites ?? []) {
25
- const applies =
26
- (overwrite.type === OverwriteType.Role && memberRoles.includes(overwrite.id)) ||
27
- (overwrite.type === OverwriteType.Member && overwrite.id === memberId);
28
- if (!applies) continue;
29
- const allow = BigInt(overwrite.allow || '0');
30
- const deny = BigInt(overwrite.deny || '0');
31
- perms = (perms & ~deny) | allow;
32
- }
33
- return (perms & PermissionFlags.Administrator) !== 0n ? ALL_PERMISSIONS_BIGINT : perms;
34
- }
35
-
36
- /**
37
- * Check if a permission bitfield has a specific permission.
38
- * Administrator (bit 3) implies all permissions per Fluxer/Discord convention.
39
- */
40
- export function hasPermission(bitfield: bigint, permission: bigint): boolean {
41
- if ((bitfield & PermissionFlags.Administrator) !== 0n) return true;
42
- return (bitfield & permission) !== 0n;
43
- }
@@ -1,9 +0,0 @@
1
- {
2
- "extends": "../../tsconfig.json",
3
- "compilerOptions": {
4
- "outDir": "./dist",
5
- "rootDir": "./src",
6
- "noEmit": false
7
- },
8
- "include": ["src/**/*"]
9
- }
@@ -1,9 +0,0 @@
1
- import { defineConfig } from 'tsup';
2
-
3
- export default defineConfig({
4
- entry: ['src/index.ts'],
5
- format: ['cjs', 'esm'],
6
- dts: true,
7
- splitting: false,
8
- clean: true,
9
- });
@@ -1,9 +0,0 @@
1
- import { defineConfig } from 'vitest/config';
2
-
3
- export default defineConfig({
4
- test: {
5
- globals: true,
6
- environment: 'node',
7
- include: ['src/**/*.test.ts'],
8
- },
9
- });
@@ -1,52 +0,0 @@
1
- {
2
- "name": "@erinjs/rest",
3
- "publishConfig": {
4
- "access": "public"
5
- },
6
- "version": "1.2.4",
7
- "description": "REST client for the Fluxer API",
8
- "repository": {
9
- "type": "git",
10
- "url": "https://github.com/blstmo-abandoned-us-for-the-milk/core.git",
11
- "directory": "packages/rest"
12
- },
13
- "bugs": "https://github.com/blstmo-abandoned-us-for-the-milk/core/issues",
14
- "homepage": "https://github.com/blstmo-abandoned-us-for-the-milk/core#readme",
15
- "keywords": [
16
- "fluxer",
17
- "api",
18
- "rest",
19
- "sdk",
20
- "bot"
21
- ],
22
- "license": "Apache-2.0",
23
- "main": "./dist/index.js",
24
- "module": "./dist/index.mjs",
25
- "types": "./dist/index.d.ts",
26
- "exports": {
27
- ".": {
28
- "types": "./dist/index.d.ts",
29
- "import": "./dist/index.mjs",
30
- "require": "./dist/index.js"
31
- }
32
- },
33
- "files": [
34
- "dist"
35
- ],
36
- "scripts": {
37
- "build": "tsup",
38
- "clean": "rm -rf dist",
39
- "lint": "eslint src --max-warnings 0 --config ../../eslint.config.js",
40
- "lint:fix": "eslint src --fix --config ../../eslint.config.js",
41
- "test": "vitest run --passWithNoTests",
42
- "test:coverage": "vitest run --coverage --passWithNoTests"
43
- },
44
- "dependencies": {
45
- "@erinjs/types": "workspace:*"
46
- },
47
- "devDependencies": {
48
- "@types/node": "^20.0.0",
49
- "tsup": "^8.3.0",
50
- "typescript": "^5.6.0"
51
- }
52
- }
@@ -1,64 +0,0 @@
1
- import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
2
- import { REST } from './REST.js';
3
- import { Routes } from '@erinjs/types';
4
-
5
- describe('REST', () => {
6
- let fetchMock: ReturnType<typeof vi.fn>;
7
-
8
- beforeEach(() => {
9
- fetchMock = vi.fn();
10
- vi.stubGlobal('fetch', fetchMock);
11
- });
12
-
13
- afterEach(() => {
14
- vi.unstubAllGlobals();
15
- });
16
-
17
- it('constructor and setToken', () => {
18
- const rest = new REST();
19
- expect(rest.token).toBeNull();
20
- rest.setToken('abc123');
21
- expect(rest.token).toBe('abc123');
22
- rest.setToken(null);
23
- expect(rest.token).toBeNull();
24
- });
25
-
26
- it('setToken returns this for chaining', () => {
27
- const rest = new REST();
28
- expect(rest.setToken('x')).toBe(rest);
29
- });
30
-
31
- it('get delegates to requestManager', async () => {
32
- const rest = new REST();
33
- fetchMock.mockResolvedValueOnce({
34
- ok: true,
35
- status: 200,
36
- text: () => Promise.resolve('{"id":"1"}'),
37
- headers: new Headers(),
38
- });
39
- const result = await rest.get('/channels/1');
40
- expect(result).toEqual({ id: '1' });
41
- });
42
-
43
- it('post sends body', async () => {
44
- const rest = new REST();
45
- fetchMock.mockResolvedValueOnce({
46
- ok: true,
47
- status: 200,
48
- text: () => Promise.resolve('{}'),
49
- headers: new Headers(),
50
- });
51
- await rest.post('/channels', { body: { name: 'test' } });
52
- expect(fetchMock).toHaveBeenCalledWith(
53
- expect.any(String),
54
- expect.objectContaining({
55
- method: 'POST',
56
- body: '{"name":"test"}',
57
- }),
58
- );
59
- });
60
-
61
- it('Routes is exposed', () => {
62
- expect(REST.Routes).toBe(Routes);
63
- });
64
- });
@@ -1,90 +0,0 @@
1
- import { EventEmitter } from 'events';
2
- import { RequestManager } from './RequestManager.js';
3
- import { Routes } from '@erinjs/types';
4
-
5
- /** Options for the REST client. */
6
- export interface RESTOptions {
7
- api?: string;
8
- version?: string;
9
- authPrefix?: 'Bot' | 'Bearer';
10
- timeout?: number;
11
- retries?: number;
12
- userAgent?: string;
13
- }
14
-
15
- /** HTTP client for the Fluxer API. Used by Client internally. */
16
- export class REST extends EventEmitter {
17
- private readonly requestManager: RequestManager;
18
- private _token: string | null = null;
19
-
20
- constructor(options: RESTOptions = {}) {
21
- super();
22
- this.requestManager = new RequestManager({
23
- api: options.api ?? 'https://api.fluxer.app',
24
- version: options.version ?? '1',
25
- authPrefix: options.authPrefix ?? 'Bot',
26
- timeout: options.timeout ?? 15000,
27
- retries: options.retries ?? 3,
28
- userAgent: options.userAgent ?? 'erin.js',
29
- });
30
- }
31
-
32
- /** Set the bot token for authenticated requests. */
33
- setToken(token: string | null): this {
34
- this._token = token;
35
- this.requestManager.setToken(token);
36
- return this;
37
- }
38
-
39
- /** Current bot token, or null. */
40
- get token(): string | null {
41
- return this._token;
42
- }
43
-
44
- /** Send a GET request. */
45
- async get<T>(route: string, options?: { auth?: boolean }): Promise<T> {
46
- return this.requestManager.request<T>('GET', route, { auth: options?.auth });
47
- }
48
-
49
- /** Send a POST request. */
50
- async post<T>(
51
- route: string,
52
- options?: {
53
- body?: unknown;
54
- auth?: boolean;
55
- files?: Array<{ name: string; data: Blob | ArrayBuffer | Uint8Array; filename?: string }>;
56
- },
57
- ): Promise<T> {
58
- return this.requestManager.request<T>('POST', route, {
59
- body: options?.body,
60
- auth: options?.auth,
61
- files: options?.files,
62
- });
63
- }
64
-
65
- /** Send a PATCH request. */
66
- async patch<T>(route: string, options?: { body?: unknown; auth?: boolean }): Promise<T> {
67
- return this.requestManager.request<T>('PATCH', route, {
68
- body: options?.body,
69
- auth: options?.auth,
70
- });
71
- }
72
-
73
- /** Send a PUT request. */
74
- async put<T>(route: string, options?: { body?: unknown; auth?: boolean }): Promise<T> {
75
- return this.requestManager.request<T>('PUT', route, {
76
- body: options?.body,
77
- auth: options?.auth,
78
- });
79
- }
80
-
81
- /** Send a DELETE request. */
82
- async delete<T>(route: string, options?: { auth?: boolean }): Promise<T> {
83
- return this.requestManager.request<T>('DELETE', route, { auth: options?.auth });
84
- }
85
-
86
- /** Route helpers (from @erinjs/types) for building paths. */
87
- static get Routes(): typeof Routes {
88
- return Routes;
89
- }
90
- }
@@ -1,71 +0,0 @@
1
- import { describe, it, expect, beforeEach } from 'vitest';
2
- import { RateLimitManager } from './RateLimitManager.js';
3
-
4
- describe('RateLimitManager', () => {
5
- let manager: RateLimitManager;
6
-
7
- beforeEach(() => {
8
- manager = new RateLimitManager();
9
- });
10
-
11
- it('getBucket returns undefined for unknown route', () => {
12
- expect(manager.getBucket('unknown')).toBeUndefined();
13
- });
14
-
15
- it('setBucket and getBucket', () => {
16
- manager.setBucket('/channels/:id', 5, 3, Date.now() + 60000);
17
- const bucket = manager.getBucket('/channels/:id');
18
- expect(bucket).toEqual({ limit: 5, remaining: 3, resetAt: expect.any(Number) });
19
- });
20
-
21
- it('getWaitTime returns 0 when no limit', () => {
22
- expect(manager.getWaitTime('any')).toBe(0);
23
- });
24
-
25
- it('getWaitTime returns wait when bucket exhausted', () => {
26
- const resetAt = Date.now() + 5000;
27
- manager.setBucket('route', 5, 0, resetAt);
28
- const wait = manager.getWaitTime('route');
29
- expect(wait).toBeGreaterThan(0);
30
- expect(wait).toBeLessThanOrEqual(5000);
31
- });
32
-
33
- it('getWaitTime returns 0 when resetAt in past', () => {
34
- manager.setBucket('route', 5, 0, Date.now() - 1000);
35
- expect(manager.getWaitTime('route')).toBe(0);
36
- });
37
-
38
- it('setGlobalReset and getGlobalReset', () => {
39
- const resetAt = Date.now() + 10000;
40
- manager.setGlobalReset(resetAt);
41
- expect(manager.getGlobalReset()).toBe(resetAt);
42
- });
43
-
44
- it('getWaitTime considers global reset', () => {
45
- const resetAt = Date.now() + 3000;
46
- manager.setGlobalReset(resetAt);
47
- const wait = manager.getWaitTime('any-route');
48
- expect(wait).toBeGreaterThan(0);
49
- });
50
-
51
- it('updateFromHeaders parses X-RateLimit headers', () => {
52
- const resetSeconds = Math.floor(Date.now() / 1000) + 60;
53
- const headers = new Headers({
54
- 'X-RateLimit-Limit': '5',
55
- 'X-RateLimit-Remaining': '2',
56
- 'X-RateLimit-Reset': String(resetSeconds),
57
- });
58
- manager.updateFromHeaders('/channels/123', headers);
59
- const bucket = manager.getBucket('/channels/123');
60
- expect(bucket?.limit).toBe(5);
61
- expect(bucket?.remaining).toBe(2);
62
- });
63
-
64
- it('updateFromHeaders handles Retry-After', () => {
65
- const headers = new Headers({ 'Retry-After': '10' });
66
- manager.updateFromHeaders('/global', headers);
67
- const bucket = manager.getBucket('/global');
68
- expect(bucket?.limit).toBe(1);
69
- expect(bucket?.remaining).toBe(0);
70
- });
71
- });
@@ -1,60 +0,0 @@
1
- /**
2
- * Tracks rate limit state per bucket (route hash).
3
- * Delays requests when limit is exceeded.
4
- */
5
- export interface RateLimitState {
6
- limit: number;
7
- remaining: number;
8
- resetAt: number;
9
- }
10
-
11
- export class RateLimitManager {
12
- private buckets = new Map<string, RateLimitState>();
13
- private globalResetAt = 0;
14
-
15
- getBucket(route: string): RateLimitState | undefined {
16
- return this.buckets.get(route);
17
- }
18
-
19
- setBucket(route: string, limit: number, remaining: number, resetAt: number): void {
20
- this.buckets.set(route, { limit, remaining, resetAt });
21
- }
22
-
23
- setGlobalReset(resetAt: number): void {
24
- this.globalResetAt = resetAt;
25
- }
26
-
27
- getGlobalReset(): number {
28
- return this.globalResetAt;
29
- }
30
-
31
- /** Returns ms to wait before we can send again (0 if no wait). */
32
- getWaitTime(route: string): number {
33
- const now = Date.now();
34
- const globalWait = this.globalResetAt > now ? this.globalResetAt - now : 0;
35
- const bucket = this.buckets.get(route);
36
- const bucketWait =
37
- bucket && bucket.remaining <= 0 && bucket.resetAt > now ? bucket.resetAt - now : 0;
38
- return Math.max(globalWait, bucketWait);
39
- }
40
-
41
- /** Parse rate limit headers and update state. */
42
- updateFromHeaders(route: string, headers: Headers): void {
43
- const limit = headers.get('X-RateLimit-Limit');
44
- const remaining = headers.get('X-RateLimit-Remaining');
45
- const reset = headers.get('X-RateLimit-Reset');
46
- if (limit !== null && remaining !== null && reset !== null) {
47
- const resetMs = parseInt(reset, 10);
48
- if (!Number.isNaN(resetMs)) {
49
- const resetAt = resetMs * 1000;
50
- this.setBucket(route, parseInt(limit, 10), parseInt(remaining, 10), resetAt);
51
- }
52
- }
53
- const retryAfter = headers.get('Retry-After');
54
- if (retryAfter !== null) {
55
- const sec = parseInt(retryAfter, 10);
56
- const resetAt = Date.now() + (Number.isNaN(sec) ? 0 : sec * 1000);
57
- this.setBucket(route, 1, 0, resetAt);
58
- }
59
- }
60
- }