Add leaderboard

late-submissions
Elnu 2 years ago
parent 62a26cda2e
commit a2ff904532

@ -1,4 +1,5 @@
DISCORD_TOKEN=
PREFIX="-h "
HUGO=/path/to/hugo
GUILD_DATA=/path/to/guilds.json
GUILD_DATA=/path/to/guilds.json
LEADERBOARD=/path/to/leaderboard.html

@ -66,7 +66,7 @@ async fn submit(ctx: &Context, msg: &Message) -> CommandResult {
let path = Path::new(&submission_images_dir);
std::fs::create_dir_all(path)?;
let mut submission_data = get_submission_data();
let mut submission_data = get_current_submission_data();
let mut existing_submitter = false;
let mut invalid_types = false;
let mut requires_rebuild = false;
@ -164,6 +164,7 @@ async fn submit(ctx: &Context, msg: &Message) -> CommandResult {
if invalid_types {
message.push_str("\nSome of your attachments could not be uploaded; only **.png**, **.jpg**, and **.jpeg** files are permitted.");
}
leaderboard(&ctx).await?;
rebuild_site();
} else if invalid_types {
message.push_str("Sorry, your submission could not be uploaded; only **.png**, **.jpg**, and **.jpeg** files are permitted.");
@ -175,7 +176,7 @@ async fn submit(ctx: &Context, msg: &Message) -> CommandResult {
#[command]
async fn images(ctx: &Context, msg: &Message) -> CommandResult {
let submission_data = get_submission_data();
let submission_data = get_current_submission_data();
let images: Vec<String> = {
let mut images = Vec::new();
for submission in submission_data.iter() {
@ -237,7 +238,7 @@ async fn imageDelete(ctx: &Context, msg: &Message, mut args: Args) -> CommandRes
return Ok(());
}
let challenge_number = get_challenge_number();
let mut submission_data = get_submission_data();
let mut submission_data = get_current_submission_data();
for (i, submission) in submission_data.iter_mut().enumerate() {
if !is_matching_submission(&submission, &msg) {
continue;
@ -366,11 +367,7 @@ async fn suggest(ctx: &Context, msg: &Message, args: Args) -> CommandResult {
));
let mut author = serenity::builder::CreateEmbedAuthor::default();
author
.icon_url(format!(
"https://cdn.discordapp.com/avatars/{}/{}.webp",
msg.author.id,
msg.author.avatar.as_ref().unwrap()
))
.icon_url(get_avatar(&msg.author))
.name(username)
.url(format!("https://discord.com/users/{}", msg.author.id));
e.set_author(author);
@ -385,4 +382,4 @@ async fn suggest(ctx: &Context, msg: &Message, args: Args) -> CommandResult {
.unwrap();
msg.reply(&ctx.http, "Suggestion sent! Thank you for making a suggestion. If it is chosen to be used in a future challenge, you will be mentioned in the challenge description!").await?;
Ok(())
}
}

@ -11,6 +11,7 @@ use std::fs::OpenOptions;
use std::io::Read;
use std::io::Write;
use std::process::Command;
use std::collections::HashMap;
pub fn get_challenge_number() -> i32 {
let challenge_dir = format!("{}/content/challenges", env::var("HUGO").unwrap());
@ -41,16 +42,24 @@ pub fn get_submission_images_dir() -> String {
format!("{}/assets/{}", get_hugo_path(), get_challenge_number())
}
pub fn get_submission_data_path() -> String {
pub fn get_submission_data_path(challenge: i32) -> String {
format!(
"{}/data/challenges/{}.json",
get_hugo_path(),
get_challenge_number()
challenge
)
}
pub fn get_submission_data() -> Vec<Value> {
let submission_data_path = get_submission_data_path();
pub fn get_current_submission_data_path() -> String {
get_submission_data_path(get_challenge_number())
}
pub fn get_current_submission_data() -> Vec<Value> {
get_submission_data(get_challenge_number())
}
pub fn get_submission_data(challenge: i32) -> Vec<Value> {
let submission_data_path = get_submission_data_path(challenge);
let submission_data_json = match File::open(&submission_data_path) {
Ok(mut file) => {
let mut json = String::new();
@ -74,7 +83,7 @@ pub fn set_submission_data(submission_data: Vec<Value>) {
let mut submission_data_file = OpenOptions::new()
.write(true)
.truncate(true)
.open(get_submission_data_path())
.open(get_current_submission_data_path())
.unwrap();
submission_data_file
.write_all(
@ -364,3 +373,42 @@ pub async fn random_kanji(
}
Ok(())
}
pub fn get_avatar(user: &User) -> String {
match user.avatar_url() {
Some(avatar_url) => avatar_url,
None => user.default_avatar_url()
}
}
pub async fn leaderboard(ctx: &Context) -> CommandResult {
const LENGTH: usize = 5;
let mut submission_counts: HashMap<String, u32> = HashMap::new();
for challenge in 1..get_challenge_number() + 1 {
let submission_data = get_submission_data(challenge);
for submission in submission_data.iter() {
let user = submission_counts.entry(String::from(submission.as_object().unwrap()["id"].as_str().unwrap())).or_insert(0);
*user += 1;
}
}
let mut top_submitters: Vec<(&String, &u32)> = submission_counts.iter().collect();
top_submitters.sort_by(|a, b| b.1.cmp(a.1));
let mut leaderboard_html = String::from("<table id=\"leaderboard\">");
for (i, (id, count)) in top_submitters[0..LENGTH].iter().enumerate() {
let place = i + 1;
let user = UserId(id.parse::<u64>().unwrap()).to_user(&ctx.http).await?;
let avatar = get_avatar(&user);
let profile = format!("https://discord.com/users/{id}");
let name = user.name;
let discriminator = user.discriminator;
leaderboard_html.push_str(&format!("<tr><td>{place}</td><td><img src=\"{avatar}\" alt=\"avatar\"></td><td><a href=\"{profile}\" target=\"_blank\">{name}<span class=\"muted\">#{discriminator}</span></a></td><td>{count}</td></tr>"));
}
leaderboard_html.push_str("</table>");
let mut file = std::fs::OpenOptions::new()
.write(true)
.open(env::var("LEADERBOARD").unwrap())
.unwrap();
file.write_all(leaderboard_html.as_bytes())?;
file.flush()?;
Ok(())
}
Loading…
Cancel
Save