djs-selfbot-v13 3.1.8 → 3.7.1

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 (364) hide show
  1. package/LICENSE +1 -1
  2. package/README.md +48 -37
  3. package/package.json +44 -26
  4. package/src/WebSocket.js +39 -39
  5. package/src/client/BaseClient.js +86 -86
  6. package/src/client/Client.js +934 -765
  7. package/src/client/WebhookClient.js +61 -61
  8. package/src/client/actions/Action.js +116 -120
  9. package/src/client/actions/ActionsManager.js +80 -78
  10. package/src/client/actions/ApplicationCommandPermissionsUpdate.js +34 -34
  11. package/src/client/actions/AutoModerationActionExecution.js +27 -27
  12. package/src/client/actions/AutoModerationRuleCreate.js +28 -28
  13. package/src/client/actions/AutoModerationRuleDelete.js +32 -32
  14. package/src/client/actions/AutoModerationRuleUpdate.js +30 -30
  15. package/src/client/actions/ChannelCreate.js +23 -23
  16. package/src/client/actions/ChannelDelete.js +39 -39
  17. package/src/client/actions/ChannelUpdate.js +43 -43
  18. package/src/client/actions/GuildAuditLogEntryCreate.js +29 -29
  19. package/src/client/actions/GuildBanAdd.js +20 -20
  20. package/src/client/actions/GuildBanRemove.js +25 -25
  21. package/src/client/actions/GuildChannelsPositionUpdate.js +21 -21
  22. package/src/client/actions/GuildDelete.js +65 -65
  23. package/src/client/actions/GuildEmojiCreate.js +20 -20
  24. package/src/client/actions/GuildEmojiDelete.js +21 -21
  25. package/src/client/actions/GuildEmojiUpdate.js +20 -20
  26. package/src/client/actions/GuildEmojisUpdate.js +34 -34
  27. package/src/client/actions/GuildIntegrationsUpdate.js +19 -19
  28. package/src/client/actions/GuildMemberRemove.js +33 -33
  29. package/src/client/actions/GuildMemberUpdate.js +44 -44
  30. package/src/client/actions/GuildRoleCreate.js +25 -25
  31. package/src/client/actions/GuildRoleDelete.js +31 -31
  32. package/src/client/actions/GuildRoleUpdate.js +39 -39
  33. package/src/client/actions/GuildRolesPositionUpdate.js +21 -21
  34. package/src/client/actions/GuildScheduledEventCreate.js +27 -27
  35. package/src/client/actions/GuildScheduledEventDelete.js +31 -31
  36. package/src/client/actions/GuildScheduledEventUpdate.js +30 -30
  37. package/src/client/actions/GuildScheduledEventUserAdd.js +32 -32
  38. package/src/client/actions/GuildScheduledEventUserRemove.js +32 -32
  39. package/src/client/actions/GuildStickerCreate.js +20 -20
  40. package/src/client/actions/GuildStickerDelete.js +21 -21
  41. package/src/client/actions/GuildStickerUpdate.js +20 -20
  42. package/src/client/actions/GuildStickersUpdate.js +34 -34
  43. package/src/client/actions/GuildUpdate.js +33 -33
  44. package/src/client/actions/InviteCreate.js +28 -28
  45. package/src/client/actions/InviteDelete.js +30 -30
  46. package/src/client/actions/MessageCreate.js +50 -46
  47. package/src/client/actions/MessageDelete.js +32 -32
  48. package/src/client/actions/MessageDeleteBulk.js +46 -46
  49. package/src/client/actions/MessagePollVoteAdd.js +33 -0
  50. package/src/client/actions/MessagePollVoteRemove.js +33 -0
  51. package/src/client/actions/MessageReactionAdd.js +68 -56
  52. package/src/client/actions/MessageReactionRemove.js +50 -45
  53. package/src/client/actions/MessageReactionRemoveAll.js +33 -33
  54. package/src/client/actions/MessageReactionRemoveEmoji.js +28 -28
  55. package/src/client/actions/MessageUpdate.js +26 -26
  56. package/src/client/actions/PresenceUpdate.js +50 -46
  57. package/src/client/actions/StageInstanceCreate.js +28 -28
  58. package/src/client/actions/StageInstanceDelete.js +33 -33
  59. package/src/client/actions/StageInstanceUpdate.js +30 -30
  60. package/src/client/actions/ThreadCreate.js +24 -24
  61. package/src/client/actions/ThreadDelete.js +32 -32
  62. package/src/client/actions/ThreadListSync.js +59 -59
  63. package/src/client/actions/ThreadMemberUpdate.js +30 -30
  64. package/src/client/actions/ThreadMembersUpdate.js +34 -34
  65. package/src/client/actions/TypingStart.js +29 -29
  66. package/src/client/actions/UserUpdate.js +35 -35
  67. package/src/client/actions/VoiceStateUpdate.js +50 -57
  68. package/src/client/actions/WebhooksUpdate.js +20 -20
  69. package/src/client/voice/ClientVoiceManager.js +151 -51
  70. package/src/client/voice/VoiceConnection.js +1249 -0
  71. package/src/client/voice/dispatcher/AnnexBDispatcher.js +120 -0
  72. package/src/client/voice/dispatcher/AudioDispatcher.js +145 -0
  73. package/src/client/voice/dispatcher/BaseDispatcher.js +459 -0
  74. package/src/client/voice/dispatcher/VPxDispatcher.js +54 -0
  75. package/src/client/voice/dispatcher/VideoDispatcher.js +68 -0
  76. package/src/client/voice/networking/VoiceUDPClient.js +173 -0
  77. package/src/client/voice/networking/VoiceWebSocket.js +286 -0
  78. package/src/client/voice/player/MediaPlayer.js +321 -0
  79. package/src/client/voice/player/processing/AnnexBNalSplitter.js +244 -0
  80. package/src/client/voice/player/processing/IvfSplitter.js +106 -0
  81. package/src/client/voice/player/processing/PCMInsertSilence.js +37 -0
  82. package/src/client/voice/receiver/PacketHandler.js +260 -0
  83. package/src/client/voice/receiver/Receiver.js +96 -0
  84. package/src/client/voice/receiver/Recorder.js +173 -0
  85. package/src/client/voice/util/Function.js +116 -0
  86. package/src/client/voice/util/PlayInterface.js +122 -0
  87. package/src/client/voice/util/Secretbox.js +64 -0
  88. package/src/client/voice/util/Silence.js +16 -0
  89. package/src/client/voice/util/Socket.js +62 -0
  90. package/src/client/voice/util/VolumeInterface.js +104 -0
  91. package/src/client/websocket/WebSocketManager.js +392 -392
  92. package/src/client/websocket/WebSocketShard.js +907 -906
  93. package/src/client/websocket/handlers/APPLICATION_COMMAND_CREATE.js +18 -18
  94. package/src/client/websocket/handlers/APPLICATION_COMMAND_DELETE.js +20 -20
  95. package/src/client/websocket/handlers/APPLICATION_COMMAND_PERMISSIONS_UPDATE.js +5 -5
  96. package/src/client/websocket/handlers/APPLICATION_COMMAND_UPDATE.js +20 -20
  97. package/src/client/websocket/handlers/AUTO_MODERATION_ACTION_EXECUTION.js +5 -5
  98. package/src/client/websocket/handlers/AUTO_MODERATION_RULE_CREATE.js +5 -5
  99. package/src/client/websocket/handlers/AUTO_MODERATION_RULE_DELETE.js +5 -5
  100. package/src/client/websocket/handlers/AUTO_MODERATION_RULE_UPDATE.js +5 -5
  101. package/src/client/websocket/handlers/CALL_CREATE.js +14 -14
  102. package/src/client/websocket/handlers/CALL_DELETE.js +11 -11
  103. package/src/client/websocket/handlers/CALL_UPDATE.js +11 -11
  104. package/src/client/websocket/handlers/CHANNEL_CREATE.js +5 -5
  105. package/src/client/websocket/handlers/CHANNEL_DELETE.js +5 -5
  106. package/src/client/websocket/handlers/CHANNEL_PINS_UPDATE.js +22 -22
  107. package/src/client/websocket/handlers/CHANNEL_RECIPIENT_ADD.js +19 -19
  108. package/src/client/websocket/handlers/CHANNEL_RECIPIENT_REMOVE.js +16 -16
  109. package/src/client/websocket/handlers/CHANNEL_UPDATE.js +16 -16
  110. package/src/client/websocket/handlers/GUILD_AUDIT_LOG_ENTRY_CREATE.js +5 -5
  111. package/src/client/websocket/handlers/GUILD_BAN_ADD.js +5 -5
  112. package/src/client/websocket/handlers/GUILD_BAN_REMOVE.js +5 -5
  113. package/src/client/websocket/handlers/GUILD_CREATE.js +52 -53
  114. package/src/client/websocket/handlers/GUILD_DELETE.js +5 -5
  115. package/src/client/websocket/handlers/GUILD_EMOJIS_UPDATE.js +5 -5
  116. package/src/client/websocket/handlers/GUILD_INTEGRATIONS_UPDATE.js +5 -5
  117. package/src/client/websocket/handlers/GUILD_MEMBERS_CHUNK.js +39 -39
  118. package/src/client/websocket/handlers/GUILD_MEMBER_ADD.js +20 -20
  119. package/src/client/websocket/handlers/GUILD_MEMBER_REMOVE.js +5 -5
  120. package/src/client/websocket/handlers/GUILD_MEMBER_UPDATE.js +5 -5
  121. package/src/client/websocket/handlers/GUILD_ROLE_CREATE.js +5 -5
  122. package/src/client/websocket/handlers/GUILD_ROLE_DELETE.js +5 -5
  123. package/src/client/websocket/handlers/GUILD_ROLE_UPDATE.js +5 -5
  124. package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_CREATE.js +5 -5
  125. package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_DELETE.js +5 -5
  126. package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_UPDATE.js +5 -5
  127. package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_USER_ADD.js +5 -5
  128. package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_USER_REMOVE.js +5 -5
  129. package/src/client/websocket/handlers/GUILD_STICKERS_UPDATE.js +5 -5
  130. package/src/client/websocket/handlers/GUILD_UPDATE.js +5 -5
  131. package/src/client/websocket/handlers/INTERACTION_MODAL_CREATE.js +12 -12
  132. package/src/client/websocket/handlers/INVITE_CREATE.js +5 -5
  133. package/src/client/websocket/handlers/INVITE_DELETE.js +5 -5
  134. package/src/client/websocket/handlers/MESSAGE_CREATE.js +5 -5
  135. package/src/client/websocket/handlers/MESSAGE_DELETE.js +5 -5
  136. package/src/client/websocket/handlers/MESSAGE_DELETE_BULK.js +5 -5
  137. package/src/client/websocket/handlers/MESSAGE_POLL_VOTE_ADD.js +5 -0
  138. package/src/client/websocket/handlers/MESSAGE_POLL_VOTE_REMOVE.js +5 -0
  139. package/src/client/websocket/handlers/MESSAGE_REACTION_ADD.js +5 -5
  140. package/src/client/websocket/handlers/MESSAGE_REACTION_REMOVE.js +5 -5
  141. package/src/client/websocket/handlers/MESSAGE_REACTION_REMOVE_ALL.js +5 -5
  142. package/src/client/websocket/handlers/MESSAGE_REACTION_REMOVE_EMOJI.js +5 -5
  143. package/src/client/websocket/handlers/MESSAGE_UPDATE.js +16 -16
  144. package/src/client/websocket/handlers/PRESENCE_UPDATE.js +5 -5
  145. package/src/client/websocket/handlers/READY.js +121 -80
  146. package/src/client/websocket/handlers/RELATIONSHIP_ADD.js +19 -19
  147. package/src/client/websocket/handlers/RELATIONSHIP_REMOVE.js +17 -17
  148. package/src/client/websocket/handlers/RELATIONSHIP_UPDATE.js +41 -41
  149. package/src/client/websocket/handlers/RESUMED.js +14 -14
  150. package/src/client/websocket/handlers/STAGE_INSTANCE_CREATE.js +5 -5
  151. package/src/client/websocket/handlers/STAGE_INSTANCE_DELETE.js +5 -5
  152. package/src/client/websocket/handlers/STAGE_INSTANCE_UPDATE.js +5 -5
  153. package/src/client/websocket/handlers/THREAD_CREATE.js +5 -5
  154. package/src/client/websocket/handlers/THREAD_DELETE.js +5 -5
  155. package/src/client/websocket/handlers/THREAD_LIST_SYNC.js +5 -5
  156. package/src/client/websocket/handlers/THREAD_MEMBERS_UPDATE.js +5 -5
  157. package/src/client/websocket/handlers/THREAD_MEMBER_UPDATE.js +5 -5
  158. package/src/client/websocket/handlers/THREAD_UPDATE.js +16 -16
  159. package/src/client/websocket/handlers/TYPING_START.js +5 -5
  160. package/src/client/websocket/handlers/USER_GUILD_SETTINGS_UPDATE.js +6 -6
  161. package/src/client/websocket/handlers/USER_NOTE_UPDATE.js +5 -5
  162. package/src/client/websocket/handlers/USER_REQUIRED_ACTION_UPDATE.js +78 -78
  163. package/src/client/websocket/handlers/USER_SETTINGS_UPDATE.js +5 -5
  164. package/src/client/websocket/handlers/USER_UPDATE.js +5 -5
  165. package/src/client/websocket/handlers/VOICE_CHANNEL_EFFECT_SEND.js +16 -0
  166. package/src/client/websocket/handlers/VOICE_CHANNEL_STATUS_UPDATE.js +12 -12
  167. package/src/client/websocket/handlers/VOICE_SERVER_UPDATE.js +6 -6
  168. package/src/client/websocket/handlers/VOICE_STATE_UPDATE.js +5 -5
  169. package/src/client/websocket/handlers/WEBHOOKS_UPDATE.js +5 -5
  170. package/src/client/websocket/handlers/index.js +84 -81
  171. package/src/errors/DJSError.js +61 -61
  172. package/src/errors/Messages.js +217 -182
  173. package/src/errors/index.js +4 -4
  174. package/src/index.js +172 -158
  175. package/src/managers/ApplicationCommandManager.js +264 -264
  176. package/src/managers/ApplicationCommandPermissionsManager.js +417 -417
  177. package/src/managers/AutoModerationRuleManager.js +296 -296
  178. package/src/managers/BaseGuildEmojiManager.js +80 -80
  179. package/src/managers/BaseManager.js +19 -19
  180. package/src/managers/BillingManager.js +66 -66
  181. package/src/managers/CachedManager.js +71 -71
  182. package/src/managers/ChannelManager.js +148 -138
  183. package/src/managers/ClientUserSettingManager.js +372 -372
  184. package/src/managers/DataManager.js +61 -61
  185. package/src/managers/GuildBanManager.js +250 -204
  186. package/src/managers/GuildChannelManager.js +488 -504
  187. package/src/managers/GuildEmojiManager.js +171 -171
  188. package/src/managers/GuildEmojiRoleManager.js +118 -118
  189. package/src/managers/GuildForumThreadManager.js +108 -108
  190. package/src/managers/GuildInviteManager.js +213 -213
  191. package/src/managers/GuildManager.js +338 -304
  192. package/src/managers/GuildMemberManager.js +599 -593
  193. package/src/managers/GuildMemberRoleManager.js +195 -191
  194. package/src/managers/GuildScheduledEventManager.js +314 -296
  195. package/src/managers/GuildSettingManager.js +155 -155
  196. package/src/managers/GuildStickerManager.js +179 -179
  197. package/src/managers/GuildTextThreadManager.js +98 -98
  198. package/src/managers/InteractionManager.js +39 -39
  199. package/src/managers/MessageManager.js +423 -391
  200. package/src/managers/PermissionOverwriteManager.js +164 -166
  201. package/src/managers/PresenceManager.js +71 -58
  202. package/src/managers/ReactionManager.js +67 -67
  203. package/src/managers/ReactionUserManager.js +73 -71
  204. package/src/managers/RelationshipManager.js +278 -265
  205. package/src/managers/RoleManager.js +448 -352
  206. package/src/managers/SessionManager.js +66 -0
  207. package/src/managers/StageInstanceManager.js +162 -162
  208. package/src/managers/ThreadManager.js +175 -174
  209. package/src/managers/ThreadMemberManager.js +186 -186
  210. package/src/managers/UserManager.js +136 -146
  211. package/src/managers/UserNoteManager.js +53 -53
  212. package/src/managers/VoiceStateManager.js +59 -37
  213. package/src/rest/APIRequest.js +154 -154
  214. package/src/rest/APIRouter.js +53 -53
  215. package/src/rest/DiscordAPIError.js +119 -104
  216. package/src/rest/HTTPError.js +62 -62
  217. package/src/rest/RESTManager.js +67 -62
  218. package/src/rest/RateLimitError.js +55 -55
  219. package/src/rest/RequestHandler.js +466 -444
  220. package/src/sharding/Shard.js +444 -443
  221. package/src/sharding/ShardClientUtil.js +279 -275
  222. package/src/sharding/ShardingManager.js +319 -318
  223. package/src/structures/AnonymousGuild.js +98 -98
  224. package/src/structures/ApplicationCommand.js +593 -593
  225. package/src/structures/ApplicationRoleConnectionMetadata.js +48 -48
  226. package/src/structures/AutoModerationActionExecution.js +89 -89
  227. package/src/structures/AutoModerationRule.js +294 -294
  228. package/src/structures/AutocompleteInteraction.js +107 -107
  229. package/src/structures/Base.js +43 -43
  230. package/src/structures/BaseCommandInteraction.js +211 -211
  231. package/src/structures/BaseGuild.js +116 -116
  232. package/src/structures/BaseGuildEmoji.js +56 -56
  233. package/src/structures/BaseGuildTextChannel.js +191 -191
  234. package/src/structures/BaseGuildVoiceChannel.js +241 -241
  235. package/src/structures/BaseMessageComponent.js +181 -114
  236. package/src/structures/ButtonInteraction.js +11 -11
  237. package/src/structures/CallState.js +63 -63
  238. package/src/structures/CategoryChannel.js +85 -85
  239. package/src/structures/Channel.js +284 -270
  240. package/src/structures/ClientPresence.js +77 -88
  241. package/src/structures/ClientUser.js +479 -450
  242. package/src/structures/CommandInteraction.js +41 -41
  243. package/src/structures/CommandInteractionOptionResolver.js +276 -276
  244. package/src/structures/ContainerComponent.js +68 -0
  245. package/src/structures/ContextMenuInteraction.js +65 -65
  246. package/src/structures/DMChannel.js +219 -217
  247. package/src/structures/DirectoryChannel.js +20 -20
  248. package/src/structures/Emoji.js +148 -148
  249. package/src/structures/FileComponent.js +49 -0
  250. package/src/structures/ForumChannel.js +31 -261
  251. package/src/structures/GroupDMChannel.js +394 -387
  252. package/src/structures/Guild.js +1643 -1608
  253. package/src/structures/GuildAuditLogs.js +746 -729
  254. package/src/structures/GuildBan.js +59 -59
  255. package/src/structures/GuildBoost.js +108 -108
  256. package/src/structures/GuildChannel.js +470 -468
  257. package/src/structures/GuildEmoji.js +161 -161
  258. package/src/structures/GuildMember.js +636 -568
  259. package/src/structures/GuildPreview.js +191 -191
  260. package/src/structures/GuildPreviewEmoji.js +27 -27
  261. package/src/structures/GuildScheduledEvent.js +536 -441
  262. package/src/structures/GuildTemplate.js +236 -236
  263. package/src/structures/Integration.js +188 -188
  264. package/src/structures/IntegrationApplication.js +96 -96
  265. package/src/structures/Interaction.js +290 -290
  266. package/src/structures/InteractionCollector.js +248 -248
  267. package/src/structures/InteractionWebhook.js +43 -43
  268. package/src/structures/Invite.js +358 -358
  269. package/src/structures/InviteGuild.js +23 -23
  270. package/src/structures/InviteStageInstance.js +86 -86
  271. package/src/structures/MediaChannel.js +11 -0
  272. package/src/structures/MediaGalleryComponent.js +41 -0
  273. package/src/structures/MediaGalleryItem.js +47 -0
  274. package/src/structures/Message.js +1252 -1236
  275. package/src/structures/MessageActionRow.js +105 -103
  276. package/src/structures/MessageAttachment.js +216 -204
  277. package/src/structures/MessageButton.js +166 -165
  278. package/src/structures/MessageCollector.js +146 -146
  279. package/src/structures/MessageComponentInteraction.js +120 -120
  280. package/src/structures/MessageContextMenuInteraction.js +20 -20
  281. package/src/structures/MessageEmbed.js +596 -586
  282. package/src/structures/MessageMentions.js +273 -273
  283. package/src/structures/MessagePayload.js +354 -316
  284. package/src/structures/MessageReaction.js +181 -171
  285. package/src/structures/MessageSelectMenu.js +141 -140
  286. package/src/structures/Modal.js +161 -174
  287. package/src/structures/ModalSubmitFieldsResolver.js +53 -53
  288. package/src/structures/ModalSubmitInteraction.js +119 -119
  289. package/src/structures/NewsChannel.js +32 -32
  290. package/src/structures/OAuth2Guild.js +28 -28
  291. package/src/structures/PermissionOverwrites.js +198 -196
  292. package/src/structures/Poll.js +108 -0
  293. package/src/structures/PollAnswer.js +88 -0
  294. package/src/structures/Presence.js +1105 -443
  295. package/src/structures/ReactionCollector.js +229 -229
  296. package/src/structures/ReactionEmoji.js +31 -31
  297. package/src/structures/Role.js +590 -531
  298. package/src/structures/SectionComponent.js +48 -0
  299. package/src/structures/SelectMenuInteraction.js +21 -21
  300. package/src/structures/SeparatorComponent.js +48 -0
  301. package/src/structures/Session.js +81 -0
  302. package/src/structures/StageChannel.js +104 -104
  303. package/src/structures/StageInstance.js +208 -208
  304. package/src/structures/Sticker.js +310 -310
  305. package/src/structures/StickerPack.js +95 -95
  306. package/src/structures/StoreChannel.js +56 -56
  307. package/src/structures/Team.js +118 -118
  308. package/src/structures/TeamMember.js +80 -71
  309. package/src/structures/TextChannel.js +33 -33
  310. package/src/structures/TextDisplayComponent.js +40 -0
  311. package/src/structures/TextInputComponent.js +132 -131
  312. package/src/structures/ThreadChannel.js +605 -607
  313. package/src/structures/ThreadMember.js +105 -105
  314. package/src/structures/ThreadOnlyChannel.js +249 -0
  315. package/src/structures/ThumbnailComponent.js +57 -0
  316. package/src/structures/Typing.js +74 -74
  317. package/src/structures/UnfurledMediaItem.js +29 -0
  318. package/src/structures/User.js +640 -509
  319. package/src/structures/UserContextMenuInteraction.js +29 -29
  320. package/src/structures/VoiceChannel.js +110 -110
  321. package/src/structures/VoiceChannelEffect.js +69 -0
  322. package/src/structures/VoiceRegion.js +53 -53
  323. package/src/structures/VoiceState.js +354 -341
  324. package/src/structures/WebEmbed.js +373 -373
  325. package/src/structures/Webhook.js +478 -467
  326. package/src/structures/WelcomeChannel.js +60 -60
  327. package/src/structures/WelcomeScreen.js +48 -48
  328. package/src/structures/Widget.js +87 -87
  329. package/src/structures/WidgetMember.js +99 -99
  330. package/src/structures/interfaces/Application.js +825 -313
  331. package/src/structures/interfaces/Collector.js +300 -300
  332. package/src/structures/interfaces/InteractionResponses.js +313 -313
  333. package/src/structures/interfaces/TextBasedChannel.js +759 -721
  334. package/src/util/APITypes.js +59 -0
  335. package/src/util/ActivityFlags.js +44 -44
  336. package/src/util/ApplicationFlags.js +76 -76
  337. package/src/util/AttachmentFlags.js +38 -38
  338. package/src/util/BitField.js +170 -170
  339. package/src/util/ChannelFlags.js +45 -45
  340. package/src/util/Constants.js +1914 -1760
  341. package/src/util/DataResolver.js +146 -145
  342. package/src/util/Formatters.js +228 -228
  343. package/src/util/GuildMemberFlags.js +43 -43
  344. package/src/util/Intents.js +74 -74
  345. package/src/util/InviteFlags.js +34 -29
  346. package/src/util/LimitedCollection.js +131 -131
  347. package/src/util/MessageFlags.js +63 -54
  348. package/src/util/Options.js +358 -342
  349. package/src/util/Permissions.js +202 -192
  350. package/src/util/PremiumUsageFlags.js +31 -31
  351. package/src/util/PurchasedFlags.js +33 -33
  352. package/src/util/RemoteAuth.js +382 -379
  353. package/src/util/RoleFlags.js +37 -37
  354. package/src/util/SnowflakeUtil.js +92 -92
  355. package/src/util/Speaking.js +33 -0
  356. package/src/util/Sweepers.js +466 -466
  357. package/src/util/SystemChannelFlags.js +55 -55
  358. package/src/util/ThreadMemberFlags.js +30 -30
  359. package/src/util/UserFlags.js +104 -104
  360. package/src/util/Util.js +1048 -803
  361. package/typings/enums.d.ts +439 -291
  362. package/typings/index.d.ts +8247 -7390
  363. package/typings/rawDataTypes.d.ts +403 -342
  364. package/src/structures/RichPresence.js +0 -702
