Files
discordeno/module/client.ts
2020-02-25 22:56:15 -05:00

141 lines
4.5 KiB
TypeScript

import { endpoints } from "../constants/discord.ts"
import DiscordRequestManager from "./discord-request-manager.ts"
import { DiscordBotGatewayData, DiscordPayload, DiscordHeartbeatPayload, GatewayOpcode } from "../types/discord.ts"
import ShardingManager from "./sharding-manager.ts"
import {
connectWebSocket,
isWebSocketCloseEvent,
isWebSocketPingEvent,
isWebSocketPongEvent
} from "https://deno.land/std/ws/mod.ts"
import Gateway from "./gateway.ts"
import { ClientOptions, FulfilledClientOptions } from "../types/options.ts"
import { CollectedMessageType } from "../types/message-type.ts"
class Client {
bot_id: string
/** The bot's token. This should never be used by end users. It is meant to be used internally to make requests to the Discord API. */
token: string
/** The Rate limit manager to handle all outgoing requests to discord. Not meant to be used by users. */
discordRequestManager: DiscordRequestManager
/** Creates and handles all the shards necessary for the bot. */
shardingManager: ShardingManager
/** The options (with defaults) passed to the `Client` constructor. */
options: FulfilledClientOptions
protected authorization: string
constructor(options: ClientOptions) {
// Assign some defaults to the options to make them fulfilled / not annoying to use.
this.options = Object.assign(
{
properties: {
$os: "...",
$browser: "...",
$device: "..."
},
compress: false
},
options
)
this.bot_id = options.bot_id
this.token = options.token
this.authorization = `Bot ${this.options.token}`
this.discordRequestManager = new DiscordRequestManager(this, this.authorization)
this.shardingManager = new ShardingManager()
}
getGatewayData() {
return this.discordRequestManager.get(endpoints.GATEWAY_BOT) as Promise<DiscordBotGatewayData>
}
createWebsocketConnection(data: DiscordBotGatewayData) {
console.log({ data })
return connectWebSocket(data.url)
}
async bootstrap() {
const data = await this.getGatewayData()
const socket = await this.createWebsocketConnection(data)
const gateway = new Gateway(socket)
const messages = this.collectMessages(gateway)
await gateway.identify(this.options)
return {
data,
socket,
gateway,
messages,
connection: this.connect(gateway, data)
}
}
async *collectMessages(gateway: Gateway) {
const { socket } = gateway
for await (const message of socket.receive()) {
if (typeof message === "string") {
yield {
type: CollectedMessageType.Message,
data: JSON.parse(message)
}
} else if (isWebSocketCloseEvent(message)) {
yield { type: CollectedMessageType.Close, ...message }
return
} else if (isWebSocketPingEvent(message)) {
yield { type: CollectedMessageType.Ping }
} else if (isWebSocketPongEvent(message)) {
yield { type: CollectedMessageType.Pong }
}
}
}
/** Begins initial handshake, creates the websocket with Discord and spawns all necessary shards. */
async *connect(
gateway: Gateway,
data: DiscordBotGatewayData
): AsyncGenerator<{ type: CollectedMessageType; data?: DiscordPayload; action?: Promise<void> }> {
for await (const message of this.collectMessages(gateway)) {
switch (message.type) {
case CollectedMessageType.Ping:
console.log("Ping!")
yield message
break
case CollectedMessageType.Pong:
console.log("Pong!")
yield message
break
case CollectedMessageType.Close:
console.log("Close :(", message)
yield message
break
case CollectedMessageType.Message:
await this.handleDiscordPayload(message.data, gateway)
yield message
console.log({ yay: true, ...message })
break
}
}
// Begin spawning all necessary shards
this.spawnShards(data.shards)
}
handleDiscordPayload(data: DiscordPayload, gateway: Gateway) {
switch (data.op) {
case GatewayOpcode.Hello:
console.log("heartbeating...")
return gateway.sendConstantHeartbeats((data.d as DiscordHeartbeatPayload).heartbeat_interval, data.s)
}
// Make all code paths return a promise for consistency.
return Promise.resolve(undefined)
}
spawnShards(total: number, id = 1) {
// this.ShardingManager.spawnShard(id);
if (id < total) this.spawnShards(total, id + 1)
}
}
export default Client