# frozen_string_literal: true require "async/http" require "async/websocket" require "async/barrier" require "json" require "zlib" module Discorb # # A module for Discord Gateway. # This module is internal use only. # module Gateway # # Represents an event. # class GatewayEvent # @private def initialize(data) @data = data end end # # Represents a reaction event. # class ReactionEvent < GatewayEvent # @return [Hash] The raw data of the event. attr_reader :data # @return [Discorb::Snowflake] The ID of the user who reacted. attr_reader :user_id alias member_id user_id # @return [Discorb::Snowflake] The ID of the channel the message was sent in. attr_reader :channel_id # @return [Discorb::Snowflake] The ID of the message. attr_reader :message_id # @return [Discorb::Snowflake] The ID of the guild the message was sent in. attr_reader :guild_id # @macro client_cache # @return [Discorb::User] The user who reacted. attr_reader :user # @macro client_cache # @return [Discorb::Channel] The channel the message was sent in. attr_reader :channel # @macro client_cache # @return [Discorb::Guild] The guild the message was sent in. attr_reader :guild # @macro client_cache # @return [Discorb::Message] The message the reaction was sent in. attr_reader :message # @macro client_cache # @return [Discorb::Member] The member who reacted. attr_reader :member # @return [Discorb::UnicodeEmoji, Discorb::PartialEmoji] The emoji that was reacted with. attr_reader :emoji # @macro client_cache # @return [Discorb::Member, Discorb::User] The user or member who reacted. attr_reader :fired_by alias reactor fired_by alias from fired_by # @private def initialize(client, data) @client = client @data = data if data.key?(:user_id) @user_id = Snowflake.new(data[:user_id]) else @member_data = data[:member] end @channel_id = Snowflake.new(data[:channel_id]) @message_id = Snowflake.new(data[:message_id]) @guild_id = Snowflake.new(data[:guild_id]) @guild = client.guilds[data[:guild_id]] @channel = client.channels[data[:channel_id]] unless @guild.nil? @user = client.users[data[:user_id]] if data.key?(:user_id) unless @guild.nil? @member = if data.key?(:member) @guild.members[data[:member][:user][:id]] || Member.new(@client, @guild_id, data[:member][:user], data[:member]) else @guild.members[data[:user_id]] end end @fired_by = @member || @user || @client.users[data[:user_id]] @message = client.messages[data[:message_id]] @emoji = data[:emoji][:id].nil? ? UnicodeEmoji.new(data[:emoji][:name]) : PartialEmoji.new(data[:emoji]) end # Fetch the message. # If message is cached, it will be returned. # @async # # @param [Boolean] force Whether to force fetching the message. # # @return [Async::Task] The message. def fetch_message(force: false) Async do next @message if !force && @message @message = @channel.fetch_message(@message_id).wait end end end # # Represents a `MESSAGE_REACTION_REMOVE_ALL` event. # class ReactionRemoveAllEvent < GatewayEvent # @return [Discorb::Snowflake] The ID of the channel the message was sent in. attr_reader :channel_id # @return [Discorb::Snowflake] The ID of the message. attr_reader :message_id # @return [Discorb::Snowflake] The ID of the guild the message was sent in. attr_reader :guild_id # @macro client_cache # @return [Discorb::Channel] The channel the message was sent in. attr_reader :channel # @macro client_cache # @return [Discorb::Guild] The guild the message was sent in. attr_reader :guild # @macro client_cache # @return [Discorb::Message] The message the reaction was sent in. attr_reader :message # @private def initialize(client, data) @client = client @data = data @guild_id = Snowflake.new(data[:guild_id]) @channel_id = Snowflake.new(data[:channel_id]) @message_id = Snowflake.new(data[:message_id]) @guild = client.guilds[data[:guild_id]] @channel = client.channels[data[:channel_id]] @message = client.messages[data[:message_id]] end # Fetch the message. # If message is cached, it will be returned. # @async # # @param [Boolean] force Whether to force fetching the message. # # @return [Async::Task] The message. def fetch_message(force: false) Async do next @message if !force && @message @message = @channel.fetch_message(@message_id).wait end end end # # Represents a `MESSAGE_REACTION_REMOVE_EMOJI` event. # class ReactionRemoveEmojiEvent < GatewayEvent # @return [Discorb::Snowflake] The ID of the channel the message was sent in. attr_reader :channel_id # @return [Discorb::Snowflake] The ID of the message. attr_reader :message_id # @return [Discorb::Snowflake] The ID of the guild the message was sent in. attr_reader :guild_id # @macro client_cache # @return [Discorb::Channel] The channel the message was sent in. attr_reader :channel # @macro client_cache # @return [Discorb::Guild] The guild the message was sent in. attr_reader :guild # @macro client_cache # @return [Discorb::Message] The message the reaction was sent in. attr_reader :message # @return [Discorb::UnicodeEmoji, Discorb::PartialEmoji] The emoji that was reacted with. attr_reader :emoji # @private def initialize(client, data) @client = client @data = data @guild_id = Snowflake.new(data[:guild_id]) @channel_id = Snowflake.new(data[:channel_id]) @message_id = Snowflake.new(data[:message_id]) @guild = client.guilds[data[:guild_id]] @channel = client.channels[data[:channel_id]] @message = client.messages[data[:message_id]] @emoji = data[:emoji][:id].nil? ? DiscordEmoji.new(data[:emoji][:name]) : PartialEmoji.new(data[:emoji]) end # Fetch the message. # If message is cached, it will be returned. # @async # # @param [Boolean] force Whether to force fetching the message. # # @return [Async::Task] The message. def fetch_message(force: false) Async do next @message if !force && @message @message = @channel.fetch_message(@message_id).wait end end end # # Represents a `GUILD_SCHEDULED_EVENT_USER_ADD` and `GUILD_SCHEDULED_EVENT_USER_REMOVE` event. # class ScheduledEventUserEvent < GatewayEvent # @return [Discorb::User] The user that triggered the event. attr_reader :user # @return [Discorb::Guild] The guild the event was triggered in. attr_reader :guild # @return [Discorb::ScheduledEvent] The scheduled event. attr_reader :scheduled_event # @private def initialize(client, data) @client = client @scheduled_event_id = Snowflake.new(data[:scheduled_event_id]) @user_id = Snowflake.new(data[:user_id]) @guild_id = Snowflake.new(data[:guild_id]) @guild = client.guilds[data[:guild_id]] @scheduled_event = @guild.scheduled_events[@scheduled_event_id] @user = client.users[data[:user_id]] end end # # Represents a `MESSAGE_UPDATE` event. # class MessageUpdateEvent < GatewayEvent # @return [Discorb::Message] The message before update. attr_reader :before # @return [Discorb::Message] The message after update. attr_reader :after # @return [Discorb::Snowflake] The ID of the message. attr_reader :id # @return [Discorb::Snowflake] The ID of the channel the message was sent in. attr_reader :channel_id # @return [Discorb::Snowflake] The ID of the guild the message was sent in. attr_reader :guild_id # @return [String] The new content of the message. attr_reader :content # @return [Time] The time the message was edited. attr_reader :timestamp # @return [Boolean] Whether the message pings @everyone. attr_reader :mention_everyone # @macro client_cache # @return [Array] The roles mentioned in the message. attr_reader :mention_roles # @return [Array] The attachments in the message. attr_reader :attachments # @return [Array] The embeds in the message. attr_reader :embeds # @!attribute [r] channel # @macro client_cache # @return [Discorb::Channel] The channel the message was sent in. # @!attribute [r] guild # @macro client_cache # @return [Discorb::Guild] The guild the message was sent in. def initialize(client, data, before, after) @client = client @data = data @before = before @after = after @id = Snowflake.new(data[:id]) @channel_id = Snowflake.new(data[:channel_id]) @guild_id = Snowflake.new(data[:guild_id]) if data.key?(:guild_id) @content = data[:content] @timestamp = Time.iso8601(data[:edited_timestamp]) @mention_everyone = data[:mention_everyone] @mention_roles = data[:mention_roles].map { |r| guild.roles[r] } if data.key?(:mention_roles) @attachments = data[:attachments].map { |a| Attachment.new(a) } if data.key?(:attachments) @embeds = data[:embeds] ? data[:embeds].map { |e| Embed.new(data: e) } : [] if data.key?(:embeds) end def channel @client.channels[@channel_id] end def guild @client.guilds[@guild_id] end # Fetch the message. # @async # # @return [Async::Task] The message. def fetch_message Async do channel.fetch_message(@id).wait end end end # # Represents a message but it has only ID. # class UnknownDeleteBulkMessage < GatewayEvent # @return [Discorb::Snowflake] The ID of the message. attr_reader :id # @!attribute [r] channel # @macro client_cache # @return [Discorb::Channel] The channel the message was sent in. # @!attribute [r] guild # @macro client_cache # @return [Discorb::Guild] The guild the message was sent in. # @private def initialize(client, id, data) @client = client @id = id @data = data @channel_id = Snowflake.new(data[:channel_id]) @guild_id = Snowflake.new(data[:guild_id]) if data.key?(:guild_id) end def channel @client.channels[@channel_id] end def guild @client.guilds[@guild_id] end end # # Represents a `INVITE_DELETE` event. # class InviteDeleteEvent < GatewayEvent # @return [String] The invite code. attr_reader :code # @!attribute [r] channel # @macro client_cache # @return [Discorb::Channel] The channel the message was sent in. # @!attribute [r] guild # @macro client_cache # @return [Discorb::Guild] The guild the message was sent in. # @private def initialize(client, data) @client = client @data = data @channel_id = Snowflake.new(data[:channel]) @guild_id = Snowflake.new(data[:guild_id]) @code = data[:code] end def channel @client.channels[@channel_id] end def guild @client.guilds[@guild_id] end end class GuildIntegrationsUpdateEvent < GatewayEvent def initialize(client, data) @client = client @data = data @guild_id = Snowflake.new(data[:guild_id]) end def guild @client.guilds[@guild_id] end end # # Represents a `TYPING_START` event. # class TypingStartEvent < GatewayEvent # @return [Discorb::Snowflake] The ID of the channel the user is typing in. attr_reader :user_id # @macro client_cache # @return [Discorb::Member] The member that is typing. attr_reader :member # @!attribute [r] channel # @macro client_cache # @return [Discorb::Channel] The channel the user is typing in. # @!attribute [r] guild # @macro client_cache # @return [Discorb::Guild] The guild the user is typing in. # @!attribute [r] user # @macro client_cache # @return [Discorb::User] The user that is typing. # @!attribute [r] fired_by # @macro client_cache # @return [Discorb::Member, Discorb::User] The member or user that started typing. # @private def initialize(client, data) @client = client @data = data @channel_id = Snowflake.new(data[:channel_id]) @guild_id = Snowflake.new(data[:guild_id]) if data.key?(:guild_id) @user_id = Snowflake.new(data[:user_id]) @timestamp = Time.at(data[:timestamp]) @member = guild.members[@user_id] || Member.new(@client, @guild_id, @client.users[@user_id].instance_variable_get(:@data), data[:member]) if guild end def user @client.users[@user_id] end def channel @client.channels[@channel_id] end def guild @client.guilds[@guild_id] end def fired_by @member || user end alias from fired_by end # # Represents a message pin event. # class MessagePinEvent < GatewayEvent # @return [Discorb::Message] The message that was pinned. attr_reader :message # @return [:pinned, :unpinned] The type of event. attr_reader :type # @!attribute [r] pinned? # @return [Boolean] Whether the message was pinned. # @!attribute [r] unpinned? # @return [Boolean] Whether the message was unpinned. def initialize(client, data, message) @client = client @data = data @message = message @type = if message.nil? :unknown elsif @message.pinned? :pinned else :unpinned end end def pinned? @type == :pinned end def unpinned? @type = :unpinned end end # # Represents a `WEBHOOKS_UPDATE` event. # class WebhooksUpdateEvent < GatewayEvent # @!attribute [r] channel # @macro client_cache # @return [Discorb::Channel] The channel where the webhook was updated. # @!attribute [r] guild # @macro client_cache # @return [Discorb::Guild] The guild where the webhook was updated. # @private def initialize(client, data) @client = client @data = data @guild_id = Snowflake.new(data[:guild_id]) @channel_id = Snowflake.new(data[:channel_id]) end def guild @client.guilds[@guild_id] end def channel @client.channels[@channel_id] end end # # A module to handle gateway events. # module Handler private def connect_gateway(reconnect, no_close: false) if reconnect @log.info "Reconnecting to gateway..." else @log.info "Connecting to gateway..." end Async do @connection&.close rescue nil unless no_close @http = HTTP.new(self) _, gateway_response = @http.get("/gateway").wait gateway_url = gateway_response[:url] endpoint = Async::HTTP::Endpoint.parse("#{gateway_url}?v=9&encoding=json&compress=zlib-stream", alpn_protocols: Async::HTTP::Protocol::HTTP11.names) begin @connection = Async::WebSocket::Client.connect(endpoint, headers: [["User-Agent", Discorb::USER_AGENT]], handler: RawConnection) @zlib_stream = Zlib::Inflate.new(Zlib::MAX_WBITS) buffer = +"" begin while (message = @connection.read) buffer << message if message.end_with?((+"\x00\x00\xff\xff").force_encoding("ASCII-8BIT")) begin data = @zlib_stream.inflate(buffer) buffer = +"" message = JSON.parse(data, symbolize_names: true) rescue JSON::ParserError buffer = +"" @log.error "Received invalid JSON from gateway." @log.debug "#{data}" else handle_gateway(message, reconnect) end end end rescue Async::Wrapper::Cancelled, OpenSSL::SSL::SSLError, Async::Wrapper::WaitError, EOFError, Errno::EPIPE => e @log.error "Gateway connection closed accidentally: #{e.class}: #{e.message}" connect_gateway(true, no_close: true) else # should never happen connect_gateway(true) end rescue Protocol::WebSocket::ClosedError => e @tasks.map(&:stop) case e.code when 4004 raise ClientError.new("Authentication failed"), cause: nil when 4009 @log.info "Session timed out, reconnecting." connect_gateway(true) when 4014 raise ClientError.new("Disallowed intents were specified"), cause: nil when 4002, 4003, 4005, 4007 raise ClientError.new(<<~EOS), cause: e Disconnected from gateway, probably due to library issues. #{e.message} Please report this to the library issue tracker. https://github.com/discorb-lib/discorb/issues EOS when 1001 @log.info "Gateway closed with code 1001, reconnecting." connect_gateway(true) else @log.error "Discord WebSocket closed with code #{e.code}." @log.debug "#{e.message}" connect_gateway(false) end rescue => e @log.error "Discord WebSocket error: #{e.full_message}" connect_gateway(false) end end end def send_gateway(opcode, **value) @connection.write({ op: opcode, d: value }.to_json) @connection.flush @log.debug "Sent message #{{ op: opcode, d: value }.to_json.gsub(@token, "[Token]")}" end def handle_gateway(payload, reconnect) Async do |task| data = payload[:d] @last_s = payload[:s] if payload[:s] @log.debug "Received message with opcode #{payload[:op]} from gateway:" @log.debug "#{payload.to_json.gsub(@token, "[Token]")}" case payload[:op] when 10 @heartbeat_interval = data[:heartbeat_interval] if reconnect payload = { token: @token, session_id: @session_id, seq: @last_s, } send_gateway(6, **payload) else payload = { token: @token, intents: @intents.value, compress: false, properties: { "$os" => RUBY_PLATFORM, "$browser" => "discorb", "$device" => "discorb" }, } payload[:presence] = @identify_presence if @identify_presence send_gateway(2, **payload) end when 7 @log.info "Received opcode 7, reconnecting" @tasks.map(&:stop) when 9 @log.warn "Received opcode 9, closed connection" @tasks.map(&:stop) if data @log.info "Connection is resumable, reconnecting" @connection.close connect_gateway(true) else @log.info "Connection is not resumable, reconnecting with opcode 2" @connection.close sleep(2) connect_gateway(false) end when 11 @log.debug "Received opcode 11" @ping = Time.now.to_f - @heartbeat_before when 0 handle_event(payload[:t], data) end end end def handle_heartbeat Async do |task| interval = @heartbeat_interval sleep((interval / 1000.0 - 1) * rand) loop do unless @connection.closed? @heartbeat_before = Time.now.to_f @connection.write({ op: 1, d: @last_s }.to_json) @connection.flush @log.debug "Sent opcode 1." @log.debug "Waiting for heartbeat." end sleep(interval / 1000.0 - 1) end end end def handle_event(event_name, data) return @log.debug "Client isn't ready; event #{event_name} wasn't handled" if @wait_until_ready && !@ready && !%w[READY GUILD_CREATE].include?(event_name) dispatch(:event_receive, event_name, data) @log.debug "Handling event #{event_name}" case event_name when "READY" @api_version = data[:v] @session_id = data[:session_id] @user = ClientUser.new(self, data[:user]) @uncached_guilds = data[:guilds].map { |g| g[:id] } if @uncached_guilds == [] or !@intents.guilds ready end dispatch(:ready) @tasks << handle_heartbeat when "GUILD_CREATE" if @uncached_guilds.include?(data[:id]) Guild.new(self, data, true) @uncached_guilds.delete(data[:id]) if @uncached_guilds == [] @log.debug "All guilds cached" ready end elsif @guilds.has?(data[:id]) @guilds[data[:id]].send(:_set_data, data) dispatch(:guild_available, guild) else guild = Guild.new(self, data, true) dispatch(:guild_join, guild) end dispatch(:guild_create, @guilds[data[:id]]) when "MESSAGE_CREATE" message = Message.new(self, data) dispatch(:message, message) when "GUILD_UPDATE" if @guilds.has?(data[:id]) current = @guilds[data[:id]] before = Guild.new(self, current.instance_variable_get(:@data).merge(no_cache: true), false) current.send(:_set_data, data, false) dispatch(:guild_update, before, current) else @log.warn "Unknown guild id #{data[:id]}, ignoring" end when "GUILD_DELETE" return @log.warn "Unknown guild id #{data[:id]}, ignoring" unless (guild = @guilds.delete(data[:id])) dispatch(:guild_delete, guild) if guild.has?(:unavailable) dispatch(:guild_destroy, guild) else dispatch(:guild_leave, guild) end when "GUILD_ROLE_CREATE" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) nr = Role.new(@client, guild, data[:role]) guild.roles[data[:role][:id]] = nr dispatch(:role_create, nr) when "GUILD_ROLE_UPDATE" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) return @log.warn "Unknown role id #{data[:role][:id]}, ignoring" unless guild.roles.has?(data[:role][:id]) current = guild.roles[data[:role][:id]] before = Role.new(@client, guild, current.instance_variable_get(:@data).update({ no_cache: true })) current.send(:_set_data, data[:role]) dispatch(:role_update, before, current) when "GUILD_ROLE_DELETE" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) return @log.warn "Unknown role id #{data[:role_id]}, ignoring" unless (role = guild.roles.delete(data[:role_id])) dispatch(:role_delete, role) when "CHANNEL_CREATE" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) nc = Channel.make_channel(self, data) guild.channels[data[:id]] = nc dispatch(:channel_create, nc) when "CHANNEL_UPDATE" return @log.warn "Unknown channel id #{data[:id]}, ignoring" unless (current = @channels[data[:id]]) before = Channel.make_channel(self, current.instance_variable_get(:@data), no_cache: true) current.send(:_set_data, data) dispatch(:channel_update, before, current) when "CHANNEL_DELETE" return @log.warn "Unknown channel id #{data[:id]}, ignoring" unless (channel = @channels.delete(data[:id])) @guilds[data[:guild_id]]&.channels&.delete(data[:id]) dispatch(:channel_delete, channel) when "CHANNEL_PINS_UPDATE" nil # do in MESSAGE_UPDATE when "THREAD_CREATE" thread = Channel.make_channel(self, data) dispatch(:thread_create, thread) if data.key?(:member) dispatch(:thread_join, thread) else dispatch(:thread_new, thread) end when "THREAD_UPDATE" return @log.warn "Unknown thread id #{data[:id]}, ignoring" unless (thread = @channels[data[:id]]) before = Channel.make_channel(self, thread.instance_variable_get(:@data), no_cache: true) thread.send(:_set_data, data) dispatch(:thread_update, before, thread) when "THREAD_DELETE" return @log.warn "Unknown thread id #{data[:id]}, ignoring" unless (thread = @channels.delete(data[:id])) @guilds[data[:guild_id]]&.channels&.delete(data[:id]) dispatch(:thread_delete, thread) when "THREAD_LIST_SYNC" data[:threads].each do |raw_thread| thread = Channel.make_channel(self, raw_thread.merge({ member: raw_thread[:members].find { |m| m[:id] == raw_thread[:id] } })) @channels[thread.id] = thread end when "THREAD_MEMBER_UPDATE" return @log.warn "Unknown thread id #{data[:id]}, ignoring" unless (thread = @channels[data[:id]]) if (member = thread.members[data[:id]]) old = ThreadChannel::Member.new(self, member.instance_variable_get(:@data)) member._set_data(data) else old = nil member = ThreadChannel::Member.new(self, data) thread.members[data[:user_id]] = member end dispatch(:thread_member_update, thread, old, member) when "THREAD_MEMBERS_UPDATE" return @log.warn "Unknown thread id #{data[:id]}, ignoring" unless (thread = @channels[data[:id]]) thread.instance_variable_set(:@member_count, data[:member_count]) members = [] (data[:added_members] || []).each do |raw_member| member = ThreadChannel::Member.new(self, raw_member) thread.members[member.id] = member members << member end removed_members = [] (data[:removed_member_ids] || []).each do |id| removed_members << thread.members.delete(id) end dispatch(:thread_members_update, thread, members, removed_members) when "STAGE_INSTANCE_CREATE" instance = StageInstance.new(self, data) dispatch(:stage_instance_create, instance) when "STAGE_INSTANCE_UPDATE" return @log.warn "Unknown channel id #{data[:channel_id]} , ignoring" unless (channel = @channels[data[:channel_id]]) return @log.warn "Unknown stage instance id #{data[:id]}, ignoring" unless (instance = channel.stage_instances[data[:id]]) old = StageInstance.new(self, instance.instance_variable_get(:@data), no_cache: true) current.send(:_set_data, data) dispatch(:stage_instance_update, old, current) when "STAGE_INSTANCE_DELETE" return @log.warn "Unknown channel id #{data[:channel_id]} , ignoring" unless (channel = @channels[data[:channel_id]]) return @log.warn "Unknown stage instance id #{data[:id]}, ignoring" unless (instance = channel.stage_instances.delete(data[:id])) dispatch(:stage_instance_delete, instance) when "GUILD_MEMBER_ADD" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) nm = Member.new(self, data[:guild_id], data[:user].update({ no_cache: true }), data) guild.members[nm.id] = nm dispatch(:member_add, nm) when "GUILD_MEMBER_UPDATE" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) return @log.warn "Unknown member id #{data[:user][:id]}, ignoring" unless (nm = guild.members[data[:user][:id]]) old = Member.new(self, data[:guild_id], data[:user], data.update({ no_cache: true })) nm.send(:_set_data, data[:user], data) dispatch(:member_update, old, nm) when "GUILD_MEMBER_REMOVE" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) return @log.warn "Unknown member id #{data[:user][:id]}, ignoring" unless (member = guild.members.delete(data[:user][:id])) dispatch(:member_remove, member) when "GUILD_BAN_ADD" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) user = if @users.has? data[:user][:id] @users[data[:user][:id]] else User.new(self, data[:user].update({ no_cache: true })) end dispatch(:guild_ban_add, guild, user) when "GUILD_BAN_REMOVE" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) user = if @users.has? data[:user][:id] @users[data[:user][:id]] else User.new(self, data[:user].update({ no_cache: true })) end dispatch(:guild_ban_remove, guild, user) when "GUILD_EMOJIS_UPDATE" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) before_emojis = guild.emojis.values.map(&:id).to_set data[:emojis].each do |emoji| guild.emojis[emoji[:id]] = CustomEmoji.new(self, guild, emoji) end deleted_emojis = before_emojis - guild.emojis.values.map(&:id).to_set deleted_emojis.each do |emoji| guild.emojis.delete(emoji) end when "GUILD_INTEGRATIONS_UPDATE" # dispatch(:guild_integrations_update, GuildIntegrationsUpdateEvent.new(self, data)) # Currently not implemented when "INTEGRATION_CREATE" dispatch(:integration_create, Integration.new(self, data, data[:guild_id])) when "INTEGRATION_UPDATE" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) before = Integration.new(self, data, data[:guild_id]) dispatch(:integration_update, integration) when "INTEGRATION_DELETE" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) return @log.warn "Unknown integration id #{data[:id]}, ignoring" unless (integration = guild.integrations.delete(data[:id])) dispatch(:integration_delete, integration) when "WEBHOOKS_UPDATE" dispatch(:webhooks_update, WebhooksUpdateEvent.new(self, data)) when "INVITE_CREATE" dispatch(:invite_create, Invite.new(self, data, true)) when "INVITE_DELETE" dispatch(:invite_delete, InviteDeleteEvent.new(self, data)) when "VOICE_STATE_UPDATE" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) current = guild.voice_states[data[:user_id]] if current.nil? old = nil current = VoiceState.new(self, data) guild.voice_states[data[:user_id]] = current else guild.voice_states.remove(data[:user_id]) if data[:channel_id].nil? old = VoiceState.new(self, current.instance_variable_get(:@data)) current.send(:_set_data, data) end dispatch(:voice_state_update, old, current) if old&.channel != current&.channel dispatch(:voice_channel_update, old, current) case [old&.channel.nil?, current&.channel.nil?] when [true, false] dispatch(:voice_channel_connect, current) when [false, true] dispatch(:voice_channel_disconnect, old) when [false, false] dispatch(:voice_channel_move, old, current) end end if old&.mute? != current&.mute? dispatch(:voice_mute_update, old, current) case [old&.mute?, current&.mute?] when [false, true] dispatch(:voice_mute_enable, current) when [true, false] dispatch(:voice_mute_disable, old) end end if old&.deaf? != current&.deaf? dispatch(:voice_deaf_update, old, current) case [old&.deaf?, current&.deaf?] when [false, true] dispatch(:voice_deaf_enable, current) when [true, false] dispatch(:voice_deaf_disable, old) end end if old&.self_mute? != current&.self_mute? dispatch(:voice_self_mute_update, old, current) case [old&.self_mute?, current&.self_mute?] when [false, true] dispatch(:voice_self_mute_enable, current) when [true, false] dispatch(:voice_self_mute_disable, old) end end if old&.self_deaf? != current&.self_deaf? dispatch(:voice_self_deaf_update, old, current) case [old&.self_deaf?, current&.self_deaf?] when [false, true] dispatch(:voice_self_deaf_enable, current) when [true, false] dispatch(:voice_self_deaf_disable, old) end end if old&.server_mute? != current&.server_mute? dispatch(:voice_server_mute_update, old, current) case [old&.server_mute?, current&.server_mute?] when [false, true] dispatch(:voice_server_mute_enable, current) when [true, false] dispatch(:voice_server_mute_disable, old) end end if old&.server_deaf? != current&.server_deaf? dispatch(:voice_server_deaf_update, old, current) case [old&.server_deaf?, current&.server_deaf?] when [false, true] dispatch(:voice_server_deaf_enable, current) when [true, false] dispatch(:voice_server_deaf_disable, old) end end if old&.video? != current&.video? dispatch(:voice_video_update, old, current) case [old&.video?, current&.video?] when [false, true] dispatch(:voice_video_start, current) when [true, false] dispatch(:voice_video_end, old) end end if old&.stream? != current&.stream? dispatch(:voice_stream_update, old, current) case [old&.stream?, current&.stream?] when [false, true] dispatch(:voice_stream_start, current) when [true, false] dispatch(:voice_stream_end, old) end end when "PRESENCE_UPDATE" return @log.warn "Unknown guild id #{data[:guild_id]}, ignoring" unless (guild = @guilds[data[:guild_id]]) guild.presences[data[:user][:id]] = Presence.new(self, data) when "MESSAGE_UPDATE" if (message = @messages[data[:id]]) before = Message.new(self, message.instance_variable_get(:@data), no_cache: true) message.send(:_set_data, message.instance_variable_get(:@data).merge(data)) else before = nil message = nil end if data[:edited_timestamp].nil? if message.nil? nil elsif message.pinned? message.instance_variable_set(:@pinned, false) else message.instance_variable_set(:@pinned, true) end dispatch(:message_pin_update, MessagePinEvent.new(self, data, message)) else dispatch(:message_update, MessageUpdateEvent.new(self, data, before, current)) end when "MESSAGE_DELETE" message.instance_variable_set(:@deleted, true) if (message = @messages[data[:id]]) dispatch(:message_delete_id, Snowflake.new(data[:id]), channels[data[:channel_id]], data[:guild_id] && guilds[data[:guild_id]]) dispatch(:message_delete, message, channels[data[:channel_id]], data[:guild_id] && guilds[data[:guild_id]]) when "MESSAGE_DELETE_BULK" messages = [] data[:ids].each do |id| if (message = @messages[id]) message.instance_variable_set(:@deleted, true) messages.push(message) else messages.push(UnknownDeleteBulkMessage.new(self, id, data)) end end dispatch(:message_delete_bulk, messages) when "MESSAGE_REACTION_ADD" if (target_message = @messages[data[:message_id]]) if (target_reaction = target_message.reactions.find do |r| r.emoji.is_a?(UnicodeEmoji) ? r.emoji.value == data[:emoji][:name] : r.emoji.id == data[:emoji][:id] end) target_reaction.instance_variable_set(:@count, target_reaction.count + 1) else target_message.reactions << Reaction.new( self, { count: 1, me: @user.id == data[:user_id], emoji: data[:emoji], } ) end end dispatch(:reaction_add, ReactionEvent.new(self, data)) when "MESSAGE_REACTION_REMOVE" if (target_message = @messages[data[:message_id]]) && (target_reaction = target_message.reactions.find do |r| data[:emoji][:id].nil? ? r.emoji.name == data[:emoji][:name] : r.emoji.id == data[:emoji][:id] end) target_reaction.instance_variable_set(:@count, target_reaction.count - 1) target_message.reactions.delete(target_reaction) if target_reaction.count.zero? end dispatch(:reaction_remove, ReactionEvent.new(self, data)) when "MESSAGE_REACTION_REMOVE_ALL" if (target_message = @messages[data[:message_id]]) target_message.reactions = [] end dispatch(:reaction_remove_all, ReactionRemoveAllEvent.new(self, data)) when "MESSAGE_REACTION_REMOVE_EMOJI" if (target_message = @messages[data[:message_id]]) && (target_reaction = target_message.reactions.find { |r| data[:emoji][:id].nil? ? r.name == data[:emoji][:name] : r.id == data[:emoji][:id] }) target_message.reactions.delete(target_reaction) end dispatch(:reaction_remove_emoji, ReactionRemoveEmojiEvent.new(data)) when "TYPING_START" dispatch(:typing_start, TypingStartEvent.new(self, data)) when "INTERACTION_CREATE" interaction = Interaction.make_interaction(self, data) dispatch(:integration_create, interaction) dispatch(interaction.class.event_name, interaction) when "RESUMED" @log.info("Successfully resumed connection") @tasks << handle_heartbeat dispatch(:resumed) when "GUILD_SCHEDULED_EVENT_CREATE" @log.warn("Unknown guild id #{data[:guild_id]}, ignoring") unless (guild = @guilds[data[:guild_id]]) event = ScheduledEvent.new(self, data) guild.scheduled_events[data[:id]] = event dispatch(:scheduled_event_create, event) when "GUILD_SCHEDULED_EVENT_UPDATE" @log.warn("Unknown guild id #{data[:guild_id]}, ignoring") unless (guild = @guilds[data[:guild_id]]) @log.warn("Unknown scheduled event id #{data[:id]}, ignoring") unless (event = guild.scheduled_events[data[:id]]) old = event.dup event.send(:_set_data, data) dispatch(:scheduled_event_update, old, event) if old.status != event.status case event.status when :active dispatch(:scheduled_event_start, event) when :completed dispatch(:scheduled_event_end, event) end else dispatch(:scheduled_event_edit, old, event) end when "GUILD_SCHEDULED_EVENT_DELETE" @log.warn("Unknown guild id #{data[:guild_id]}, ignoring") unless (guild = @guilds[data[:guild_id]]) @log.warn("Unknown scheduled event id #{data[:id]}, ignoring") unless (event = guild.scheduled_events[data[:id]]) guild.scheduled_events.remove(data[:id]) dispatch(:scheduled_event_delete, event) dispatch(:scheduled_event_cancel, event) when "GUILD_SCHEDULED_EVENT_USER_ADD" @log.warn("Unknown guild id #{data[:guild_id]}, ignoring") unless (guild = @guilds[data[:guild_id]]) dispatch(:scheduled_event_user_add, ScheduledEventUserEvent.new(self, data)) when "GUILD_SCHEDULED_EVENT_USER_REMOVE" @log.warn("Unknown guild id #{data[:guild_id]}, ignoring") unless (guild = @guilds[data[:guild_id]]) dispatch(:scheduled_event_user_remove, ScheduledEventUserEvent.new(self, data)) else if respond_to?("event_" + event_name.downcase) __send__("event_" + event_name.downcase, data) else @log.debug "Unhandled event: #{event_name}\n#{data.inspect}" end end end def ready Async do if @fetch_member @log.debug "Fetching members" barrier = Async::Barrier.new @guilds.each do |guild| barrier.async(parent: barrier) do guild.fetch_members end end barrier.wait end @ready = true dispatch(:standby) @log.info("Client is ready!") end end end # # A class for connecting websocket with raw bytes data. # @private # class RawConnection < Async::WebSocket::Connection def initialize(*, **) super @closed = false end def closed? @closed end def close super @closed = true end def parse(buffer) # noop buffer.to_s end def dump(object) # noop object.to_s end end end end