1use std::{io, ops::Deref};
16
17use bao_tree::io::EncodeError;
18use iroh::Endpoint;
19use n0_error::{e, stack_error};
20use proto::{ShutdownRequest, SyncDbRequest};
21use ref_cast::RefCast;
22use serde::{Deserialize, Serialize};
23use tags::Tags;
24
25pub mod blobs;
26pub mod downloader;
27pub mod proto;
28pub mod remote;
29pub mod tags;
30use crate::{api::proto::WaitIdleRequest, provider::events::ProgressError};
31pub use crate::{store::util::Tag, util::temp_tag::TempTag};
32
33pub(crate) type ApiClient = irpc::Client<proto::Request>;
34
35#[allow(missing_docs)]
36#[non_exhaustive]
37#[stack_error(derive, add_meta)]
38pub enum RequestError {
39 #[error("rpc error: {source}")]
41 Rpc { source: irpc::Error },
42 #[error("inner error: {source}")]
44 Inner {
45 #[error(std_err)]
46 source: Error,
47 },
48}
49
50impl From<irpc::Error> for RequestError {
51 fn from(value: irpc::Error) -> Self {
52 e!(RequestError::Rpc, value)
53 }
54}
55
56impl From<Error> for RequestError {
57 fn from(value: Error) -> Self {
58 e!(RequestError::Inner, value)
59 }
60}
61
62impl From<io::Error> for RequestError {
63 fn from(value: io::Error) -> Self {
64 e!(RequestError::Inner, value.into())
65 }
66}
67
68impl From<irpc::channel::mpsc::RecvError> for RequestError {
69 fn from(value: irpc::channel::mpsc::RecvError) -> Self {
70 e!(RequestError::Rpc, value.into())
71 }
72}
73
74pub type RequestResult<T> = std::result::Result<T, RequestError>;
75
76#[allow(missing_docs)]
77#[non_exhaustive]
78#[stack_error(derive, add_meta, from_sources)]
79pub enum ExportBaoError {
80 #[error("send error")]
81 Send { source: irpc::channel::SendError },
82 #[error("mpsc recv e api.acp.pro-channelsrror")]
83 MpscRecv {
84 source: irpc::channel::mpsc::RecvError,
85 },
86 #[error("oneshot recv error")]
87 OneshotRecv {
88 source: irpc::channel::oneshot::RecvError,
89 },
90 #[error("request error")]
91 Request { source: irpc::RequestError },
92 #[error("io error")]
93 ExportBaoIo {
94 #[error(std_err)]
95 source: io::Error,
96 },
97 #[error("encode error")]
98 ExportBaoInner {
99 #[error(std_err)]
100 source: bao_tree::io::EncodeError,
101 },
102 #[error("client error")]
103 ClientError { source: ProgressError },
104}
105
106impl From<ExportBaoError> for Error {
107 fn from(e: ExportBaoError) -> Self {
108 match e {
109 ExportBaoError::Send { source, .. } => Self::Io(source.into()),
110 ExportBaoError::MpscRecv { source, .. } => Self::Io(source.into()),
111 ExportBaoError::OneshotRecv { source, .. } => Self::Io(source.into()),
112 ExportBaoError::Request { source, .. } => Self::Io(source.into()),
113 ExportBaoError::ExportBaoIo { source, .. } => Self::Io(source),
114 ExportBaoError::ExportBaoInner { source, .. } => Self::Io(source.into()),
115 ExportBaoError::ClientError { source, .. } => Self::Io(source.into()),
116 }
117 }
118}
119
120impl From<irpc::Error> for ExportBaoError {
121 fn from(e: irpc::Error) -> Self {
122 match e {
123 irpc::Error::MpscRecv { source: e, .. } => e!(ExportBaoError::MpscRecv, e),
124 irpc::Error::OneshotRecv { source: e, .. } => e!(ExportBaoError::OneshotRecv, e),
125 irpc::Error::Send { source: e, .. } => e!(ExportBaoError::Send, e),
126 irpc::Error::Request { source: e, .. } => e!(ExportBaoError::Request, e),
127 #[cfg(feature = "rpc")]
128 irpc::Error::Write { source: e, .. } => e!(ExportBaoError::ExportBaoIo, e.into()),
129 }
130 }
131}
132
133pub type ExportBaoResult<T> = std::result::Result<T, ExportBaoError>;
134
135#[derive(Serialize, Deserialize)]
136#[stack_error(derive, std_sources, from_sources)]
137pub enum Error {
138 #[serde(with = "crate::util::serde::io_error_serde")]
139 Io(#[error(source)] io::Error),
140}
141
142impl Error {
143 pub fn io(
144 kind: io::ErrorKind,
145 msg: impl Into<Box<dyn std::error::Error + Send + Sync>>,
146 ) -> Self {
147 Self::Io(io::Error::new(kind, msg.into()))
148 }
149
150 pub fn other<E>(msg: E) -> Self
151 where
152 E: Into<Box<dyn std::error::Error + Send + Sync>>,
153 {
154 Self::Io(io::Error::other(msg.into()))
155 }
156}
157
158impl From<irpc::Error> for Error {
159 fn from(e: irpc::Error) -> Self {
160 Self::Io(e.into())
161 }
162}
163
164impl From<RequestError> for Error {
165 fn from(e: RequestError) -> Self {
166 match e {
167 RequestError::Rpc { source, .. } => Self::Io(source.into()),
168 RequestError::Inner { source, .. } => source,
169 }
170 }
171}
172
173impl From<irpc::channel::mpsc::RecvError> for Error {
174 fn from(e: irpc::channel::mpsc::RecvError) -> Self {
175 Self::Io(e.into())
176 }
177}
178
179#[cfg(feature = "rpc")]
180impl From<irpc::rpc::WriteError> for Error {
181 fn from(e: irpc::rpc::WriteError) -> Self {
182 Self::Io(e.into())
183 }
184}
185
186impl From<irpc::RequestError> for Error {
187 fn from(e: irpc::RequestError) -> Self {
188 Self::Io(e.into())
189 }
190}
191
192impl From<irpc::channel::SendError> for Error {
193 fn from(e: irpc::channel::SendError) -> Self {
194 Self::Io(e.into())
195 }
196}
197
198impl From<EncodeError> for Error {
199 fn from(value: EncodeError) -> Self {
200 match value {
201 EncodeError::Io(cause) => Self::Io(cause),
202 _ => Self::other(value),
203 }
204 }
205}
206
207pub type Result<T> = std::result::Result<T, Error>;
208
209#[derive(Debug, Clone, ref_cast::RefCast)]
211#[repr(transparent)]
212pub struct Store {
213 client: ApiClient,
214}
215
216impl Deref for Store {
217 type Target = blobs::Blobs;
218
219 fn deref(&self) -> &Self::Target {
220 blobs::Blobs::ref_from_sender(&self.client)
221 }
222}
223
224impl Store {
225 pub fn tags(&self) -> &Tags {
227 Tags::ref_from_sender(&self.client)
228 }
229
230 pub fn blobs(&self) -> &blobs::Blobs {
232 blobs::Blobs::ref_from_sender(&self.client)
233 }
234
235 pub fn remote(&self) -> &remote::Remote {
237 remote::Remote::ref_from_sender(&self.client)
238 }
239
240 pub fn downloader(&self, endpoint: &Endpoint) -> downloader::Downloader {
246 downloader::Downloader::new(self, endpoint)
247 }
248
249 #[cfg(feature = "rpc")]
251 pub fn connect(endpoint: quinn::Endpoint, addr: std::net::SocketAddr) -> Self {
252 let sender = irpc::Client::quinn(endpoint, addr);
253 Store::from_sender(sender)
254 }
255
256 #[cfg(feature = "rpc")]
258 pub async fn listen(self, endpoint: quinn::Endpoint) {
259 use irpc::rpc::RemoteService;
260
261 use self::proto::Request;
262 let local = self.client.as_local().unwrap().clone();
263 let handler = Request::remote_handler(local);
264 irpc::rpc::listen::<Request>(endpoint, handler).await
265 }
266
267 pub async fn sync_db(&self) -> RequestResult<()> {
268 let msg = SyncDbRequest;
269 self.client.rpc(msg).await??;
270 Ok(())
271 }
272
273 pub async fn shutdown(&self) -> irpc::Result<()> {
274 let msg = ShutdownRequest;
275 self.client.rpc(msg).await?;
276 Ok(())
277 }
278
279 pub async fn wait_idle(&self) -> irpc::Result<()> {
291 let msg = WaitIdleRequest;
292 self.client.rpc(msg).await?;
293 Ok(())
294 }
295
296 pub(crate) fn from_sender(client: ApiClient) -> Self {
297 Self { client }
298 }
299
300 pub(crate) fn ref_from_sender(client: &ApiClient) -> &Self {
301 Self::ref_cast(client)
302 }
303}