use std::ffi::CString;
use std::slice;
use libc::{c_char, c_uchar, c_void, size_t};
use crate::{ffi, ffi_util::CStrLike};
pub struct SliceTransform {
pub inner: *mut ffi::rocksdb_slicetransform_t,
}
impl SliceTransform {
pub fn create(
name: impl CStrLike,
transform_fn: TransformFn,
in_domain_fn: Option<InDomainFn>,
) -> SliceTransform {
let cb = Box::into_raw(Box::new(TransformCallback {
name: name.into_c_string().unwrap(),
transform_fn,
in_domain_fn,
}));
let st = unsafe {
ffi::rocksdb_slicetransform_create(
cb as *mut c_void,
Some(slice_transform_destructor_callback),
Some(transform_callback),
Some(in_domain_callback),
None,
Some(slice_transform_name_callback),
)
};
SliceTransform { inner: st }
}
pub fn create_fixed_prefix(len: size_t) -> SliceTransform {
SliceTransform {
inner: unsafe { ffi::rocksdb_slicetransform_create_fixed_prefix(len) },
}
}
pub fn create_noop() -> SliceTransform {
SliceTransform {
inner: unsafe { ffi::rocksdb_slicetransform_create_noop() },
}
}
}
pub type TransformFn<'a> = fn(&'a [u8]) -> &'a [u8];
pub type InDomainFn = fn(&[u8]) -> bool;
pub struct TransformCallback<'a> {
pub name: CString,
pub transform_fn: TransformFn<'a>,
pub in_domain_fn: Option<InDomainFn>,
}
pub unsafe extern "C" fn slice_transform_destructor_callback(raw_cb: *mut c_void) {
Box::from_raw(raw_cb as *mut TransformCallback);
}
pub unsafe extern "C" fn slice_transform_name_callback(raw_cb: *mut c_void) -> *const c_char {
let cb = &mut *(raw_cb as *mut TransformCallback);
cb.name.as_ptr()
}
pub unsafe extern "C" fn transform_callback(
raw_cb: *mut c_void,
raw_key: *const c_char,
key_len: size_t,
dst_length: *mut size_t,
) -> *mut c_char {
let cb = &mut *(raw_cb as *mut TransformCallback);
let key = slice::from_raw_parts(raw_key as *const u8, key_len as usize);
let prefix = (cb.transform_fn)(key);
*dst_length = prefix.len() as size_t;
prefix.as_ptr() as *mut c_char
}
pub unsafe extern "C" fn in_domain_callback(
raw_cb: *mut c_void,
raw_key: *const c_char,
key_len: size_t,
) -> c_uchar {
let cb = &mut *(raw_cb as *mut TransformCallback);
let key = slice::from_raw_parts(raw_key as *const u8, key_len as usize);
c_uchar::from(cb.in_domain_fn.map_or(true, |in_domain| in_domain(key)))
}