~aleteoryx/lfm_embed

ab822e81f8e06fbdc274090fc0a3fb31ef7f7ed1 — alyx 1 year, 4 months ago 70029b0
pull down user info
2 files changed, 46 insertions(+), 19 deletions(-)

M src/cache.rs
M src/config.rs
M src/cache.rs => src/cache.rs +8 -7
@@ 1,4 1,5 @@
use std::{future::Future, time::*, collections::HashMap, hash::Hash};
use reqwest::StatusCode;
#[derive(Debug)]
pub struct AsyncCache<K, V, F> {
    func: F,


@@ 8,9 9,9 @@ pub struct AsyncCache<K, V, F> {

impl<K, V, F, Fut> AsyncCache<K, V, F>
where
    F: for<'a> FnMut(&'a K) -> Fut,
    for<'a> F: FnMut(&'a K) -> Fut + 'a,
    K: Hash + PartialEq + Eq + Clone,
    Fut: Future<Output = Result<V, &'static str>>
    Fut: Future<Output = Result<V, (StatusCode, &'static str)>>
{
    pub fn new(interval: Duration, func: F) -> Self {
        Self{


@@ 19,7 20,7 @@ where
        }
    }
    
    pub async fn get(&mut self, key: &K) -> Result<&V, &'static str> {
    pub async fn get(&mut self, key: &K) -> Result<&V, (StatusCode, &'static str)> {
        if self.is_stale(&key) {
            self.renew(&key).await
        } else {


@@ 27,7 28,7 @@ where
        }
    }

    pub async fn renew(&mut self, key: &K) -> Result<&V, &'static str> {
    pub async fn renew(&mut self, key: &K) -> Result<&V, (StatusCode, &'static str)> {
        let val = (self.func)(&key).await?;
        self.cache.insert(key.clone(), (Instant::now(), val));
        Ok(&self.cache.get(key).unwrap().1)


@@ 51,12 52,12 @@ where

impl<K, V, F, Fut> AsyncCache<K, V, F>
where
    F: for<'a> FnMut(&'a K) -> Fut,
    for<'a> F: FnMut(&'a K) -> Fut + 'a,
    K: Hash + PartialEq + Eq + Clone,
    V: Clone,
    Fut: Future<Output = Result<V, &'static str>>
    Fut: Future<Output = Result<V, (StatusCode, &'static str)>>
{
    pub async fn get_owned(&mut self, key: &K) -> Result<V, &'static str> {
    pub async fn get_owned(&mut self, key: &K) -> Result<V, (StatusCode, &'static str)> {
        self.get(key).await.cloned()
    }
}

M src/config.rs => src/config.rs +38 -12
@@ 11,19 11,42 @@ use duration_str as ds;
use super::cache::AsyncCache;
use super::deserialize::{GetRecentTracks, GetUserInfo, Track, User};

use reqwest::Client;
use reqwest::{Client, StatusCode};
use dotenv::var;
use tokio::sync::RwLock;

static INTERNAL_THEMES: &[(&'static str, &'static str)] = &[("plain", "")];

pub static STATE: LazyLock<Arc<State>> = LazyLock::new(|| {
    State::new()
});

fn getter(username: &String) -> Pin<Box<dyn Future<Output = Result<(User, Track), &'static str>>>> {
    Box::pin(async{Err("nope")})
}
fn getter(username: &String) -> Pin<Box<dyn Future<Output = Result<(User, Track), (StatusCode, &'static str)>>>> {
    let username = username.clone();
    Box::pin(async move {
        let userreq = STATE.http.get(format!("https://ws.audioscrobbler.com/2.0/?method=user.getInfo&format=json&user={username}&api_key={}", STATE.api_key))
            .send().await
            .map_err(|e| {log::error!("Failed to get info for user `{username}`: {e}"); (StatusCode::SERVICE_UNAVAILABLE, "Couldn't connect to last.fm!")})?;
        if userreq.status() == StatusCode::NOT_FOUND { return Err((StatusCode::NOT_FOUND, "User does not exist!")); }

        let userinfo = userreq.json::<GetUserInfo>().await
            .map_err(|e| {log::error!("Couldn't parse user.getInfo for `{username}`: {e}"); (StatusCode::INTERNAL_SERVER_ERROR, "Couldn't parse user.getInfo!")})?.user;
        
        let tracksreq = STATE.http.get(format!("https://ws.audioscrobbler.com/2.0/?method=user.getRecentTracks&format=json&extended=1&limit=1&user={username}&api_key={}", STATE.api_key))
            .send().await
            .map_err(|e| {log::error!("Failed to get tracks for user `{username}`: {e}"); (StatusCode::SERVICE_UNAVAILABLE, "Couldn't connect to last.fm!")})?;
        if tracksreq.status() == StatusCode::NOT_FOUND { return Err((StatusCode::NOT_FOUND, "User does not exist!")); }
        if tracksreq.status() == StatusCode::FORBIDDEN { return Err((StatusCode::FORBIDDEN, "You need to unprivate your song history!")); }

type Cache = Arc<RwLock<AsyncCache<String, (User, Track), fn(&String) -> Pin<Box<dyn Future<Output = Result<(User, Track), &'static str>>>>>>>;
        let tracksinfo = tracksreq.json::<GetRecentTracks>().await
            .map_err(|e| {log::error!("Couldn't parse user.getRecentTracks for `{username}`: {e}"); (StatusCode::INTERNAL_SERVER_ERROR, "Couldn't parse user.getRecentTracks!")})?
            .recenttracks.track.into_iter().nth(0).ok_or((StatusCode::UNPROCESSABLE_ENTITY, "You need to listen to some songs first!"))?;

        Ok((userinfo, tracksinfo))
    })
}
type Getter = fn(&String) -> Pin<Box<(dyn Future<Output = Result<(User, Track), (StatusCode, &'static str)>>)>>;
type Cache = Arc<RwLock<AsyncCache<String, (User, Track), Getter>>>;
#[derive(Debug)]
enum Whitelist {
    Exclusive{cache: Cache, whitelist: HashSet<String>},


@@ 34,8 57,9 @@ pub struct State {
    api_key: Arc<str>,
    whitelist: Whitelist,
    port: u16,
    custom_themes: HashMap<String, Arc<str>>,
    send_refresh_header: bool
    themes: HashMap<String, Arc<str>>,
    send_refresh_header: bool,
    http: Client
}

impl State {


@@ 44,10 68,11 @@ impl State {
            api_key: var("LFME_API_KEY").expect("API key must be set").into(),
            port: var("LFME_PORT").map(|p| p.parse().expect("cannot parse as a port number")).unwrap_or(9999),
            send_refresh_header: var("LFME_SET_HEADER").map(|h| &h == "1").unwrap_or(false),
            http: Client::builder().https_only(true).user_agent(concat!(env!("CARGO_PKG_NAME"), "/", env!("CARGO_PKG_VERSION"))).build().unwrap(),
            
            custom_themes: {
            themes: {
                if let Ok(themes_dir) = var("LFME_THEMES_DIR") {
                    fs::read_dir(themes_dir).expect("error reading LFME_THEMES_DIR")
                    INTERNAL_THEMES.iter().map(|(k, v)| (k.to_string(), (*v).into())).chain(fs::read_dir(themes_dir).expect("error reading LFME_THEMES_DIR")
                        .map(|a| a.expect("error reading LFME_THEMES_DIR"))
                        .filter_map(|a| {
                            let path = a.path();


@@ 56,7 81,7 @@ impl State {
                                    Some((path.file_stem().unwrap().to_str().expect("bad filename").to_string(), fs::read_to_string(&path).expect("couldn't read theme CSS").into()))
                                }
                            else { None }
                        })
                        }))
                        .collect()
                }
                else { HashMap::new() }


@@ 65,7 90,7 @@ impl State {
            whitelist: {
                let cache_from_var = |v: &str, d: u64| -> Cache {
                    let refresh = var(v).map(|r| ds::parse(&r).expect("bad duration string")).unwrap_or_else(|_| Duration::from_secs(d));
                    Arc::new(RwLock::new(AsyncCache::new(refresh, getter as _)))
                    Arc::new(RwLock::new(AsyncCache::new(refresh, getter as Getter) as AsyncCache<String, (User, Track), Getter>))
                };
                let default_cache = || cache_from_var("LFME_DEFAULT_REFRESH", 300);
                let whitelist_cache = || cache_from_var("LFME_WHITELIST_REFRESH", 60);


@@ 91,6 116,7 @@ impl State {
        })
    }
    
    pub fn api_key(&self) -> Arc<str> { self.api_key.clone() }
    pub fn port(&self) -> u16 { self.port }
    pub fn send_refresh_header(&self) -> bool { self.send_refresh_header }
    pub fn get_theme(&self, theme: &str) -> Option<Arc<str>> { self.themes.get(theme).cloned() }
}