use crate::{
get_blob,
sqlite::{Sqlite, SqliteAction, SqliteError, SqliteRequest, SqliteResponse},
Message, PackageId, Request,
};
use std::collections::HashMap;
impl Sqlite {
pub fn read(
&self,
query: String,
params: Vec<serde_json::Value>,
) -> anyhow::Result<Vec<HashMap<String, serde_json::Value>>> {
let res = Request::new()
.target(("our", "sqlite", "distro", "sys"))
.body(serde_json::to_vec(&SqliteRequest {
package_id: self.package_id.clone(),
db: self.db.clone(),
action: SqliteAction::Query(query),
})?)
.blob_bytes(serde_json::to_vec(¶ms)?)
.send_and_await_response(self.timeout)?;
match res {
Ok(Message::Response { body, .. }) => {
let response = serde_json::from_slice::<SqliteResponse>(&body)?;
match response {
SqliteResponse::Read => {
let blob = get_blob().ok_or_else(|| SqliteError::MalformedRequest)?;
let values = serde_json::from_slice::<
Vec<HashMap<String, serde_json::Value>>,
>(&blob.bytes)
.map_err(|_| SqliteError::MalformedRequest)?;
Ok(values)
}
SqliteResponse::Err(error) => Err(error.into()),
_ => Err(anyhow::anyhow!(
"sqlite: unexpected response {:?}",
response
)),
}
}
_ => Err(anyhow::anyhow!("sqlite: unexpected message: {:?}", res)),
}
}
pub fn write(
&self,
statement: String,
params: Vec<serde_json::Value>,
tx_id: Option<u64>,
) -> anyhow::Result<()> {
let res = Request::new()
.target(("our", "sqlite", "distro", "sys"))
.body(serde_json::to_vec(&SqliteRequest {
package_id: self.package_id.clone(),
db: self.db.clone(),
action: SqliteAction::Write { statement, tx_id },
})?)
.blob_bytes(serde_json::to_vec(¶ms)?)
.send_and_await_response(self.timeout)?;
match res {
Ok(Message::Response { body, .. }) => {
let response = serde_json::from_slice::<SqliteResponse>(&body)?;
match response {
SqliteResponse::Ok => Ok(()),
SqliteResponse::Err(error) => Err(error.into()),
_ => Err(anyhow::anyhow!(
"sqlite: unexpected response {:?}",
response
)),
}
}
_ => Err(anyhow::anyhow!("sqlite: unexpected message: {:?}", res)),
}
}
pub fn begin_tx(&self) -> anyhow::Result<u64> {
let res = Request::new()
.target(("our", "sqlite", "distro", "sys"))
.body(serde_json::to_vec(&SqliteRequest {
package_id: self.package_id.clone(),
db: self.db.clone(),
action: SqliteAction::BeginTx,
})?)
.send_and_await_response(self.timeout)?;
match res {
Ok(Message::Response { body, .. }) => {
let response = serde_json::from_slice::<SqliteResponse>(&body)?;
match response {
SqliteResponse::BeginTx { tx_id } => Ok(tx_id),
SqliteResponse::Err(error) => Err(error.into()),
_ => Err(anyhow::anyhow!(
"sqlite: unexpected response {:?}",
response
)),
}
}
_ => Err(anyhow::anyhow!("sqlite: unexpected message: {:?}", res)),
}
}
pub fn commit_tx(&self, tx_id: u64) -> anyhow::Result<()> {
let res = Request::new()
.target(("our", "sqlite", "distro", "sys"))
.body(serde_json::to_vec(&SqliteRequest {
package_id: self.package_id.clone(),
db: self.db.clone(),
action: SqliteAction::Commit { tx_id },
})?)
.send_and_await_response(self.timeout)?;
match res {
Ok(Message::Response { body, .. }) => {
let response = serde_json::from_slice::<SqliteResponse>(&body)?;
match response {
SqliteResponse::Ok => Ok(()),
SqliteResponse::Err(error) => Err(error.into()),
_ => Err(anyhow::anyhow!(
"sqlite: unexpected response {:?}",
response
)),
}
}
_ => Err(anyhow::anyhow!("sqlite: unexpected message: {:?}", res)),
}
}
}
pub fn open(package_id: PackageId, db: &str, timeout: Option<u64>) -> anyhow::Result<Sqlite> {
let timeout = timeout.unwrap_or(5);
let res = Request::new()
.target(("our", "sqlite", "distro", "sys"))
.body(serde_json::to_vec(&SqliteRequest {
package_id: package_id.clone(),
db: db.to_string(),
action: SqliteAction::Open,
})?)
.send_and_await_response(timeout)?;
match res {
Ok(Message::Response { body, .. }) => {
let response = serde_json::from_slice::<SqliteResponse>(&body)?;
match response {
SqliteResponse::Ok => Ok(Sqlite {
package_id,
db: db.to_string(),
timeout,
}),
SqliteResponse::Err(error) => Err(error.into()),
_ => Err(anyhow::anyhow!(
"sqlite: unexpected response {:?}",
response
)),
}
}
_ => Err(anyhow::anyhow!("sqlite: unexpected message: {:?}", res)),
}
}
pub fn remove_db(package_id: PackageId, db: &str, timeout: Option<u64>) -> anyhow::Result<()> {
let timeout = timeout.unwrap_or(5);
let res = Request::new()
.target(("our", "sqlite", "distro", "sys"))
.body(serde_json::to_vec(&SqliteRequest {
package_id: package_id.clone(),
db: db.to_string(),
action: SqliteAction::RemoveDb,
})?)
.send_and_await_response(timeout)?;
match res {
Ok(Message::Response { body, .. }) => {
let response = serde_json::from_slice::<SqliteResponse>(&body)?;
match response {
SqliteResponse::Ok => Ok(()),
SqliteResponse::Err(error) => Err(error.into()),
_ => Err(anyhow::anyhow!(
"sqlite: unexpected response {:?}",
response
)),
}
}
_ => Err(anyhow::anyhow!("sqlite: unexpected message: {:?}", res)),
}
}