extern crate libsqlite3_sys as ffi;
use super::raw::RawConnection;
use super::row::PrivateSqliteRow;
use super::{Sqlite, SqliteAggregateFunction, SqliteBindValue};
use crate::backend::Backend;
use crate::deserialize::{FromSqlRow, StaticallySizedRow};
use crate::result::{DatabaseErrorKind, Error, QueryResult};
use crate::row::{Field, PartialRow, Row, RowIndex, RowSealed};
use crate::serialize::{IsNull, Output, ToSql};
use crate::sql_types::HasSqlType;
use crate::sqlite::connection::bind_collector::InternalSqliteBindValue;
use crate::sqlite::connection::sqlite_value::OwnedSqliteValue;
use crate::sqlite::SqliteValue;
use std::cell::{Ref, RefCell};
use std::marker::PhantomData;
use std::mem::ManuallyDrop;
use std::ops::DerefMut;
use std::rc::Rc;
pub(super) fn register<ArgsSqlType, RetSqlType, Args, Ret, F>(
conn: &RawConnection,
fn_name: &str,
deterministic: bool,
mut f: F,
) -> QueryResult<()>
where
F: FnMut(&RawConnection, Args) -> Ret + std::panic::UnwindSafe + Send + 'static,
Args: FromSqlRow<ArgsSqlType, Sqlite> + StaticallySizedRow<ArgsSqlType, Sqlite>,
Ret: ToSql<RetSqlType, Sqlite>,
Sqlite: HasSqlType<RetSqlType>,
{
let fields_needed = Args::FIELD_COUNT;
if fields_needed > 127 {
return Err(Error::DatabaseError(
DatabaseErrorKind::UnableToSendCommand,
Box::new("SQLite functions cannot take more than 127 parameters".to_string()),
));
}
conn.register_sql_function(fn_name, fields_needed, deterministic, move |conn, args| {
let args = build_sql_function_args::<ArgsSqlType, Args>(args)?;
Ok(f(conn, args))
})?;
Ok(())
}
pub(super) fn register_noargs<RetSqlType, Ret, F>(
conn: &RawConnection,
fn_name: &str,
deterministic: bool,
mut f: F,
) -> QueryResult<()>
where
F: FnMut() -> Ret + std::panic::UnwindSafe + Send + 'static,
Ret: ToSql<RetSqlType, Sqlite>,
Sqlite: HasSqlType<RetSqlType>,
{
conn.register_sql_function(fn_name, 0, deterministic, move |_, _| Ok(f()))?;
Ok(())
}
pub(super) fn register_aggregate<ArgsSqlType, RetSqlType, Args, Ret, A>(
conn: &RawConnection,
fn_name: &str,
) -> QueryResult<()>
where
A: SqliteAggregateFunction<Args, Output = Ret> + 'static + Send + std::panic::UnwindSafe,
Args: FromSqlRow<ArgsSqlType, Sqlite> + StaticallySizedRow<ArgsSqlType, Sqlite>,
Ret: ToSql<RetSqlType, Sqlite>,
Sqlite: HasSqlType<RetSqlType>,
{
let fields_needed = Args::FIELD_COUNT;
if fields_needed > 127 {
return Err(Error::DatabaseError(
DatabaseErrorKind::UnableToSendCommand,
Box::new("SQLite functions cannot take more than 127 parameters".to_string()),
));
}
conn.register_aggregate_function::<ArgsSqlType, RetSqlType, Args, Ret, A>(
fn_name,
fields_needed,
)?;
Ok(())
}
pub(super) fn build_sql_function_args<ArgsSqlType, Args>(
args: &mut [*mut ffi::sqlite3_value],
) -> Result<Args, Error>
where
Args: FromSqlRow<ArgsSqlType, Sqlite>,
{
let row = FunctionRow::new(args);
Args::build_from_row(&row).map_err(Error::DeserializationError)
}
#[allow(clippy::let_unit_value)]
pub(super) fn process_sql_function_result<RetSqlType, Ret>(
result: &'_ Ret,
) -> QueryResult<InternalSqliteBindValue<'_>>
where
Ret: ToSql<RetSqlType, Sqlite>,
Sqlite: HasSqlType<RetSqlType>,
{
let mut metadata_lookup = ();
let value = SqliteBindValue {
inner: InternalSqliteBindValue::Null,
};
let mut buf = Output::new(value, &mut metadata_lookup);
let is_null = result.to_sql(&mut buf).map_err(Error::SerializationError)?;
if let IsNull::Yes = is_null {
Ok(InternalSqliteBindValue::Null)
} else {
Ok(buf.into_inner().inner)
}
}
struct FunctionRow<'a> {
args: Rc<RefCell<ManuallyDrop<PrivateSqliteRow<'a, 'static>>>>,
field_count: usize,
marker: PhantomData<&'a ffi::sqlite3_value>,
}
impl<'a> Drop for FunctionRow<'a> {
#[allow(unsafe_code)] fn drop(&mut self) {
if let Some(args) = Rc::get_mut(&mut self.args) {
if let PrivateSqliteRow::Duplicated { column_names, .. } =
DerefMut::deref_mut(RefCell::get_mut(args))
{
if Rc::strong_count(column_names) == 1 {
unsafe { std::ptr::drop_in_place(column_names as *mut _) }
}
}
}
}
}
impl<'a> FunctionRow<'a> {
#[allow(unsafe_code)] fn new(args: &mut [*mut ffi::sqlite3_value]) -> Self {
let lengths = args.len();
let args = unsafe {
Vec::from_raw_parts(
args as *mut [*mut ffi::sqlite3_value] as *mut ffi::sqlite3_value
as *mut Option<OwnedSqliteValue>,
lengths,
lengths,
)
};
Self {
field_count: lengths,
args: Rc::new(RefCell::new(ManuallyDrop::new(
PrivateSqliteRow::Duplicated {
values: args,
column_names: Rc::from(vec![None; lengths]),
},
))),
marker: PhantomData,
}
}
}
impl RowSealed for FunctionRow<'_> {}
impl<'a> Row<'a, Sqlite> for FunctionRow<'a> {
type Field<'f> = FunctionArgument<'f> where 'a: 'f, Self: 'f;
type InnerPartialRow = Self;
fn field_count(&self) -> usize {
self.field_count
}
fn get<'b, I>(&'b self, idx: I) -> Option<Self::Field<'b>>
where
'a: 'b,
Self: crate::row::RowIndex<I>,
{
let idx = self.idx(idx)?;
Some(FunctionArgument {
args: self.args.borrow(),
col_idx: idx as i32,
})
}
fn partial_row(&self, range: std::ops::Range<usize>) -> PartialRow<'_, Self::InnerPartialRow> {
PartialRow::new(self, range)
}
}
impl<'a> RowIndex<usize> for FunctionRow<'a> {
fn idx(&self, idx: usize) -> Option<usize> {
if idx < self.field_count() {
Some(idx)
} else {
None
}
}
}
impl<'a, 'b> RowIndex<&'a str> for FunctionRow<'b> {
fn idx(&self, _idx: &'a str) -> Option<usize> {
None
}
}
struct FunctionArgument<'a> {
args: Ref<'a, ManuallyDrop<PrivateSqliteRow<'a, 'static>>>,
col_idx: i32,
}
impl<'a> Field<'a, Sqlite> for FunctionArgument<'a> {
fn field_name(&self) -> Option<&str> {
None
}
fn is_null(&self) -> bool {
self.value().is_none()
}
fn value(&self) -> Option<<Sqlite as Backend>::RawValue<'_>> {
SqliteValue::new(
Ref::map(Ref::clone(&self.args), |drop| std::ops::Deref::deref(drop)),
self.col_idx,
)
}
}