@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,87 +0,0 @@
1
- import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
2
- import { RequestManager } from './RequestManager.js';
3
- import { HTTPError, FluxerAPIError } from './errors/index.js';
4
-
5
- describe('RequestManager', () => {
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 uses defaults', () => {
18
- const rm = new RequestManager({});
19
- expect(rm.baseUrl).toBe('https://api.fluxer.app/v1');
20
- });
21
-
22
- it('constructor accepts overrides', () => {
23
- const rm = new RequestManager({ api: 'https://test', version: '2' });
24
- expect(rm.baseUrl).toBe('https://test/v2');
25
- });
26
-
27
- it('request succeeds with JSON body', async () => {
28
- const rm = new RequestManager({ retries: 0 });
29
- fetchMock.mockResolvedValueOnce({
30
- ok: true,
31
- status: 200,
32
- text: () => Promise.resolve('{"id":"123"}'),
33
- headers: new Headers(),
34
- });
35
- const result = await rm.request('GET', '/channels/123');
36
- expect(result).toEqual({ id: '123' });
37
- });
38
-
39
- it('request returns undefined for 204', async () => {
40
- const rm = new RequestManager({ retries: 0 });
41
- fetchMock.mockResolvedValueOnce({
42
- ok: true,
43
- status: 204,
44
- text: () => Promise.resolve(''),
45
- headers: new Headers(),
46
- });
47
- const result = await rm.request('DELETE', '/channels/123');
48
- expect(result).toBeUndefined();
49
- });
50
-
51
- it('request throws FluxerAPIError for non-ok with JSON body', async () => {
52
- const rm = new RequestManager({ retries: 0 });
53
- fetchMock.mockResolvedValueOnce({
54
- ok: false,
55
- status: 404,
56
- text: () => Promise.resolve('{"code":10003,"message":"Unknown Channel"}'),
57
- headers: new Headers(),
58
- });
59
- await expect(rm.request('GET', '/channels/999')).rejects.toThrow(FluxerAPIError);
60
- });
61
-
62
- it('request throws HTTPError for non-JSON error body', async () => {
63
- const rm = new RequestManager({ retries: 0 });
64
- fetchMock.mockResolvedValueOnce({
65
- ok: false,
66
- status: 500,
67
- text: () => Promise.resolve('Internal Server Error'),
68
- headers: new Headers(),
69
- });
70
- await expect(rm.request('GET', '/channels/1')).rejects.toThrow(HTTPError);
71
- });
72
-
73
- it('request uses full URL when route starts with http', async () => {
74
- const rm = new RequestManager({ retries: 0 });
75
- fetchMock.mockResolvedValueOnce({
76
- ok: true,
77
- status: 200,
78
- text: () => Promise.resolve('{}'),
79
- headers: new Headers(),
80
- });
81
- await rm.request('GET', 'https://cdn.example.com/asset/123');
82
- expect(fetchMock).toHaveBeenCalledWith(
83
- 'https://cdn.example.com/asset/123',
84
- expect.objectContaining({ method: 'GET' }),
85
- );
86
- });
87
- });
@@ -1,172 +0,0 @@
1
- import { RateLimitManager } from './RateLimitManager.js';
2
- import { FluxerAPIError, RateLimitError, HTTPError } from './errors/index.js';
3
- import { APIErrorBody, RateLimitErrorBody } from '@erinjs/types';
4
- import { buildFormData } from './utils/files.js';
5
-
6
- export interface RequestOptions {
7
- body?: unknown | FormData;
8
- headers?: Record<string, string>;
9
- files?: Array<{
10
- name: string;
11
- data: Blob | ArrayBuffer | Uint8Array | Buffer;
12
- filename?: string;
13
- }>;
14
- auth?: boolean;
15
- }
16
-
17
- export interface RestOptions {
18
- api: string;
19
- version: string;
20
- authPrefix: 'Bot' | 'Bearer';
21
- timeout: number;
22
- retries: number;
23
- userAgent: string;
24
- }
25
-
26
- export class RequestManager {
27
- private token: string | null = null;
28
- private readonly options: RestOptions;
29
- private readonly rateLimiter = new RateLimitManager();
30
-
31
- constructor(options: Partial<RestOptions>) {
32
- this.options = {
33
- api: options.api ?? 'https://api.fluxer.app',
34
- version: options.version ?? '1',
35
- authPrefix: options.authPrefix ?? 'Bot',
36
- timeout: options.timeout ?? 15000,
37
- retries: options.retries ?? 3,
38
- userAgent: options.userAgent ?? 'erin.js',
39
- };
40
- }
41
-
42
- setToken(token: string | null): void {
43
- this.token = token;
44
- }
45
-
46
- get baseUrl(): string {
47
- return `${this.options.api}/v${this.options.version}`;
48
- }
49
-
50
- /** Hash route for rate limit bucket (use path without ids for grouping). */
51
- private getRouteHash(route: string): string {
52
- return route.replace(/\d{17,19}/g, ':id');
53
- }
54
-
55
- private async waitForRateLimit(routeHash: string): Promise<void> {
56
- const wait = this.rateLimiter.getWaitTime(routeHash);
57
- if (wait > 0) await new Promise((r) => setTimeout(r, wait));
58
- }
59
-
60
- private buildHeaders(
61
- _route: string,
62
- options: RequestOptions,
63
- body: string | FormData | undefined,
64
- ): Record<string, string> {
65
- const headers: Record<string, string> = {
66
- 'User-Agent': this.options.userAgent,
67
- ...options.headers,
68
- };
69
- if (options.auth !== false && this.token) {
70
- headers['Authorization'] = `${this.options.authPrefix} ${this.token}`;
71
- }
72
- if (body !== undefined && !(body instanceof FormData)) {
73
- headers['Content-Type'] = 'application/json';
74
- }
75
- return headers;
76
- }
77
-
78
- async request<T>(method: string, route: string, options: RequestOptions = {}): Promise<T> {
79
- const routeHash = this.getRouteHash(route);
80
- const url = route.startsWith('http') ? route : `${this.baseUrl}${route}`;
81
-
82
- await this.waitForRateLimit(routeHash);
83
-
84
- let body: string | FormData | undefined;
85
- if (options.body !== undefined) {
86
- if (options.body instanceof FormData) {
87
- body = options.body;
88
- } else if (
89
- options.files?.length &&
90
- typeof options.body === 'object' &&
91
- options.body !== null
92
- ) {
93
- body = buildFormData(options.body as Record<string, unknown>, options.files);
94
- } else {
95
- body = JSON.stringify(options.body);
96
- }
97
- }
98
-
99
- const headers = this.buildHeaders(route, options, body);
100
-
101
- let lastError: Error | null = null;
102
- for (let attempt = 0; attempt <= this.options.retries; attempt++) {
103
- const controller = new AbortController();
104
- const timeoutId = setTimeout(() => controller.abort(), this.options.timeout);
105
- try {
106
- const response = await fetch(url, {
107
- method,
108
- headers,
109
- body,
110
- signal: controller.signal,
111
- });
112
-
113
- this.rateLimiter.updateFromHeaders(routeHash, response.headers);
114
-
115
- if (response.status === 429) {
116
- const data = (await response.json().catch(() => ({}))) as RateLimitErrorBody;
117
- const retryAfter =
118
- (data.retry_after ?? parseInt(response.headers.get('Retry-After') ?? '0', 10)) * 1000;
119
- this.rateLimiter.setBucket(routeHash, 1, 0, Date.now() + retryAfter);
120
- if (data.global) this.rateLimiter.setGlobalReset(Date.now() + retryAfter);
121
- throw new RateLimitError(
122
- {
123
- ...data,
124
- code: 'RATE_LIMITED',
125
- message: data.message ?? 'Rate limited',
126
- retry_after: data.retry_after ?? 0,
127
- },
128
- response.status,
129
- );
130
- }
131
-
132
- const text = await response.text();
133
- if (!response.ok) {
134
- let parsed: APIErrorBody;
135
- try {
136
- parsed = JSON.parse(text) as APIErrorBody;
137
- } catch {
138
- throw new HTTPError(response.status, text);
139
- }
140
- throw new FluxerAPIError(parsed, response.status);
141
- }
142
-
143
- if (response.status === 204 || text.length === 0) return undefined as T;
144
- return JSON.parse(text) as T;
145
- } catch (err) {
146
- const wrapped = err instanceof Error ? err : new Error(String(err));
147
- lastError =
148
- attempt > 0
149
- ? new Error(`Retry ${attempt} failed: ${wrapped.message}`, {
150
- cause: wrapped,
151
- })
152
- : wrapped;
153
- if (err instanceof RateLimitError && attempt < this.options.retries) {
154
- const retryMs = err.retryAfter * 1000;
155
- if (Number.isFinite(retryMs)) {
156
- await new Promise((r) => setTimeout(r, retryMs));
157
- continue;
158
- }
159
- }
160
- if (err instanceof FluxerAPIError || err instanceof HTTPError) throw err;
161
- if (attempt < this.options.retries) {
162
- await new Promise((r) => setTimeout(r, 500 * (attempt + 1)));
163
- continue;
164
- }
165
- throw lastError;
166
- } finally {
167
- clearTimeout(timeoutId);
168
- }
169
- }
170
- throw lastError ?? new Error('Request failed');
171
- }
172
- }
@@ -1,57 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { FluxerAPIError } from './FluxerAPIError.js';
3
-
4
- describe('FluxerAPIError', () => {
5
- it('creates error with message from body', () => {
6
- const err = new FluxerAPIError(
7
- { message: 'Channel not found', code: 'CHANNEL_NOT_FOUND' },
8
- 404,
9
- );
10
- expect(err.message).toBe('Channel not found');
11
- expect(err.name).toBe('FluxerAPIError');
12
- });
13
-
14
- it('stores code and statusCode', () => {
15
- const err = new FluxerAPIError({ message: 'Rate limited', code: 'RATE_LIMITED' }, 429);
16
- expect(err.code).toBe('RATE_LIMITED');
17
- expect(err.statusCode).toBe(429);
18
- });
19
-
20
- it('stores optional errors field', () => {
21
- const errors = { field: ['invalid value'] };
22
- const err = new FluxerAPIError(
23
- { message: 'Validation failed', code: 'VALIDATION_ERROR', errors },
24
- 400,
25
- );
26
- expect(err.errors).toEqual(errors);
27
- });
28
-
29
- it('isRetryable returns true for 429', () => {
30
- const err = new FluxerAPIError({ message: 'Rate limited', code: 'RATE_LIMITED' }, 429);
31
- expect(err.isRetryable).toBe(true);
32
- });
33
-
34
- it('isRetryable returns true for 5xx', () => {
35
- expect(new FluxerAPIError({ message: 'Server error', code: 'INTERNAL' }, 500).isRetryable).toBe(
36
- true,
37
- );
38
- expect(
39
- new FluxerAPIError({ message: 'Bad gateway', code: 'BAD_GATEWAY' }, 502).isRetryable,
40
- ).toBe(true);
41
- expect(
42
- new FluxerAPIError({ message: 'Unavailable', code: 'UNAVAILABLE' }, 503).isRetryable,
43
- ).toBe(true);
44
- });
45
-
46
- it('isRetryable returns false for 4xx (except 429)', () => {
47
- expect(new FluxerAPIError({ message: 'Not found', code: 'NOT_FOUND' }, 404).isRetryable).toBe(
48
- false,
49
- );
50
- expect(new FluxerAPIError({ message: 'Forbidden', code: 'FORBIDDEN' }, 403).isRetryable).toBe(
51
- false,
52
- );
53
- expect(
54
- new FluxerAPIError({ message: 'Bad request', code: 'BAD_REQUEST' }, 400).isRetryable,
55
- ).toBe(false);
56
- });
57
- });
@@ -1,21 +0,0 @@
1
- import { APIErrorBody } from '@erinjs/types';
2
-
3
- export class FluxerAPIError extends Error {
4
- readonly code: string;
5
- readonly statusCode: number;
6
- readonly errors?: APIErrorBody['errors'];
7
-
8
- constructor(body: APIErrorBody, statusCode: number) {
9
- super(body.message);
10
- this.name = 'FluxerAPIError';
11
- this.code = body.code;
12
- this.statusCode = statusCode;
13
- this.errors = body.errors;
14
- Object.setPrototypeOf(this, FluxerAPIError.prototype);
15
- }
16
-
17
- /** True if the error is retryable (429 rate limit, 5xx server errors). */
18
- get isRetryable(): boolean {
19
- return this.statusCode === 429 || (this.statusCode >= 500 && this.statusCode < 600);
20
- }
21
- }
@@ -1,55 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { HTTPError } from './HTTPError.js';
3
-
4
- describe('HTTPError', () => {
5
- it('creates error with status and body', () => {
6
- const err = new HTTPError(404, '{"error":"not found"}');
7
- expect(err.message).toContain('404');
8
- expect(err.message).toContain('not found');
9
- expect(err.name).toBe('HTTPError');
10
- });
11
-
12
- it('stores statusCode and body', () => {
13
- const err = new HTTPError(500, 'Internal Server Error');
14
- expect(err.statusCode).toBe(500);
15
- expect(err.body).toBe('Internal Server Error');
16
- });
17
-
18
- it('accepts null body', () => {
19
- const err = new HTTPError(502, null);
20
- expect(err.body).toBeNull();
21
- expect(err.message).toContain('502');
22
- });
23
-
24
- it('uses status hint when body is empty', () => {
25
- const err = new HTTPError(503, '');
26
- expect(err.message).toContain('Service Unavailable');
27
- });
28
-
29
- it('isRetryable returns true for 429', () => {
30
- const err = new HTTPError(429, 'Too Many Requests');
31
- expect(err.isRetryable).toBe(true);
32
- });
33
-
34
- it('isRetryable returns true for 5xx', () => {
35
- expect(new HTTPError(500, '').isRetryable).toBe(true);
36
- expect(new HTTPError(502, '').isRetryable).toBe(true);
37
- expect(new HTTPError(503, '').isRetryable).toBe(true);
38
- expect(new HTTPError(504, '').isRetryable).toBe(true);
39
- });
40
-
41
- it('isRetryable returns false for 4xx (except 429)', () => {
42
- expect(new HTTPError(400, '').isRetryable).toBe(false);
43
- expect(new HTTPError(403, '').isRetryable).toBe(false);
44
- expect(new HTTPError(404, '').isRetryable).toBe(false);
45
- });
46
-
47
- it('isRetryable returns false for 599 (upper boundary)', () => {
48
- expect(new HTTPError(599, '').isRetryable).toBe(true);
49
- });
50
-
51
- it('uses body when present over status hint', () => {
52
- const err = new HTTPError(502, 'Custom error body');
53
- expect(err.message).toContain('Custom error body');
54
- });
55
- });
@@ -1,25 +0,0 @@
1
- const STATUS_MESSAGES: Record<number, string> = {
2
- 502: 'Bad Gateway — Fluxer API may be temporarily unavailable.',
3
- 503: 'Service Unavailable — Fluxer API is down or overloaded. Try again later.',
4
- 504: 'Gateway Timeout — Fluxer API did not respond in time.',
5
- };
6
-
7
- export class HTTPError extends Error {
8
- readonly statusCode: number;
9
- readonly body: string | null;
10
-
11
- constructor(statusCode: number, body: string | null) {
12
- const hint = STATUS_MESSAGES[statusCode];
13
- const detail = body?.trim() || (hint ?? 'No body');
14
- super(`HTTP ${statusCode}: ${detail}`);
15
- this.name = 'HTTPError';
16
- this.statusCode = statusCode;
17
- this.body = body;
18
- Object.setPrototypeOf(this, HTTPError.prototype);
19
- }
20
-
21
- /** True if the error is retryable (429 rate limit, 5xx server errors). */
22
- get isRetryable(): boolean {
23
- return this.statusCode === 429 || (this.statusCode >= 500 && this.statusCode < 600);
24
- }
25
- }
@@ -1,41 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { RateLimitError } from './RateLimitError.js';
3
-
4
- describe('RateLimitError', () => {
5
- it('extends FluxerAPIError with retryAfter and global', () => {
6
- const err = new RateLimitError(
7
- { message: 'Rate limited', code: 'RATE_LIMITED', retry_after: 5 },
8
- 429,
9
- );
10
- expect(err.message).toBe('Rate limited');
11
- expect(err.name).toBe('RateLimitError');
12
- expect(err.statusCode).toBe(429);
13
- expect(err.retryAfter).toBe(5);
14
- expect(err.global).toBe(false);
15
- });
16
-
17
- it('sets global from body when true', () => {
18
- const err = new RateLimitError(
19
- { message: 'Global rate limit', code: 'RATE_LIMITED', retry_after: 10, global: true },
20
- 429,
21
- );
22
- expect(err.global).toBe(true);
23
- expect(err.retryAfter).toBe(10);
24
- });
25
-
26
- it('defaults global to false when omitted', () => {
27
- const err = new RateLimitError(
28
- { message: 'Limited', code: 'RATE_LIMITED', retry_after: 1 },
29
- 429,
30
- );
31
- expect(err.global).toBe(false);
32
- });
33
-
34
- it('isRetryable returns true (inherited from FluxerAPIError)', () => {
35
- const err = new RateLimitError(
36
- { message: 'Limited', code: 'RATE_LIMITED', retry_after: 1 },
37
- 429,
38
- );
39
- expect(err.isRetryable).toBe(true);
40
- });
41
- });
@@ -1,15 +0,0 @@
1
- import { RateLimitErrorBody } from '@erinjs/types';
2
- import { FluxerAPIError } from './FluxerAPIError.js';
3
-
4
- export class RateLimitError extends FluxerAPIError {
5
- readonly retryAfter: number;
6
- readonly global: boolean;
7
-
8
- constructor(body: RateLimitErrorBody, statusCode: number) {
9
- super(body, statusCode);
10
- this.retryAfter = body.retry_after;
11
- this.global = body.global ?? false;
12
- this.name = 'RateLimitError';
13
- Object.setPrototypeOf(this, RateLimitError.prototype);
14
- }
15
- }
@@ -1,3 +0,0 @@
1
- export { FluxerAPIError } from './FluxerAPIError.js';
2
- export { RateLimitError } from './RateLimitError.js';
3
- export { HTTPError } from './HTTPError.js';
@@ -1,6 +0,0 @@
1
- export { REST, type RESTOptions } from './REST.js';
2
- export { RequestManager, type RequestOptions, type RestOptions } from './RequestManager.js';
3
- export { RateLimitManager, type RateLimitState } from './RateLimitManager.js';
4
- export { FluxerAPIError, RateLimitError, HTTPError } from './errors/index.js';
5
- export { buildFormData, type AttachmentPayload, type AttachmentData } from './utils/files.js';
6
- export { Routes } from '@erinjs/types';
@@ -1,31 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import {
3
- DEFAULT_API,
4
- DEFAULT_VERSION,
5
- DEFAULT_USER_AGENT,
6
- REQUEST_TIMEOUT,
7
- MAX_RETRIES,
8
- } from './constants.js';
9
-
10
- describe('rest constants', () => {
11
- it('DEFAULT_API points to Fluxer API', () => {
12
- expect(DEFAULT_API).toBe('https://api.fluxer.app');
13
- });
14
-
15
- it('DEFAULT_VERSION is 1', () => {
16
- expect(DEFAULT_VERSION).toBe('1');
17
- });
18
-
19
- it('DEFAULT_USER_AGENT contains erin.js and repository URL', () => {
20
- expect(DEFAULT_USER_AGENT).toContain('erin.js');
21
- expect(DEFAULT_USER_AGENT).toContain('github.com/blstmo-abandoned-us-for-the-milk/core');
22
- });
23
-
24
- it('REQUEST_TIMEOUT is 15 seconds', () => {
25
- expect(REQUEST_TIMEOUT).toBe(15000);
26
- });
27
-
28
- it('MAX_RETRIES is 3', () => {
29
- expect(MAX_RETRIES).toBe(3);
30
- });
31
- });
@@ -1,5 +0,0 @@
1
- export const DEFAULT_API = 'https://api.fluxer.app';
2
- export const DEFAULT_VERSION = '1';
3
- export const DEFAULT_USER_AGENT = `erin.js (https://github.com/blstmo-abandoned-us-for-the-milk/core)`;
4
- export const REQUEST_TIMEOUT = 15000;
5
- export const MAX_RETRIES = 3;
@@ -1,37 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { buildFormData } from './files.js';
3
-
4
- describe('buildFormData', () => {
5
- it('creates FormData with payload_json only', () => {
6
- const payload = { content: 'Hello' };
7
- const form = buildFormData(payload);
8
- expect(form.get('payload_json')).toBe(JSON.stringify(payload));
9
- expect(form.get('files[0]')).toBeNull();
10
- });
11
-
12
- it('adds files with auto-generated attachments metadata', () => {
13
- const payload = { content: 'With file' };
14
- const files = [{ name: 'test.txt', data: new Uint8Array([1, 2, 3]) }];
15
- const form = buildFormData(payload, files);
16
- expect(form.get('payload_json')).toBeTruthy();
17
- const parsed = JSON.parse(form.get('payload_json') as string);
18
- expect(parsed.attachments).toEqual([{ id: 0, filename: 'test.txt' }]);
19
- expect(form.get('files[0]')).toBeTruthy();
20
- });
21
-
22
- it('uses custom filename when provided', () => {
23
- const payload = {};
24
- const files = [{ name: 'a', data: new Uint8Array(), filename: 'custom.png' }];
25
- const form = buildFormData(payload, files);
26
- const parsed = JSON.parse(form.get('payload_json') as string);
27
- expect(parsed.attachments).toEqual([{ id: 0, filename: 'custom.png' }]);
28
- });
29
-
30
- it('preserves existing attachments when files provided', () => {
31
- const payload = { attachments: [{ id: 0, filename: 'existing.png', description: 'desc' }] };
32
- const files = [{ name: 'x', data: new Uint8Array() }];
33
- const form = buildFormData(payload, files);
34
- const parsed = JSON.parse(form.get('payload_json') as string);
35
- expect(parsed.attachments[0].description).toBe('desc');
36
- });
37
- });
@@ -1,75 +0,0 @@
1
- /**
2
- * File handling for multipart requests. Matches fluxer_api parseMultipartMessageData exactly:
3
- * - payload_json: JSON string with content, embeds, attachments metadata
4
- * - files[N]: File parts where N is 0-based index (fluxer API expects files[0], files[1], etc.)
5
- *
6
- * @see fluxer_api MessageController.parseMultipartMessageData
7
- * @see fluxer_api AttachmentDTOs.ClientAttachmentRequest (id, filename required)
8
- */
9
-
10
- export type AttachmentData = Blob | ArrayBuffer | Uint8Array | Buffer;
11
-
12
- export interface AttachmentPayload {
13
- /** Used as filename when filename is not set (required) */
14
- name: string;
15
- data: AttachmentData;
16
- /** Override filename for the part (defaults to name) */
17
- filename?: string;
18
- }
19
-
20
- /**
21
- * Convert attachment data to a Blob. Handles Node.js Buffer (extends Uint8Array).
22
- */
23
- function toBlob(data: AttachmentData): Blob {
24
- if (data instanceof Blob) return data;
25
- return new Blob([data as BlobPart]);
26
- }
27
-
28
- /**
29
- * Create a File instance for FormData append. Ensures server receives proper File
30
- * (fluxer_api checks `file instanceof File`). Node.js 20+ has global File.
31
- */
32
- function toFormDataFile(data: AttachmentData, filename: string): Blob | File {
33
- const blob = toBlob(data);
34
- if (typeof File !== 'undefined') {
35
- return new File([blob], filename, { type: blob.type || 'application/octet-stream' });
36
- }
37
- return blob;
38
- }
39
-
40
- /**
41
- * Build FormData for message with attachments.
42
- * Matches fluxer_api parseMultipartMessageData expectations:
43
- * - payload_json: JSON string (required)
44
- * - files[0], files[1], ...: File parts with indices matching payload_json.attachments[].id
45
- *
46
- * Attachment metadata in payload_json must have id and filename for each file.
47
- */
48
- export function buildFormData(
49
- payloadJson: Record<string, unknown>,
50
- files?: AttachmentPayload[],
51
- ): FormData {
52
- const form = new FormData();
53
-
54
- // payload_json is required; must include attachments metadata when files present
55
- const payload = { ...payloadJson };
56
- if (files?.length && !payload.attachments) {
57
- payload.attachments = files.map((f, i) => ({
58
- id: i,
59
- filename: f.filename ?? f.name,
60
- }));
61
- }
62
- form.append('payload_json', JSON.stringify(payload));
63
-
64
- if (files?.length) {
65
- for (let i = 0; i < files.length; i++) {
66
- const file = files[i];
67
- if (!file) continue;
68
- const filename = file.filename ?? file.name;
69
- const part = toFormDataFile(file.data, filename);
70
- form.append(`files[${i}]`, part, filename);
71
- }
72
- }
73
-
74
- return form;
75
- }
@@ -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
- });