mirror of
https://github.com/natsukagami/youmubot.git
synced 2025-04-16 07:18:54 +00:00
112 lines
3.2 KiB
Rust
112 lines
3.2 KiB
Rust
use crate::db::{CfSavedUsers, CfUser};
|
|
use announcer::MemberToChannels;
|
|
use chrono::Utc;
|
|
use codeforces::{RatingChange, User};
|
|
use serenity::{
|
|
framework::standard::{CommandError, CommandResult},
|
|
http::CacheHttp,
|
|
model::id::{ChannelId, UserId},
|
|
CacheAndHttp,
|
|
};
|
|
use std::sync::Arc;
|
|
use youmubot_prelude::*;
|
|
|
|
type Reqwest = <HTTPClient as TypeMapKey>::Value;
|
|
|
|
/// Updates the rating and rating changes of the users.
|
|
pub fn updates(
|
|
http: Arc<CacheAndHttp>,
|
|
data: AppData,
|
|
channels: MemberToChannels,
|
|
) -> CommandResult {
|
|
let mut users = CfSavedUsers::open(&*data.read()).borrow()?.clone();
|
|
let reqwest = data.get_cloned::<HTTPClient>();
|
|
|
|
for (user_id, cfu) in users.iter_mut() {
|
|
if let Err(e) = update_user(http.clone(), &channels, &reqwest, *user_id, cfu) {
|
|
dbg!((*user_id, e));
|
|
}
|
|
}
|
|
|
|
*CfSavedUsers::open(&*data.read()).borrow_mut()? = users;
|
|
Ok(())
|
|
}
|
|
|
|
fn update_user(
|
|
http: Arc<CacheAndHttp>,
|
|
channels: &MemberToChannels,
|
|
reqwest: &Reqwest,
|
|
user_id: UserId,
|
|
cfu: &mut CfUser,
|
|
) -> CommandResult {
|
|
// Ensure this takes 200ms
|
|
let after = crossbeam_channel::after(std::time::Duration::from_secs_f32(0.2));
|
|
let info = User::info(reqwest, &[cfu.handle.as_str()])?
|
|
.into_iter()
|
|
.next()
|
|
.ok_or(CommandError::from("Not found"))?;
|
|
|
|
let rating_changes = info.rating_changes(reqwest)?;
|
|
|
|
let mut channels_list: Option<Vec<ChannelId>> = None;
|
|
cfu.last_update = Utc::now();
|
|
// Update the rating
|
|
cfu.rating = info.rating;
|
|
|
|
let mut send_message = |rc: RatingChange| -> CommandResult {
|
|
let channels =
|
|
channels_list.get_or_insert_with(|| channels.channels_of(http.clone(), user_id));
|
|
if channels.is_empty() {
|
|
return Ok(());
|
|
}
|
|
let (contest, _, _) =
|
|
codeforces::Contest::standings(reqwest, rc.contest_id, |f| f.limit(1, 1))?;
|
|
for channel in channels {
|
|
if let Err(e) = channel.send_message(http.http(), |e| {
|
|
e.content(format!("Rating change for {}!", user_id.mention()))
|
|
.embed(|c| {
|
|
crate::embed::rating_change_embed(
|
|
&rc,
|
|
&info,
|
|
&contest,
|
|
&user_id.mention(),
|
|
c,
|
|
)
|
|
})
|
|
}) {
|
|
dbg!(e);
|
|
}
|
|
}
|
|
Ok(())
|
|
};
|
|
|
|
let rating_changes = match cfu.last_contest_id {
|
|
None => rating_changes,
|
|
Some(v) => {
|
|
let mut v: Vec<_> = rating_changes
|
|
.into_iter()
|
|
// Skip instead of take because sometimes Codeforces
|
|
// performs rollback.
|
|
.skip_while(|rc| rc.contest_id != v)
|
|
.skip(1)
|
|
.collect();
|
|
v.reverse();
|
|
v
|
|
}
|
|
};
|
|
|
|
cfu.last_contest_id = rating_changes
|
|
.first()
|
|
.map(|v| v.contest_id)
|
|
.or(cfu.last_contest_id);
|
|
|
|
// Check for any good announcements to make
|
|
for rc in rating_changes {
|
|
if let Err(v) = send_message(rc) {
|
|
dbg!(v);
|
|
}
|
|
}
|
|
after.recv().ok();
|
|
|
|
Ok(())
|
|
}
|