@@ -1,721 +1,759 @@
1
- 'use strict';
2
-
3
- /* eslint-disable import/order */
4
- const MessageCollector = require('../MessageCollector');
5
- const MessagePayload = require('../MessagePayload');
6
- const { InteractionTypes, ApplicationCommandOptionTypes, Events } = require('../../util/Constants');
7
- const { Error } = require('../../errors');
8
- const SnowflakeUtil = require('../../util/SnowflakeUtil');
9
- const { setTimeout } = require('node:timers');
10
- const { s } = require('@sapphire/shapeshift');
11
- const Util = require('../../util/Util');
12
- const validateName = stringName =>
13
- s.string
14
- .lengthGreaterThanOrEqual(1)
15
- .lengthLessThanOrEqual(32)
16
- .regex(/^[\p{Ll}\p{Lm}\p{Lo}\p{N}\p{sc=Devanagari}\p{sc=Thai}_-]+$/u)
17
- .setValidationEnabled(true)
18
- .parse(stringName);
19
-
20
- /**
21
- * Interface for classes that have text-channel-like features.
22
- * @interface
23
- */
24
- class TextBasedChannel {
25
- constructor() {
26
- /**
27
- * A manager of the messages sent to this channel
28
- * @type {MessageManager}
29
- */
30
- this.messages = new MessageManager(this);
31
-
32
- /**
33
- * The channel's last message id, if one was sent
34
- * @type {?Snowflake}
35
- */
36
- this.lastMessageId = null;
37
-
38
- /**
39
- * The timestamp when the last pinned message was pinned, if there was one
40
- * @type {?number}
41
- */
42
- this.lastPinTimestamp = null;
43
- }
44
-
45
- /**
46
- * The Message object of the last message in the channel, if one was sent
47
- * @type {?Message}
48
- * @readonly
49
- */
50
- get lastMessage() {
51
- return this.messages.resolve(this.lastMessageId);
52
- }
53
-
54
- /**
55
- * The date when the last pinned message was pinned, if there was one
56
- * @type {?Date}
57
- * @readonly
58
- */
59
- get lastPinAt() {
60
- return this.lastPinTimestamp ? new Date(this.lastPinTimestamp) : null;
61
- }
62
-
63
- /**
64
- * Base options provided when sending.
65
- * @typedef {Object} BaseMessageOptions
66
- * @property {MessageActivity} [activity] Group activity
67
- * @property {boolean} [tts=false] Whether or not the message should be spoken aloud
68
- * @property {string} [nonce=''] The nonce for the message
69
- * @property {string} [content=''] The content for the message
70
- * @property {Array<(MessageEmbed|APIEmbed)>} [embeds] The embeds for the message
71
- * (see [here](https://discord.com/developers/docs/resources/channel#embed-object) for more details)
72
- * @property {MessageMentionOptions} [allowedMentions] Which mentions should be parsed from the message content
73
- * (see [here](https://discord.com/developers/docs/resources/channel#allowed-mentions-object) for more details)
74
- * @property {Array<(FileOptions|BufferResolvable|MessageAttachment[])>} [files] Files to send with the message
75
- * @property {Array<(MessageActionRow|MessageActionRowOptions)>} [components]
76
- * Action rows containing interactive components for the message (buttons, select menus)
77
- * @property {MessageAttachment[]} [attachments] Attachments to send in the message
78
- */
79
-
80
- /**
81
- * Options provided when sending or editing a message.
82
- * @typedef {BaseMessageOptions} MessageOptions
83
- * @property {ReplyOptions} [reply] The options for replying to a message
84
- * @property {StickerResolvable[]} [stickers=[]] Stickers to send in the message
85
- * @property {MessageFlags} [flags] Which flags to set for the message.
86
- * Only `SUPPRESS_EMBEDS`, `SUPPRESS_NOTIFICATIONS` and `IS_VOICE_MESSAGE` can be set.
87
- */
88
-
89
- /**
90
- * Options provided to control parsing of mentions by Discord
91
- * @typedef {Object} MessageMentionOptions
92
- * @property {MessageMentionTypes[]} [parse] Types of mentions to be parsed
93
- * @property {Snowflake[]} [users] Snowflakes of Users to be parsed as mentions
94
- * @property {Snowflake[]} [roles] Snowflakes of Roles to be parsed as mentions
95
- * @property {boolean} [repliedUser=true] Whether the author of the Message being replied to should be pinged
96
- */
97
-
98
- /**
99
- * Types of mentions to enable in MessageMentionOptions.
100
- * - `roles`
101
- * - `users`
102
- * - `everyone`
103
- * @typedef {string} MessageMentionTypes
104
- */
105
-
106
- /**
107
- * @typedef {Object} FileOptions
108
- * @property {BufferResolvable} attachment File to attach
109
- * @property {string} [name='file.jpg'] Filename of the attachment
110
- * @property {string} description The description of the file
111
- */
112
-
113
- /**
114
- * Options for sending a message with a reply.
115
- * @typedef {Object} ReplyOptions
116
- * @property {MessageResolvable} messageReference The message to reply to (must be in the same channel and not system)
117
- * @property {boolean} [failIfNotExists=true] Whether to error if the referenced message
118
- * does not exist (creates a standard message in this case when false)
119
- */
120
-
121
- /**
122
- * Sends a message to this channel.
123
- * @param {string|MessagePayload|MessageOptions} options The options to provide
124
- * @returns {Promise<Message>}
125
- * @example
126
- * // Send a basic message
127
- * channel.send('hello!')
128
- * .then(message => console.log(`Sent message: ${message.content}`))
129
- * .catch(console.error);
130
- * @example
131
- * // Send a remote file
132
- * channel.send({
133
- * files: ['https://cdn.discordapp.com/icons/222078108977594368/6e1019b3179d71046e463a75915e7244.png?size=2048']
134
- * })
135
- * .then(console.log)
136
- * .catch(console.error);
137
- * @example
138
- * // Send a local file
139
- * channel.send({
140
- * files: [{
141
- * attachment: 'entire/path/to/file.jpg',
142
- * name: 'file.jpg',
143
- * description: 'A description of the file'
144
- * }]
145
- * })
146
- * .then(console.log)
147
- * .catch(console.error);
148
- */
149
- async send(options) {
150
- const User = require('../User');
151
- const { GuildMember } = require('../GuildMember');
152
-
153
- if (this instanceof User || this instanceof GuildMember) {
154
- const dm = await this.createDM();
155
- return dm.send(options);
156
- }
157
-
158
- let messagePayload;
159
-
160
- if (options instanceof MessagePayload) {
161
- messagePayload = options.resolveData();
162
- } else {
163
- messagePayload = MessagePayload.create(this, options).resolveData();
164
- }
165
-
166
- const { data, files } = await messagePayload.resolveFiles();
167
- // New API
168
- const attachments = await Util.getUploadURL(this.client, this.id, files);
169
- const requestPromises = attachments.map(async attachment => {
170
- await Util.uploadFile(files[attachment.id].file, attachment.upload_url);
171
- return {
172
- id: attachment.id,
173
- filename: files[attachment.id].name,
174
- uploaded_filename: attachment.upload_filename,
175
- description: files[attachment.id].description,
176
- duration_secs: files[attachment.id].duration_secs,
177
- waveform: files[attachment.id].waveform,
178
- };
179
- });
180
- const attachmentsData = await Promise.all(requestPromises);
181
- attachmentsData.sort((a, b) => parseInt(a.id) - parseInt(b.id));
182
- data.attachments = attachmentsData;
183
- // Empty Files
184
- const d = await this.client.api.channels[this.id].messages.post({ data });
185
-
186
- return this.messages.cache.get(d.id) ?? this.messages._add(d);
187
- }
188
-
189
- searchInteraction() {
190
- // Support Slash / ContextMenu
191
- // API https://canary.discord.com/api/v9/guilds/:id/application-command-index // Guild
192
- // https://canary.discord.com/api/v9/channels/:id/application-command-index // DM Channel
193
- // Updated: 07/01/2023
194
- return this.client.api[this.guild ? 'guilds' : 'channels'][this.guild?.id || this.id][
195
- 'application-command-index'
196
- ].get();
197
- }
198
-
199
- async sendSlash(botOrApplicationId, commandNameString, ...args) {
200
- // Parse commandName /role add user
201
- const cmd = commandNameString.trim().split(' ');
202
- // Ex: role add user => [role, add, user]
203
- // Parse: name, subGr, sub
204
- const commandName = validateName(cmd[0]);
205
- // Parse: role
206
- const sub = cmd.slice(1);
207
- // Parse: [add, user]
208
- for (let i = 0; i < sub.length; i++) {
209
- if (sub.length > 2) {
210
- throw new Error('INVALID_COMMAND_NAME', cmd);
211
- }
212
- validateName(sub[i]);
213
- }
214
- // Search all
215
- const data = await this.searchInteraction();
216
- // Find command...
217
- const filterCommand = data.application_commands.filter(obj =>
218
- // Filter: name | name_default
219
- [obj.name, obj.name_default].includes(commandName),
220
- );
221
- // Filter Bot
222
- botOrApplicationId = this.client.users.resolveId(botOrApplicationId);
223
- const application = data.applications.find(
224
- obj => obj.id == botOrApplicationId || obj.bot?.id == botOrApplicationId,
225
- );
226
- // Find Command with application
227
- const command = filterCommand.find(command => command.application_id == application.id);
228
-
229
- args = args.flat(2);
230
- let optionFormat = [];
231
- let attachments = [];
232
- let optionsMaxdepth, subGroup, subCommand;
233
- if (sub.length == 2) {
234
- // Subcommand Group > Subcommand
235
- // Find Sub group
236
- subGroup = command.options.find(
237
- obj =>
238
- obj.type == ApplicationCommandOptionTypes.SUB_COMMAND_GROUP && [obj.name, obj.name_default].includes(sub[0]),
239
- );
240
- if (!subGroup) throw new Error('SLASH_COMMAND_SUB_COMMAND_GROUP_INVALID', sub[0]);
241
- // Find Sub
242
- subCommand = subGroup.options.find(
243
- obj => obj.type == ApplicationCommandOptionTypes.SUB_COMMAND && [obj.name, obj.name_default].includes(sub[1]),
244
- );
245
- if (!subCommand) throw new Error('SLASH_COMMAND_SUB_COMMAND_INVALID', sub[1]);
246
- // Options
247
- optionsMaxdepth = subCommand.options;
248
- } else if (sub.length == 1) {
249
- // Subcommand
250
- subCommand = command.options.find(
251
- obj => obj.type == ApplicationCommandOptionTypes.SUB_COMMAND && [obj.name, obj.name_default].includes(sub[0]),
252
- );
253
- if (!subCommand) throw new Error('SLASH_COMMAND_SUB_COMMAND_INVALID', sub[0]);
254
- // Options
255
- optionsMaxdepth = subCommand.options;
256
- } else {
257
- optionsMaxdepth = command.options;
258
- }
259
- const valueRequired = optionsMaxdepth?.filter(o => o.required).length || 0;
260
- for (let i = 0; i < Math.min(args.length, optionsMaxdepth?.length || 0); i++) {
261
- const optionInput = optionsMaxdepth[i];
262
- const value = args[i];
263
- const parseData = await parseOption(
264
- this.client,
265
- optionInput,
266
- value,
267
- optionFormat,
268
- attachments,
269
- command,
270
- application.id,
271
- this.guild?.id,
272
- this.id,
273
- subGroup,
274
- subCommand,
275
- );
276
- optionFormat = parseData.optionFormat;
277
- attachments = parseData.attachments;
278
- }
279
- if (valueRequired > args.length) {
280
- throw new Error('SLASH_COMMAND_REQUIRED_OPTIONS_MISSING', valueRequired, optionFormat.length);
281
- }
282
- // Post
283
- let postData;
284
- if (subGroup) {
285
- postData = [
286
- {
287
- type: ApplicationCommandOptionTypes.SUB_COMMAND_GROUP,
288
- name: subGroup.name,
289
- options: [
290
- {
291
- type: ApplicationCommandOptionTypes.SUB_COMMAND,
292
- name: subCommand.name,
293
- options: optionFormat,
294
- },
295
- ],
296
- },
297
- ];
298
- } else if (subCommand) {
299
- postData = [
300
- {
301
- type: ApplicationCommandOptionTypes.SUB_COMMAND,
302
- name: subCommand.name,
303
- options: optionFormat,
304
- },
305
- ];
306
- } else {
307
- postData = optionFormat;
308
- }
309
- const nonce = SnowflakeUtil.generate();
310
- const body = createPostData(
311
- this.client,
312
- false,
313
- application.id,
314
- nonce,
315
- this.guild?.id,
316
- Boolean(command.guild_id),
317
- this.id,
318
- command.version,
319
- command.id,
320
- command.name_default || command.name,
321
- command.type,
322
- postData,
323
- attachments,
324
- );
325
- this.client.api.interactions.post({
326
- data: body,
327
- usePayloadJSON: true,
328
- });
329
- return new Promise((resolve, reject) => {
330
- const timeoutMs = 5_000;
331
- // Waiting for MsgCreate / ModalCreate
332
- const handler = data => {
333
- if (data.nonce !== nonce) return;
334
- clearTimeout(timeout);
335
- this.client.removeListener(Events.MESSAGE_CREATE, handler);
336
- this.client.removeListener(Events.INTERACTION_MODAL_CREATE, handler);
337
- this.client.decrementMaxListeners();
338
- resolve(data);
339
- };
340
- const timeout = setTimeout(() => {
341
- this.client.removeListener(Events.MESSAGE_CREATE, handler);
342
- this.client.removeListener(Events.INTERACTION_MODAL_CREATE, handler);
343
- this.client.decrementMaxListeners();
344
- reject(new Error('INTERACTION_FAILED'));
345
- }, timeoutMs).unref();
346
- this.client.incrementMaxListeners();
347
- this.client.on(Events.MESSAGE_CREATE, handler);
348
- this.client.on(Events.INTERACTION_MODAL_CREATE, handler);
349
- });
350
- }
351
-
352
- /**
353
- * Sends a typing indicator in the channel.
354
- * @returns {Promise<void>} Resolves upon the typing status being sent
355
- * @example
356
- * // Start typing in a channel
357
- * channel.sendTyping();
358
- */
359
- async sendTyping() {
360
- await this.client.api.channels(this.id).typing.post();
361
- }
362
-
363
- /**
364
- * Creates a Message Collector.
365
- * @param {MessageCollectorOptions} [options={}] The options to pass to the collector
366
- * @returns {MessageCollector}
367
- * @example
368
- * // Create a message collector
369
- * const filter = m => m.content.includes('discord');
370
- * const collector = channel.createMessageCollector({ filter, time: 15_000 });
371
- * collector.on('collect', m => console.log(`Collected ${m.content}`));
372
- * collector.on('end', collected => console.log(`Collected ${collected.size} items`));
373
- */
374
- createMessageCollector(options = {}) {
375
- return new MessageCollector(this, options);
376
- }
377
-
378
- /**
379
- * An object containing the same properties as CollectorOptions, but a few more:
380
- * @typedef {MessageCollectorOptions} AwaitMessagesOptions
381
- * @property {string[]} [errors] Stop/end reasons that cause the promise to reject
382
- */
383
-
384
- /**
385
- * Similar to createMessageCollector but in promise form.
386
- * Resolves with a collection of messages that pass the specified filter.
387
- * @param {AwaitMessagesOptions} [options={}] Optional options to pass to the internal collector
388
- * @returns {Promise<Collection<Snowflake, Message>>}
389
- * @example
390
- * // Await !vote messages
391
- * const filter = m => m.content.startsWith('!vote');
392
- * // Errors: ['time'] treats ending because of the time limit as an error
393
- * channel.awaitMessages({ filter, max: 4, time: 60_000, errors: ['time'] })
394
- * .then(collected => console.log(collected.size))
395
- * .catch(collected => console.log(`After a minute, only ${collected.size} out of 4 voted.`));
396
- */
397
- awaitMessages(options = {}) {
398
- return new Promise((resolve, reject) => {
399
- const collector = this.createMessageCollector(options);
400
- collector.once('end', (collection, reason) => {
401
- if (options.errors?.includes(reason)) {
402
- reject(collection);
403
- } else {
404
- resolve(collection);
405
- }
406
- });
407
- });
408
- }
409
-
410
- /**
411
- * Fetches all webhooks for the channel.
412
- * @returns {Promise<Collection<Snowflake, Webhook>>}
413
- * @example
414
- * // Fetch webhooks
415
- * channel.fetchWebhooks()
416
- * .then(hooks => console.log(`This channel has ${hooks.size} hooks`))
417
- * .catch(console.error);
418
- */
419
- fetchWebhooks() {
420
- return this.guild.channels.fetchWebhooks(this.id);
421
- }
422
-
423
- /**
424
- * Options used to create a {@link Webhook} in a guild text-based channel.
425
- * @typedef {Object} ChannelWebhookCreateOptions
426
- * @property {?(BufferResolvable|Base64Resolvable)} [avatar] Avatar for the webhook
427
- * @property {string} [reason] Reason for creating the webhook
428
- */
429
-
430
- /**
431
- * Creates a webhook for the channel.
432
- * @param {string} name The name of the webhook
433
- * @param {ChannelWebhookCreateOptions} [options] Options for creating the webhook
434
- * @returns {Promise<Webhook>} Returns the created Webhook
435
- * @example
436
- * // Create a webhook for the current channel
437
- * channel.createWebhook('Snek', {
438
- * avatar: 'https://i.imgur.com/mI8XcpG.jpg',
439
- * reason: 'Needed a cool new Webhook'
440
- * })
441
- * .then(console.log)
442
- * .catch(console.error)
443
- */
444
- createWebhook(name, options = {}) {
445
- return this.guild.channels.createWebhook(this.id, name, options);
446
- }
447
-
448
- /**
449
- * Sets the rate limit per user (slowmode) for this channel.
450
- * @param {number} rateLimitPerUser The new rate limit in seconds
451
- * @param {string} [reason] Reason for changing the channel's rate limit
452
- * @returns {Promise<this>}
453
- */
454
- setRateLimitPerUser(rateLimitPerUser, reason) {
455
- return this.edit({ rateLimitPerUser }, reason);
456
- }
457
-
458
- /**
459
- * Sets whether this channel is flagged as NSFW.
460
- * @param {boolean} [nsfw=true] Whether the channel should be considered NSFW
461
- * @param {string} [reason] Reason for changing the channel's NSFW flag
462
- * @returns {Promise<this>}
463
- */
464
- setNSFW(nsfw = true, reason) {
465
- return this.edit({ nsfw }, reason);
466
- }
467
-
468
- static applyToClass(structure, full = false, ignore = []) {
469
- const props = ['send'];
470
- if (full) {
471
- props.push(
472
- 'sendSlash',
473
- 'searchInteraction',
474
- 'lastMessage',
475
- 'lastPinAt',
476
- 'sendTyping',
477
- 'createMessageCollector',
478
- 'awaitMessages',
479
- 'fetchWebhooks',
480
- 'createWebhook',
481
- 'setRateLimitPerUser',
482
- 'setNSFW',
483
- );
484
- }
485
- for (const prop of props) {
486
- if (ignore.includes(prop)) continue;
487
- Object.defineProperty(
488
- structure.prototype,
489
- prop,
490
- Object.getOwnPropertyDescriptor(TextBasedChannel.prototype, prop),
491
- );
492
- }
493
- }
494
- }
495
-
496
- module.exports = TextBasedChannel;
497
-
498
- // Fixes Circular
499
- const MessageManager = require('../../managers/MessageManager');
500
-
501
- // Utils
502
- function parseChoices(parent, list_choices, value) {
503
- if (value !== undefined) {
504
- if (Array.isArray(list_choices) && list_choices.length) {
505
- const choice = list_choices.find(c => [c.name, c.value].includes(value));
506
- if (choice) {
507
- return choice.value;
508
- } else {
509
- throw new Error('INVALID_SLASH_COMMAND_CHOICES', parent, value);
510
- }
511
- } else {
512
- return value;
513
- }
514
- } else {
515
- return undefined;
516
- }
517
- }
518
-
519
- async function addDataFromAttachment(value, client, channelId, attachments) {
520
- value = await MessagePayload.resolveFile(value);
521
- if (!value?.file) {
522
- throw new TypeError('The attachment data must be a BufferResolvable or Stream or FileOptions of MessageAttachment');
523
- }
524
- const data = await Util.getUploadURL(client, channelId, [value]);
525
- await Util.uploadFile(value.file, data[0].upload_url);
526
- const id = attachments.length;
527
- attachments.push({
528
- id,
529
- filename: value.name,
530
- uploaded_filename: data[0].upload_filename,
531
- });
532
- return {
533
- id,
534
- attachments,
535
- };
536
- }
537
-
538
- async function parseOption(
539
- client,
540
- optionCommand,
541
- value,
542
- optionFormat,
543
- attachments,
544
- command,
545
- applicationId,
546
- guildId,
547
- channelId,
548
- subGroup,
549
- subCommand,
550
- ) {
551
- const data = {
552
- type: optionCommand.type,
553
- name: optionCommand.name,
554
- };
555
- if (value !== undefined) {
556
- switch (optionCommand.type) {
557
- case ApplicationCommandOptionTypes.BOOLEAN:
558
- case 'BOOLEAN': {
559
- data.value = Boolean(value);
560
- break;
561
- }
562
- case ApplicationCommandOptionTypes.INTEGER:
563
- case 'INTEGER': {
564
- data.value = Number(value);
565
- break;
566
- }
567
- case ApplicationCommandOptionTypes.ATTACHMENT:
568
- case 'ATTACHMENT': {
569
- const parseData = await addDataFromAttachment(value, client, channelId, attachments);
570
- data.value = parseData.id;
571
- attachments = parseData.attachments;
572
- break;
573
- }
574
- case ApplicationCommandOptionTypes.SUB_COMMAND_GROUP:
575
- case 'SUB_COMMAND_GROUP': {
576
- break;
577
- }
578
- default: {
579
- value = parseChoices(optionCommand.name, optionCommand.choices, value);
580
- if (optionCommand.autocomplete) {
581
- const nonce = SnowflakeUtil.generate();
582
- // Post
583
- let postData;
584
- if (subGroup) {
585
- postData = [
586
- {
587
- type: ApplicationCommandOptionTypes.SUB_COMMAND_GROUP,
588
- name: subGroup.name,
589
- options: [
590
- {
591
- type: ApplicationCommandOptionTypes.SUB_COMMAND,
592
- name: subCommand.name,
593
- options: [
594
- {
595
- type: optionCommand.type,
596
- name: optionCommand.name,
597
- value,
598
- focused: true,
599
- },
600
- ],
601
- },
602
- ],
603
- },
604
- ];
605
- } else if (subCommand) {
606
- postData = [
607
- {
608
- type: ApplicationCommandOptionTypes.SUB_COMMAND,
609
- name: subCommand.name,
610
- options: [
611
- {
612
- type: optionCommand.type,
613
- name: optionCommand.name,
614
- value,
615
- focused: true,
616
- },
617
- ],
618
- },
619
- ];
620
- } else {
621
- postData = [
622
- {
623
- type: optionCommand.type,
624
- name: optionCommand.name,
625
- value,
626
- focused: true,
627
- },
628
- ];
629
- }
630
- const body = createPostData(
631
- client,
632
- true,
633
- applicationId,
634
- nonce,
635
- guildId,
636
- Boolean(command.guild_id),
637
- channelId,
638
- command.version,
639
- command.id,
640
- command.name_default || command.name,
641
- command.type,
642
- postData,
643
- [],
644
- );
645
- await client.api.interactions.post({
646
- data: body,
647
- });
648
- data.value = await awaitAutocomplete(client, nonce, value);
649
- } else {
650
- data.value = value;
651
- }
652
- }
653
- }
654
- optionFormat.push(data);
655
- }
656
- return {
657
- optionFormat,
658
- attachments,
659
- };
660
- }
661
-
662
- function awaitAutocomplete(client, nonce, defaultValue) {
663
- return new Promise(resolve => {
664
- const handler = data => {
665
- if (data.t !== 'APPLICATION_COMMAND_AUTOCOMPLETE_RESPONSE') return;
666
- if (data.d?.nonce !== nonce) return;
667
- clearTimeout(timeout);
668
- client.removeListener(Events.UNHANDLED_PACKET, handler);
669
- client.decrementMaxListeners();
670
- if (data.d.choices.length >= 1) {
671
- resolve(data.d.choices[0].value);
672
- } else {
673
- resolve(defaultValue);
674
- }
675
- };
676
- const timeout = setTimeout(() => {
677
- client.removeListener(Events.UNHANDLED_PACKET, handler);
678
- client.decrementMaxListeners();
679
- resolve(defaultValue);
680
- }, 5_000).unref();
681
- client.incrementMaxListeners();
682
- client.on(Events.UNHANDLED_PACKET, handler);
683
- });
684
- }
685
-
686
- function createPostData(
687
- client,
688
- isAutocomplete = false,
689
- applicationId,
690
- nonce,
691
- guildId,
692
- isGuildCommand,
693
- channelId,
694
- commandVersion,
695
- commandId,
696
- commandName,
697
- commandType,
698
- postData,
699
- attachments = [],
700
- ) {
701
- const data = {
702
- type: isAutocomplete ? InteractionTypes.APPLICATION_COMMAND_AUTOCOMPLETE : InteractionTypes.APPLICATION_COMMAND,
703
- application_id: applicationId,
704
- guild_id: guildId,
705
- channel_id: channelId,
706
- session_id: client.sessionId,
707
- data: {
708
- version: commandVersion,
709
- id: commandId,
710
- name: commandName,
711
- type: commandType,
712
- options: postData,
713
- attachments: attachments,
714
- },
715
- nonce,
716
- };
717
- if (isGuildCommand) {
718
- data.data.guild_id = guildId;
719
- }
720
- return data;
721
- }
1
+ 'use strict';
2
+
3
+ /* eslint-disable import/order */
4
+ const MessageCollector = require('../MessageCollector');
5
+ const MessagePayload = require('../MessagePayload');
6
+ const { InteractionTypes, ApplicationCommandOptionTypes, Events } = require('../../util/Constants');
7
+ const { Error } = require('../../errors');
8
+ const SnowflakeUtil = require('../../util/SnowflakeUtil');
9
+ const { setTimeout } = require('node:timers');
10
+ const { s } = require('@sapphire/shapeshift');
11
+ const Util = require('../../util/Util');
12
+ const validateName = stringName =>
13
+ s
14
+ .string()
15
+ .lengthGreaterThanOrEqual(1)
16
+ .lengthLessThanOrEqual(32)
17
+ .regex(/^[\p{Ll}\p{Lm}\p{Lo}\p{N}\p{sc=Devanagari}\p{sc=Thai}_-]+$/u)
18
+ .setValidationEnabled(true)
19
+ .parse(stringName);
20
+
21
+ /**
22
+ * Interface for classes that have text-channel-like features.
23
+ * @interface
24
+ */
25
+ class TextBasedChannel {
26
+ constructor() {
27
+ /**
28
+ * A manager of the messages sent to this channel
29
+ * @type {MessageManager}
30
+ */
31
+ this.messages = new MessageManager(this);
32
+
33
+ /**
34
+ * The channel's last message id, if one was sent
35
+ * @type {?Snowflake}
36
+ */
37
+ this.lastMessageId = null;
38
+
39
+ /**
40
+ * The timestamp when the last pinned message was pinned, if there was one
41
+ * @type {?number}
42
+ */
43
+ this.lastPinTimestamp = null;
44
+ }
45
+
46
+ /**
47
+ * The Message object of the last message in the channel, if one was sent
48
+ * @type {?Message}
49
+ * @readonly
50
+ */
51
+ get lastMessage() {
52
+ return this.messages.resolve(this.lastMessageId);
53
+ }
54
+
55
+ /**
56
+ * The date when the last pinned message was pinned, if there was one
57
+ * @type {?Date}
58
+ * @readonly
59
+ */
60
+ get lastPinAt() {
61
+ return this.lastPinTimestamp ? new Date(this.lastPinTimestamp) : null;
62
+ }
63
+
64
+ /**
65
+ * Represents the data for a poll answer.
66
+ * @typedef {Object} PollAnswerData
67
+ * @property {string} text The text for the poll answer
68
+ * @property {EmojiIdentifierResolvable} [emoji] The emoji for the poll answer
69
+ */
70
+
71
+ /**
72
+ * Represents the data for a poll.
73
+ * @typedef {Object} PollData
74
+ * @property {PollQuestionMedia} question The question for the poll
75
+ * @property {PollAnswerData[]} answers The answers for the poll
76
+ * @property {number} duration The duration in hours for the poll
77
+ * @property {boolean} allowMultiselect Whether the poll allows multiple answers
78
+ * @property {PollLayoutType} [layoutType] The layout type for the poll
79
+ */
80
+
81
+ /**
82
+ * Base options provided when sending.
83
+ * @typedef {Object} BaseMessageOptions
84
+ * @property {MessageActivity} [activity] Group activity
85
+ * @property {boolean} [tts=false] Whether or not the message should be spoken aloud
86
+ * @property {string} [nonce=''] The nonce for the message
87
+ * @property {string} [content=''] The content for the message
88
+ * @property {Array<(MessageEmbed|APIEmbed)>} [embeds] The embeds for the message
89
+ * (see [here](https://discord.com/developers/docs/resources/channel#embed-object) for more details)
90
+ * @property {MessageMentionOptions} [allowedMentions] Which mentions should be parsed from the message content
91
+ * (see [here](https://discord.com/developers/docs/resources/channel#allowed-mentions-object) for more details)
92
+ * @property {Array<(FileOptions|BufferResolvable|MessageAttachment[])>} [files] Files to send with the message
93
+ * @property {Array<(MessageActionRow|MessageActionRowOptions)>} [components]
94
+ * Action rows containing interactive components for the message (buttons, select menus)
95
+ * @property {MessageAttachment[]} [attachments] Attachments to send in the message
96
+ */
97
+
98
+ /**
99
+ * The base message options for messages including a poll.
100
+ * @typedef {BaseMessageOptions} BaseMessageOptionsWithPoll
101
+ * @property {PollData} [poll] The poll to send with the message
102
+ */
103
+
104
+ /**
105
+ * @typedef {Object} ForwardOptions
106
+ * @property {MessageResolvable} message The originating message
107
+ * @property {TextBasedChannelResolvable} [channel] The channel of the originating message
108
+ * @property {GuildResolvable} [guild] The guild of the originating message
109
+ */
110
+
111
+ /**
112
+ * Options provided when sending or editing a message.
113
+ * @typedef {BaseMessageOptions} MessageOptions
114
+ * @property {ReplyOptions} [reply] The options for replying to a message
115
+ * @property {ForwardOptions} [forward] The options for forwarding a message
116
+ * @property {StickerResolvable[]} [stickers=[]] Stickers to send in the message
117
+ * @property {MessageFlags} [flags] Which flags to set for the message.
118
+ * Only `SUPPRESS_EMBEDS`, `SUPPRESS_NOTIFICATIONS` and `IS_VOICE_MESSAGE` can be set.
119
+ */
120
+
121
+ /**
122
+ * Options provided to control parsing of mentions by Discord
123
+ * @typedef {Object} MessageMentionOptions
124
+ * @property {MessageMentionTypes[]} [parse] Types of mentions to be parsed
125
+ * @property {Snowflake[]} [users] Snowflakes of Users to be parsed as mentions
126
+ * @property {Snowflake[]} [roles] Snowflakes of Roles to be parsed as mentions
127
+ * @property {boolean} [repliedUser=true] Whether the author of the Message being replied to should be pinged
128
+ */
129
+
130
+ /**
131
+ * Types of mentions to enable in MessageMentionOptions.
132
+ * - `roles`
133
+ * - `users`
134
+ * - `everyone`
135
+ * @typedef {string} MessageMentionTypes
136
+ */
137
+
138
+ /**
139
+ * @typedef {Object} FileOptions
140
+ * @property {BufferResolvable} attachment File to attach
141
+ * @property {string} [name='file.jpg'] Filename of the attachment
142
+ * @property {string} description The description of the file
143
+ */
144
+
145
+ /**
146
+ * Options for sending a message with a reply.
147
+ * @typedef {Object} ReplyOptions
148
+ * @property {MessageResolvable} messageReference The message to reply to (must be in the same channel and not system)
149
+ * @property {boolean} [failIfNotExists=true] Whether to error if the referenced message
150
+ * does not exist (creates a standard message in this case when false)
151
+ */
152
+
153
+ /**
154
+ * Sends a message to this channel.
155
+ * @param {string|MessagePayload|MessageOptions} options The options to provide
156
+ * @returns {Promise<Message>}
157
+ * @example
158
+ * // Send a basic message
159
+ * channel.send('hello!')
160
+ * .then(message => console.log(`Sent message: ${message.content}`))
161
+ * .catch(console.error);
162
+ * @example
163
+ * // Send a remote file
164
+ * channel.send({
165
+ * files: ['https://cdn.discordapp.com/icons/222078108977594368/6e1019b3179d71046e463a75915e7244.png?size=2048']
166
+ * })
167
+ * .then(console.log)
168
+ * .catch(console.error);
169
+ * @example
170
+ * // Send a local file
171
+ * channel.send({
172
+ * files: [{
173
+ * attachment: 'entire/path/to/file.jpg',
174
+ * name: 'file.jpg',
175
+ * description: 'A description of the file'
176
+ * }]
177
+ * })
178
+ * .then(console.log)
179
+ * .catch(console.error);
180
+ */
181
+ async send(options) {
182
+ const User = require('../User');
183
+ const { GuildMember } = require('../GuildMember');
184
+
185
+ if (this instanceof User || this instanceof GuildMember) {
186
+ const dm = await this.createDM();
187
+ return dm.send(options);
188
+ }
189
+
190
+ let messagePayload;
191
+
192
+ if (options instanceof MessagePayload) {
193
+ messagePayload = options.resolveData();
194
+ } else {
195
+ messagePayload = MessagePayload.create(this, options).resolveData();
196
+ }
197
+
198
+ const { data, files } = await messagePayload.resolveFiles();
199
+ // New API
200
+ const attachments = await Util.getUploadURL(this.client, this.id, files);
201
+ const requestPromises = attachments.map(async attachment => {
202
+ await Util.uploadFile(files[attachment.id].file, attachment.upload_url);
203
+ return {
204
+ id: attachment.id,
205
+ filename: files[attachment.id].name,
206
+ uploaded_filename: attachment.upload_filename,
207
+ description: files[attachment.id].description,
208
+ duration_secs: files[attachment.id].duration_secs,
209
+ waveform: files[attachment.id].waveform,
210
+ };
211
+ });
212
+ const attachmentsData = await Promise.all(requestPromises);
213
+ attachmentsData.sort((a, b) => parseInt(a.id) - parseInt(b.id));
214
+ data.attachments = attachmentsData;
215
+ // Empty Files
216
+ const d = await this.client.api.channels[this.id].messages.post({ data });
217
+
218
+ return this.messages.cache.get(d.id) ?? this.messages._add(d);
219
+ }
220
+
221
+ searchInteractionFromGuildAndPrivateChannel() {
222
+ // Support Slash / ContextMenu
223
+ // API https://canary.discord.com/api/v9/guilds/:id/application-command-index // Guild
224
+ // https://canary.discord.com/api/v9/channels/:id/application-command-index // DM Channel
225
+ // Updated: 07/01/2023
226
+ return this.client.api[this.guild ? 'guilds' : 'channels'][this.guild?.id || this.id]['application-command-index']
227
+ .get()
228
+ .catch(() => ({
229
+ application_commands: [],
230
+ applications: [],
231
+ version: '',
232
+ }));
233
+ }
234
+
235
+ searchInteractionUserApps() {
236
+ return this.client.api.users['@me']['application-command-index'].get().catch(() => ({
237
+ application_commands: [],
238
+ applications: [],
239
+ version: '',
240
+ }));
241
+ }
242
+
243
+ searchInteraction() {
244
+ return Promise.all([this.searchInteractionFromGuildAndPrivateChannel(), this.searchInteractionUserApps()]).then(
245
+ ([dataA, dataB]) => ({
246
+ applications: [...dataA.applications, ...dataB.applications],
247
+ application_commands: [...dataA.application_commands, ...dataB.application_commands],
248
+ }),
249
+ );
250
+ }
251
+
252
+ async sendSlash(botOrApplicationId, commandNameString, ...args) {
253
+ // Parse commandName /role add user
254
+ const cmd = commandNameString.trim().split(' ');
255
+ // Ex: role add user => [role, add, user]
256
+ // Parse: name, subGr, sub
257
+ const commandName = validateName(cmd[0]);
258
+ // Parse: role
259
+ const sub = cmd.slice(1);
260
+ // Parse: [add, user]
261
+ for (let i = 0; i < sub.length; i++) {
262
+ if (sub.length > 2) {
263
+ throw new Error('INVALID_COMMAND_NAME', cmd);
264
+ }
265
+ validateName(sub[i]);
266
+ }
267
+ // Search all
268
+ const data = await this.searchInteraction();
269
+ // Find command...
270
+ const filterCommand = data.application_commands.filter(obj =>
271
+ // Filter: name | name_default
272
+ [obj.name, obj.name_default].includes(commandName),
273
+ );
274
+ // Filter Bot
275
+ botOrApplicationId = this.client.users.resolveId(botOrApplicationId);
276
+ const application = data.applications.find(obj => obj.id == botOrApplicationId || obj.bot_id == botOrApplicationId);
277
+ if (!application) {
278
+ throw new Error('INVALID_APPLICATION_COMMAND', "Bot/Application doesn't exist");
279
+ }
280
+ // Find Command with application
281
+ const command = filterCommand.find(command => command.application_id == application.id);
282
+ if (!command) {
283
+ throw new Error('INVALID_APPLICATION_COMMAND', application.id);
284
+ }
285
+ args = args.flat(2);
286
+ let optionFormat = [];
287
+ let attachments = [];
288
+ let optionsMaxdepth, subGroup, subCommand;
289
+ if (sub.length == 2) {
290
+ // Subcommand Group > Subcommand
291
+ // Find Sub group
292
+ subGroup = command.options.find(
293
+ obj =>
294
+ obj.type == ApplicationCommandOptionTypes.SUB_COMMAND_GROUP && [obj.name, obj.name_default].includes(sub[0]),
295
+ );
296
+ if (!subGroup) throw new Error('SLASH_COMMAND_SUB_COMMAND_GROUP_INVALID', sub[0]);
297
+ // Find Sub
298
+ subCommand = subGroup.options.find(
299
+ obj => obj.type == ApplicationCommandOptionTypes.SUB_COMMAND && [obj.name, obj.name_default].includes(sub[1]),
300
+ );
301
+ if (!subCommand) throw new Error('SLASH_COMMAND_SUB_COMMAND_INVALID', sub[1]);
302
+ // Options
303
+ optionsMaxdepth = subCommand.options;
304
+ } else if (sub.length == 1) {
305
+ // Subcommand
306
+ subCommand = command.options.find(
307
+ obj => obj.type == ApplicationCommandOptionTypes.SUB_COMMAND && [obj.name, obj.name_default].includes(sub[0]),
308
+ );
309
+ if (!subCommand) throw new Error('SLASH_COMMAND_SUB_COMMAND_INVALID', sub[0]);
310
+ // Options
311
+ optionsMaxdepth = subCommand.options;
312
+ } else {
313
+ optionsMaxdepth = command.options;
314
+ }
315
+ const valueRequired = optionsMaxdepth?.filter(o => o.required).length || 0;
316
+ for (let i = 0; i < Math.min(args.length, optionsMaxdepth?.length || 0); i++) {
317
+ const optionInput = optionsMaxdepth[i];
318
+ const value = args[i];
319
+ const parseData = await parseOption(
320
+ this.client,
321
+ optionInput,
322
+ value,
323
+ optionFormat,
324
+ attachments,
325
+ command,
326
+ application.id,
327
+ this.guild?.id,
328
+ this.id,
329
+ subGroup,
330
+ subCommand,
331
+ );
332
+ optionFormat = parseData.optionFormat;
333
+ attachments = parseData.attachments;
334
+ }
335
+ if (valueRequired > args.length) {
336
+ throw new Error('SLASH_COMMAND_REQUIRED_OPTIONS_MISSING', valueRequired, optionFormat.length);
337
+ }
338
+ // Post
339
+ let postData;
340
+ if (subGroup) {
341
+ postData = [
342
+ {
343
+ type: ApplicationCommandOptionTypes.SUB_COMMAND_GROUP,
344
+ name: subGroup.name,
345
+ options: [
346
+ {
347
+ type: ApplicationCommandOptionTypes.SUB_COMMAND,
348
+ name: subCommand.name,
349
+ options: optionFormat,
350
+ },
351
+ ],
352
+ },
353
+ ];
354
+ } else if (subCommand) {
355
+ postData = [
356
+ {
357
+ type: ApplicationCommandOptionTypes.SUB_COMMAND,
358
+ name: subCommand.name,
359
+ options: optionFormat,
360
+ },
361
+ ];
362
+ } else {
363
+ postData = optionFormat;
364
+ }
365
+ const nonce = SnowflakeUtil.generate();
366
+ const body = createPostData(
367
+ this.client,
368
+ false,
369
+ application.id,
370
+ nonce,
371
+ this.guild?.id,
372
+ Boolean(command.guild_id),
373
+ this.id,
374
+ command.version,
375
+ command.id,
376
+ command.name_default || command.name,
377
+ command.type,
378
+ postData,
379
+ attachments,
380
+ );
381
+ this.client.api.interactions.post({
382
+ data: body,
383
+ usePayloadJSON: true,
384
+ });
385
+ return Util.createPromiseInteraction(this.client, nonce, 5000);
386
+ }
387
+
388
+ /**
389
+ * Sends a typing indicator in the channel.
390
+ * @returns {Promise<{ message_send_cooldown_ms: number, thread_create_cooldown_ms: number }|void>} Resolves upon the typing status being sent
391
+ * @example
392
+ * // Start typing in a channel
393
+ * channel.sendTyping();
394
+ */
395
+ sendTyping() {
396
+ return this.client.api.channels(this.id).typing.post();
397
+ }
398
+
399
+ /**
400
+ * Creates a Message Collector.
401
+ * @param {MessageCollectorOptions} [options={}] The options to pass to the collector
402
+ * @returns {MessageCollector}
403
+ * @example
404
+ * // Create a message collector
405
+ * const filter = m => m.content.includes('discord');
406
+ * const collector = channel.createMessageCollector({ filter, time: 15_000 });
407
+ * collector.on('collect', m => console.log(`Collected ${m.content}`));
408
+ * collector.on('end', collected => console.log(`Collected ${collected.size} items`));
409
+ */
410
+ createMessageCollector(options = {}) {
411
+ return new MessageCollector(this, options);
412
+ }
413
+
414
+ /**
415
+ * An object containing the same properties as CollectorOptions, but a few more:
416
+ * @typedef {MessageCollectorOptions} AwaitMessagesOptions
417
+ * @property {string[]} [errors] Stop/end reasons that cause the promise to reject
418
+ */
419
+
420
+ /**
421
+ * Similar to createMessageCollector but in promise form.
422
+ * Resolves with a collection of messages that pass the specified filter.
423
+ * @param {AwaitMessagesOptions} [options={}] Optional options to pass to the internal collector
424
+ * @returns {Promise<Collection<Snowflake, Message>>}
425
+ * @example
426
+ * // Await !vote messages
427
+ * const filter = m => m.content.startsWith('!vote');
428
+ * // Errors: ['time'] treats ending because of the time limit as an error
429
+ * channel.awaitMessages({ filter, max: 4, time: 60_000, errors: ['time'] })
430
+ * .then(collected => console.log(collected.size))
431
+ * .catch(collected => console.log(`After a minute, only ${collected.size} out of 4 voted.`));
432
+ */
433
+ awaitMessages(options = {}) {
434
+ return new Promise((resolve, reject) => {
435
+ const collector = this.createMessageCollector(options);
436
+ collector.once('end', (collection, reason) => {
437
+ if (options.errors?.includes(reason)) {
438
+ reject(collection);
439
+ } else {
440
+ resolve(collection);
441
+ }
442
+ });
443
+ });
444
+ }
445
+
446
+ /**
447
+ * Fetches all webhooks for the channel.
448
+ * @returns {Promise<Collection<Snowflake, Webhook>>}
449
+ * @example
450
+ * // Fetch webhooks
451
+ * channel.fetchWebhooks()
452
+ * .then(hooks => console.log(`This channel has ${hooks.size} hooks`))
453
+ * .catch(console.error);
454
+ */
455
+ fetchWebhooks() {
456
+ return this.guild.channels.fetchWebhooks(this.id);
457
+ }
458
+
459
+ /**
460
+ * Options used to create a {@link Webhook} in a guild text-based channel.
461
+ * @typedef {Object} ChannelWebhookCreateOptions
462
+ * @property {?(BufferResolvable|Base64Resolvable)} [avatar] Avatar for the webhook
463
+ * @property {string} [reason] Reason for creating the webhook
464
+ */
465
+
466
+ /**
467
+ * Creates a webhook for the channel.
468
+ * @param {string} name The name of the webhook
469
+ * @param {ChannelWebhookCreateOptions} [options] Options for creating the webhook
470
+ * @returns {Promise<Webhook>} Returns the created Webhook
471
+ * @example
472
+ * // Create a webhook for the current channel
473
+ * channel.createWebhook('Snek', {
474
+ * avatar: 'https://i.imgur.com/mI8XcpG.jpg',
475
+ * reason: 'Needed a cool new Webhook'
476
+ * })
477
+ * .then(console.log)
478
+ * .catch(console.error)
479
+ */
480
+ createWebhook(name, options = {}) {
481
+ return this.guild.channels.createWebhook(this.id, name, options);
482
+ }
483
+
484
+ /**
485
+ * Sets the rate limit per user (slowmode) for this channel.
486
+ * @param {number} rateLimitPerUser The new rate limit in seconds
487
+ * @param {string} [reason] Reason for changing the channel's rate limit
488
+ * @returns {Promise<this>}
489
+ */
490
+ setRateLimitPerUser(rateLimitPerUser, reason) {
491
+ return this.edit({ rateLimitPerUser }, reason);
492
+ }
493
+
494
+ /**
495
+ * Sets whether this channel is flagged as NSFW.
496
+ * @param {boolean} [nsfw=true] Whether the channel should be considered NSFW
497
+ * @param {string} [reason] Reason for changing the channel's NSFW flag
498
+ * @returns {Promise<this>}
499
+ */
500
+ setNSFW(nsfw = true, reason) {
501
+ return this.edit({ nsfw }, reason);
502
+ }
503
+
504
+ static applyToClass(structure, full = false, ignore = []) {
505
+ const props = ['send'];
506
+ if (full) {
507
+ props.push(
508
+ 'sendSlash',
509
+ 'searchInteraction',
510
+ 'searchInteractionFromGuildAndPrivateChannel',
511
+ 'searchInteractionUserApps',
512
+ 'lastMessage',
513
+ 'lastPinAt',
514
+ 'sendTyping',
515
+ 'createMessageCollector',
516
+ 'awaitMessages',
517
+ 'fetchWebhooks',
518
+ 'createWebhook',
519
+ 'setRateLimitPerUser',
520
+ 'setNSFW',
521
+ );
522
+ }
523
+ for (const prop of props) {
524
+ if (ignore.includes(prop)) continue;
525
+ Object.defineProperty(
526
+ structure.prototype,
527
+ prop,
528
+ Object.getOwnPropertyDescriptor(TextBasedChannel.prototype, prop),
529
+ );
530
+ }
531
+ }
532
+ }
533
+
534
+ module.exports = TextBasedChannel;
535
+
536
+ // Fixes Circular
537
+ const MessageManager = require('../../managers/MessageManager');
538
+
539
+ // Utils
540
+ function parseChoices(parent, list_choices, value) {
541
+ if (value !== undefined) {
542
+ if (Array.isArray(list_choices) && list_choices.length) {
543
+ const choice = list_choices.find(c => [c.name, c.value].includes(value));
544
+ if (choice) {
545
+ return choice.value;
546
+ } else {
547
+ throw new Error('INVALID_SLASH_COMMAND_CHOICES', parent, value);
548
+ }
549
+ } else {
550
+ return value;
551
+ }
552
+ } else {
553
+ return undefined;
554
+ }
555
+ }
556
+
557
+ async function addDataFromAttachment(value, client, channelId, attachments) {
558
+ value = await MessagePayload.resolveFile(value);
559
+ if (!value?.file) {
560
+ throw new TypeError('The attachment data must be a BufferResolvable or Stream or FileOptions of MessageAttachment');
561
+ }
562
+ const data = await Util.getUploadURL(client, channelId, [value]);
563
+ await Util.uploadFile(value.file, data[0].upload_url);
564
+ const id = attachments.length;
565
+ attachments.push({
566
+ id,
567
+ filename: value.name,
568
+ uploaded_filename: data[0].upload_filename,
569
+ });
570
+ return {
571
+ id,
572
+ attachments,
573
+ };
574
+ }
575
+
576
+ async function parseOption(
577
+ client,
578
+ optionCommand,
579
+ value,
580
+ optionFormat,
581
+ attachments,
582
+ command,
583
+ applicationId,
584
+ guildId,
585
+ channelId,
586
+ subGroup,
587
+ subCommand,
588
+ ) {
589
+ const data = {
590
+ type: optionCommand.type,
591
+ name: optionCommand.name,
592
+ };
593
+ if (value !== undefined) {
594
+ switch (optionCommand.type) {
595
+ case ApplicationCommandOptionTypes.BOOLEAN:
596
+ case 'BOOLEAN': {
597
+ data.value = Boolean(value);
598
+ break;
599
+ }
600
+ case ApplicationCommandOptionTypes.INTEGER:
601
+ case 'INTEGER': {
602
+ data.value = Number(value);
603
+ break;
604
+ }
605
+ case ApplicationCommandOptionTypes.ATTACHMENT:
606
+ case 'ATTACHMENT': {
607
+ const parseData = await addDataFromAttachment(value, client, channelId, attachments);
608
+ data.value = parseData.id;
609
+ attachments = parseData.attachments;
610
+ break;
611
+ }
612
+ case ApplicationCommandOptionTypes.SUB_COMMAND_GROUP:
613
+ case 'SUB_COMMAND_GROUP': {
614
+ break;
615
+ }
616
+ default: {
617
+ value = parseChoices(optionCommand.name, optionCommand.choices, value);
618
+ if (optionCommand.autocomplete) {
619
+ const nonce = SnowflakeUtil.generate();
620
+ // Post
621
+ let postData;
622
+ if (subGroup) {
623
+ postData = [
624
+ {
625
+ type: ApplicationCommandOptionTypes.SUB_COMMAND_GROUP,
626
+ name: subGroup.name,
627
+ options: [
628
+ {
629
+ type: ApplicationCommandOptionTypes.SUB_COMMAND,
630
+ name: subCommand.name,
631
+ options: [
632
+ {
633
+ type: optionCommand.type,
634
+ name: optionCommand.name,
635
+ value,
636
+ focused: true,
637
+ },
638
+ ],
639
+ },
640
+ ],
641
+ },
642
+ ];
643
+ } else if (subCommand) {
644
+ postData = [
645
+ {
646
+ type: ApplicationCommandOptionTypes.SUB_COMMAND,
647
+ name: subCommand.name,
648
+ options: [
649
+ {
650
+ type: optionCommand.type,
651
+ name: optionCommand.name,
652
+ value,
653
+ focused: true,
654
+ },
655
+ ],
656
+ },
657
+ ];
658
+ } else {
659
+ postData = [
660
+ {
661
+ type: optionCommand.type,
662
+ name: optionCommand.name,
663
+ value,
664
+ focused: true,
665
+ },
666
+ ];
667
+ }
668
+ const body = createPostData(
669
+ client,
670
+ true,
671
+ applicationId,
672
+ nonce,
673
+ guildId,
674
+ Boolean(command.guild_id),
675
+ channelId,
676
+ command.version,
677
+ command.id,
678
+ command.name_default || command.name,
679
+ command.type,
680
+ postData,
681
+ [],
682
+ );
683
+ await client.api.interactions.post({
684
+ data: body,
685
+ });
686
+ data.value = await awaitAutocomplete(client, nonce, value);
687
+ } else {
688
+ data.value = value;
689
+ }
690
+ }
691
+ }
692
+ optionFormat.push(data);
693
+ }
694
+ return {
695
+ optionFormat,
696
+ attachments,
697
+ };
698
+ }
699
+
700
+ function awaitAutocomplete(client, nonce, defaultValue) {
701
+ return new Promise(resolve => {
702
+ const handler = data => {
703
+ if (data.t !== 'APPLICATION_COMMAND_AUTOCOMPLETE_RESPONSE') return;
704
+ if (data.d?.nonce !== nonce) return;
705
+ clearTimeout(timeout);
706
+ client.removeListener(Events.UNHANDLED_PACKET, handler);
707
+ client.decrementMaxListeners();
708
+ if (data.d.choices.length >= 1) {
709
+ resolve(data.d.choices[0].value);
710
+ } else {
711
+ resolve(defaultValue);
712
+ }
713
+ };
714
+ const timeout = setTimeout(() => {
715
+ client.removeListener(Events.UNHANDLED_PACKET, handler);
716
+ client.decrementMaxListeners();
717
+ resolve(defaultValue);
718
+ }, 5_000).unref();
719
+ client.incrementMaxListeners();
720
+ client.on(Events.UNHANDLED_PACKET, handler);
721
+ });
722
+ }
723
+
724
+ function createPostData(
725
+ client,
726
+ isAutocomplete = false,
727
+ applicationId,
728
+ nonce,
729
+ guildId,
730
+ isGuildCommand,
731
+ channelId,
732
+ commandVersion,
733
+ commandId,
734
+ commandName,
735
+ commandType,
736
+ postData,
737
+ attachments = [],
738
+ ) {
739
+ const data = {
740
+ type: isAutocomplete ? InteractionTypes.APPLICATION_COMMAND_AUTOCOMPLETE : InteractionTypes.APPLICATION_COMMAND,
741
+ application_id: applicationId,
742
+ guild_id: guildId,
743
+ channel_id: channelId,
744
+ session_id: client.sessionId,
745
+ data: {
746
+ version: commandVersion,
747
+ id: commandId,
748
+ name: commandName,
749
+ type: commandType,
750
+ options: postData,
751
+ attachments: attachments,
752
+ },
753
+ nonce,
754
+ };
755
+ if (isGuildCommand) {
756
+ data.data.guild_id = guildId;
757
+ }
758
+ return data;
759
+ }