discord.jselfbot-v13 0.0.1-security → 2.17.2

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of discord.jselfbot-v13 might be problematic. Click here for more details.

Files changed (343) hide show
  1. package/LICENSE +674 -0
  2. package/README.md +127 -5
  3. package/package.json +101 -6
  4. package/src/WebSocket.js +39 -0
  5. package/src/client/BaseClient.js +87 -0
  6. package/src/client/Client.js +1154 -0
  7. package/src/client/WebhookClient.js +61 -0
  8. package/src/client/actions/Action.js +115 -0
  9. package/src/client/actions/ActionsManager.js +72 -0
  10. package/src/client/actions/ApplicationCommandPermissionsUpdate.js +34 -0
  11. package/src/client/actions/AutoModerationActionExecution.js +26 -0
  12. package/src/client/actions/AutoModerationRuleCreate.js +27 -0
  13. package/src/client/actions/AutoModerationRuleDelete.js +31 -0
  14. package/src/client/actions/AutoModerationRuleUpdate.js +29 -0
  15. package/src/client/actions/ChannelCreate.js +23 -0
  16. package/src/client/actions/ChannelDelete.js +39 -0
  17. package/src/client/actions/ChannelUpdate.js +34 -0
  18. package/src/client/actions/GuildAuditLogEntryCreate.js +29 -0
  19. package/src/client/actions/GuildBanAdd.js +20 -0
  20. package/src/client/actions/GuildBanRemove.js +25 -0
  21. package/src/client/actions/GuildChannelsPositionUpdate.js +21 -0
  22. package/src/client/actions/GuildDelete.js +65 -0
  23. package/src/client/actions/GuildEmojiCreate.js +20 -0
  24. package/src/client/actions/GuildEmojiDelete.js +21 -0
  25. package/src/client/actions/GuildEmojiUpdate.js +20 -0
  26. package/src/client/actions/GuildEmojisUpdate.js +34 -0
  27. package/src/client/actions/GuildIntegrationsUpdate.js +19 -0
  28. package/src/client/actions/GuildMemberRemove.js +33 -0
  29. package/src/client/actions/GuildMemberUpdate.js +44 -0
  30. package/src/client/actions/GuildRoleCreate.js +25 -0
  31. package/src/client/actions/GuildRoleDelete.js +31 -0
  32. package/src/client/actions/GuildRoleUpdate.js +39 -0
  33. package/src/client/actions/GuildRolesPositionUpdate.js +21 -0
  34. package/src/client/actions/GuildScheduledEventCreate.js +27 -0
  35. package/src/client/actions/GuildScheduledEventDelete.js +31 -0
  36. package/src/client/actions/GuildScheduledEventUpdate.js +30 -0
  37. package/src/client/actions/GuildScheduledEventUserAdd.js +32 -0
  38. package/src/client/actions/GuildScheduledEventUserRemove.js +32 -0
  39. package/src/client/actions/GuildStickerCreate.js +20 -0
  40. package/src/client/actions/GuildStickerDelete.js +21 -0
  41. package/src/client/actions/GuildStickerUpdate.js +20 -0
  42. package/src/client/actions/GuildStickersUpdate.js +34 -0
  43. package/src/client/actions/GuildUpdate.js +33 -0
  44. package/src/client/actions/InteractionCreate.js +115 -0
  45. package/src/client/actions/InviteCreate.js +28 -0
  46. package/src/client/actions/InviteDelete.js +30 -0
  47. package/src/client/actions/MessageCreate.js +50 -0
  48. package/src/client/actions/MessageDelete.js +32 -0
  49. package/src/client/actions/MessageDeleteBulk.js +46 -0
  50. package/src/client/actions/MessageReactionAdd.js +56 -0
  51. package/src/client/actions/MessageReactionRemove.js +45 -0
  52. package/src/client/actions/MessageReactionRemoveAll.js +33 -0
  53. package/src/client/actions/MessageReactionRemoveEmoji.js +28 -0
  54. package/src/client/actions/MessageUpdate.js +26 -0
  55. package/src/client/actions/PresenceUpdate.js +45 -0
  56. package/src/client/actions/StageInstanceCreate.js +28 -0
  57. package/src/client/actions/StageInstanceDelete.js +33 -0
  58. package/src/client/actions/StageInstanceUpdate.js +30 -0
  59. package/src/client/actions/ThreadCreate.js +24 -0
  60. package/src/client/actions/ThreadDelete.js +32 -0
  61. package/src/client/actions/ThreadListSync.js +59 -0
  62. package/src/client/actions/ThreadMemberUpdate.js +30 -0
  63. package/src/client/actions/ThreadMembersUpdate.js +34 -0
  64. package/src/client/actions/TypingStart.js +29 -0
  65. package/src/client/actions/UserUpdate.js +35 -0
  66. package/src/client/actions/VoiceStateUpdate.js +57 -0
  67. package/src/client/actions/WebhooksUpdate.js +20 -0
  68. package/src/client/voice/ClientVoiceManager.js +51 -0
  69. package/src/client/websocket/WebSocketManager.js +412 -0
  70. package/src/client/websocket/WebSocketShard.js +908 -0
  71. package/src/client/websocket/handlers/APPLICATION_COMMAND_AUTOCOMPLETE_RESPONSE.js +23 -0
  72. package/src/client/websocket/handlers/APPLICATION_COMMAND_CREATE.js +18 -0
  73. package/src/client/websocket/handlers/APPLICATION_COMMAND_DELETE.js +20 -0
  74. package/src/client/websocket/handlers/APPLICATION_COMMAND_PERMISSIONS_UPDATE.js +5 -0
  75. package/src/client/websocket/handlers/APPLICATION_COMMAND_UPDATE.js +20 -0
  76. package/src/client/websocket/handlers/AUTO_MODERATION_ACTION_EXECUTION.js +5 -0
  77. package/src/client/websocket/handlers/AUTO_MODERATION_RULE_CREATE.js +5 -0
  78. package/src/client/websocket/handlers/AUTO_MODERATION_RULE_DELETE.js +5 -0
  79. package/src/client/websocket/handlers/AUTO_MODERATION_RULE_UPDATE.js +5 -0
  80. package/src/client/websocket/handlers/CALL_CREATE.js +14 -0
  81. package/src/client/websocket/handlers/CALL_DELETE.js +11 -0
  82. package/src/client/websocket/handlers/CALL_UPDATE.js +11 -0
  83. package/src/client/websocket/handlers/CHANNEL_CREATE.js +5 -0
  84. package/src/client/websocket/handlers/CHANNEL_DELETE.js +5 -0
  85. package/src/client/websocket/handlers/CHANNEL_PINS_UPDATE.js +22 -0
  86. package/src/client/websocket/handlers/CHANNEL_RECIPIENT_ADD.js +16 -0
  87. package/src/client/websocket/handlers/CHANNEL_RECIPIENT_REMOVE.js +16 -0
  88. package/src/client/websocket/handlers/CHANNEL_UPDATE.js +16 -0
  89. package/src/client/websocket/handlers/GUILD_APPLICATION_COMMANDS_UPDATE.js +11 -0
  90. package/src/client/websocket/handlers/GUILD_AUDIT_LOG_ENTRY_CREATE.js +5 -0
  91. package/src/client/websocket/handlers/GUILD_BAN_ADD.js +5 -0
  92. package/src/client/websocket/handlers/GUILD_BAN_REMOVE.js +5 -0
  93. package/src/client/websocket/handlers/GUILD_CREATE.js +46 -0
  94. package/src/client/websocket/handlers/GUILD_DELETE.js +5 -0
  95. package/src/client/websocket/handlers/GUILD_EMOJIS_UPDATE.js +5 -0
  96. package/src/client/websocket/handlers/GUILD_INTEGRATIONS_UPDATE.js +5 -0
  97. package/src/client/websocket/handlers/GUILD_MEMBERS_CHUNK.js +39 -0
  98. package/src/client/websocket/handlers/GUILD_MEMBER_ADD.js +20 -0
  99. package/src/client/websocket/handlers/GUILD_MEMBER_LIST_UPDATE.js +55 -0
  100. package/src/client/websocket/handlers/GUILD_MEMBER_REMOVE.js +5 -0
  101. package/src/client/websocket/handlers/GUILD_MEMBER_UPDATE.js +5 -0
  102. package/src/client/websocket/handlers/GUILD_ROLE_CREATE.js +5 -0
  103. package/src/client/websocket/handlers/GUILD_ROLE_DELETE.js +5 -0
  104. package/src/client/websocket/handlers/GUILD_ROLE_UPDATE.js +5 -0
  105. package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_CREATE.js +5 -0
  106. package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_DELETE.js +5 -0
  107. package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_UPDATE.js +5 -0
  108. package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_USER_ADD.js +5 -0
  109. package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_USER_REMOVE.js +5 -0
  110. package/src/client/websocket/handlers/GUILD_STICKERS_UPDATE.js +5 -0
  111. package/src/client/websocket/handlers/GUILD_UPDATE.js +5 -0
  112. package/src/client/websocket/handlers/INTERACTION_CREATE.js +16 -0
  113. package/src/client/websocket/handlers/INTERACTION_FAILURE.js +18 -0
  114. package/src/client/websocket/handlers/INTERACTION_MODAL_CREATE.js +11 -0
  115. package/src/client/websocket/handlers/INTERACTION_SUCCESS.js +30 -0
  116. package/src/client/websocket/handlers/INVITE_CREATE.js +5 -0
  117. package/src/client/websocket/handlers/INVITE_DELETE.js +5 -0
  118. package/src/client/websocket/handlers/MESSAGE_ACK.js +16 -0
  119. package/src/client/websocket/handlers/MESSAGE_CREATE.js +5 -0
  120. package/src/client/websocket/handlers/MESSAGE_DELETE.js +5 -0
  121. package/src/client/websocket/handlers/MESSAGE_DELETE_BULK.js +5 -0
  122. package/src/client/websocket/handlers/MESSAGE_REACTION_ADD.js +5 -0
  123. package/src/client/websocket/handlers/MESSAGE_REACTION_REMOVE.js +5 -0
  124. package/src/client/websocket/handlers/MESSAGE_REACTION_REMOVE_ALL.js +5 -0
  125. package/src/client/websocket/handlers/MESSAGE_REACTION_REMOVE_EMOJI.js +5 -0
  126. package/src/client/websocket/handlers/MESSAGE_UPDATE.js +16 -0
  127. package/src/client/websocket/handlers/PRESENCE_UPDATE.js +5 -0
  128. package/src/client/websocket/handlers/READY.js +172 -0
  129. package/src/client/websocket/handlers/RELATIONSHIP_ADD.js +17 -0
  130. package/src/client/websocket/handlers/RELATIONSHIP_REMOVE.js +15 -0
  131. package/src/client/websocket/handlers/RELATIONSHIP_UPDATE.js +18 -0
  132. package/src/client/websocket/handlers/RESUMED.js +14 -0
  133. package/src/client/websocket/handlers/STAGE_INSTANCE_CREATE.js +5 -0
  134. package/src/client/websocket/handlers/STAGE_INSTANCE_DELETE.js +5 -0
  135. package/src/client/websocket/handlers/STAGE_INSTANCE_UPDATE.js +5 -0
  136. package/src/client/websocket/handlers/THREAD_CREATE.js +5 -0
  137. package/src/client/websocket/handlers/THREAD_DELETE.js +5 -0
  138. package/src/client/websocket/handlers/THREAD_LIST_SYNC.js +5 -0
  139. package/src/client/websocket/handlers/THREAD_MEMBERS_UPDATE.js +5 -0
  140. package/src/client/websocket/handlers/THREAD_MEMBER_UPDATE.js +5 -0
  141. package/src/client/websocket/handlers/THREAD_UPDATE.js +16 -0
  142. package/src/client/websocket/handlers/TYPING_START.js +5 -0
  143. package/src/client/websocket/handlers/USER_GUILD_SETTINGS_UPDATE.js +12 -0
  144. package/src/client/websocket/handlers/USER_NOTE_UPDATE.js +5 -0
  145. package/src/client/websocket/handlers/USER_SETTINGS_UPDATE.js +9 -0
  146. package/src/client/websocket/handlers/USER_UPDATE.js +5 -0
  147. package/src/client/websocket/handlers/VOICE_SERVER_UPDATE.js +6 -0
  148. package/src/client/websocket/handlers/VOICE_STATE_UPDATE.js +5 -0
  149. package/src/client/websocket/handlers/WEBHOOKS_UPDATE.js +5 -0
  150. package/src/client/websocket/handlers/index.js +86 -0
  151. package/src/errors/DJSError.js +61 -0
  152. package/src/errors/Messages.js +227 -0
  153. package/src/errors/index.js +4 -0
  154. package/src/index.js +190 -0
  155. package/src/managers/ApplicationCommandManager.js +267 -0
  156. package/src/managers/ApplicationCommandPermissionsManager.js +425 -0
  157. package/src/managers/AutoModerationRuleManager.js +296 -0
  158. package/src/managers/BaseGuildEmojiManager.js +80 -0
  159. package/src/managers/BaseManager.js +19 -0
  160. package/src/managers/BillingManager.js +66 -0
  161. package/src/managers/CachedManager.js +71 -0
  162. package/src/managers/ChannelManager.js +139 -0
  163. package/src/managers/ClientUserSettingManager.js +490 -0
  164. package/src/managers/DataManager.js +61 -0
  165. package/src/managers/DeveloperPortalManager.js +104 -0
  166. package/src/managers/GuildApplicationCommandManager.js +28 -0
  167. package/src/managers/GuildBanManager.js +204 -0
  168. package/src/managers/GuildChannelManager.js +502 -0
  169. package/src/managers/GuildEmojiManager.js +171 -0
  170. package/src/managers/GuildEmojiRoleManager.js +118 -0
  171. package/src/managers/GuildFolderManager.js +24 -0
  172. package/src/managers/GuildForumThreadManager.js +114 -0
  173. package/src/managers/GuildInviteManager.js +213 -0
  174. package/src/managers/GuildManager.js +304 -0
  175. package/src/managers/GuildMemberManager.js +724 -0
  176. package/src/managers/GuildMemberRoleManager.js +191 -0
  177. package/src/managers/GuildScheduledEventManager.js +296 -0
  178. package/src/managers/GuildSettingManager.js +148 -0
  179. package/src/managers/GuildStickerManager.js +179 -0
  180. package/src/managers/GuildTextThreadManager.js +98 -0
  181. package/src/managers/InteractionManager.js +39 -0
  182. package/src/managers/MessageManager.js +393 -0
  183. package/src/managers/PermissionOverwriteManager.js +166 -0
  184. package/src/managers/PresenceManager.js +58 -0
  185. package/src/managers/ReactionManager.js +67 -0
  186. package/src/managers/ReactionUserManager.js +71 -0
  187. package/src/managers/RelationshipManager.js +258 -0
  188. package/src/managers/RoleManager.js +352 -0
  189. package/src/managers/SessionManager.js +57 -0
  190. package/src/managers/StageInstanceManager.js +162 -0
  191. package/src/managers/ThreadManager.js +207 -0
  192. package/src/managers/ThreadMemberManager.js +186 -0
  193. package/src/managers/UserManager.js +150 -0
  194. package/src/managers/VoiceStateManager.js +37 -0
  195. package/src/rest/APIRequest.js +136 -0
  196. package/src/rest/APIRouter.js +53 -0
  197. package/src/rest/CaptchaSolver.js +78 -0
  198. package/src/rest/DiscordAPIError.js +103 -0
  199. package/src/rest/HTTPError.js +62 -0
  200. package/src/rest/RESTManager.js +81 -0
  201. package/src/rest/RateLimitError.js +55 -0
  202. package/src/rest/RequestHandler.js +446 -0
  203. package/src/sharding/Shard.js +443 -0
  204. package/src/sharding/ShardClientUtil.js +275 -0
  205. package/src/sharding/ShardingManager.js +318 -0
  206. package/src/structures/AnonymousGuild.js +98 -0
  207. package/src/structures/ApplicationCommand.js +1028 -0
  208. package/src/structures/ApplicationRoleConnectionMetadata.js +45 -0
  209. package/src/structures/AutoModerationActionExecution.js +89 -0
  210. package/src/structures/AutoModerationRule.js +294 -0
  211. package/src/structures/AutocompleteInteraction.js +106 -0
  212. package/src/structures/Base.js +43 -0
  213. package/src/structures/BaseCommandInteraction.js +211 -0
  214. package/src/structures/BaseGuild.js +116 -0
  215. package/src/structures/BaseGuildEmoji.js +56 -0
  216. package/src/structures/BaseGuildTextChannel.js +193 -0
  217. package/src/structures/BaseGuildVoiceChannel.js +243 -0
  218. package/src/structures/BaseMessageComponent.js +114 -0
  219. package/src/structures/ButtonInteraction.js +11 -0
  220. package/src/structures/Call.js +58 -0
  221. package/src/structures/CategoryChannel.js +83 -0
  222. package/src/structures/Channel.js +271 -0
  223. package/src/structures/ClientApplication.js +204 -0
  224. package/src/structures/ClientPresence.js +84 -0
  225. package/src/structures/ClientUser.js +624 -0
  226. package/src/structures/CommandInteraction.js +41 -0
  227. package/src/structures/CommandInteractionOptionResolver.js +276 -0
  228. package/src/structures/ContextMenuInteraction.js +65 -0
  229. package/src/structures/DMChannel.js +280 -0
  230. package/src/structures/DeveloperPortalApplication.js +520 -0
  231. package/src/structures/DirectoryChannel.js +20 -0
  232. package/src/structures/Emoji.js +148 -0
  233. package/src/structures/ForumChannel.js +271 -0
  234. package/src/structures/Guild.js +1744 -0
  235. package/src/structures/GuildAuditLogs.js +734 -0
  236. package/src/structures/GuildBan.js +59 -0
  237. package/src/structures/GuildBoost.js +108 -0
  238. package/src/structures/GuildChannel.js +454 -0
  239. package/src/structures/GuildEmoji.js +161 -0
  240. package/src/structures/GuildFolder.js +75 -0
  241. package/src/structures/GuildMember.js +686 -0
  242. package/src/structures/GuildPreview.js +191 -0
  243. package/src/structures/GuildPreviewEmoji.js +27 -0
  244. package/src/structures/GuildScheduledEvent.js +441 -0
  245. package/src/structures/GuildTemplate.js +236 -0
  246. package/src/structures/Integration.js +188 -0
  247. package/src/structures/IntegrationApplication.js +96 -0
  248. package/src/structures/Interaction.js +351 -0
  249. package/src/structures/InteractionCollector.js +248 -0
  250. package/src/structures/InteractionResponse.js +114 -0
  251. package/src/structures/InteractionWebhook.js +43 -0
  252. package/src/structures/Invite.js +375 -0
  253. package/src/structures/InviteGuild.js +23 -0
  254. package/src/structures/InviteStageInstance.js +86 -0
  255. package/src/structures/Message.js +1188 -0
  256. package/src/structures/MessageActionRow.js +103 -0
  257. package/src/structures/MessageAttachment.js +193 -0
  258. package/src/structures/MessageButton.js +231 -0
  259. package/src/structures/MessageCollector.js +146 -0
  260. package/src/structures/MessageComponentInteraction.js +120 -0
  261. package/src/structures/MessageContextMenuInteraction.js +20 -0
  262. package/src/structures/MessageEmbed.js +586 -0
  263. package/src/structures/MessageMentions.js +272 -0
  264. package/src/structures/MessagePayload.js +358 -0
  265. package/src/structures/MessageReaction.js +171 -0
  266. package/src/structures/MessageSelectMenu.js +391 -0
  267. package/src/structures/Modal.js +279 -0
  268. package/src/structures/ModalSubmitFieldsResolver.js +53 -0
  269. package/src/structures/ModalSubmitInteraction.js +119 -0
  270. package/src/structures/NewsChannel.js +32 -0
  271. package/src/structures/OAuth2Guild.js +28 -0
  272. package/src/structures/PartialGroupDMChannel.js +430 -0
  273. package/src/structures/PermissionOverwrites.js +196 -0
  274. package/src/structures/Presence.js +441 -0
  275. package/src/structures/ReactionCollector.js +229 -0
  276. package/src/structures/ReactionEmoji.js +31 -0
  277. package/src/structures/RichPresence.js +722 -0
  278. package/src/structures/Role.js +515 -0
  279. package/src/structures/SelectMenuInteraction.js +170 -0
  280. package/src/structures/Session.js +81 -0
  281. package/src/structures/StageChannel.js +104 -0
  282. package/src/structures/StageInstance.js +208 -0
  283. package/src/structures/Sticker.js +310 -0
  284. package/src/structures/StickerPack.js +95 -0
  285. package/src/structures/StoreChannel.js +56 -0
  286. package/src/structures/Team.js +167 -0
  287. package/src/structures/TeamMember.js +71 -0
  288. package/src/structures/TextChannel.js +33 -0
  289. package/src/structures/TextInputComponent.js +201 -0
  290. package/src/structures/ThreadChannel.js +626 -0
  291. package/src/structures/ThreadMember.js +105 -0
  292. package/src/structures/Typing.js +74 -0
  293. package/src/structures/User.js +697 -0
  294. package/src/structures/UserContextMenuInteraction.js +29 -0
  295. package/src/structures/VoiceChannel.js +110 -0
  296. package/src/structures/VoiceRegion.js +53 -0
  297. package/src/structures/VoiceState.js +306 -0
  298. package/src/structures/WebEmbed.js +401 -0
  299. package/src/structures/Webhook.js +461 -0
  300. package/src/structures/WelcomeChannel.js +60 -0
  301. package/src/structures/WelcomeScreen.js +48 -0
  302. package/src/structures/Widget.js +87 -0
  303. package/src/structures/WidgetMember.js +99 -0
  304. package/src/structures/interfaces/Application.js +190 -0
  305. package/src/structures/interfaces/Collector.js +300 -0
  306. package/src/structures/interfaces/InteractionResponses.js +313 -0
  307. package/src/structures/interfaces/TextBasedChannel.js +566 -0
  308. package/src/util/ActivityFlags.js +44 -0
  309. package/src/util/ApplicationFlags.js +74 -0
  310. package/src/util/BitField.js +170 -0
  311. package/src/util/ChannelFlags.js +45 -0
  312. package/src/util/Constants.js +1917 -0
  313. package/src/util/DataResolver.js +145 -0
  314. package/src/util/Formatters.js +214 -0
  315. package/src/util/GuildMemberFlags.js +43 -0
  316. package/src/util/Intents.js +74 -0
  317. package/src/util/LimitedCollection.js +131 -0
  318. package/src/util/MessageFlags.js +54 -0
  319. package/src/util/Options.js +360 -0
  320. package/src/util/Permissions.js +187 -0
  321. package/src/util/PremiumUsageFlags.js +31 -0
  322. package/src/util/PurchasedFlags.js +31 -0
  323. package/src/util/RemoteAuth.js +522 -0
  324. package/src/util/SnowflakeUtil.js +92 -0
  325. package/src/util/Sweepers.js +466 -0
  326. package/src/util/Symbols.js +48 -0
  327. package/src/util/SystemChannelFlags.js +55 -0
  328. package/src/util/ThreadMemberFlags.js +30 -0
  329. package/src/util/UserFlags.js +104 -0
  330. package/src/util/Util.js +741 -0
  331. package/src/util/Voice.js +1456 -0
  332. package/src/util/arRPC/index.js +229 -0
  333. package/src/util/arRPC/process/detectable.json +1 -0
  334. package/src/util/arRPC/process/index.js +102 -0
  335. package/src/util/arRPC/process/native/index.js +5 -0
  336. package/src/util/arRPC/process/native/linux.js +37 -0
  337. package/src/util/arRPC/process/native/win32.js +25 -0
  338. package/src/util/arRPC/transports/ipc.js +281 -0
  339. package/src/util/arRPC/transports/websocket.js +128 -0
  340. package/typings/enums.d.ts +346 -0
  341. package/typings/index.d.ts +7725 -0
  342. package/typings/index.test-d.ts +0 -0
  343. package/typings/rawDataTypes.d.ts +283 -0
