1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
use crate::{
    media_container::{
        preferences::{Preferences as MediaContainerPreferences, Setting, Value},
        MediaContainerWrapper,
    },
    url::SERVER_PREFS,
    HttpClient, Result,
};
use std::mem::discriminant;

#[derive(Debug, Clone)]
pub struct Preferences<'a> {
    client: HttpClient,
    settings: Vec<Setting>,
    changed: Vec<&'a str>,
}

impl<'a> Preferences<'a> {
    #[tracing::instrument(level = "debug", skip(client))]
    pub async fn new<C: Into<HttpClient>>(client: C) -> Result<Preferences<'a>> {
        let client = client.into();
        let mc: MediaContainerWrapper<MediaContainerPreferences> =
            client.get(SERVER_PREFS).json().await?;

        Ok(Preferences {
            client,
            settings: mc.media_container.settings,
            changed: vec![],
        })
    }

    pub fn get(&self, key: &str) -> Option<&Setting> {
        self.settings.iter().find(|s| s.id == key)
    }

    pub fn all(&self) -> &[Setting] {
        &self.settings
    }

    pub fn set(&mut self, key: &'a str, value: Value) -> Result<&mut Self> {
        if let Some(setting) = self.settings.iter_mut().find(|s| s.id == key) {
            if discriminant(&setting.value) == discriminant(&value) {
                setting.value = value;
                self.changed.push(key);
                Ok(self)
            } else {
                Err(crate::Error::IncompatibleSettingValues)
            }
        } else {
            Err(crate::Error::RequestedSettingNotFound(key.to_string()))
        }
    }

    #[tracing::instrument(level = "debug")]
    pub async fn commit(self) -> Result<Preferences<'a>> {
        if self.changed.is_empty() {
            return Ok(self);
        }

        let mut params = vec![];
        for s in self
            .settings
            .iter()
            .filter(|s| self.changed.contains(&s.id.as_str()))
        {
            params.push((s.id.as_str(), s.value.to_string()));
        }

        let uri = format!("{}?{}", SERVER_PREFS, serde_urlencoded::to_string(params)?);

        self.client.put(uri).consume().await?;

        Ok(Preferences {
            client: self.client,
            settings: self.settings,
            changed: vec![],
        })
    }
}