use std::intrinsics::transmute;
use crate::{
handles::{AsStatementRef, SqlText, Statement},
parameter::Blob,
CursorImpl, Error, ParameterCollectionRef,
};
pub fn execute_with_parameters<S>(
lazy_statement: impl FnOnce() -> Result<S, Error>,
query: Option<&SqlText>,
mut params: impl ParameterCollectionRef,
) -> Result<Option<CursorImpl<S>>, Error>
where
S: AsStatementRef,
{
let parameter_set_size = params.parameter_set_size();
if parameter_set_size == 0 {
return Ok(None);
}
let mut statement = lazy_statement()?;
let mut stmt = statement.as_stmt_ref();
stmt.reset_parameters().into_result(&stmt)?;
unsafe {
stmt.set_paramset_size(parameter_set_size)
.into_result(&stmt)?;
params.bind_parameters_to(&mut stmt)?;
execute(statement, query)
}
}
pub unsafe fn execute<S>(
mut statement: S,
query: Option<&SqlText>,
) -> Result<Option<CursorImpl<S>>, Error>
where
S: AsStatementRef,
{
let mut stmt = statement.as_stmt_ref();
let need_data = if let Some(sql) = query {
stmt.exec_direct(sql).into_result(&stmt)?
} else {
stmt.execute().into_result(&stmt)?
};
if need_data {
while let Some(blob_ptr) = stmt.param_data().into_result(&stmt)? {
let blob_ptr: *mut &mut dyn Blob = transmute(blob_ptr);
let blob_ref = &mut *blob_ptr;
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 fn execute_columns<S>(
mut statement: S,
catalog_name: &SqlText,
schema_name: &SqlText,
table_name: &SqlText,
column_name: &SqlText,
) -> Result<CursorImpl<S>, Error>
where
S: AsStatementRef,
{
let mut stmt = statement.as_stmt_ref();
stmt.columns(catalog_name, schema_name, table_name, column_name)
.into_result(&stmt)?;
debug_assert_ne!(stmt.num_result_cols().unwrap(), 0);
let cursor = unsafe { CursorImpl::new(statement) };
Ok(cursor)
}
pub fn execute_tables<S>(
mut statement: S,
catalog_name: &SqlText,
schema_name: &SqlText,
table_name: &SqlText,
column_name: &SqlText,
) -> Result<CursorImpl<S>, Error>
where
S: AsStatementRef,
{
let mut stmt = statement.as_stmt_ref();
stmt.tables(catalog_name, schema_name, table_name, column_name)
.into_result(&stmt)?;
debug_assert_ne!(stmt.num_result_cols().unwrap(), 0);
let cursor = unsafe { CursorImpl::new(statement) };
Ok(cursor)
}