mirror of
https://github.com/isar/rusqlite.git
synced 2025-04-04 22:37:44 +08:00
Wrap the unsafe functions to the special preupdate hook functions in a safe wrapper.
This commit is contained in:
parent
ceff6cb8b1
commit
9187b3e960
97
src/hooks.rs
97
src/hooks.rs
@ -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);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user