feat(gateway): split out event handler to separate file, remove twilight cache

This commit is contained in:
spiral 2022-04-13 08:48:06 -04:00
parent c2094e3b7a
commit 6be8dd0773
No known key found for this signature in database
GPG Key ID: 244A11E4B0BCF40E
4 changed files with 119 additions and 207 deletions

97
gateway/Cargo.lock generated
View File

@ -235,17 +235,6 @@ dependencies = [
"subtle", "subtle",
] ]
[[package]]
name = "dashmap"
version = "5.2.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "4c8858831f7781322e539ea39e72449c46b059638250c14344fec8d0aa6e539c"
dependencies = [
"cfg-if",
"num_cpus",
"parking_lot 0.12.0",
]
[[package]] [[package]]
name = "deadpool" name = "deadpool"
version = "0.9.2" version = "0.9.2"
@ -819,17 +808,7 @@ checksum = "7d17b78036a60663b797adeaee46f5c9dfebb86948d1255007a1d6be0271ff99"
dependencies = [ dependencies = [
"instant", "instant",
"lock_api", "lock_api",
"parking_lot_core 0.8.5", "parking_lot_core",
]
[[package]]
name = "parking_lot"
version = "0.12.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "87f5ec2493a61ac0506c0f4199f99070cbe83857b0337006a30f3e6719b8ef58"
dependencies = [
"lock_api",
"parking_lot_core 0.9.2",
] ]
[[package]] [[package]]
@ -846,19 +825,6 @@ dependencies = [
"winapi", "winapi",
] ]
[[package]]
name = "parking_lot_core"
version = "0.9.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "995f667a6c822200b0433ac218e05582f0e2efa1b922a3fd2fbaadc5f87bab37"
dependencies = [
"cfg-if",
"libc",
"redox_syscall",
"smallvec",
"windows-sys",
]
[[package]] [[package]]
name = "percent-encoding" name = "percent-encoding"
version = "2.1.0" version = "2.1.0"
@ -910,6 +876,7 @@ dependencies = [
"deadpool", "deadpool",
"deadpool-postgres", "deadpool-postgres",
"futures", "futures",
"lazy_static",
"libc", "libc",
"postgres-types", "postgres-types",
"procfs", "procfs",
@ -920,7 +887,6 @@ dependencies = [
"tokio-stream", "tokio-stream",
"tracing", "tracing",
"tracing-subscriber", "tracing-subscriber",
"twilight-cache-inmemory",
"twilight-gateway", "twilight-gateway",
"twilight-gateway-queue", "twilight-gateway-queue",
"twilight-http", "twilight-http",
@ -1434,7 +1400,7 @@ dependencies = [
"mio", "mio",
"num_cpus", "num_cpus",
"once_cell", "once_cell",
"parking_lot 0.11.2", "parking_lot",
"pin-project-lite", "pin-project-lite",
"signal-hook-registry", "signal-hook-registry",
"tokio-macros", "tokio-macros",
@ -1464,7 +1430,7 @@ dependencies = [
"fallible-iterator", "fallible-iterator",
"futures", "futures",
"log", "log",
"parking_lot 0.11.2", "parking_lot",
"percent-encoding", "percent-encoding",
"phf", "phf",
"pin-project-lite", "pin-project-lite",
@ -1618,18 +1584,6 @@ dependencies = [
"webpki", "webpki",
] ]
[[package]]
name = "twilight-cache-inmemory"
version = "0.10.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "644c73ca822657936e6da14b96c11808d6ec848798d1ea6ecc15aef1fc12a383"
dependencies = [
"bitflags",
"dashmap",
"serde",
"twilight-model",
]
[[package]] [[package]]
name = "twilight-gateway" name = "twilight-gateway"
version = "0.10.1" version = "0.10.1"
@ -1898,46 +1852,3 @@ name = "winapi-x86_64-pc-windows-gnu"
version = "0.4.0" version = "0.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f"
[[package]]
name = "windows-sys"
version = "0.34.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "5acdd78cb4ba54c0045ac14f62d8f94a03d10047904ae2a40afa1e99d8f70825"
dependencies = [
"windows_aarch64_msvc",
"windows_i686_gnu",
"windows_i686_msvc",
"windows_x86_64_gnu",
"windows_x86_64_msvc",
]
[[package]]
name = "windows_aarch64_msvc"
version = "0.34.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "17cffbe740121affb56fad0fc0e421804adf0ae00891205213b5cecd30db881d"
[[package]]
name = "windows_i686_gnu"
version = "0.34.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2564fde759adb79129d9b4f54be42b32c89970c18ebf93124ca8870a498688ed"
[[package]]
name = "windows_i686_msvc"
version = "0.34.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9cd9d32ba70453522332c14d38814bceeb747d80b3958676007acadd7e166956"
[[package]]
name = "windows_x86_64_gnu"
version = "0.34.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "cfce6deae227ee8d356d19effc141a509cc503dfd1f850622ec4b0f84428e1f4"
[[package]]
name = "windows_x86_64_msvc"
version = "0.34.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d19538ccc21819d01deaf88d6a17eae6596a12e9aafdbb97916fb49896d89de9"

View File

@ -13,12 +13,12 @@ tracing = "0.1"
tracing-subscriber = "0.3" tracing-subscriber = "0.3"
tokio = { version = "1", features = ["full"] } tokio = { version = "1", features = ["full"] }
tokio-stream = { version = "0.1", features = ["sync"] } tokio-stream = { version = "0.1", features = ["sync"] }
lazy_static = "1.4.0"
procfs = "0.12.0" procfs = "0.12.0"
libc = "0.2.122" libc = "0.2.122"
# Twilight # Twilight
twilight-cache-inmemory = "0.10.0"
twilight-gateway = "0.10.0" twilight-gateway = "0.10.0"
twilight-gateway-queue = "0.10.0" twilight-gateway-queue = "0.10.0"
twilight-http = "0.10.0" twilight-http = "0.10.0"

108
gateway/src/evt.rs Normal file
View File

@ -0,0 +1,108 @@
use deadpool_postgres::Pool;
use redis::AsyncCommands;
use twilight_model::gateway::event::GatewayEventDeserializer;
use std::sync::Arc;
use tracing::info;
use twilight_gateway::Event;
use twilight_http::Client as HttpClient;
lazy_static::lazy_static! {
static ref ALLOWED_EVENTS: Vec<&'static str> = [
"INTERACTION_CREATE",
"MESSAGE_CREATE",
"MESSAGE_DELETE",
"MESSAGE_DELETE_BULK",
"MESSAGE_UPDATE",
"MESSAGE_REACTION_ADD",
].to_vec();
}
pub async fn handle_event<'a>(
shard_id: u64,
event: Event,
http: Arc<HttpClient>,
_db: Pool,
rconn: redis::Client
) -> anyhow::Result<()> {
match event {
Event::GatewayInvalidateSession(resumable) => {
info!("shard {} session invalidated, resumable? {}", shard_id, resumable);
}
Event::ShardConnected(_) => {
info!("shard {} connected", shard_id);
}
Event::ShardDisconnected(info) => {
info!("shard {} disconnected, code: {:?}, reason: {:?}", shard_id, info.code, info.reason);
}
Event::ShardPayload(payload) => {
let deserializer = GatewayEventDeserializer::from_json(std::str::from_utf8(&payload.bytes)?).unwrap();
if deserializer.op() == 0 && ALLOWED_EVENTS.contains(&deserializer.event_type_ref().unwrap()) {
let mut conn = rconn.get_async_connection().await?;
conn.publish::<&str, Vec<u8>, i32>("evt", payload.bytes).await?;
}
}
Event::MessageCreate(msg) => {
if msg.content == "pkt;test" {
// let message_context = db::get_message_context(
// &db,
// msg.author.id.get(),
// msg.guild_id.map(|x| x.get()).unwrap_or(0),
// msg.channel_id.get(),
// )
// .await?;
// let content = format!("message context:\n```\n{:#?}\n```", message_context);
// http.create_message(msg.channel_id)
// .reply(msg.id)
// .content(&content)?
// .exec()
// .await?;
// let proxy_members = db::get_proxy_members(
// &db,
// msg.author.id.get(),
// msg.guild_id.map(|x| x.get()).unwrap_or(0),
// )
// .await?;
// let content = format!("proxy members:\n```\n{:#?}\n```", proxy_members);
// info!("{}", content);
// http.create_message(msg.channel_id)
// .reply(msg.id)
// .content(&content)?
// .exec()
// .await?;
// let cache_stats = cache.stats();
// let pid = unsafe { libc::getpid() };
// let pagesize = {
// unsafe {
// libc::sysconf(libc::_SC_PAGESIZE)
// }
// };
// let p = procfs::process::Process::new(pid)?;
// let content = format!(
// "[rust]\nguilds:{}\nchannels:{}\nroles:{}\nusers:{}\nmembers:{}\n\nmemory usage: {}",
// cache_stats.guilds(),
// cache_stats.channels(),
// cache_stats.roles(),
// cache_stats.users(),
// cache_stats.members(),
// p.stat.rss * pagesize
// );
// http.create_message(msg.channel_id)
// .reply(msg.id)
// .content(&content)?
// .exec()
// .await?;
}
}
_ => {}
}
Ok(())
}

View File

@ -1,17 +1,16 @@
use deadpool_postgres::Pool; use deadpool_postgres::Pool;
use futures::StreamExt; use futures::StreamExt;
use redis::AsyncCommands;
use std::{sync::Arc, env}; use std::{sync::Arc, env};
use tracing::{error, info, Level}; use tracing::{error, info, Level};
use twilight_cache_inmemory::{InMemoryCache, ResourceType};
use twilight_gateway::{ use twilight_gateway::{
cluster::{Events, ShardScheme}, cluster::{Events, ShardScheme},
Cluster, Event, EventTypeFlags, Intents, Cluster, EventTypeFlags, Intents,
}; };
use twilight_http::Client as HttpClient; use twilight_http::Client as HttpClient;
mod config; mod config;
mod evt;
mod db; mod db;
mod util; mod util;
@ -25,10 +24,10 @@ async fn main() -> anyhow::Result<()> {
let http = Arc::new(HttpClient::new(cfg.token.clone())); let http = Arc::new(HttpClient::new(cfg.token.clone()));
let rconn = redis::Client::open(cfg.redis_addr.clone()).unwrap(); let rconn = redis::Client::open(cfg.redis_addr.clone()).unwrap();
let (_cluster, events) = init_gateway(&cfg, rconn.clone()).await?; let (_cluster, events) = init_gateway(&cfg, rconn.clone()).await?;
let cache = init_cache(); // let cache = init_cache();
let db = db::init_db(&cfg).await?; let db = db::init_db(&cfg).await?;
run(http, events, cache, db, rconn).await?; run(http, events, db, rconn).await?;
Ok(()) Ok(())
} }
@ -36,25 +35,22 @@ async fn main() -> anyhow::Result<()> {
async fn run( async fn run(
http: Arc<HttpClient>, http: Arc<HttpClient>,
mut events: Events, mut events: Events,
cache: Arc<InMemoryCache>,
db: Pool, db: Pool,
rconn: redis::Client, rconn: redis::Client,
) -> anyhow::Result<()> { ) -> anyhow::Result<()> {
while let Some((shard_id, event)) = events.next().await { while let Some((shard_id, event)) = events.next().await {
cache.update(&event); // cache.update(&event);
let http_cloned = http.clone(); let http_cloned = http.clone();
let cache_cloned = cache.clone();
let db_cloned = db.clone(); let db_cloned = db.clone();
let rconn_cloned = rconn.clone(); let rconn_cloned = rconn.clone();
tokio::spawn(async move { tokio::spawn(async move {
let result = handle_event( let result = evt::handle_event(
shard_id, shard_id,
event, event,
http_cloned, http_cloned,
cache_cloned,
db_cloned, db_cloned,
rconn_cloned rconn_cloned
) )
@ -68,93 +64,6 @@ async fn run(
Ok(()) Ok(())
} }
async fn handle_event<'a>(
shard_id: u64,
event: Event,
http: Arc<HttpClient>,
cache: Arc<InMemoryCache>,
_db: Pool,
rconn: redis::Client
) -> anyhow::Result<()> {
match event {
Event::GatewayInvalidateSession(resumable) => {
info!("shard {} session invalidated, resumable? {}", shard_id, resumable);
}
Event::ShardConnected(_) => {
info!("shard {} connected", shard_id);
}
Event::ShardDisconnected(info) => {
info!("shard {} disconnected, code: {:?}, reason: {:?}", shard_id, info.code, info.reason);
}
Event::ShardPayload(payload) => {
let mut conn = rconn.get_async_connection().await?;
conn.publish::<&str, Vec<u8>, i32>("evt", payload.bytes).await?;
}
Event::MessageCreate(msg) => {
if msg.content == "pkt;test" {
// let message_context = db::get_message_context(
// &db,
// msg.author.id.get(),
// msg.guild_id.map(|x| x.get()).unwrap_or(0),
// msg.channel_id.get(),
// )
// .await?;
// let content = format!("message context:\n```\n{:#?}\n```", message_context);
// http.create_message(msg.channel_id)
// .reply(msg.id)
// .content(&content)?
// .exec()
// .await?;
// let proxy_members = db::get_proxy_members(
// &db,
// msg.author.id.get(),
// msg.guild_id.map(|x| x.get()).unwrap_or(0),
// )
// .await?;
// let content = format!("proxy members:\n```\n{:#?}\n```", proxy_members);
// info!("{}", content);
// http.create_message(msg.channel_id)
// .reply(msg.id)
// .content(&content)?
// .exec()
// .await?;
let cache_stats = cache.stats();
let pid = unsafe { libc::getpid() };
let pagesize = {
unsafe {
libc::sysconf(libc::_SC_PAGESIZE)
}
};
let p = procfs::process::Process::new(pid)?;
let content = format!(
"[rust]\nguilds:{}\nchannels:{}\nroles:{}\nusers:{}\nmembers:{}\n\nmemory usage: {}",
cache_stats.guilds(),
cache_stats.channels(),
cache_stats.roles(),
cache_stats.users(),
cache_stats.members(),
p.stat.rss * pagesize
);
http.create_message(msg.channel_id)
.reply(msg.id)
.content(&content)?
.exec()
.await?;
}
}
_ => {}
}
Ok(())
}
fn init_tracing() { fn init_tracing() {
tracing_subscriber::fmt() tracing_subscriber::fmt()
.with_max_level(Level::INFO) .with_max_level(Level::INFO)
@ -203,10 +112,7 @@ async fn init_gateway(
| EventTypeFlags::SHARD_PAYLOAD | EventTypeFlags::SHARD_PAYLOAD
| EventTypeFlags::SHARD_CONNECTED | EventTypeFlags::SHARD_CONNECTED
| EventTypeFlags::SHARD_DISCONNECTED | EventTypeFlags::SHARD_DISCONNECTED
| EventTypeFlags::GUILD_CREATE
| EventTypeFlags::CHANNEL_CREATE
| EventTypeFlags::MESSAGE_CREATE | EventTypeFlags::MESSAGE_CREATE
// | EventTypeFlags::MESSAGE_UPDATE
) )
.queue(Arc::new(queue)) .queue(Arc::new(queue))
.build() .build()
@ -220,16 +126,3 @@ async fn init_gateway(
Ok((cluster, events)) Ok((cluster, events))
} }
fn init_cache() -> Arc<InMemoryCache> {
let cache = InMemoryCache::builder()
.resource_types(
ResourceType::GUILD
| ResourceType::CHANNEL
| ResourceType::ROLE
| ResourceType::USER
// | ResourceType::MEMBER
)
.build();
Arc::new(cache)
}