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