Sort by score and oppai pp

This commit is contained in:
Natsu Kagami 2020-12-01 03:56:14 -05:00
parent a728119fbe
commit 8d4c3e01d8
Signed by: nki
GPG key ID: 73376E117CD20735
2 changed files with 98 additions and 21 deletions

View file

@ -8,7 +8,7 @@ use serenity::{builder::CreateEmbed, utils::MessageBuilder};
use youmubot_prelude::*; use youmubot_prelude::*;
/// Writes a number grouped in groups of 3. /// Writes a number grouped in groups of 3.
fn grouped_number(num: u64) -> String { pub(crate) fn grouped_number(num: u64) -> String {
let s = num.to_string(); let s = num.to_string();
let mut b = MessageBuilder::new(); let mut b = MessageBuilder::new();
let mut i = if s.len() % 3 == 0 { 3 } else { s.len() % 3 }; let mut i = if s.len() % 3 == 0 { 3 } else { s.len() % 3 };

View file

@ -4,8 +4,8 @@ use super::{
ModeArg, OsuClient, ModeArg, OsuClient,
}; };
use crate::{ use crate::{
discord::BeatmapWithMode, discord::{oppai_cache::BeatmapCache, BeatmapWithMode},
models::{Mode, Score}, models::{Mode, Mods, Score},
request::UserID, request::UserID,
}; };
use serenity::{ use serenity::{
@ -139,11 +139,30 @@ pub(crate) mod update_lock {
} }
} }
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
enum OrderBy {
PP,
Score,
}
impl From<&str> for OrderBy {
fn from(s: &str) -> Self {
if s == "--score" {
Self::Score
} else {
Self::PP
}
}
}
#[command("updatelb")] #[command("updatelb")]
#[description = "Update the leaderboard on the last seen beatmap"] #[description = "Update the leaderboard on the last seen beatmap"]
#[max_args(0)] #[usage = "[--score to sort by score, default to sort by pp]"]
#[max_args(1)]
#[only_in(guilds)] #[only_in(guilds)]
pub async fn update_leaderboard(ctx: &Context, m: &Message, mut _args: Args) -> CommandResult { pub async fn update_leaderboard(ctx: &Context, m: &Message, args: Args) -> CommandResult {
let sort_order = OrderBy::from(args.rest());
let guild = m.guild_id.unwrap(); let guild = m.guild_id.unwrap();
let data = ctx.data.read().await; let data = ctx.data.read().await;
let update_lock = data.get::<update_lock::UpdateLock>().unwrap(); let update_lock = data.get::<update_lock::UpdateLock>().unwrap();
@ -161,6 +180,7 @@ pub async fn update_leaderboard(ctx: &Context, m: &Message, mut _args: Args) ->
return Ok(()); return Ok(());
} }
}; };
let mode = bm.1;
let member_cache = data.get::<MemberCache>().unwrap(); let member_cache = data.get::<MemberCache>().unwrap();
// Signal that we are running. // Signal that we are running.
let running_reaction = m.react(&ctx, '⌛').await?; let running_reaction = m.react(&ctx, '⌛').await?;
@ -186,7 +206,7 @@ pub async fn update_leaderboard(ctx: &Context, m: &Message, mut _args: Args) ->
.filter_map(future::ready) .filter_map(future::ready)
.filter_map(|(member, osu_id)| async move { .filter_map(|(member, osu_id)| async move {
let scores = osu let scores = osu
.scores(beatmap_id, |f| f.user(UserID::ID(osu_id))) .scores(beatmap_id, |f| f.user(UserID::ID(osu_id)).mode(mode))
.await .await
.ok(); .ok();
scores scores
@ -211,22 +231,26 @@ pub async fn update_leaderboard(ctx: &Context, m: &Message, mut _args: Args) ->
m.reply( m.reply(
&ctx, &ctx,
format!( format!(
"update for beatmap ({}, {}) complete, {} users updated.", "update for beatmap (`{}`) complete, {} users updated.",
bm.0.beatmap_id, bm.1, updated_users bm.0.short_link(if bm.mode() != bm.1 { Some(bm.1) } else { None }, None),
updated_users
), ),
) )
.await .await
.ok(); .ok();
drop(update_lock); drop(update_lock);
show_leaderboard(ctx, m, bm).await show_leaderboard(ctx, m, bm, sort_order).await
} }
#[command("leaderboard")] #[command("leaderboard")]
#[aliases("lb", "bmranks", "br", "cc")] #[aliases("lb", "bmranks", "br", "cc")]
#[usage = "[--score to sort by score, default to sort by pp]"]
#[description = "See the server's ranks on the last seen beatmap"] #[description = "See the server's ranks on the last seen beatmap"]
#[max_args(0)] #[max_args(1)]
#[only_in(guilds)] #[only_in(guilds)]
pub async fn leaderboard(ctx: &Context, m: &Message, mut _args: Args) -> CommandResult { pub async fn leaderboard(ctx: &Context, m: &Message, args: Args) -> CommandResult {
let sort_order = OrderBy::from(args.rest());
let data = ctx.data.read().await; let data = ctx.data.read().await;
let bm = match get_beatmap(&*data, m.channel_id)? { let bm = match get_beatmap(&*data, m.channel_id)? {
Some(bm) => bm, Some(bm) => bm,
@ -235,13 +259,36 @@ pub async fn leaderboard(ctx: &Context, m: &Message, mut _args: Args) -> Command
return Ok(()); return Ok(());
} }
}; };
show_leaderboard(ctx, m, bm).await show_leaderboard(ctx, m, bm, sort_order).await
} }
async fn show_leaderboard(ctx: &Context, m: &Message, bm: BeatmapWithMode) -> CommandResult { async fn show_leaderboard(
ctx: &Context,
m: &Message,
bm: BeatmapWithMode,
order: OrderBy,
) -> CommandResult {
let data = ctx.data.read().await; let data = ctx.data.read().await;
let mut osu_user_bests = OsuUserBests::open(&*data); let mut osu_user_bests = OsuUserBests::open(&*data);
// Get oppai map.
let mode = bm.1;
let oppai = data.get::<BeatmapCache>().unwrap();
let oppai_map = oppai.get_beatmap(bm.0.beatmap_id).await?;
let get_oppai_pp = move |combo: u64, misses: u64, acc: f64, mods: Mods| {
mode.to_oppai_mode().and_then(|mode| {
oppai_map
.get_pp_from(
oppai_rs::Combo::non_fc(combo as u32, misses as u32),
acc as f32,
Some(mode),
mods,
)
.ok()
.map(|v| v as f64)
})
};
// Run a check on the user once too! // Run a check on the user once too!
{ {
let osu_users = OsuSavedUsers::open(&*data); let osu_users = OsuSavedUsers::open(&*data);
@ -266,8 +313,7 @@ async fn show_leaderboard(ctx: &Context, m: &Message, bm: BeatmapWithMode) -> Co
let guild = m.guild_id.expect("Guild-only command"); let guild = m.guild_id.expect("Guild-only command");
let member_cache = data.get::<MemberCache>().unwrap(); let member_cache = data.get::<MemberCache>().unwrap();
let scores = { let scores = {
const NO_SCORES: &'static str = const NO_SCORES: &'static str = "No scores have been recorded for this beatmap.";
"No scores have been recorded for this beatmap. Run `osu check` to scan for yours!";
let users = osu_user_bests let users = osu_user_bests
.borrow()? .borrow()?
@ -300,11 +346,29 @@ async fn show_leaderboard(ctx: &Context, m: &Message, bm: BeatmapWithMode) -> Co
.map(move |v| future::ready((user.clone(), v.clone()))) .map(move |v| future::ready((user.clone(), v.clone())))
.collect::<stream::FuturesUnordered<_>>() .collect::<stream::FuturesUnordered<_>>()
}) })
.filter_map(|(user, score)| future::ready(score.pp.map(|v| (v, user, score)))) .filter_map(|(user, score)| {
future::ready(
score
.pp
.or_else(|| {
get_oppai_pp(
score.max_combo,
score.count_miss,
score.accuracy(mode),
score.mods,
)
})
.map(|v| (v, user, score)),
)
})
.collect::<Vec<_>>() .collect::<Vec<_>>()
.await; .await;
scores match order {
.sort_by(|(a, _, _), (b, _, _)| b.partial_cmp(a).unwrap_or(std::cmp::Ordering::Equal)); OrderBy::PP => scores.sort_by(|(a, _, _), (b, _, _)| {
b.partial_cmp(a).unwrap_or(std::cmp::Ordering::Equal)
}),
OrderBy::Score => scores.sort_by(|(_, _, a), (_, _, b)| b.score.cmp(&a.score)),
};
scores scores
}; };
@ -350,11 +414,18 @@ async fn show_leaderboard(ctx: &Context, m: &Message, bm: BeatmapWithMode) -> Co
.map(|(_, _, v)| v.rank.to_string()) .map(|(_, _, v)| v.rank.to_string())
.collect::<Vec<_>>(); .collect::<Vec<_>>();
let rw = ranks.iter().map(|v| v.len()).max().unwrap().max(4); let rw = ranks.iter().map(|v| v.len()).max().unwrap().max(4);
let pp_label = match order {
OrderBy::PP => "pp",
OrderBy::Score => "score",
};
let pp = scores let pp = scores
.iter() .iter()
.map(|(pp, _, _)| format!("{:.2}", pp)) .map(|(pp, _, s)| match order {
OrderBy::PP => format!("{:.2}", pp),
OrderBy::Score => format!("{}", crate::discord::embeds::grouped_number(s.score)),
})
.collect::<Vec<_>>(); .collect::<Vec<_>>();
let pw = pp.iter().map(|v| v.len()).max().unwrap_or(2); let pw = pp.iter().map(|v| v.len()).max().unwrap_or(pp_label.len());
/*mods width*/ /*mods width*/
let mdw = scores let mdw = scores
.iter() .iter()
@ -377,7 +448,7 @@ async fn show_leaderboard(ctx: &Context, m: &Message, bm: BeatmapWithMode) -> Co
.push_line("```") .push_line("```")
.push_line(format!( .push_line(format!(
"rank | {:>pw$} | {:mdw$} | {:rw$} | {:>aw$} | {:>cw$} | {:mw$} | {:uw$}", "rank | {:>pw$} | {:mdw$} | {:rw$} | {:>aw$} | {:>cw$} | {:mw$} | {:uw$}",
"pp", pp_label,
"mods", "mods",
"rank", "rank",
"acc", "acc",
@ -434,6 +505,12 @@ async fn show_leaderboard(ctx: &Context, m: &Message, bm: BeatmapWithMode) -> Co
page + 1, page + 1,
(total_len + ITEMS_PER_PAGE - 1) / ITEMS_PER_PAGE, (total_len + ITEMS_PER_PAGE - 1) / ITEMS_PER_PAGE,
)); ));
if let crate::models::ApprovalStatus::Ranked(_) = bm.0.approval {
} else {
if order == OrderBy::PP {
content.push_line("PP was calculated by `oppai-rs`, **not** official values.");
}
}
m.edit(&ctx, |f| f.content(content.build())).await?; m.edit(&ctx, |f| f.content(content.build())).await?;
Ok(true) Ok(true)
}) })