#[cfg(feature = "chrono")]
use chrono::{
DateTime,
Utc,
FixedOffset,
};
#[cfg(feature = "jiff")]
use jiff::{
Zoned,
Timestamp,
};
pub trait GoodErrorQuery<T> {
fn to_good_error_query(self, query: &str) -> Result<T, loga::Error>;
}
impl<T> GoodErrorQuery<T> for Result<T, rusqlite::Error> {
fn to_good_error_query(self, query: &str) -> Result<T, loga::Error> {
match self {
Ok(v) => Ok(v),
Err(e) => Err(loga::err(e).context(format!("Error executing query: {}", query))),
}
}
}
pub trait SqliteConnection {
fn execute(&mut self, query: &str, params: impl rusqlite::Params) -> rusqlite::Result<usize>;
fn query<
T,
F: FnMut(&rusqlite::Row<'_>) -> rusqlite::Result<T>,
>(&mut self, query: &str, params: impl rusqlite::Params, f: F) -> rusqlite::Result<Vec<T>>;
fn load_array_module(&mut self) -> rusqlite::Result<()>;
}
impl SqliteConnection for rusqlite::Connection {
fn execute(&mut self, query: &str, params: impl rusqlite::Params) -> rusqlite::Result<usize> {
rusqlite::Connection::execute(self, query, params)
}
fn query<
T,
F: FnMut(&rusqlite::Row<'_>) -> rusqlite::Result<T>,
>(&mut self, query: &str, params: impl rusqlite::Params, mut f: F) -> rusqlite::Result<Vec<T>> {
let mut stmt = self.prepare(query)?;
let rows = stmt.query_map(params, |row| f(row))?;
let mut res = vec![];
for row in rows {
res.push(row?);
}
Ok(res)
}
fn load_array_module(&mut self) -> rusqlite::Result<()> {
rusqlite::vtab::array::load_module(self)
}
}
impl SqliteConnection for rusqlite::Transaction<'_> {
fn execute(&mut self, query: &str, params: impl rusqlite::Params) -> rusqlite::Result<usize> {
rusqlite::Connection::execute(self, query, params)
}
fn query<
T,
F: FnMut(&rusqlite::Row<'_>) -> rusqlite::Result<T>,
>(&mut self, query: &str, params: impl rusqlite::Params, mut f: F) -> rusqlite::Result<Vec<T>> {
let mut stmt = self.prepare(query)?;
let rows = stmt.query_map(params, |row| f(row))?;
let mut res = vec![];
for row in rows {
res.push(row?);
}
Ok(res)
}
fn load_array_module(&mut self) -> rusqlite::Result<()> {
Ok(())
}
}
pub enum GoodOrmningSqliteTimestamp {
String(String),
I64(i64),
}
impl rusqlite::types::ToSql for GoodOrmningSqliteTimestamp {
fn to_sql(&self) -> rusqlite::Result<rusqlite::types::ToSqlOutput<'_>> {
match self {
GoodOrmningSqliteTimestamp::String(s) => Ok(
rusqlite::types::ToSqlOutput::Owned(rusqlite::types::Value::Text(s.clone())),
),
GoodOrmningSqliteTimestamp::I64(i) => Ok(
rusqlite::types::ToSqlOutput::Owned(rusqlite::types::Value::Integer(*i)),
),
}
}
}
impl rusqlite::types::FromSql for GoodOrmningSqliteTimestamp {
fn column_result(value: rusqlite::types::ValueRef<'_>) -> Result<Self, rusqlite::types::FromSqlError> {
match value {
rusqlite::types::ValueRef::Text(s) => {
let s = std::str::from_utf8(s).map_err(|e| rusqlite::types::FromSqlError::Other(Box::new(e)))?;
Ok(GoodOrmningSqliteTimestamp::String(s.to_string()))
},
rusqlite::types::ValueRef::Integer(i) => {
Ok(GoodOrmningSqliteTimestamp::I64(i))
},
_ => Err(rusqlite::types::FromSqlError::InvalidType),
}
}
}
impl From<GoodOrmningSqliteTimestamp> for rusqlite::types::Value {
fn from(val: GoodOrmningSqliteTimestamp) -> Self {
match val {
GoodOrmningSqliteTimestamp::String(s) => rusqlite::types::Value::Text(s),
GoodOrmningSqliteTimestamp::I64(i) => rusqlite::types::Value::Integer(i),
}
}
}
pub trait GoodOrmningCustomAuto<T> {
fn to_sql(value: &T) -> i64;
fn from_sql(value: i64) -> Result<T, String>;
}
pub trait GoodOrmningCustomBool<T> {
fn to_sql(value: &T) -> bool;
fn from_sql(value: bool) -> Result<T, String>;
}
pub trait GoodOrmningCustomI16<T> {
fn to_sql(value: &T) -> i16;
fn from_sql(value: i16) -> Result<T, String>;
}
pub trait GoodOrmningCustomI32<T> {
fn to_sql(value: &T) -> i32;
fn from_sql(value: i32) -> Result<T, String>;
}
pub trait GoodOrmningCustomI64<T> {
fn to_sql(value: &T) -> i64;
fn from_sql(value: i64) -> Result<T, String>;
}
pub trait GoodOrmningCustomU32<T> {
fn to_sql(value: &T) -> u32;
fn from_sql(value: u32) -> Result<T, String>;
}
pub trait GoodOrmningCustomF32<T> {
fn to_sql(value: &T) -> f32;
fn from_sql(value: f32) -> Result<T, String>;
}
pub trait GoodOrmningCustomF64<T> {
fn to_sql(value: &T) -> f64;
fn from_sql(value: f64) -> Result<T, String>;
}
pub trait GoodOrmningCustomString<T> {
fn to_sql<'a>(value: &'a T) -> String;
fn from_sql(value: String) -> Result<T, String>;
}
pub trait GoodOrmningCustomBytes<T> {
fn to_sql<'a>(value: &'a T) -> Vec<u8>;
fn from_sql(value: Vec<u8>) -> Result<T, String>;
}
#[cfg(feature = "chrono")]
pub trait GoodOrmningCustomUtcTimeChrono<T> {
fn to_sql(value: &T) -> DateTime<Utc>;
fn from_sql(value: DateTime<Utc>) -> Result<T, String>;
}
#[cfg(feature = "chrono")]
pub trait GoodOrmningCustomFixedOffsetTimeChrono<T> {
fn to_sql(value: &T) -> DateTime<FixedOffset>;
fn from_sql(value: DateTime<FixedOffset>) -> Result<T, String>;
}
#[cfg(feature = "jiff")]
pub trait GoodOrmningCustomUtcTimeJiff<T> {
fn to_sql(value: &T) -> Timestamp;
fn from_sql(value: Timestamp) -> Result<T, String>;
}
#[cfg(feature = "jiff")]
pub trait GoodOrmningCustomFixedOffsetTimeJiff<T> {
fn to_sql(value: &T) -> Zoned;
fn from_sql(value: Zoned) -> Result<T, String>;
}