use crate::broker::VmgsBrokerRpc;
use inspect::Inspect;
use mesh_channel::rpc::RpcError;
use mesh_channel::rpc::RpcSend;
use thiserror::Error;
use tracing::instrument;
use vmgs::VmgsFileInfo;
use vmgs_format::FileId;
#[derive(Debug, Error)]
#[non_exhaustive]
pub enum VmgsClientError {
#[error("broker is offline")]
BrokerOffline(#[from] RpcError),
#[error("vmgs error")]
Vmgs(#[from] vmgs::Error),
}
impl From<RpcError<vmgs::Error>> for VmgsClientError {
fn from(value: RpcError<vmgs::Error>) -> Self {
match value {
RpcError::Call(e) => VmgsClientError::Vmgs(e),
RpcError::Channel(e) => VmgsClientError::BrokerOffline(RpcError::Channel(e)),
}
}
}
#[derive(Clone)]
pub struct VmgsClient {
pub(crate) control: mesh_channel::Sender<VmgsBrokerRpc>,
}
impl Inspect for VmgsClient {
fn inspect(&self, req: inspect::Request<'_>) {
self.control.send(VmgsBrokerRpc::Inspect(req.defer()));
}
}
impl VmgsClient {
#[instrument(skip_all, fields(file_id))]
pub async fn get_file_info(&self, file_id: FileId) -> Result<VmgsFileInfo, VmgsClientError> {
let res = self
.control
.call_failable(VmgsBrokerRpc::GetFileInfo, file_id)
.await?;
Ok(res)
}
#[instrument(skip_all, fields(file_id))]
pub async fn read_file(&self, file_id: FileId) -> Result<Vec<u8>, VmgsClientError> {
let res = self
.control
.call_failable(VmgsBrokerRpc::ReadFile, file_id)
.await?;
Ok(res)
}
#[instrument(skip_all, fields(file_id))]
pub async fn write_file(&self, file_id: FileId, buf: Vec<u8>) -> Result<(), VmgsClientError> {
self.control
.call_failable(VmgsBrokerRpc::WriteFile, (file_id, buf))
.await?;
Ok(())
}
#[cfg(with_encryption)]
#[instrument(skip_all, fields(file_id))]
pub async fn write_file_encrypted(
&self,
file_id: FileId,
buf: Vec<u8>,
) -> Result<(), VmgsClientError> {
self.control
.call_failable(VmgsBrokerRpc::WriteFileEncrypted, (file_id, buf))
.await?;
Ok(())
}
pub async fn save(&self) -> Result<vmgs::save_restore::state::SavedVmgsState, VmgsClientError> {
let res = self.control.call(VmgsBrokerRpc::Save, ()).await?;
Ok(res)
}
}