djs-selfbot-v13 3.2.2 → 3.7.2

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/README.md +63 -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 +941 -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/QuestManager.js +376 -0
  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 -160
  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 -85
  241. package/src/structures/ClientUser.js +653 -448
  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 +1697 -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 -1227
  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 -318
  284. package/src/structures/MessageReaction.js +181 -171
  285. package/src/structures/MessageSelectMenu.js +141 -140
  286. package/src/structures/Modal.js +161 -161
  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 +1137 -1101
  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 -543
  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 -719
  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 -1773
  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 -336
  349. package/src/util/Permissions.js +202 -202
  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 -889
  361. package/typings/enums.d.ts +439 -297
  362. package/typings/index.d.ts +8392 -7432
  363. package/typings/rawDataTypes.d.ts +403 -342
  364. 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
+ }