aboutsummaryrefslogtreecommitdiff
path: root/nix-rust/src/store/binary_cache_store.rs
blob: 9e1e88b7c854ee7c80d172622f1b4d929dc5d123 (plain)
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
use super::{PathInfo, Store, StorePath};
use crate::Error;
use hyper::client::Client;

pub struct BinaryCacheStore {
    base_uri: String,
    client: Client<hyper::client::HttpConnector, hyper::Body>,
}

impl BinaryCacheStore {
    pub fn new(base_uri: String) -> Self {
        Self {
            base_uri,
            client: Client::new(),
        }
    }
}

impl Store for BinaryCacheStore {
    fn query_path_info(
        &self,
        path: &StorePath,
    ) -> std::pin::Pin<Box<dyn std::future::Future<Output = Result<PathInfo, Error>> + Send>> {
        let uri = format!("{}/{}.narinfo", self.base_uri.clone(), path.hash);
        let path = path.clone();
        let client = self.client.clone();
        let store_dir = self.store_dir().to_string();

        Box::pin(async move {
            let response = client.get(uri.parse::<hyper::Uri>().unwrap()).await?;

            if response.status() == hyper::StatusCode::NOT_FOUND
                || response.status() == hyper::StatusCode::FORBIDDEN
            {
                return Err(Error::InvalidPath(path));
            }

            let mut body = response.into_body();

            let mut bytes = Vec::new();
            while let Some(next) = body.next().await {
                bytes.extend(next?);
            }

            PathInfo::parse_nar_info(std::str::from_utf8(&bytes).unwrap(), &store_dir)
        })
    }
}