diff --git a/crates/backend/src/local.rs b/crates/backend/src/local.rs index 872877f7..a60e2d04 100644 --- a/crates/backend/src/local.rs +++ b/crates/backend/src/local.rs @@ -417,6 +417,13 @@ impl ReadBackend for LocalBackend { Ok(vec.into()) } + + /// [`LocalBackend`] doesn't use `async`, even under the hood. + /// + /// So it can be called from `async` features. + fn is_async_compatible(&self) -> bool { + true + } } impl WriteBackend for LocalBackend { diff --git a/crates/backend/src/opendal.rs b/crates/backend/src/opendal.rs index 049e16b0..acfb7c92 100644 --- a/crates/backend/src/opendal.rs +++ b/crates/backend/src/opendal.rs @@ -386,6 +386,15 @@ impl ReadBackend for OpenDALBackend { )? .to_bytes()) } + + /// [`OpenDALBackend`] is `sync` and uses `block_on(async Fn)` under the hood. + /// + /// When implementing `rustic_core` using this backend in some `async` features will not work. + /// + /// see + fn is_async_compatible(&self) -> bool { + false + } } impl WriteBackend for OpenDALBackend { diff --git a/crates/backend/src/rclone.rs b/crates/backend/src/rclone.rs index 94bb4b3b..c8522900 100644 --- a/crates/backend/src/rclone.rs +++ b/crates/backend/src/rclone.rs @@ -353,6 +353,11 @@ impl ReadBackend for RcloneBackend { ) -> RusticResult { self.rest.read_partial(tpe, id, cacheable, offset, length) } + + /// [`RcloneBackend`] uses [`RestBackend`]. + fn is_async_compatible(&self) -> bool { + self.rest.is_async_compatible() + } } impl WriteBackend for RcloneBackend { diff --git a/crates/backend/src/rest.rs b/crates/backend/src/rest.rs index 5d5841e9..8a7e0223 100644 --- a/crates/backend/src/rest.rs +++ b/crates/backend/src/rest.rs @@ -350,6 +350,16 @@ impl ReadBackend for RestBackend { }) .map_err(construct_backoff_error) } + + /// [`RestBackend`] uses `reqwest` which blocking implementation + /// uses an `async` runtime under the hood. + /// + /// When implementing `rustic_core` using this backend in some `async` features will not work. + /// + /// see + fn is_async_compatible(&self) -> bool { + false + } } fn construct_join_url_error( diff --git a/crates/core/src/backend.rs b/crates/core/src/backend.rs index 2179671b..25896f89 100644 --- a/crates/core/src/backend.rs +++ b/crates/core/src/backend.rs @@ -174,6 +174,21 @@ pub trait ReadBackend: Send + Sync + 'static { fn warm_up(&self, _tpe: FileType, _id: &Id) -> RusticResult<()> { Ok(()) } + + /// Getter to determine if some backend is compatible + /// with async features in `rustic_core` implementations. + /// + /// ## Default impl + /// + /// A default impl allow these change to be non-breaking. + /// By default it is `false`, async compatibility is opt-in. + /// + /// ## Temporary + /// + /// see + fn is_async_compatible(&self) -> bool { + false + } } /// Trait for Searching in a backend. @@ -343,7 +358,7 @@ pub trait WriteBackend: ReadBackend { mock! { Backend {} - impl ReadBackend for Backend{ + impl ReadBackend for Backend { fn location(&self) -> String; fn list_with_size(&self, tpe: FileType) -> RusticResult>; fn read_full(&self, tpe: FileType, id: &Id) -> RusticResult; @@ -355,6 +370,7 @@ mock! { offset: u32, length: u32, ) -> RusticResult; + fn is_async_compatible(&self) -> bool; } impl WriteBackend for Backend { @@ -400,6 +416,9 @@ impl ReadBackend for Arc { self.deref() .read_partial(tpe, id, cacheable, offset, length) } + fn is_async_compatible(&self) -> bool { + self.deref().is_async_compatible() + } } impl std::fmt::Debug for dyn WriteBackend { diff --git a/crates/core/src/backend/cache.rs b/crates/core/src/backend/cache.rs index d534cdfd..06844799 100644 --- a/crates/core/src/backend/cache.rs +++ b/crates/core/src/backend/cache.rs @@ -177,6 +177,10 @@ impl ReadBackend for CachedBackend { fn warm_up(&self, tpe: FileType, id: &Id) -> RusticResult<()> { self.be.warm_up(tpe, id) } + + fn is_async_compatible(&self) -> bool { + self.be.is_async_compatible() + } } impl WriteBackend for CachedBackend { diff --git a/crates/core/src/backend/decrypt.rs b/crates/core/src/backend/decrypt.rs index 414084f4..fa1cc7c6 100644 --- a/crates/core/src/backend/decrypt.rs +++ b/crates/core/src/backend/decrypt.rs @@ -216,7 +216,7 @@ pub trait DecryptWriteBackend: WriteBackend + Clone + 'static { /// /// # Returns /// - /// The processed data, the original data length and when compression is used, the uncomressed length + /// The processed data, the original data length and when compression is used, the uncompressed length fn process_data(&self, data: &[u8]) -> RusticResult<(Vec, u32, Option)>; /// Writes the given data to the backend without compression and returns the id of the data. @@ -559,7 +559,7 @@ impl DecryptWriteBackend for DecryptBackend { /// /// # Arguments /// - /// * `extra_echeck` - The compression level to use for zstd. + /// * `extra_verify` - The compression level to use for zstd. fn set_extra_verify(&mut self, extra_verify: bool) { self.extra_verify = extra_verify; } @@ -622,6 +622,10 @@ impl ReadBackend for DecryptBackend { ) -> RusticResult { self.be.read_partial(tpe, id, cacheable, offset, length) } + + fn is_async_compatible(&self) -> bool { + self.be.is_async_compatible() + } } impl WriteBackend for DecryptBackend { diff --git a/crates/core/src/backend/dry_run.rs b/crates/core/src/backend/dry_run.rs index 5bd07cb8..300656fe 100644 --- a/crates/core/src/backend/dry_run.rs +++ b/crates/core/src/backend/dry_run.rs @@ -103,6 +103,10 @@ impl ReadBackend for DryRunBackend { ) -> RusticResult { self.be.read_partial(tpe, id, cacheable, offset, length) } + + fn is_async_compatible(&self) -> bool { + self.be.is_async_compatible() + } } impl DecryptWriteBackend for DryRunBackend { diff --git a/crates/core/src/backend/hotcold.rs b/crates/core/src/backend/hotcold.rs index e5306e2f..594e1851 100644 --- a/crates/core/src/backend/hotcold.rs +++ b/crates/core/src/backend/hotcold.rs @@ -75,6 +75,10 @@ impl ReadBackend for HotColdBackend { fn warm_up(&self, tpe: FileType, id: &Id) -> RusticResult<()> { self.be.warm_up(tpe, id) } + + fn is_async_compatible(&self) -> bool { + self.be.is_async_compatible() + } } impl WriteBackend for HotColdBackend { diff --git a/crates/core/src/backend/warm_up.rs b/crates/core/src/backend/warm_up.rs index 3e25d49a..8406176d 100644 --- a/crates/core/src/backend/warm_up.rs +++ b/crates/core/src/backend/warm_up.rs @@ -59,6 +59,10 @@ impl ReadBackend for WarmUpAccessBackend { _ = self.be.read_partial(tpe, id, false, 0, 1); Ok(()) } + + fn is_async_compatible(&self) -> bool { + self.be.is_async_compatible() + } } impl WriteBackend for WarmUpAccessBackend { diff --git a/crates/core/src/repository.rs b/crates/core/src/repository.rs index d15d8f21..a0299ddd 100644 --- a/crates/core/src/repository.rs +++ b/crates/core/src/repository.rs @@ -708,6 +708,19 @@ impl Repository { pub fn list(&self) -> RusticResult> { Ok(self.be.list(T::TYPE)?.into_iter().map(Into::into)) } + + /// Check if one of this repository backend is incompatible + /// with async features in `rustic_core` implementations. + /// + /// see + pub fn is_async_compatible(&self) -> bool { + // check if be or be_hot is incompatible with async + self.be.is_async_compatible() + && self + .be_hot + .as_ref() + .map_or(true, ReadBackend::is_async_compatible) + } } impl Repository { @@ -1896,7 +1909,7 @@ impl Repository { impl Repository { /// Run a backup of `source` using the given options. /// - /// You have to give a preflled [`SnapshotFile`] which is modified and saved. + /// You have to give a prefilled [`SnapshotFile`] which is modified and saved. /// /// # Arguments /// diff --git a/crates/testing/src/backend.rs b/crates/testing/src/backend.rs index 418bd320..64e9b070 100644 --- a/crates/testing/src/backend.rs +++ b/crates/testing/src/backend.rs @@ -58,6 +58,11 @@ pub mod in_memory_backend { ) -> RusticResult { Ok(self.0.read().unwrap()[tpe][id].slice(offset as usize..(offset + length) as usize)) } + + /// [`InMemoryBackend`] doesn't use `async`, even under the hood. + fn is_async_compatible(&self) -> bool { + true + } } impl WriteBackend for InMemoryBackend {