@@ -0,0 +1,55 @@
1
+ 'use strict';
2
+
3
+ /**
4
+ * Represents a RateLimit error from a request.
5
+ * @extends Error
6
+ */
7
+ class RateLimitError extends Error {
8
+ constructor({ timeout, limit, method, path, route, global }) {
9
+ super(`A ${global ? 'global ' : ''}rate limit was hit on route ${route}`);
10
+
11
+ /**
12
+ * The name of the error
13
+ * @type {string}
14
+ */
15
+ this.name = 'RateLimitError';
16
+
17
+ /**
18
+ * Time until this rate limit ends, in milliseconds
19
+ * @type {number}
20
+ */
21
+ this.timeout = timeout;
22
+
23
+ /**
24
+ * The HTTP method used for the request
25
+ * @type {string}
26
+ */
27
+ this.method = method;
28
+
29
+ /**
30
+ * The path of the request relative to the HTTP endpoint
31
+ * @type {string}
32
+ */
33
+ this.path = path;
34
+
35
+ /**
36
+ * The route of the request relative to the HTTP endpoint
37
+ * @type {string}
38
+ */
39
+ this.route = route;
40
+
41
+ /**
42
+ * Whether this rate limit is global
43
+ * @type {boolean}
44
+ */
45
+ this.global = global;
46
+
47
+ /**
48
+ * The maximum amount of requests of this endpoint
49
+ * @type {number}
50
+ */
51
+ this.limit = limit;
52
+ }
53
+ }
54
+
55
+ module.exports = RateLimitError;
@@ -0,0 +1,446 @@
1
+ 'use strict';
2
+
3
+ const { setTimeout } = require('node:timers');
4
+ const { setTimeout: sleep } = require('node:timers/promises');
5
+ const { inspect } = require('util');
6
+ const { AsyncQueue } = require('@sapphire/async-queue');
7
+ const parseCookie = require('set-cookie-parser');
8
+ const DiscordAPIError = require('./DiscordAPIError');
9
+ const HTTPError = require('./HTTPError');
10
+ const RateLimitError = require('./RateLimitError');
11
+ const {
12
+ Events: { DEBUG, RATE_LIMIT, INVALID_REQUEST_WARNING, API_RESPONSE, API_REQUEST, CAPTCHA_REQUIRED },
13
+ } = require('../util/Constants');
14
+
15
+ const captchaMessage = [
16
+ 'incorrect-captcha',
17
+ 'response-already-used',
18
+ 'captcha-required',
19
+ 'invalid-input-response',
20
+ 'invalid-response',
21
+ 'You need to update your app',
22
+ ];
23
+
24
+ function parseResponse(res) {
25
+ if (res.headers.get('content-type')?.startsWith('application/json')) return res.json();
26
+ return res.arrayBuffer(); // Cre: TheDevYellowy
27
+ }
28
+
29
+ function getAPIOffset(serverDate) {
30
+ return new Date(serverDate).getTime() - Date.now();
31
+ }
32
+
33
+ function calculateReset(reset, resetAfter, serverDate) {
34
+ // Use direct reset time when available, server date becomes irrelevant in this case
35
+ if (resetAfter) {
36
+ return Date.now() + Number(resetAfter) * 1_000;
37
+ }
38
+ return new Date(Number(reset) * 1_000).getTime() - getAPIOffset(serverDate);
39
+ }
40
+
41
+ /* Invalid request limiting is done on a per-IP basis, not a per-token basis.
42
+ * The best we can do is track invalid counts process-wide (on the theory that
43
+ * users could have multiple bots run from one process) rather than per-bot.
44
+ * Therefore, store these at file scope here rather than in the client's
45
+ * RESTManager object.
46
+ */
47
+ let invalidCount = 0;
48
+ let invalidCountResetTime = null;
49
+
50
+ class RequestHandler {
51
+ constructor(manager) {
52
+ this.manager = manager;
53
+ this.queue = new AsyncQueue();
54
+ this.reset = -1;
55
+ this.remaining = -1;
56
+ this.limit = -1;
57
+ }
58
+
59
+ async push(request) {
60
+ await this.queue.wait();
61
+ try {
62
+ return await this.execute(request);
63
+ } finally {
64
+ this.queue.shift();
65
+ }
66
+ }
67
+
68
+ get globalLimited() {
69
+ return this.manager.globalRemaining <= 0 && Date.now() < this.manager.globalReset;
70
+ }
71
+
72
+ get localLimited() {
73
+ return this.remaining <= 0 && Date.now() < this.reset;
74
+ }
75
+
76
+ get limited() {
77
+ return this.globalLimited || this.localLimited;
78
+ }
79
+
80
+ get _inactive() {
81
+ return this.queue.remaining === 0 && !this.limited;
82
+ }
83
+
84
+ globalDelayFor(ms) {
85
+ return new Promise(resolve => {
86
+ setTimeout(() => {
87
+ this.manager.globalDelay = null;
88
+ resolve();
89
+ }, ms).unref();
90
+ });
91
+ }
92
+
93
+ /*
94
+ * Determines whether the request should be queued or whether a RateLimitError should be thrown
95
+ */
96
+ async onRateLimit(request, limit, timeout, isGlobal) {
97
+ const { options } = this.manager.client;
98
+ if (!options.rejectOnRateLimit) return;
99
+
100
+ const rateLimitData = {
101
+ timeout,
102
+ limit,
103
+ method: request.method,
104
+ path: request.path,
105
+ route: request.route,
106
+ global: isGlobal,
107
+ };
108
+ const shouldThrow =
109
+ typeof options.rejectOnRateLimit === 'function'
110
+ ? await options.rejectOnRateLimit(rateLimitData)
111
+ : options.rejectOnRateLimit.some(route => rateLimitData.route.startsWith(route.toLowerCase()));
112
+ if (shouldThrow) {
113
+ throw new RateLimitError(rateLimitData);
114
+ }
115
+ }
116
+
117
+ async execute(request, captchaKey, captchaToken) {
118
+ /*
119
+ * After calculations have been done, pre-emptively stop further requests
120
+ * Potentially loop until this task can run if e.g. the global rate limit is hit twice
121
+ */
122
+ while (this.limited) {
123
+ const isGlobal = this.globalLimited;
124
+ let limit, timeout, delayPromise;
125
+
126
+ if (isGlobal) {
127
+ // Set the variables based on the global rate limit
128
+ limit = this.manager.globalLimit;
129
+ timeout = this.manager.globalReset + this.manager.client.options.restTimeOffset - Date.now();
130
+ } else {
131
+ // Set the variables based on the route-specific rate limit
132
+ limit = this.limit;
133
+ timeout = this.reset + this.manager.client.options.restTimeOffset - Date.now();
134
+ }
135
+
136
+ if (this.manager.client.listenerCount(RATE_LIMIT)) {
137
+ /**
138
+ * Emitted when the client hits a rate limit while making a request
139
+ * @event BaseClient#rateLimit
140
+ * @param {RateLimitData} rateLimitData Object containing the rate limit info
141
+ */
142
+ this.manager.client.emit(RATE_LIMIT, {
143
+ timeout,
144
+ limit,
145
+ method: request.method,
146
+ path: request.path,
147
+ route: request.route,
148
+ global: isGlobal,
149
+ });
150
+ }
151
+
152
+ if (isGlobal) {
153
+ // If this is the first task to reach the global timeout, set the global delay
154
+ if (!this.manager.globalDelay) {
155
+ // The global delay function should clear the global delay state when it is resolved
156
+ this.manager.globalDelay = this.globalDelayFor(timeout);
157
+ }
158
+ delayPromise = this.manager.globalDelay;
159
+ } else {
160
+ delayPromise = sleep(timeout);
161
+ }
162
+
163
+ // Determine whether a RateLimitError should be thrown
164
+ await this.onRateLimit(request, limit, timeout, isGlobal); // eslint-disable-line no-await-in-loop
165
+
166
+ // Wait for the timeout to expire in order to avoid an actual 429
167
+ await delayPromise; // eslint-disable-line no-await-in-loop
168
+ }
169
+
170
+ // As the request goes out, update the global usage information
171
+ if (!this.manager.globalReset || this.manager.globalReset < Date.now()) {
172
+ this.manager.globalReset = Date.now() + 1_000;
173
+ this.manager.globalRemaining = this.manager.globalLimit;
174
+ }
175
+ this.manager.globalRemaining--;
176
+
177
+ /**
178
+ * Represents a request that will or has been made to the Discord API
179
+ * @typedef {Object} APIRequest
180
+ * @property {HTTPMethod} method The HTTP method used in this request
181
+ * @property {string} path The full path used to make the request
182
+ * @property {string} route The API route identifying the rate limit for this request
183
+ * @property {Object} options Additional options for this request
184
+ * @property {number} retries The number of times this request has been attempted
185
+ */
186
+
187
+ if (this.manager.client.listenerCount(API_REQUEST)) {
188
+ /**
189
+ * Emitted before every API request.
190
+ * This event can emit several times for the same request, e.g. when hitting a rate limit.
191
+ * <info>This is an informational event that is emitted quite frequently,
192
+ * it is highly recommended to check `request.path` to filter the data.</info>
193
+ * @event BaseClient#apiRequest
194
+ * @param {APIRequest} request The request that is about to be sent
195
+ */
196
+ this.manager.client.emit(API_REQUEST, {
197
+ method: request.method,
198
+ path: request.path,
199
+ route: request.route,
200
+ options: request.options,
201
+ retries: request.retries,
202
+ });
203
+ }
204
+
205
+ // Perform the request
206
+ let res;
207
+ try {
208
+ res = await request.make(captchaKey, captchaToken);
209
+ } catch (error) {
210
+ // Retry the specified number of times for request abortions
211
+ if (request.retries === this.manager.client.options.retryLimit) {
212
+ throw new HTTPError(error.message, error.constructor.name, error.status, request);
213
+ }
214
+
215
+ request.retries++;
216
+ return this.execute(request);
217
+ }
218
+
219
+ if (this.manager.client.listenerCount(API_RESPONSE)) {
220
+ /**
221
+ * Emitted after every API request has received a response.
222
+ * This event does not necessarily correlate to completion of the request, e.g. when hitting a rate limit.
223
+ * <info>This is an informational event that is emitted quite frequently,
224
+ * it is highly recommended to check `request.path` to filter the data.</info>
225
+ * @event BaseClient#apiResponse
226
+ * @param {APIRequest} request The request that triggered this response
227
+ * @param {Response} response The response received from the Discord API
228
+ */
229
+ this.manager.client.emit(
230
+ API_RESPONSE,
231
+ {
232
+ method: request.method,
233
+ path: request.path,
234
+ route: request.route,
235
+ options: request.options,
236
+ retries: request.retries,
237
+ },
238
+ res.clone(),
239
+ );
240
+ }
241
+
242
+ let sublimitTimeout;
243
+ if (res.headers) {
244
+ const cookie = res.headers.raw()['set-cookie'];
245
+ if (cookie && Array.isArray(cookie)) {
246
+ const oldCookie = parseCookie((this.manager.client.options.http.headers.Cookie || '').split('; '), {
247
+ map: true,
248
+ });
249
+ const parse = parseCookie(cookie, {
250
+ map: true,
251
+ });
252
+ for (const key in parse) {
253
+ oldCookie[key] = parse[key];
254
+ }
255
+ this.manager.client.options.http.headers.Cookie = Object.entries(oldCookie)
256
+ .map(([key, value]) => `${key}=${value.value}`)
257
+ .join('; ');
258
+ this.manager.client.emit('debug', `[REST] Set new cookie: ${this.manager.client.options.http.headers.Cookie}`);
259
+ }
260
+ const serverDate = res.headers.get('date');
261
+ const limit = res.headers.get('x-ratelimit-limit');
262
+ const remaining = res.headers.get('x-ratelimit-remaining');
263
+ const reset = res.headers.get('x-ratelimit-reset');
264
+ const resetAfter = res.headers.get('x-ratelimit-reset-after');
265
+ this.limit = limit ? Number(limit) : Infinity;
266
+ this.remaining = remaining ? Number(remaining) : 1;
267
+
268
+ this.reset = reset || resetAfter ? calculateReset(reset, resetAfter, serverDate) : Date.now();
269
+
270
+ // https://github.com/discord/discord-api-docs/issues/182
271
+ if (!resetAfter && request.route.includes('reactions')) {
272
+ this.reset = new Date(serverDate).getTime() - getAPIOffset(serverDate) + 250;
273
+ }
274
+
275
+ // Handle retryAfter, which means we have actually hit a rate limit
276
+ let retryAfter = res.headers.get('retry-after');
277
+ retryAfter = retryAfter ? Number(retryAfter) * 1_000 : -1;
278
+ if (retryAfter > 0) {
279
+ // If the global rate limit header is set, that means we hit the global rate limit
280
+ if (res.headers.get('x-ratelimit-global')) {
281
+ this.manager.globalRemaining = 0;
282
+ this.manager.globalReset = Date.now() + retryAfter;
283
+ } else if (!this.localLimited) {
284
+ /*
285
+ * This is a sublimit (e.g. 2 channel name changes/10 minutes) since the headers don't indicate a
286
+ * route-wide rate limit. Don't update remaining or reset to avoid rate limiting the whole
287
+ * endpoint, just set a reset time on the request itself to avoid retrying too soon.
288
+ */
289
+ sublimitTimeout = retryAfter;
290
+ }
291
+ }
292
+ }
293
+
294
+ // Count the invalid requests
295
+ if (res.status === 401 || res.status === 403 || res.status === 429) {
296
+ if (!invalidCountResetTime || invalidCountResetTime < Date.now()) {
297
+ invalidCountResetTime = Date.now() + 1_000 * 60 * 10;
298
+ invalidCount = 0;
299
+ }
300
+ invalidCount++;
301
+
302
+ const emitInvalid =
303
+ this.manager.client.listenerCount(INVALID_REQUEST_WARNING) &&
304
+ this.manager.client.options.invalidRequestWarningInterval > 0 &&
305
+ invalidCount % this.manager.client.options.invalidRequestWarningInterval === 0;
306
+ if (emitInvalid) {
307
+ /**
308
+ * @typedef {Object} InvalidRequestWarningData
309
+ * @property {number} count Number of invalid requests that have been made in the window
310
+ * @property {number} remainingTime Time in milliseconds remaining before the count resets
311
+ */
312
+
313
+ /**
314
+ * Emitted periodically when the process sends invalid requests to let users avoid the
315
+ * 10k invalid requests in 10 minutes threshold that causes a ban
316
+ * @event BaseClient#invalidRequestWarning
317
+ * @param {InvalidRequestWarningData} invalidRequestWarningData Object containing the invalid request info
318
+ */
319
+ this.manager.client.emit(INVALID_REQUEST_WARNING, {
320
+ count: invalidCount,
321
+ remainingTime: invalidCountResetTime - Date.now(),
322
+ });
323
+ }
324
+ }
325
+
326
+ // Handle 2xx and 3xx responses
327
+ if (res.ok) {
328
+ // Nothing wrong with the request, proceed with the next one
329
+ return parseResponse(res);
330
+ }
331
+
332
+ // Handle 4xx responses
333
+ if (res.status >= 400 && res.status < 500) {
334
+ // Handle ratelimited requests
335
+ if (res.status === 429) {
336
+ const isGlobal = this.globalLimited;
337
+ let limit, timeout;
338
+ if (isGlobal) {
339
+ // Set the variables based on the global rate limit
340
+ limit = this.manager.globalLimit;
341
+ timeout = this.manager.globalReset + this.manager.client.options.restTimeOffset - Date.now();
342
+ } else {
343
+ // Set the variables based on the route-specific rate limit
344
+ limit = this.limit;
345
+ timeout = this.reset + this.manager.client.options.restTimeOffset - Date.now();
346
+ }
347
+
348
+ this.manager.client.emit(
349
+ DEBUG,
350
+ `Hit a 429 while executing a request.
351
+ Global : ${isGlobal}
352
+ Method : ${request.method}
353
+ Path : ${request.path}
354
+ Route : ${request.route}
355
+ Limit : ${limit}
356
+ Timeout : ${timeout}ms
357
+ Sublimit: ${sublimitTimeout ? `${sublimitTimeout}ms` : 'None'}`,
358
+ );
359
+
360
+ await this.onRateLimit(request, limit, timeout, isGlobal);
361
+
362
+ // If caused by a sublimit, wait it out here so other requests on the route can be handled
363
+ if (sublimitTimeout) {
364
+ await sleep(sublimitTimeout);
365
+ }
366
+ return this.execute(request);
367
+ }
368
+
369
+ // Handle possible malformed requests
370
+ let data;
371
+ try {
372
+ data = await parseResponse(res);
373
+ if (data?.captcha_service) {
374
+ /**
375
+ * Emitted when a request is blocked by a captcha
376
+ * @event Client#captchaRequired
377
+ * @param {Request} request The request that was blocked
378
+ * @param {Captcha} data The data returned by Discord
379
+ */
380
+ this.manager.client.emit(CAPTCHA_REQUIRED, request, data);
381
+ }
382
+ if (
383
+ data?.captcha_service &&
384
+ this.manager.client.options.captchaService &&
385
+ request.retries < this.manager.client.options.captchaRetryLimit &&
386
+ captchaMessage.some(s => data.captcha_key[0].includes(s))
387
+ ) {
388
+ // Retry the request after a captcha is solved
389
+ this.manager.client.emit(
390
+ DEBUG,
391
+ `Hit a captcha while executing a request. Solving captcha ...
392
+ Method : ${request.method}
393
+ Path : ${request.path}
394
+ Route : ${request.route}
395
+ Info : ${inspect(data, { depth: null })}`,
396
+ );
397
+ const captcha = await this.manager.captchaService.solve(
398
+ data,
399
+ this.manager.client.options.http.headers['User-Agent'],
400
+ );
401
+ // Sleep: await this.manager.client.sleep(5_000);
402
+ this.manager.client.emit(
403
+ DEBUG,
404
+ `Captcha details:
405
+ Method : ${request.method}
406
+ Path : ${request.path}
407
+ Route : ${request.route}
408
+ Key : ${captcha ? `${captcha.slice(0, 30)}...` : '[Captcha not solved]'}
409
+ rqToken : ${data.captcha_rqtoken}`,
410
+ );
411
+ request.retries++;
412
+ return this.execute(request, captcha, data.captcha_rqtoken);
413
+ }
414
+ } catch (err) {
415
+ throw new HTTPError(err.message, err.constructor.name, err.status, request);
416
+ }
417
+ throw new DiscordAPIError(data, res.status, request);
418
+ }
419
+
420
+ // Handle 5xx responses
421
+ if (res.status >= 500 && res.status < 600) {
422
+ // Retry the specified number of times for possible serverside issues
423
+ if (request.retries === this.manager.client.options.retryLimit) {
424
+ throw new HTTPError(res.statusText, res.constructor.name, res.status, request);
425
+ }
426
+
427
+ request.retries++;
428
+ return this.execute(request);
429
+ }
430
+
431
+ // Fallback in the rare case a status code outside the range 200..=599 is returned
432
+ return null;
433
+ }
434
+ }
435
+
436
+ module.exports = RequestHandler;
437
+
438
+ /**
439
+ * @external HTTPMethod
440
+ * @see {@link https://developer.mozilla.org/en-US/docs/Web/HTTP/Methods}
441
+ */
442
+
443
+ /**
444
+ * @external Response
445
+ * @see {@link https://developer.mozilla.org/en-US/docs/Web/API/Response}
446
+ */