use arrow::datatypes::DataType;
use super::Result;
use crate::{
ffi,
types::{FromSqlError, Type},
};
use std::{error, ffi::CStr, fmt, path::PathBuf, str};
#[derive(Debug)]
#[allow(clippy::enum_variant_names)]
#[non_exhaustive]
pub enum Error {
DuckDBFailure(ffi::Error, Option<String>),
FromSqlConversionFailure(usize, Type, Box<dyn error::Error + Send + Sync + 'static>),
IntegralValueOutOfRange(usize, i128),
Utf8Error(str::Utf8Error),
NulError(::std::ffi::NulError),
InvalidParameterName(String),
InvalidPath(PathBuf),
ExecuteReturnedResults,
QueryReturnedNoRows,
QueryReturnedMoreThanOneRow,
InvalidColumnIndex(usize),
InvalidColumnName(String),
InvalidColumnType(usize, String, Type),
ArrowTypeToDuckdbType(String, DataType),
StatementChangedRows(usize),
ToSqlConversionFailure(Box<dyn error::Error + Send + Sync + 'static>),
InvalidQuery,
MultipleStatement,
InvalidParameterCount(usize, usize),
InvalidParameterIndex(usize),
AppendError,
}
impl PartialEq for Error {
fn eq(&self, other: &Self) -> bool {
match (self, other) {
(Self::DuckDBFailure(e1, s1), Self::DuckDBFailure(e2, s2)) => e1 == e2 && s1 == s2,
(Self::IntegralValueOutOfRange(i1, n1), Self::IntegralValueOutOfRange(i2, n2)) => i1 == i2 && n1 == n2,
(Self::Utf8Error(e1), Self::Utf8Error(e2)) => e1 == e2,
(Self::NulError(e1), Self::NulError(e2)) => e1 == e2,
(Self::InvalidParameterName(n1), Self::InvalidParameterName(n2)) => n1 == n2,
(Self::InvalidPath(p1), Self::InvalidPath(p2)) => p1 == p2,
(Self::ExecuteReturnedResults, Self::ExecuteReturnedResults) => true,
(Self::QueryReturnedNoRows, Self::QueryReturnedNoRows) => true,
(Self::QueryReturnedMoreThanOneRow, Self::QueryReturnedMoreThanOneRow) => true,
(Self::InvalidColumnIndex(i1), Self::InvalidColumnIndex(i2)) => i1 == i2,
(Self::InvalidColumnName(n1), Self::InvalidColumnName(n2)) => n1 == n2,
(Self::InvalidColumnType(i1, n1, t1), Self::InvalidColumnType(i2, n2, t2)) => {
i1 == i2 && t1 == t2 && n1 == n2
}
(Self::StatementChangedRows(n1), Self::StatementChangedRows(n2)) => n1 == n2,
(Self::InvalidParameterCount(i1, n1), Self::InvalidParameterCount(i2, n2)) => i1 == i2 && n1 == n2,
(Self::InvalidParameterIndex(i1), Self::InvalidParameterIndex(i2)) => i1 == i2,
(..) => false,
}
}
}
impl From<str::Utf8Error> for Error {
#[cold]
fn from(err: str::Utf8Error) -> Self {
Self::Utf8Error(err)
}
}
impl From<::std::ffi::NulError> for Error {
#[cold]
fn from(err: ::std::ffi::NulError) -> Self {
Self::NulError(err)
}
}
const UNKNOWN_COLUMN: usize = usize::MAX;
impl From<FromSqlError> for Error {
#[cold]
fn from(err: FromSqlError) -> Self {
match err {
FromSqlError::OutOfRange(val) => Self::IntegralValueOutOfRange(UNKNOWN_COLUMN, val),
#[cfg(feature = "uuid")]
FromSqlError::InvalidUuidSize(_) => {
Self::FromSqlConversionFailure(UNKNOWN_COLUMN, Type::Blob, Box::new(err))
}
FromSqlError::Other(source) => Self::FromSqlConversionFailure(UNKNOWN_COLUMN, Type::Null, source),
_ => Self::FromSqlConversionFailure(UNKNOWN_COLUMN, Type::Null, Box::new(err)),
}
}
}
impl fmt::Display for Error {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match *self {
Self::DuckDBFailure(ref err, None) => err.fmt(f),
Self::DuckDBFailure(_, Some(ref s)) => write!(f, "{s}"),
Self::FromSqlConversionFailure(i, ref t, ref err) => {
if i != UNKNOWN_COLUMN {
write!(f, "Conversion error from type {t} at index: {i}, {err}")
} else {
err.fmt(f)
}
}
Self::IntegralValueOutOfRange(col, val) => {
if col != UNKNOWN_COLUMN {
write!(f, "Integer {val} out of range at index {col}")
} else {
write!(f, "Integer {val} out of range")
}
}
Self::Utf8Error(ref err) => err.fmt(f),
Self::NulError(ref err) => err.fmt(f),
Self::InvalidParameterName(ref name) => write!(f, "Invalid parameter name: {name}"),
Self::InvalidPath(ref p) => write!(f, "Invalid path: {}", p.to_string_lossy()),
Self::ExecuteReturnedResults => {
write!(f, "Execute returned results - did you mean to call query?")
}
Self::QueryReturnedNoRows => write!(f, "Query returned no rows"),
Self::QueryReturnedMoreThanOneRow => write!(f, "Query returned more than one row"),
Self::InvalidColumnIndex(i) => write!(f, "Invalid column index: {i}"),
Self::InvalidColumnName(ref name) => write!(f, "Invalid column name: {name}"),
Self::InvalidColumnType(i, ref name, ref t) => {
write!(f, "Invalid column type {t} at index: {i}, name: {name}")
}
Self::ArrowTypeToDuckdbType(ref name, ref t) => {
write!(f, "Invalid column type {t} , name: {name}")
}
Self::InvalidParameterCount(i1, n1) => {
write!(f, "Wrong number of parameters passed to query. Got {i1}, needed {n1}")
}
Self::InvalidParameterIndex(i) => write!(f, "Invalid parameter index: {i}"),
Self::StatementChangedRows(i) => write!(f, "Query changed {i} rows"),
Self::ToSqlConversionFailure(ref err) => err.fmt(f),
Self::InvalidQuery => write!(f, "Query is not read-only"),
Self::MultipleStatement => write!(f, "Multiple statements provided"),
Self::AppendError => write!(f, "Append error"),
}
}
}
impl error::Error for Error {
fn source(&self) -> Option<&(dyn error::Error + 'static)> {
match *self {
Self::DuckDBFailure(ref err, _) => Some(err),
Self::Utf8Error(ref err) => Some(err),
Self::NulError(ref err) => Some(err),
Self::IntegralValueOutOfRange(..)
| Self::InvalidParameterName(_)
| Self::ExecuteReturnedResults
| Self::QueryReturnedNoRows
| Self::QueryReturnedMoreThanOneRow
| Self::InvalidColumnIndex(_)
| Self::InvalidColumnName(_)
| Self::InvalidColumnType(..)
| Self::InvalidPath(_)
| Self::InvalidParameterCount(..)
| Self::InvalidParameterIndex(_)
| Self::StatementChangedRows(_)
| Self::InvalidQuery
| Self::AppendError
| Self::ArrowTypeToDuckdbType(..)
| Self::MultipleStatement => None,
Self::FromSqlConversionFailure(_, _, ref err) | Self::ToSqlConversionFailure(ref err) => Some(&**err),
}
}
}
#[inline]
fn error_from_duckdb_code(code: ffi::duckdb_state, message: Option<String>) -> Result<()> {
Err(Error::DuckDBFailure(ffi::Error::new(code), message))
}
#[cold]
#[inline]
pub fn result_from_duckdb_appender(code: ffi::duckdb_state, appender: *mut ffi::duckdb_appender) -> Result<()> {
if code == ffi::DuckDBSuccess {
return Ok(());
}
unsafe {
let message = if (*appender).is_null() {
Some("appender is null".to_string())
} else {
let c_err = ffi::duckdb_appender_error(*appender);
let message = Some(CStr::from_ptr(c_err).to_string_lossy().to_string());
ffi::duckdb_appender_destroy(appender);
message
};
error_from_duckdb_code(code, message)
}
}
#[cold]
#[inline]
pub fn result_from_duckdb_prepare(code: ffi::duckdb_state, mut prepare: ffi::duckdb_prepared_statement) -> Result<()> {
if code == ffi::DuckDBSuccess {
return Ok(());
}
unsafe {
let message = if prepare.is_null() {
Some("prepare is null".to_string())
} else {
let c_err = ffi::duckdb_prepare_error(prepare);
let message = Some(CStr::from_ptr(c_err).to_string_lossy().to_string());
ffi::duckdb_destroy_prepare(&mut prepare);
message
};
error_from_duckdb_code(code, message)
}
}
#[cold]
#[inline]
pub fn result_from_duckdb_arrow(code: ffi::duckdb_state, mut out: ffi::duckdb_arrow) -> Result<()> {
if code == ffi::DuckDBSuccess {
return Ok(());
}
unsafe {
let message = if out.is_null() {
Some("out is null".to_string())
} else {
let c_err = ffi::duckdb_query_arrow_error(out);
let message = Some(CStr::from_ptr(c_err).to_string_lossy().to_string());
ffi::duckdb_destroy_arrow(&mut out);
message
};
error_from_duckdb_code(code, message)
}
}
#[cold]
#[inline]
pub fn result_from_duckdb_extract(
num_statements: ffi::idx_t,
mut extracted: ffi::duckdb_extracted_statements,
) -> Result<()> {
if num_statements > 0 {
return Ok(());
}
unsafe {
let message = if extracted.is_null() {
Some("extracted statements are null".to_string())
} else {
let c_err = ffi::duckdb_extract_statements_error(extracted);
let message = if c_err.is_null() {
None
} else {
Some(CStr::from_ptr(c_err).to_string_lossy().to_string())
};
ffi::duckdb_destroy_extracted(&mut extracted);
message
};
error_from_duckdb_code(ffi::DuckDBError, message)
}
}