#![forbid(unsafe_code)]
#[allow(clippy::single_component_path_imports)]
use qusql_sqlx_type_macro;
pub use crate::qusql_sqlx_type_macro::{query, query_as};
#[doc(hidden)]
pub struct Integer;
#[doc(hidden)]
pub struct Float;
#[doc(hidden)]
pub struct Timestamp;
#[doc(hidden)]
pub struct DateTime;
#[doc(hidden)]
pub struct Date;
#[doc(hidden)]
pub struct Time;
#[doc(hidden)]
pub struct Any;
#[doc(hidden)]
pub struct Uuid;
#[doc(hidden)]
pub trait ArgIn<T> {}
pub trait ArgOut<T, const IDX: usize> {}
macro_rules! arg_io {
( $dst: ty, $t: ty ) => {
impl ArgIn<$dst> for $t {}
impl ArgIn<$dst> for &$t {}
impl ArgIn<$dst> for Option<$t> {}
impl ArgIn<$dst> for Option<&$t> {}
impl ArgIn<$dst> for &Option<$t> {}
impl ArgIn<$dst> for &Option<&$t> {}
impl ArgIn<Option<$dst>> for $t {}
impl ArgIn<Option<$dst>> for &$t {}
impl ArgIn<Option<$dst>> for Option<$t> {}
impl ArgIn<Option<$dst>> for Option<&$t> {}
impl ArgIn<Option<$dst>> for &Option<$t> {}
impl ArgIn<Option<$dst>> for &Option<&$t> {}
impl<const IDX: usize> ArgOut<$dst, IDX> for $t {}
impl<const IDX: usize> ArgOut<Option<$dst>, IDX> for Option<$t> {}
impl<const IDX: usize> ArgOut<$dst, IDX> for Option<$t> {}
impl<const IDX: usize> ArgOut<Option<$dst>, IDX> for $t {}
};
}
arg_io!(Any, u64);
arg_io!(Any, i64);
arg_io!(Any, u32);
arg_io!(Any, i32);
arg_io!(Any, u16);
arg_io!(Any, i16);
arg_io!(Any, u8);
arg_io!(Any, i8);
arg_io!(Any, String);
arg_io!(Any, f64);
arg_io!(Any, f32);
arg_io!(Any, &str);
arg_io!(Integer, u64);
arg_io!(Integer, i64);
arg_io!(Integer, u32);
arg_io!(Integer, i32);
arg_io!(Integer, u16);
arg_io!(Integer, i16);
arg_io!(Integer, u8);
arg_io!(Integer, i8);
arg_io!(String, String);
arg_io!(Float, f64);
arg_io!(Float, f32);
arg_io!(u64, u64);
arg_io!(i64, i64);
arg_io!(u32, u32);
arg_io!(i32, i32);
arg_io!(u16, u16);
arg_io!(i16, i16);
arg_io!(u8, u8);
arg_io!(i8, i8);
arg_io!(bool, bool);
arg_io!(f32, f32);
arg_io!(f64, f64);
arg_io!(&str, &str);
arg_io!(&str, String);
arg_io!(&str, std::borrow::Cow<'_, str>);
arg_io!(&[u8], &[u8]);
arg_io!(&[u8], Vec<u8>);
arg_io!(Vec<u8>, Vec<u8>);
arg_io!(Any, Vec<u8>);
arg_io!(Any, Vec<i32>);
arg_io!(Any, Vec<i64>);
arg_io!(Any, Vec<String>);
arg_io!(Any, Vec<Vec<u8>>);
arg_io!(Any, Vec<bool>);
arg_io!(Any, Vec<f64>);
impl ArgIn<Any> for &[u8] {}
impl ArgIn<Option<Any>> for &[u8] {}
impl ArgIn<Any> for &[i32] {}
impl ArgIn<Option<Any>> for &[i32] {}
impl ArgIn<Any> for &[i64] {}
impl ArgIn<Option<Any>> for &[i64] {}
impl ArgIn<Any> for &[String] {}
impl ArgIn<Option<Any>> for &[String] {}
impl ArgIn<Option<Any>> for Option<&[String]> {}
impl ArgIn<Option<Any>> for Option<&[i32]> {}
impl ArgIn<Option<Any>> for Option<&[i64]> {}
impl ArgIn<Any> for &[f64] {}
impl ArgIn<Option<Any>> for &[f64] {}
impl ArgIn<Option<Any>> for Option<&[f64]> {}
impl ArgIn<Any> for &[bool] {}
impl ArgIn<Option<Any>> for &[bool] {}
impl<T> ArgIn<std::convert::Infallible> for T {}
impl<T> ArgIn<Option<std::convert::Infallible>> for T {}
arg_io!(Timestamp, chrono::NaiveDateTime);
arg_io!(DateTime, chrono::NaiveDateTime);
arg_io!(chrono::DateTime<chrono::Utc>, chrono::DateTime<chrono::Utc>);
arg_io!(Timestamp, chrono::DateTime<chrono::Utc>);
arg_io!(Date, chrono::NaiveDate);
arg_io!(Uuid, String);
arg_io!(Uuid, &str);
#[cfg(feature = "uuid")]
mod uuid_support {
use super::*;
arg_io!(Uuid, uuid::Uuid);
arg_io!(uuid::Uuid, uuid::Uuid);
arg_io!(Any, Vec<uuid::Uuid>);
arg_io!(Any, Vec<Option<uuid::Uuid>>);
impl ArgIn<Any> for &[uuid::Uuid] {}
impl ArgIn<Option<Any>> for &[uuid::Uuid] {}
impl ArgIn<Option<Any>> for Option<&[uuid::Uuid]> {}
}
#[cfg(feature = "uuid")]
#[doc(hidden)]
pub type UuidValue = uuid::Uuid;
#[cfg(not(feature = "uuid"))]
#[doc(hidden)]
pub type UuidValue = String;
#[cfg(feature = "json")]
mod json_support {
use super::*;
arg_io!(String, serde_json::Value);
arg_io!(serde_json::Value, serde_json::Value);
arg_io!(&str, serde_json::Value);
arg_io!(Any, Vec<serde_json::Value>);
impl ArgIn<Any> for &[serde_json::Value] {}
impl ArgIn<Option<Any>> for &[serde_json::Value] {}
}
#[cfg(feature = "json")]
#[doc(hidden)]
pub type JsonValue = serde_json::Value;
#[doc(hidden)]
pub fn check_arg<T, T2: ArgIn<T>>(_: &T2) {}
#[doc(hidden)]
pub fn check_arg_list_hack<T, T2: ArgIn<T>>(_: &[T2]) {}
#[doc(hidden)]
pub fn arg_out<T, T2: ArgOut<T, IDX>, const IDX: usize>(v: T2) -> T2 {
v
}
#[doc(hidden)]
pub fn convert_list_query(query: &str, list_sizes: &[usize]) -> String {
let mut query_iter = query.split("_LIST_");
let mut query = query_iter.next().expect("None empty query").to_string();
for size in list_sizes {
if *size == 0 {
query.push_str("NULL");
} else {
for i in 0..*size {
if i == 0 {
query.push('?');
} else {
query.push_str(", ?");
}
}
}
query.push_str(query_iter.next().expect("More _LIST_ in query"));
}
if query_iter.next().is_some() {
panic!("Too many _LIST_ in query");
}
query
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_convert_list_query() {
assert_eq!(
&convert_list_query("FOO (_LIST_) X _LIST_ O _LIST_ BAR (_LIST_)", &[0, 1, 2, 3]),
"FOO (NULL) X ? O ?, ? BAR (?, ?, ?)"
);
}
}