spin_plugins/badger/
store.rs

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
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
use std::path::PathBuf;

use anyhow::anyhow;
use serde::{Deserialize, Serialize};

const DEFAULT_STORE_DIR: &str = "spin";
const DEFAULT_STORE_FILE: &str = "plugins-notifications.json";

pub struct BadgerRecordManager {
    db_path: PathBuf,
}

#[derive(Serialize, Deserialize)]
pub struct BadgerRecord {
    name: String,
    badgered_from: semver::Version,
    badgered_to: Vec<semver::Version>,
    when: chrono::DateTime<chrono::Utc>,
}

pub enum PreviousBadger {
    Fresh,
    FromCurrent {
        to: Vec<semver::Version>,
        when: chrono::DateTime<chrono::Utc>,
    },
}

impl PreviousBadger {
    fn includes(&self, version: &semver::Version) -> bool {
        match self {
            Self::Fresh => false,
            Self::FromCurrent { to, .. } => to.contains(version),
        }
    }

    pub fn includes_any(&self, version: &[&semver::Version]) -> bool {
        version.iter().any(|version| self.includes(version))
    }
}

impl BadgerRecordManager {
    pub fn default() -> anyhow::Result<Self> {
        let base_dir = dirs::cache_dir()
            .or_else(|| dirs::home_dir().map(|p| p.join(".spin")))
            .ok_or_else(|| anyhow!("Unable to get local data directory or home directory"))?;
        let db_path = base_dir.join(DEFAULT_STORE_DIR).join(DEFAULT_STORE_FILE);
        Ok(Self { db_path })
    }

    fn load(&self) -> Vec<BadgerRecord> {
        match std::fs::read(&self.db_path) {
            Ok(v) => serde_json::from_slice(&v).unwrap_or_default(),
            Err(_) => vec![], // There's no meaningful action or recovery, so swallow the error and treat the situation as fresh badger.
        }
    }

    fn save(&self, records: Vec<BadgerRecord>) -> anyhow::Result<()> {
        if let Some(dir) = self.db_path.parent() {
            std::fs::create_dir_all(dir)?;
        }
        let json = serde_json::to_vec_pretty(&records)?;
        std::fs::write(&self.db_path, json)?;
        Ok(())
    }

    async fn last_badger(&self, name: &str) -> Option<BadgerRecord> {
        self.load().into_iter().find(|r| r.name == name)
    }

    pub async fn previous_badger(
        &self,
        name: &str,
        current_version: &semver::Version,
    ) -> PreviousBadger {
        match self.last_badger(name).await {
            Some(b) if &b.badgered_from == current_version => PreviousBadger::FromCurrent {
                to: b.badgered_to,
                when: b.when,
            },
            _ => PreviousBadger::Fresh,
        }
    }

    pub async fn record_badger(&self, name: &str, from: &semver::Version, to: &[&semver::Version]) {
        let new = BadgerRecord {
            name: name.to_owned(),
            badgered_from: from.clone(),
            badgered_to: to.iter().cloned().map(<semver::Version>::clone).collect(),
            when: chrono::Utc::now(),
        };

        // There is a potential race condition here if someone runs two plugins at
        // the same time. As this is unlikely, and the worst outcome is that a user
        // misses a badger or gets a double badger, let's not worry about it for now.
        let mut existing = self.load();
        match existing.iter().position(|r| r.name == name) {
            Some(index) => existing[index] = new,
            None => existing.push(new),
        };
        _ = self.save(existing);
    }
}