spin_loader/
cache.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
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
//! Cache for OCI registry entities.

use anyhow::{ensure, Context, Result};

use std::{
    path::PathBuf,
    sync::atomic::{AtomicBool, Ordering},
};

use crate::fs::{create_dir_all, write_file};

const CONFIG_DIR: &str = "spin";
const REGISTRY_CACHE_DIR: &str = "registry";
const MANIFESTS_DIR: &str = "manifests";
const WASM_DIR: &str = "wasm";
const DATA_DIR: &str = "data";

/// Cache for registry entities.
#[derive(Debug)]
pub struct Cache {
    /// Root directory for the cache instance.
    root: PathBuf,
    /// Whether the cache directories have been checked to exist (and
    /// created if necessary).
    dirs_ensured_once: AtomicBool,
}

impl Cache {
    /// Create a new cache given an optional root directory.
    pub async fn new(root: Option<PathBuf>) -> Result<Self> {
        let root = match root {
            Some(root) => root,
            None => dirs::cache_dir()
                .context("cannot get cache directory")?
                .join(CONFIG_DIR),
        };
        let root = root.join(REGISTRY_CACHE_DIR);

        Ok(Self {
            root,
            dirs_ensured_once: AtomicBool::new(false),
        })
    }

    /// The manifests directory for the current cache.
    pub fn manifests_dir(&self) -> PathBuf {
        self.root.join(MANIFESTS_DIR)
    }

    /// The Wasm bytes directory for the current cache.
    fn wasm_dir(&self) -> PathBuf {
        self.root.join(WASM_DIR)
    }

    /// The data directory for the current cache.
    fn data_dir(&self) -> PathBuf {
        self.root.join(DATA_DIR)
    }

    /// Return the path to a wasm file given its digest.
    pub fn wasm_file(&self, digest: impl AsRef<str>) -> Result<PathBuf> {
        // Check the expected wasm directory first; else check the data directory as a fallback.
        // (Layers with unknown media types are currently saved to the data directory in client.pull())
        // This adds a bit of futureproofing for fetching wasm layers with different/updated media types
        // (see WASM_LAYER_MEDIA_TYPE, which is subject to change in future versions).
        let mut path = self.wasm_path(&digest);
        if !path.exists() {
            path = self.data_path(&digest);
        }
        ensure!(
            path.exists(),
            "cannot find wasm file for digest {}",
            digest.as_ref()
        );
        Ok(path)
    }

    /// Return the path to a data file given its digest.
    pub fn data_file(&self, digest: impl AsRef<str>) -> Result<PathBuf> {
        let path = self.data_path(&digest);
        ensure!(
            path.exists(),
            "cannot find data file for digest {}",
            digest.as_ref()
        );
        Ok(path)
    }

    /// Write the contents in the cache's wasm directory.
    pub async fn write_wasm(&self, bytes: impl AsRef<[u8]>, digest: impl AsRef<str>) -> Result<()> {
        self.ensure_dirs().await?;
        write_file(&self.wasm_path(digest), bytes.as_ref()).await?;
        Ok(())
    }

    /// Write the contents in the cache's data directory.
    pub async fn write_data(&self, bytes: impl AsRef<[u8]>, digest: impl AsRef<str>) -> Result<()> {
        self.ensure_dirs().await?;
        write_file(&self.data_path(digest), bytes.as_ref()).await?;
        Ok(())
    }

    /// The path of contents in the cache's wasm directory, which may or may not exist.
    pub fn wasm_path(&self, digest: impl AsRef<str>) -> PathBuf {
        self.wasm_dir().join(safe_name(digest).as_ref())
    }

    /// The path of contents in the cache's wasm directory, which may or may not exist.
    pub fn data_path(&self, digest: impl AsRef<str>) -> PathBuf {
        self.data_dir().join(safe_name(digest).as_ref())
    }

    /// Ensure the expected configuration directories are found in the root.
    ///
    /// ```text
    /// └── <configuration-root>
    ///     └── registry
    ///             └──manifests
    ///             └──wasm
    ///             └──data
    /// ```
    pub async fn ensure_dirs(&self) -> Result<()> {
        tracing::debug!("using cache root directory {}", self.root.display());

        // We don't care about ordering as this function is idempotent -
        // we are using an Atomic only for interior mutability.
        if self.dirs_ensured_once.load(Ordering::Relaxed) {
            return Ok(());
        }

        let root = &self.root;

        let p = root.join(MANIFESTS_DIR);
        if !p.is_dir() {
            create_dir_all(&p).await.with_context(|| {
                format!("failed to create manifests directory `{}`", p.display())
            })?;
        }

        let p = root.join(WASM_DIR);
        if !p.is_dir() {
            create_dir_all(&p)
                .await
                .with_context(|| format!("failed to create wasm directory `{}`", p.display()))?;
        }

        let p = root.join(DATA_DIR);
        if !p.is_dir() {
            create_dir_all(&p)
                .await
                .with_context(|| format!("failed to create assets directory `{}`", p.display()))?;
        }

        self.dirs_ensured_once.store(true, Ordering::Relaxed);

        Ok(())
    }
}

#[cfg(windows)]
fn safe_name(digest: impl AsRef<str>) -> impl AsRef<std::path::Path> {
    digest.as_ref().replace(':', "_")
}

#[cfg(not(windows))]
fn safe_name(digest: impl AsRef<str>) -> impl AsRef<str> {
    digest
}

#[cfg(test)]
mod test {
    use spin_common::sha256::hex_digest_from_bytes;

    use super::*;

    #[tokio::test]
    async fn accepts_prefixed_digests() -> anyhow::Result<()> {
        let temp_dir = tempfile::tempdir()?;
        let cache = Cache::new(Some(temp_dir.path().to_owned())).await?;

        let wasm = "Wasm".as_bytes();
        let digest = format!("sha256:{}", hex_digest_from_bytes(wasm));
        cache.write_wasm(wasm, &digest).await?;
        assert_eq!(wasm, std::fs::read(cache.wasm_path(&digest))?);

        let data = "hello".as_bytes();
        let digest = format!("sha256:{}", hex_digest_from_bytes(data));
        cache.write_data(data, &digest).await?;
        assert_eq!(data, std::fs::read(cache.data_path(&digest))?);

        Ok(())
    }
}