use crate::{FromDatum, IntoDatum, direct_function_call, direct_function_call_as_datum, pg_sys};
use core::ffi::CStr;
use pgrx_pg_sys::PgTryBuilder;
use pgrx_pg_sys::errcodes::PgSqlErrorCode;
use pgrx_sql_entity_graph::metadata::{
ArgumentError, ReturnsError, ReturnsRef, SqlMappingRef, SqlTranslatable,
};
use serde::de::{Error, Visitor};
use serde::{Deserialize, Deserializer, Serialize, Serializer};
use std::fmt;
use std::ops::Deref;
#[derive(Debug, Clone, Ord, PartialOrd, Eq, PartialEq)]
pub struct Inet(pub String);
impl Deref for Inet {
type Target = str;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl Serialize for Inet {
fn serialize<S>(&self, serializer: S) -> Result<<S as Serializer>::Ok, <S as Serializer>::Error>
where
S: Serializer,
{
serializer.serialize_str(&self.0)
}
}
impl<'de> Deserialize<'de> for Inet {
fn deserialize<D>(deserializer: D) -> Result<Self, <D as Deserializer<'de>>::Error>
where
D: Deserializer<'de>,
{
struct InetVisitor;
impl<'de> Visitor<'de> for InetVisitor {
type Value = Inet;
fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
formatter.write_str("a quoted JSON string in proper inet form")
}
fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
where
E: Error,
{
self.visit_string(v.to_owned())
}
fn visit_string<E>(self, v: String) -> Result<Self::Value, E>
where
E: Error,
{
PgTryBuilder::new(|| {
let datum = Inet(v.clone()).into_datum().unwrap();
unsafe {
pg_sys::pfree(datum.cast_mut_ptr());
}
Ok(Inet(v.clone()))
})
.catch_when(PgSqlErrorCode::ERRCODE_INVALID_TEXT_REPRESENTATION, |_| {
Err(Error::custom(format!("invalid inet value: {v}")))
})
.execute()
}
}
deserializer.deserialize_str(InetVisitor)
}
}
impl FromDatum for Inet {
unsafe fn from_polymorphic_datum(
datum: pg_sys::Datum,
is_null: bool,
_typoid: pg_sys::Oid,
) -> Option<Inet> {
if is_null {
None
} else {
let cstr = direct_function_call::<&CStr>(pg_sys::inet_out, &[Some(datum)]);
Some(Inet(
cstr.unwrap().to_str().expect("unable to convert &cstr inet into &str").to_owned(),
))
}
}
}
impl IntoDatum for Inet {
fn into_datum(self) -> Option<pg_sys::Datum> {
let cstr = alloc::ffi::CString::new(self.0).expect("failed to convert inet into CString");
unsafe { direct_function_call_as_datum(pg_sys::inet_in, &[cstr.as_c_str().into_datum()]) }
}
fn type_oid() -> pg_sys::Oid {
pg_sys::INETOID
}
}
impl From<String> for Inet {
fn from(val: String) -> Self {
Inet(val)
}
}
unsafe impl SqlTranslatable for Inet {
const TYPE_IDENT: &'static str = crate::pgrx_resolved_type!(Inet);
const TYPE_ORIGIN: pgrx_sql_entity_graph::metadata::TypeOrigin =
pgrx_sql_entity_graph::metadata::TypeOrigin::External;
const ARGUMENT_SQL: Result<SqlMappingRef, ArgumentError> = Ok(SqlMappingRef::literal("inet"));
const RETURN_SQL: Result<ReturnsRef, ReturnsError> =
Ok(ReturnsRef::One(SqlMappingRef::literal("inet")));
}