use std::mem::transmute;
use crate::{
CursorImpl, CursorPolling, Error, ParameterCollectionRef, Sleep,
handles::{AsStatementRef, SqlText, Statement, StatementRef},
parameter::Blob,
sleep::wait_for,
};
pub fn execute_with_parameters<S>(
statement: S,
query: Option<&SqlText<'_>>,
params: impl ParameterCollectionRef,
) -> Result<Option<CursorImpl<S>>, Error>
where
S: AsStatementRef,
{
unsafe {
if let Some(statement) = bind_parameters(statement, params)? {
execute(statement, query)
} else {
Ok(None)
}
}
}
pub async fn execute_with_parameters_polling<S>(
statement: S,
query: Option<&SqlText<'_>>,
params: impl ParameterCollectionRef,
sleep: impl Sleep,
) -> Result<Option<CursorPolling<S>>, Error>
where
S: AsStatementRef,
{
unsafe {
if let Some(statement) = bind_parameters(statement, params)? {
execute_polling(statement, query, sleep).await
} else {
Ok(None)
}
}
}
unsafe fn bind_parameters<S>(
mut statement: S,
mut params: impl ParameterCollectionRef,
) -> Result<Option<S>, Error>
where
S: AsStatementRef,
{
unsafe {
let parameter_set_size = params.parameter_set_size();
let mut stmt = statement.as_stmt_ref();
stmt.reset_parameters().into_result(&stmt)?;
stmt.set_paramset_size(parameter_set_size)
.into_result(&stmt)?;
params.bind_parameters_to(&mut stmt)?;
Ok(Some(statement))
}
}
pub unsafe fn execute<S>(
mut statement: S,
query: Option<&SqlText<'_>>,
) -> Result<Option<CursorImpl<S>>, Error>
where
S: AsStatementRef,
{
unsafe {
let mut stmt = statement.as_stmt_ref();
let result = if let Some(sql) = query {
stmt.exec_direct(sql)
} else {
stmt.execute()
};
let need_data = result
.on_success(|| false)
.on_no_data(|| false)
.on_need_data(|| true)
.into_result(&stmt)?;
if need_data {
while let Some(blob_ref) = next_blob_param(&mut stmt)? {
while let Some(batch) = blob_ref.next_batch().map_err(Error::FailedReadingInput)? {
stmt.put_binary_batch(batch).into_result(&stmt)?;
}
}
}
if stmt.num_result_cols().into_result(&stmt)? == 0 {
Ok(None)
} else {
let cursor = CursorImpl::new(statement);
Ok(Some(cursor))
}
}
}
pub async unsafe fn execute_polling<S>(
mut statement: S,
query: Option<&SqlText<'_>>,
mut sleep: impl Sleep,
) -> Result<Option<CursorPolling<S>>, Error>
where
S: AsStatementRef,
{
unsafe {
let mut stmt = statement.as_stmt_ref();
let result = if let Some(sql) = query {
wait_for(|| stmt.exec_direct(sql), &mut sleep).await
} else {
wait_for(|| stmt.execute(), &mut sleep).await
};
let need_data = result
.on_success(|| false)
.on_no_data(|| false)
.on_need_data(|| true)
.into_result(&stmt)?;
if need_data {
while let Some(blob_ref) = next_blob_param(&mut stmt)? {
while let Some(batch) = blob_ref.next_batch().map_err(Error::FailedReadingInput)? {
let result = wait_for(|| stmt.put_binary_batch(batch), &mut sleep).await;
result.into_result(&stmt)?;
}
}
}
let num_result_cols = wait_for(|| stmt.num_result_cols(), &mut sleep)
.await
.into_result(&stmt)?;
if num_result_cols == 0 {
Ok(None)
} else {
let cursor = CursorPolling::new(statement);
Ok(Some(cursor))
}
}
}
unsafe fn next_blob_param<'a>(
stmt: &mut StatementRef<'a>,
) -> Result<Option<&'a mut dyn Blob>, Error> {
let maybe_ptr = stmt.param_data().into_result(stmt)?;
if let Some(blob_ptr) = maybe_ptr {
let blob_ptr: *mut &mut dyn Blob = unsafe { transmute(blob_ptr) };
let blob_ref = unsafe { &mut *blob_ptr };
Ok(Some(*blob_ref))
} else {
Ok(None)
}
}