Merge pull request #45 from DjDeveloperr/main

feat(gateway): added presence caching, marking events as complete!
This commit is contained in:
Helloyunho 2020-12-02 20:20:30 +09:00 committed by GitHub
commit 78c86a9bbc
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
13 changed files with 188 additions and 76 deletions

3
mod.ts
View file

@ -16,6 +16,8 @@ export * from './src/managers/guilds.ts'
export * from './src/managers/guildChannels.ts' export * from './src/managers/guildChannels.ts'
export * from './src/managers/guildEmojis.ts' export * from './src/managers/guildEmojis.ts'
export * from './src/managers/members.ts' export * from './src/managers/members.ts'
export * from './src/managers/messageReactions.ts'
export * from './src/managers/reactionUsers.ts'
export * from './src/managers/messages.ts' export * from './src/managers/messages.ts'
export * from './src/managers/roles.ts' export * from './src/managers/roles.ts'
export * from './src/managers/users.ts' export * from './src/managers/users.ts'
@ -39,6 +41,7 @@ export * from './src/structures/presence.ts'
export * from './src/structures/role.ts' export * from './src/structures/role.ts'
export * from './src/structures/snowflake.ts' export * from './src/structures/snowflake.ts'
export * from './src/structures/textChannel.ts' export * from './src/structures/textChannel.ts'
export * from './src/structures/messageReaction.ts'
export * from './src/structures/user.ts' export * from './src/structures/user.ts'
export * from './src/structures/webhook.ts' export * from './src/structures/webhook.ts'
export * from './src/types/application.ts' export * from './src/types/application.ts'

View file

