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