Wrap the unsafe functions to the special preupdate hook functions in a safe wrapper.

This commit is contained in:
Midas Lambrichts 2021-02-26 21:02:06 +01:00
parent ceff6cb8b1
commit 9187b3e960

View File

@ -48,27 +48,71 @@ mod preupdate_hook {
use crate::types::ValueRef; use crate::types::ValueRef;
use crate::{Connection, InnerConnection}; use crate::{Connection, InnerConnection};
// TODO: how to allow user access to these functions, since they should be only accessible in pub enum PreUpdateCase {
// the scope of a preupdate_hook callback. Insert(PreUpdateInsert),
pub struct PreUpdateHookFunctions { Delete(PreUpdateDelete),
Update(PreUpdateUpdate),
}
pub struct PreUpdateInsert {
db: *mut ffi::sqlite3, db: *mut ffi::sqlite3,
} }
impl PreUpdateHookFunctions { impl PreUpdateInsert {
pub unsafe fn get_count(&self) -> i32 { pub fn get_count(&self) -> i32 {
ffi::sqlite3_preupdate_count(self.db) unsafe { ffi::sqlite3_preupdate_count(self.db) }
} }
pub unsafe fn get_old(&self, i: i32) -> ValueRef { pub fn get_new<'a>(&'a self, i: i32) -> ValueRef<'a> {
let mut p_value: *mut ffi::sqlite3_value = ptr::null_mut(); let mut p_value: *mut ffi::sqlite3_value = ptr::null_mut();
ffi::sqlite3_preupdate_old(self.db, i, &mut p_value); unsafe {
ValueRef::from_value(p_value) ffi::sqlite3_preupdate_new(self.db, i, &mut p_value);
ValueRef::from_value(p_value)
}
}
}
pub struct PreUpdateDelete {
db: *mut ffi::sqlite3,
}
impl PreUpdateDelete {
pub fn get_count(&self) -> i32 {
unsafe { ffi::sqlite3_preupdate_count(self.db) }
} }
pub unsafe fn get_new(&self, i: i32) -> ValueRef { pub fn get_old<'a>(&'a self, i: i32) -> ValueRef<'a> {
let mut p_value: *mut ffi::sqlite3_value = ptr::null_mut(); let mut p_value: *mut ffi::sqlite3_value = ptr::null_mut();
ffi::sqlite3_preupdate_new(self.db, i, &mut p_value); unsafe {
ValueRef::from_value(p_value) ffi::sqlite3_preupdate_old(self.db, i, &mut p_value);
ValueRef::from_value(p_value)
}
}
}
pub struct PreUpdateUpdate {
db: *mut ffi::sqlite3,
}
impl PreUpdateUpdate {
pub fn get_count(&self) -> i32 {
unsafe { ffi::sqlite3_preupdate_count(self.db) }
}
pub fn get_old<'a>(&'a self, i: i32) -> ValueRef<'a> {
let mut p_value: *mut ffi::sqlite3_value = ptr::null_mut();
unsafe {
ffi::sqlite3_preupdate_old(self.db, i, &mut p_value);
ValueRef::from_value(p_value)
}
}
pub fn get_new<'a>(&'a self, i: i32) -> ValueRef<'a> {
let mut p_value: *mut ffi::sqlite3_value = ptr::null_mut();
unsafe {
ffi::sqlite3_preupdate_new(self.db, i, &mut p_value);
ValueRef::from_value(p_value)
}
} }
} }
@ -88,7 +132,7 @@ mod preupdate_hook {
#[inline] #[inline]
pub fn preupdate_hook<'c, F>(&'c self, hook: Option<F>) pub fn preupdate_hook<'c, F>(&'c self, hook: Option<F>)
where where
F: FnMut(Action, &str, &str, i64, i64, &PreUpdateHookFunctions) + Send + 'c, F: FnMut(Action, &str, &str, i64, i64, &PreUpdateCase) + Send + 'c,
{ {
self.db.borrow_mut().preupdate_hook(hook); self.db.borrow_mut().preupdate_hook(hook);
} }
@ -97,12 +141,12 @@ mod preupdate_hook {
impl InnerConnection { impl InnerConnection {
#[inline] #[inline]
pub fn remove_preupdate_hook(&mut self) { pub fn remove_preupdate_hook(&mut self) {
self.preupdate_hook(None::<fn(Action, &str, &str, i64, i64, &PreUpdateHookFunctions)>); self.preupdate_hook(None::<fn(Action, &str, &str, i64, i64, &PreUpdateCase)>);
} }
fn preupdate_hook<'c, F>(&'c mut self, hook: Option<F>) fn preupdate_hook<'c, F>(&'c mut self, hook: Option<F>)
where where
F: FnMut(Action, &str, &str, i64, i64, &PreUpdateHookFunctions) + Send + 'c, F: FnMut(Action, &str, &str, i64, i64, &PreUpdateCase) + Send + 'c,
{ {
unsafe extern "C" fn call_boxed_closure<F>( unsafe extern "C" fn call_boxed_closure<F>(
p_arg: *mut c_void, p_arg: *mut c_void,
@ -113,7 +157,7 @@ mod preupdate_hook {
row_id: i64, row_id: i64,
new_row_id: i64, new_row_id: i64,
) where ) where
F: FnMut(Action, &str, &str, i64, i64, &PreUpdateHookFunctions), F: FnMut(Action, &str, &str, i64, i64, &PreUpdateCase),
{ {
use std::ffi::CStr; use std::ffi::CStr;
use std::str; use std::str;
@ -128,12 +172,12 @@ mod preupdate_hook {
str::from_utf8(c_slice) str::from_utf8(c_slice)
}; };
// TODO: how to properly allow a user to use the functions let preupdate_hook_functions = match action {
// (sqlite3_preupdate_old,...) that are only in scope Action::SQLITE_INSERT => PreUpdateCase::Insert(PreUpdateInsert { db: sqlite }),
// during the callback? Action::SQLITE_DELETE => PreUpdateCase::Delete(PreUpdateDelete { db: sqlite }),
// Also how to pass in the rowids, because they can be undefined based on the Action::SQLITE_UPDATE => PreUpdateCase::Update(PreUpdateUpdate { db: sqlite }),
// action. _ => todo!(),
let preupdate_hook_functions = PreUpdateHookFunctions { db: sqlite }; };
let _ = catch_unwind(|| { let _ = catch_unwind(|| {
let boxed_hook: *mut F = p_arg as *mut F; let boxed_hook: *mut F = p_arg as *mut F;
@ -179,7 +223,7 @@ mod preupdate_hook {
#[cfg(test)] #[cfg(test)]
mod test { mod test {
use super::super::Action; use super::super::Action;
use super::PreUpdateHookFunctions; use super::PreUpdateCase;
use crate::{Connection, Result}; use crate::{Connection, Result};
#[test] #[test]
@ -188,12 +232,7 @@ mod preupdate_hook {
let mut called = false; let mut called = false;
db.preupdate_hook(Some( db.preupdate_hook(Some(
|action, |action, db: &str, tbl: &str, row_id, new_row_id, _func: &PreUpdateCase| {
db: &str,
tbl: &str,
row_id,
new_row_id,
_func: &PreUpdateHookFunctions| {
assert_eq!(Action::SQLITE_INSERT, action); assert_eq!(Action::SQLITE_INSERT, action);
assert_eq!("main", db); assert_eq!("main", db);
assert_eq!("foo", tbl); assert_eq!("foo", tbl);