@ -22,6 +22,8 @@ export const guildCreate: GatewayEventHandler = async (
await guild.roles.fromPayload(d.roles) await guild.roles.fromPayload(d.roles)
if (d.presences !== undefined) await guild.presences.fromPayload(d.presences)
if (d.voice_states !== undefined) if (d.voice_states !== undefined)
await guild.voiceStates.fromPayload(d.voice_states) await guild.voiceStates.fromPayload(d.voice_states)

View file

@ -14,6 +14,7 @@ export const guildDelte: GatewayEventHandler = async (
await guild.members.flush() await guild.members.flush()
await guild.channels.flush() await guild.channels.flush()
await guild.roles.flush() await guild.roles.flush()
await guild.presences.flush()
await gateway.client.guilds.delete(d.id) await gateway.client.guilds.delete(d.id)
gateway.client.emit('guildDelete', guild) gateway.client.emit('guildDelete', guild)

View file

@ -14,7 +14,13 @@ export const guildMembersChunk: GatewayEventHandler = async (
await guild.members.set(member.user.id, member) await guild.members.set(member.user.id, member)
} }
// TODO: Cache Presences if (d.chunk_index === 0) await guild.presences.flush()
if (d.presences !== undefined) {
for (const pres of d.presences) {
await guild.presences.set(pres.user.id, pres)
}
}
gateway.client.emit('guildMembersChunk', guild, { gateway.client.emit('guildMembersChunk', guild, {
members: d.members.map((m) => m.user.id), members: d.members.map((m) => m.user.id),

View file

@ -47,6 +47,9 @@ import { guildMembersChunk } from './guildMembersChunk.ts'
import { presenceUpdate } from './presenceUpdate.ts' import { presenceUpdate } from './presenceUpdate.ts'
import { inviteCreate } from './inviteCreate.ts' import { inviteCreate } from './inviteCreate.ts'
import { inviteDelete } from './inviteDelete.ts' import { inviteDelete } from './inviteDelete.ts'
import { MessageReaction } from '../../structures/messageReaction.ts'
import { Invite } from '../../structures/invite.ts'
import { Presence } from '../../structures/presence.ts'
export const gatewayHandlers: { export const gatewayHandlers: {
[eventCode in GatewayEvents]: GatewayEventHandler | undefined [eventCode in GatewayEvents]: GatewayEventHandler | undefined
@ -131,16 +134,23 @@ export interface ClientEvents extends EventTypes {
uncached: Set<string> uncached: Set<string>
) => void ) => void
messageUpdate: (before: Message, after: Message) => void messageUpdate: (before: Message, after: Message) => void
messageReactionAdd: (reaction: MessageReaction, user: User) => void
messageReactionRemove: (reaction: MessageReaction, user: User) => void
messageReactionRemoveAll: (message: Message) => void
messageReactionRemoveEmoji: (message: Message, emoji: Emoji) => void
typingStart: ( typingStart: (
user: User, user: User,
channel: TextChannel, channel: TextChannel,
at: Date, at: Date,
guildData?: TypingStartGuildData guildData?: TypingStartGuildData
) => void ) => void
inviteCreate: (invite: Invite) => void
inviteDelete: (invite: Invite) => void
userUpdate: (before: User, after: User) => void userUpdate: (before: User, after: User) => void
voiceServerUpdate: (data: VoiceServerUpdateData) => void voiceServerUpdate: (data: VoiceServerUpdateData) => void
voiceStateAdd: (state: VoiceState) => void voiceStateAdd: (state: VoiceState) => void
voiceStateRemove: (state: VoiceState) => void voiceStateRemove: (state: VoiceState) => void
voiceStateUpdate: (state: VoiceState, after: VoiceState) => void voiceStateUpdate: (state: VoiceState, after: VoiceState) => void
presenceUpdate: (presence: Presence) => void
webhooksUpdate: (guild: Guild, channel: GuildTextChannel) => void webhooksUpdate: (guild: Guild, channel: GuildTextChannel) => void
} }

View file

@ -2,6 +2,7 @@ import { Gateway, GatewayEventHandler } from '../index.ts'
import { Guild } from '../../structures/guild.ts' import { Guild } from '../../structures/guild.ts'
import { InviteCreatePayload } from '../../types/gateway.ts' import { InviteCreatePayload } from '../../types/gateway.ts'
import { ChannelPayload, GuildPayload, InvitePayload } from '../../../mod.ts' import { ChannelPayload, GuildPayload, InvitePayload } from '../../../mod.ts'
import { Invite } from '../../structures/invite.ts'
export const inviteCreate: GatewayEventHandler = async ( export const inviteCreate: GatewayEventHandler = async (
gateway: Gateway, gateway: Gateway,
@ -36,5 +37,5 @@ export const inviteCreate: GatewayEventHandler = async (
await guild.invites.set(d.code, dataConverted) await guild.invites.set(d.code, dataConverted)
const invite = await guild.invites.get(d.code) const invite = await guild.invites.get(d.code)
gateway.client.emit('inviteCreate', invite) gateway.client.emit('inviteCreate', (invite as unknown) as Invite)
} }

View file

@ -1,6 +1,16 @@
import { PresenceUpdatePayload } from '../../types/gateway.ts'
import { Gateway, GatewayEventHandler } from '../index.ts' import { Gateway, GatewayEventHandler } from '../index.ts'
export const presenceUpdate: GatewayEventHandler = async ( export const presenceUpdate: GatewayEventHandler = async (
gateway: Gateway, gateway: Gateway,
d: any d: PresenceUpdatePayload
) => {} ) => {
const guild = await gateway.client.guilds.get(d.guild_id)
if (guild === undefined) return
await guild.presences.set(d.user.id, d)
const presence = await guild.presences.get(d.user.id)
if (presence === undefined) return
gateway.client.emit('presenceUpdate', presence)
}

39
src/managers/presences.ts Normal file
View file

@ -0,0 +1,39 @@
import { Client } from '../models/client.ts'
import { Guild } from '../structures/guild.ts'
import { Presence } from '../structures/presence.ts'
import { User } from '../structures/user.ts'
import { PresenceUpdatePayload } from '../types/gateway.ts'
import { BaseManager } from './base.ts'
export class GuildPresencesManager extends BaseManager<
PresenceUpdatePayload,
Presence
> {
guild: Guild
constructor(client: Client, guild: Guild) {
super(client, `presences:${guild.id}`, Presence)
this.guild = guild
}
async get(id: string): Promise<Presence | undefined> {
const raw = await this._get(id)
if (raw === undefined) return
let user = await this.client.users.get(raw.user.id)
if (user === undefined) user = new User(this.client, raw.user)
const guild = await this.client.guilds.get(raw.guild_id)
if (guild === undefined) return
const presence = new Presence(this.client, raw, user, guild)
return presence
}
async fromPayload(
data: PresenceUpdatePayload[]
): Promise<GuildPresencesManager> {
await this.flush()
for (const pres of data) {
await this.set(pres.user.id, pres)
}
return this
}
}

View file

@ -7,7 +7,6 @@ import {
IntegrationAccountPayload, IntegrationAccountPayload,
IntegrationExpireBehavior, IntegrationExpireBehavior,
} from '../types/guild.ts' } from '../types/guild.ts'
import { PresenceUpdatePayload } from '../types/gateway.ts'
import { Base } from './base.ts' import { Base } from './base.ts'
import { RolesManager } from '../managers/roles.ts' import { RolesManager } from '../managers/roles.ts'
import { InviteManager } from '../managers/invites.ts' import { InviteManager } from '../managers/invites.ts'
@ -21,6 +20,7 @@ import { Application } from './application.ts'
import { GUILD_BAN, GUILD_BANS, GUILD_INTEGRATIONS } from '../types/endpoint.ts' import { GUILD_BAN, GUILD_BANS, GUILD_INTEGRATIONS } from '../types/endpoint.ts'
import { GuildVoiceStatesManager } from '../managers/guildVoiceStates.ts' import { GuildVoiceStatesManager } from '../managers/guildVoiceStates.ts'
import { RequestMembersOptions } from '../gateway/index.ts' import { RequestMembersOptions } from '../gateway/index.ts'
import { GuildPresencesManager } from '../managers/presences.ts'
export class GuildBan extends Base { export class GuildBan extends Base {
guild: Guild guild: Guild
@ -147,7 +147,7 @@ export class Guild extends Base {
voiceStates: GuildVoiceStatesManager voiceStates: GuildVoiceStatesManager
members: MembersManager members: MembersManager
channels: GuildChannelsManager channels: GuildChannelsManager
presences?: PresenceUpdatePayload[] presences: GuildPresencesManager
maxPresences?: number maxPresences?: number
maxMembers?: number maxMembers?: number
vanityURLCode?: string vanityURLCode?: string
@ -169,6 +169,7 @@ export class Guild extends Base {
this.unavailable = data.unavailable this.unavailable = data.unavailable
this.members = new MembersManager(this.client, this) this.members = new MembersManager(this.client, this)
this.voiceStates = new GuildVoiceStatesManager(client, this) this.voiceStates = new GuildVoiceStatesManager(client, this)
this.presences = new GuildPresencesManager(client, this)
this.channels = new GuildChannelsManager( this.channels = new GuildChannelsManager(
this.client, this.client,
this.client.channels, this.client.channels,
@ -203,7 +204,6 @@ export class Guild extends Base {
this.joinedAt = data.joined_at this.joinedAt = data.joined_at
this.large = data.large this.large = data.large
this.memberCount = data.member_count this.memberCount = data.member_count
this.presences = data.presences
this.maxPresences = data.max_presences this.maxPresences = data.max_presences
this.maxMembers = data.max_members this.maxMembers = data.max_members
this.vanityURLCode = data.vanity_url_code this.vanityURLCode = data.vanity_url_code
@ -252,7 +252,6 @@ export class Guild extends Base {
this.joinedAt = data.joined_at ?? this.joinedAt this.joinedAt = data.joined_at ?? this.joinedAt
this.large = data.large ?? this.large this.large = data.large ?? this.large
this.memberCount = data.member_count ?? this.memberCount this.memberCount = data.member_count ?? this.memberCount
this.presences = data.presences ?? this.presences
this.maxPresences = data.max_presences ?? this.maxPresences this.maxPresences = data.max_presences ?? this.maxPresences
this.maxMembers = data.max_members ?? this.maxMembers this.maxMembers = data.max_members ?? this.maxMembers
this.vanityURLCode = data.vanity_url_code ?? this.vanityURLCode this.vanityURLCode = data.vanity_url_code ?? this.vanityURLCode

View file

@ -1,5 +1,15 @@
import { ActivityGame, ClientActivity, StatusType } from '../types/presence.ts' import {
import { StatusUpdatePayload } from '../types/gateway.ts' ActivityGame,
ActivityPayload,
ClientActivity,
ClientStatus,
StatusType,
} from '../types/presence.ts'
import { PresenceUpdatePayload, StatusUpdatePayload } from '../types/gateway.ts'
import { Base } from './base.ts'
import { Guild } from './guild.ts'
import { User } from './user.ts'
import { Client } from '../models/client.ts'
enum ActivityTypes { enum ActivityTypes {
PLAYING = 0, PLAYING = 0,
@ -7,7 +17,37 @@ enum ActivityTypes {
LISTENING = 2, LISTENING = 2,
WATCHING = 3, WATCHING = 3,
CUSTOM_STATUS = 4, CUSTOM_STATUS = 4,
COMPETING = 5 COMPETING = 5,
}
export class Presence extends Base {
user: User
guild: Guild
status: StatusType
// TODO: Maybe a new structure for this?
activities: ActivityPayload[]
clientStatus: ClientStatus
constructor(
client: Client,
data: PresenceUpdatePayload,
user: User,
guild: Guild
) {
super(client, data)
this.user = user
this.guild = guild
this.status = data.status
this.activities = data.activities
this.clientStatus = data.client_status
}
fromPayload(data: PresenceUpdatePayload): Presence {
this.status = data.status
this.activities = data.activities
this.clientStatus = data.client_status
return this
}
} }
export class ClientPresence { export class ClientPresence {
@ -48,7 +88,7 @@ export class ClientPresence {
afk: this.afk === undefined ? false : this.afk, afk: this.afk === undefined ? false : this.afk,
activities: this.createActivity(), activities: this.createActivity(),
since: this.since === undefined ? null : this.since, since: this.since === undefined ? null : this.since,
status: this.status === undefined ? 'online' : this.status status: this.status === undefined ? 'online' : this.status,
} }
} }
@ -62,7 +102,7 @@ export class ClientPresence {
: [this.activity] : [this.activity]
if (activity === null) return activity if (activity === null) return activity
else { else {
activity.map(e => { activity.map((e) => {
if (typeof e.type === 'string') e.type = ActivityTypes[e.type] if (typeof e.type === 'string') e.type = ActivityTypes[e.type]
return e return e
}) })

View file

@ -1,4 +1,5 @@
import { Channel, Guild } from "../../mod.ts" import { Channel } from '../structures/channel.ts'
import { Guild } from '../structures/guild.ts'
import { ChannelPayload } from './channel.ts' import { ChannelPayload } from './channel.ts'
import { GuildPayload } from './guild.ts' import { GuildPayload } from './guild.ts'
import { UserPayload } from './user.ts' import { UserPayload } from './user.ts'

View file

@ -56,7 +56,7 @@ export enum ActivityFlags {
SPECTATE = 1 << 2, SPECTATE = 1 << 2,
JOIN_REQUEST = 1 << 3, JOIN_REQUEST = 1 << 3,
SYNC = 1 << 4, SYNC = 1 << 4,
PLAY = 1 << 5 PLAY = 1 << 5,
} }
export type ActivityType = export type ActivityType =