#![allow(unsafe_code)] extern crate libsqlite3_sys as ffi;
use std::ffi::{CString, NulError};
use std::io::{stderr, Write};
use std::os::raw as libc;
use std::ptr::NonNull;
use std::{mem, ptr, slice, str};
use super::functions::{build_sql_function_args, process_sql_function_result};
use super::stmt::ensure_sqlite_ok;
use super::{Sqlite, SqliteAggregateFunction};
use crate::deserialize::FromSqlRow;
use crate::result::Error::DatabaseError;
use crate::result::*;
use crate::serialize::ToSql;
use crate::sql_types::HasSqlType;
macro_rules! assert_fail {
($fmt:expr $(,$args:tt)*) => {
eprint!(concat!(
$fmt,
"If you see this message, please open an issue at https://github.com/diesel-rs/diesel/issues/new.\n",
"Source location: {}:{}\n",
), $($args,)* file!(), line!());
std::process::abort()
};
}
#[allow(missing_debug_implementations, missing_copy_implementations)]
pub(super) struct RawConnection {
pub(super) internal_connection: NonNull<ffi::sqlite3>,
}
impl RawConnection {
pub(super) fn establish(database_url: &str) -> ConnectionResult<Self> {
let mut conn_pointer = ptr::null_mut();
let database_url = if database_url.starts_with("sqlite://") {
CString::new(database_url.replacen("sqlite://", "file:", 1))?
} else {
CString::new(database_url)?
};
let flags = ffi::SQLITE_OPEN_READWRITE | ffi::SQLITE_OPEN_CREATE | ffi::SQLITE_OPEN_URI;
let connection_status = unsafe {
ffi::sqlite3_open_v2(database_url.as_ptr(), &mut conn_pointer, flags, ptr::null())
};
match connection_status {
ffi::SQLITE_OK => {
let conn_pointer = unsafe { NonNull::new_unchecked(conn_pointer) };
Ok(RawConnection {
internal_connection: conn_pointer,
})
}
err_code => {
let message = super::error_message(err_code);
Err(ConnectionError::BadConnection(message.into()))
}
}
}
pub(super) fn exec(&self, query: &str) -> QueryResult<()> {
let query = CString::new(query)?;
let callback_fn = None;
let callback_arg = ptr::null_mut();
let result = unsafe {
ffi::sqlite3_exec(
self.internal_connection.as_ptr(),
query.as_ptr(),
callback_fn,
callback_arg,
ptr::null_mut(),
)
};
ensure_sqlite_ok(result, self.internal_connection.as_ptr())
}
pub(super) fn rows_affected_by_last_query(&self) -> usize {
unsafe { ffi::sqlite3_changes(self.internal_connection.as_ptr()) as usize }
}
pub(super) fn register_sql_function<F, Ret, RetSqlType>(
&self,
fn_name: &str,
num_args: usize,
deterministic: bool,
f: F,
) -> QueryResult<()>
where
F: FnMut(&Self, &mut [*mut ffi::sqlite3_value]) -> QueryResult<Ret>
+ std::panic::UnwindSafe
+ Send
+ 'static,
Ret: ToSql<RetSqlType, Sqlite>,
Sqlite: HasSqlType<RetSqlType>,
{
let callback_fn = Box::into_raw(Box::new(CustomFunctionUserPtr {
callback: f,
function_name: fn_name.to_owned(),
}));
let fn_name = Self::get_fn_name(fn_name)?;
let flags = Self::get_flags(deterministic);
let result = unsafe {
ffi::sqlite3_create_function_v2(
self.internal_connection.as_ptr(),
fn_name.as_ptr(),
num_args as _,
flags,
callback_fn as *mut _,
Some(run_custom_function::<F, Ret, RetSqlType>),
None,
None,
Some(destroy_boxed::<CustomFunctionUserPtr<F>>),
)
};
Self::process_sql_function_result(result)
}
pub(super) fn register_aggregate_function<ArgsSqlType, RetSqlType, Args, Ret, A>(
&self,
fn_name: &str,
num_args: usize,
) -> QueryResult<()>
where
A: SqliteAggregateFunction<Args, Output = Ret> + 'static + Send + std::panic::UnwindSafe,
Args: FromSqlRow<ArgsSqlType, Sqlite>,
Ret: ToSql<RetSqlType, Sqlite>,
Sqlite: HasSqlType<RetSqlType>,
{
let fn_name = Self::get_fn_name(fn_name)?;
let flags = Self::get_flags(false);
let result = unsafe {
ffi::sqlite3_create_function_v2(
self.internal_connection.as_ptr(),
fn_name.as_ptr(),
num_args as _,
flags,
ptr::null_mut(),
None,
Some(run_aggregator_step_function::<_, _, _, _, A>),
Some(run_aggregator_final_function::<_, _, _, _, A>),
None,
)
};
Self::process_sql_function_result(result)
}
pub(super) fn register_collation_function<F>(
&self,
collation_name: &str,
collation: F,
) -> QueryResult<()>
where
F: Fn(&str, &str) -> std::cmp::Ordering + std::panic::UnwindSafe + Send + 'static,
{
let callback_fn = Box::into_raw(Box::new(CollationUserPtr {
callback: collation,
collation_name: collation_name.to_owned(),
}));
let collation_name = Self::get_fn_name(collation_name)?;
let result = unsafe {
ffi::sqlite3_create_collation_v2(
self.internal_connection.as_ptr(),
collation_name.as_ptr(),
ffi::SQLITE_UTF8,
callback_fn as *mut _,
Some(run_collation_function::<F>),
Some(destroy_boxed::<CollationUserPtr<F>>),
)
};
let result = Self::process_sql_function_result(result);
if result.is_err() {
destroy_boxed::<CollationUserPtr<F>>(callback_fn as *mut _);
}
result
}
fn get_fn_name(fn_name: &str) -> Result<CString, NulError> {
CString::new(fn_name)
}
fn get_flags(deterministic: bool) -> i32 {
let mut flags = ffi::SQLITE_UTF8;
if deterministic {
flags |= ffi::SQLITE_DETERMINISTIC;
}
flags
}
fn process_sql_function_result(result: i32) -> Result<(), Error> {
if result == ffi::SQLITE_OK {
Ok(())
} else {
let error_message = super::error_message(result);
Err(DatabaseError(
DatabaseErrorKind::Unknown,
Box::new(error_message.to_string()),
))
}
}
}
impl Drop for RawConnection {
fn drop(&mut self) {
use std::thread::panicking;
let close_result = unsafe { ffi::sqlite3_close(self.internal_connection.as_ptr()) };
if close_result != ffi::SQLITE_OK {
let error_message = super::error_message(close_result);
if panicking() {
write!(stderr(), "Error closing SQLite connection: {error_message}")
.expect("Error writing to `stderr`");
} else {
panic!("Error closing SQLite connection: {}", error_message);
}
}
}
}
enum SqliteCallbackError {
Abort(&'static str),
DieselError(crate::result::Error),
Panic(Box<dyn std::any::Any + Send>, String),
}
impl SqliteCallbackError {
fn emit(&self, ctx: *mut ffi::sqlite3_context) {
let s;
let msg = match self {
SqliteCallbackError::Abort(msg) => *msg,
SqliteCallbackError::DieselError(e) => {
s = e.to_string();
&s
}
SqliteCallbackError::Panic(_, msg) => msg,
};
unsafe {
context_error_str(ctx, msg);
}
}
}
impl From<crate::result::Error> for SqliteCallbackError {
fn from(e: crate::result::Error) -> Self {
Self::DieselError(e)
}
}
struct CustomFunctionUserPtr<F> {
callback: F,
function_name: String,
}
#[allow(warnings)]
extern "C" fn run_custom_function<F, Ret, RetSqlType>(
ctx: *mut ffi::sqlite3_context,
num_args: libc::c_int,
value_ptr: *mut *mut ffi::sqlite3_value,
) where
F: FnMut(&RawConnection, &mut [*mut ffi::sqlite3_value]) -> QueryResult<Ret>
+ std::panic::UnwindSafe
+ Send
+ 'static,
Ret: ToSql<RetSqlType, Sqlite>,
Sqlite: HasSqlType<RetSqlType>,
{
use std::ops::Deref;
static NULL_DATA_ERR: &str = "An unknown error occurred. sqlite3_user_data returned a null pointer. This should never happen.";
static NULL_CONN_ERR: &str = "An unknown error occurred. sqlite3_context_db_handle returned a null pointer. This should never happen.";
let conn = match unsafe { NonNull::new(ffi::sqlite3_context_db_handle(ctx)) } {
Some(conn) => mem::ManuallyDrop::new(RawConnection {
internal_connection: conn,
}),
None => {
unsafe { context_error_str(ctx, NULL_CONN_ERR) };
return;
}
};
let data_ptr = unsafe { ffi::sqlite3_user_data(ctx) };
let mut data_ptr = match NonNull::new(data_ptr as *mut CustomFunctionUserPtr<F>) {
None => unsafe {
context_error_str(ctx, NULL_DATA_ERR);
return;
},
Some(mut f) => f,
};
let data_ptr = unsafe { data_ptr.as_mut() };
let callback = std::panic::AssertUnwindSafe(&mut data_ptr.callback);
let result = std::panic::catch_unwind(move || {
let _ = &callback;
let args = unsafe { slice::from_raw_parts_mut(value_ptr, num_args as _) };
let res = (callback.0)(&*conn, args)?;
let value = process_sql_function_result(&res)?;
unsafe {
value.result_of(&mut *ctx);
}
Ok(())
})
.unwrap_or_else(|p| {
Err(SqliteCallbackError::Panic(
p,
data_ptr.function_name.clone(),
))
});
if let Err(e) = result {
e.emit(ctx);
}
}
#[repr(u8)]
enum OptionalAggregator<A> {
None,
Some(A),
}
#[allow(warnings)]
extern "C" fn run_aggregator_step_function<ArgsSqlType, RetSqlType, Args, Ret, A>(
ctx: *mut ffi::sqlite3_context,
num_args: libc::c_int,
value_ptr: *mut *mut ffi::sqlite3_value,
) where
A: SqliteAggregateFunction<Args, Output = Ret> + 'static + Send + std::panic::UnwindSafe,
Args: FromSqlRow<ArgsSqlType, Sqlite>,
Ret: ToSql<RetSqlType, Sqlite>,
Sqlite: HasSqlType<RetSqlType>,
{
let result = std::panic::catch_unwind(move || {
let args = unsafe { slice::from_raw_parts_mut(value_ptr, num_args as _) };
run_aggregator_step::<A, Args, ArgsSqlType>(ctx, args)
})
.unwrap_or_else(|e| {
Err(SqliteCallbackError::Panic(
e,
format!("{}::step() panicked", std::any::type_name::<A>()),
))
});
match result {
Ok(()) => {}
Err(e) => e.emit(ctx),
}
}
fn run_aggregator_step<A, Args, ArgsSqlType>(
ctx: *mut ffi::sqlite3_context,
args: &mut [*mut ffi::sqlite3_value],
) -> Result<(), SqliteCallbackError>
where
A: SqliteAggregateFunction<Args>,
Args: FromSqlRow<ArgsSqlType, Sqlite>,
{
static NULL_AG_CTX_ERR: &str = "An unknown error occurred. sqlite3_aggregate_context returned a null pointer. This should never happen.";
static NULL_CTX_ERR: &str =
"We've written the aggregator to the aggregate context, but it could not be retrieved.";
let aggregate_context = unsafe {
ffi::sqlite3_aggregate_context(ctx, std::mem::size_of::<OptionalAggregator<A>>() as i32)
};
let aggregate_context = NonNull::new(aggregate_context as *mut OptionalAggregator<A>);
let aggregator = unsafe {
match aggregate_context.map(|a| &mut *a.as_ptr()) {
Some(&mut OptionalAggregator::Some(ref mut agg)) => agg,
Some(a_ptr @ &mut OptionalAggregator::None) => {
ptr::write_unaligned(a_ptr as *mut _, OptionalAggregator::Some(A::default()));
if let OptionalAggregator::Some(ref mut agg) = a_ptr {
agg
} else {
return Err(SqliteCallbackError::Abort(NULL_CTX_ERR));
}
}
None => {
return Err(SqliteCallbackError::Abort(NULL_AG_CTX_ERR));
}
}
};
let args = build_sql_function_args::<ArgsSqlType, Args>(args)?;
aggregator.step(args);
Ok(())
}
extern "C" fn run_aggregator_final_function<ArgsSqlType, RetSqlType, Args, Ret, A>(
ctx: *mut ffi::sqlite3_context,
) where
A: SqliteAggregateFunction<Args, Output = Ret> + 'static + Send,
Args: FromSqlRow<ArgsSqlType, Sqlite>,
Ret: ToSql<RetSqlType, Sqlite>,
Sqlite: HasSqlType<RetSqlType>,
{
static NO_AGGREGATOR_FOUND: &str = "We've written to the aggregator in the xStep callback. If xStep was never called, then ffi::sqlite_aggregate_context() would have returned a NULL pointer.";
let aggregate_context = unsafe {
ffi::sqlite3_aggregate_context(ctx, 0)
};
let result = std::panic::catch_unwind(|| {
let mut aggregate_context = NonNull::new(aggregate_context as *mut OptionalAggregator<A>);
let aggregator = if let Some(a) = aggregate_context.as_mut() {
let a = unsafe { a.as_mut() };
match std::mem::replace(a, OptionalAggregator::None) {
OptionalAggregator::None => {
return Err(SqliteCallbackError::Abort(NO_AGGREGATOR_FOUND));
}
OptionalAggregator::Some(a) => Some(a),
}
} else {
None
};
let res = A::finalize(aggregator);
let value = process_sql_function_result(&res)?;
unsafe {
value.result_of(&mut *ctx);
}
Ok(())
})
.unwrap_or_else(|e| {
Err(SqliteCallbackError::Panic(
e,
format!("{}::finalize() panicked", std::any::type_name::<A>()),
))
});
if let Err(e) = result {
e.emit(ctx);
}
}
unsafe fn context_error_str(ctx: *mut ffi::sqlite3_context, error: &str) {
ffi::sqlite3_result_error(ctx, error.as_ptr() as *const _, error.len() as _);
}
struct CollationUserPtr<F> {
callback: F,
collation_name: String,
}
#[allow(warnings)]
extern "C" fn run_collation_function<F>(
user_ptr: *mut libc::c_void,
lhs_len: libc::c_int,
lhs_ptr: *const libc::c_void,
rhs_len: libc::c_int,
rhs_ptr: *const libc::c_void,
) -> libc::c_int
where
F: Fn(&str, &str) -> std::cmp::Ordering + Send + std::panic::UnwindSafe + 'static,
{
let user_ptr = user_ptr as *const CollationUserPtr<F>;
let user_ptr = std::panic::AssertUnwindSafe(unsafe { user_ptr.as_ref() });
let result = std::panic::catch_unwind(|| {
let user_ptr = user_ptr.ok_or_else(|| {
SqliteCallbackError::Abort(
"Got a null pointer as data pointer. This should never happen",
)
})?;
for (ptr, len, side) in &[(rhs_ptr, rhs_len, "rhs"), (lhs_ptr, lhs_len, "lhs")] {
if *len < 0 {
assert_fail!(
"An unknown error occurred. {}_len is negative. This should never happen.",
side
);
}
if ptr.is_null() {
assert_fail!(
"An unknown error occurred. {}_ptr is a null pointer. This should never happen.",
side
);
}
}
let (rhs, lhs) = unsafe {
(
str::from_utf8(slice::from_raw_parts(rhs_ptr as *const u8, rhs_len as _)),
str::from_utf8(slice::from_raw_parts(lhs_ptr as *const u8, lhs_len as _)),
)
};
let rhs =
rhs.map_err(|_| SqliteCallbackError::Abort("Got an invalid UTF-8 string for rhs"))?;
let lhs =
lhs.map_err(|_| SqliteCallbackError::Abort("Got an invalid UTF-8 string for lhs"))?;
Ok((user_ptr.callback)(rhs, lhs))
})
.unwrap_or_else(|p| {
Err(SqliteCallbackError::Panic(
p,
user_ptr
.map(|u| u.collation_name.clone())
.unwrap_or_default(),
))
});
match result {
Ok(std::cmp::Ordering::Less) => -1,
Ok(std::cmp::Ordering::Equal) => 0,
Ok(std::cmp::Ordering::Greater) => 1,
Err(SqliteCallbackError::Abort(a)) => {
eprintln!(
"Collation function {} failed with: {}",
user_ptr
.map(|c| &c.collation_name as &str)
.unwrap_or_default(),
a
);
std::process::abort()
}
Err(SqliteCallbackError::DieselError(e)) => {
eprintln!(
"Collation function {} failed with: {}",
user_ptr
.map(|c| &c.collation_name as &str)
.unwrap_or_default(),
e
);
std::process::abort()
}
Err(SqliteCallbackError::Panic(_, msg)) => {
eprintln!("Collation function {} panicked", msg);
std::process::abort()
}
}
}
extern "C" fn destroy_boxed<F>(data: *mut libc::c_void) {
let ptr = data as *mut F;
unsafe { std::mem::drop(Box::from_raw(ptr)) };
}