use super::error::ErrorTypes; use sqlx::mysql::MySqlPool; use serenity::voice::{ AudioSource, ffmpeg, }; use tokio::{ fs::File, process::Command, }; use std::{ env, path::Path, }; pub struct Sound { pub name: String, pub id: u32, pub plays: u32, pub public: bool, pub server_id: u64, pub uploader_id: u64, pub src: Vec, } impl Sound { pub async fn search_for_sound(query: &str, guild_id: u64, user_id: u64, db_pool: MySqlPool, strict: bool) -> Result, sqlx::Error> { fn extract_id(s: &str) -> Option { if s.len() > 3 && s.to_lowercase().starts_with("id:") { match s[3..].parse::() { Ok(id) => Some(id), Err(_) => None } } else { None } } if let Some(id) = extract_id(&query) { let sound = sqlx::query_as_unchecked!( Self, " SELECT * FROM sounds WHERE id = ? AND ( public = 1 OR uploader_id = ? OR server_id = ? ) ", id, user_id, guild_id ) .fetch_all(&db_pool) .await?; Ok(sound) } else { let name = query; let sound; if strict { sound = sqlx::query_as_unchecked!( Self, " SELECT * FROM sounds WHERE name = ? AND ( public = 1 OR uploader_id = ? OR server_id = ? ) ORDER BY rand(), public = 1, server_id = ?, uploader_id = ? ", name, user_id, guild_id, guild_id, user_id ) .fetch_all(&db_pool) .await?; } else { sound = sqlx::query_as_unchecked!( Self, " SELECT * FROM sounds WHERE name LIKE CONCAT('%', ?, '%') AND ( public = 1 OR uploader_id = ? OR server_id = ? ) ORDER BY rand(), public = 1, server_id = ?, uploader_id = ? ", name, user_id, guild_id, guild_id, user_id ) .fetch_all(&db_pool) .await?; } Ok(sound) } } pub async fn store_sound_source(&self) -> Result, Box> { let caching_location = env::var("CACHING_LOCATION").unwrap_or(String::from("/tmp")); let path_name = format!("{}/sound-{}", caching_location, self.id); let path = Path::new(&path_name); if !path.exists() { use tokio::prelude::*; let mut file = File::create(&path).await?; file.write_all(self.src.as_ref()).await?; } Ok(ffmpeg(path_name).await?) } pub async fn count_user_sounds(user_id: u64, db_pool: MySqlPool) -> Result { let c = sqlx::query!( " SELECT COUNT(1) as count FROM sounds WHERE uploader_id = ? ", user_id ) .fetch_one(&db_pool) .await?.count; Ok(c as u32) } pub async fn count_named_user_sounds(user_id: u64, name: &String, db_pool: MySqlPool) -> Result { let c = sqlx::query!( " SELECT COUNT(1) as count FROM sounds WHERE uploader_id = ? AND name = ? ", user_id, name ) .fetch_one(&db_pool) .await?.count; Ok(c as u32) } pub async fn set_as_greet(&self, user_id: u64, db_pool: MySqlPool) -> Result<(), Box> { sqlx::query!( " UPDATE users SET join_sound_id = ? WHERE user = ? ", self.id, user_id ) .execute(&db_pool) .await?; Ok(()) } pub async fn commit(&self, db_pool: MySqlPool) -> Result<(), Box> { sqlx::query!( " UPDATE sounds SET plays = ?, public = ? WHERE id = ? ", self.plays, self.public, self.id ) .execute(&db_pool) .await?; Ok(()) } pub async fn delete(&self, db_pool: MySqlPool) -> Result<(), Box> { sqlx::query!( " DELETE FROM sounds WHERE id = ? ", self.id ) .execute(&db_pool) .await?; Ok(()) } pub async fn create_anon(name: &str, src_url: &str, server_id: u64, user_id: u64, db_pool: MySqlPool) -> Result> { async fn process_src(src_url: &str) -> Option> { let future = Command::new("ffmpeg") .arg("-i") .arg(src_url) .arg("-loglevel") .arg("error") .arg("-b:a") .arg("28000") .arg("-f") .arg("opus") .arg("-fs") .arg("1048576") .arg("pipe:1") .output(); let output = future.await; match output { Ok(out) => { if out.status.success() { Some(out.stdout) } else { None } } Err(_) => None, } } let source = process_src(src_url).await; match source { Some(data) => { match sqlx::query!( " INSERT INTO sounds (name, server_id, uploader_id, public, src) VALUES (?, ?, ?, 1, ?) ", name, server_id, user_id, data ) .execute(&db_pool) .await { Ok(u) => Ok(u), Err(e) => Err(Box::new(e)) } } None => Err(Box::new(ErrorTypes::InvalidFile)) } } pub async fn get_user_sounds(user_id: u64, db_pool: MySqlPool) -> Result, Box> { let sounds = sqlx::query_as_unchecked!( Sound, " SELECT * FROM sounds WHERE uploader_id = ? ", user_id ).fetch_all(&db_pool).await?; Ok(sounds) } pub async fn get_guild_sounds(guild_id: u64, db_pool: MySqlPool) -> Result, Box> { let sounds = sqlx::query_as_unchecked!( Sound, " SELECT * FROM sounds WHERE server_id = ? ", guild_id ).fetch_all(&db_pool).await?; Ok(sounds) } }