use byteorder::{BigEndian, ByteOrder};
use std::num::{NonZeroI16, NonZeroI32, NonZeroI64};
use crate::decode::Decode;
use crate::encode::{Encode, IsNull};
use crate::error::BoxDynError;
use crate::types::Type;
use crate::{PgArgumentBuffer, PgHasArrayType, PgTypeInfo, PgValueFormat, PgValueRef, Postgres};
fn int_decode(value: PgValueRef<'_>) -> Result<i64, BoxDynError> {
Ok(match value.format() {
PgValueFormat::Text => value.as_str()?.parse()?,
PgValueFormat::Binary => {
let buf = value.as_bytes()?;
match buf.len() {
0 => {
return Err("Value Buffer found empty while decoding to integer type".into());
}
buf_len @ 9.. => {
return Err(format!(
"Value Buffer exceeds 8 bytes while decoding to integer type. Buffer size = {} bytes ", buf_len
)
.into());
}
_ => {}
}
BigEndian::read_int(buf, buf.len())
}
})
}
impl Type<Postgres> for i8 {
fn type_info() -> PgTypeInfo {
PgTypeInfo::CHAR
}
}
impl PgHasArrayType for i8 {
fn array_type_info() -> PgTypeInfo {
PgTypeInfo::CHAR_ARRAY
}
}
impl Encode<'_, Postgres> for i8 {
fn encode_by_ref(&self, buf: &mut PgArgumentBuffer) -> Result<IsNull, BoxDynError> {
buf.extend(&self.to_be_bytes());
Ok(IsNull::No)
}
}
impl Decode<'_, Postgres> for i8 {
fn decode(value: PgValueRef<'_>) -> Result<Self, BoxDynError> {
match value.format() {
PgValueFormat::Binary => int_decode(value)?.try_into().map_err(Into::into),
PgValueFormat::Text => {
let text = value.as_str()?;
if text.is_empty() {
return Ok(0);
}
if text.starts_with('\\') {
return Ok(i8::from_str_radix(text.trim_start_matches('\\'), 8)?);
}
#[allow(clippy::cast_possible_wrap)]
Ok(text.as_bytes()[0] as i8)
}
}
}
}
impl Type<Postgres> for i16 {
fn type_info() -> PgTypeInfo {
PgTypeInfo::INT2
}
}
impl PgHasArrayType for i16 {
fn array_type_info() -> PgTypeInfo {
PgTypeInfo::INT2_ARRAY
}
}
impl Encode<'_, Postgres> for i16 {
fn encode_by_ref(&self, buf: &mut PgArgumentBuffer) -> Result<IsNull, BoxDynError> {
buf.extend(&self.to_be_bytes());
Ok(IsNull::No)
}
}
impl Decode<'_, Postgres> for i16 {
fn decode(value: PgValueRef<'_>) -> Result<Self, BoxDynError> {
int_decode(value)?.try_into().map_err(Into::into)
}
}
impl Type<Postgres> for i32 {
fn type_info() -> PgTypeInfo {
PgTypeInfo::INT4
}
}
impl PgHasArrayType for i32 {
fn array_type_info() -> PgTypeInfo {
PgTypeInfo::INT4_ARRAY
}
}
impl Encode<'_, Postgres> for i32 {
fn encode_by_ref(&self, buf: &mut PgArgumentBuffer) -> Result<IsNull, BoxDynError> {
buf.extend(&self.to_be_bytes());
Ok(IsNull::No)
}
}
impl Decode<'_, Postgres> for i32 {
fn decode(value: PgValueRef<'_>) -> Result<Self, BoxDynError> {
int_decode(value)?.try_into().map_err(Into::into)
}
}
impl Type<Postgres> for i64 {
fn type_info() -> PgTypeInfo {
PgTypeInfo::INT8
}
}
impl PgHasArrayType for i64 {
fn array_type_info() -> PgTypeInfo {
PgTypeInfo::INT8_ARRAY
}
}
impl Encode<'_, Postgres> for i64 {
fn encode_by_ref(&self, buf: &mut PgArgumentBuffer) -> Result<IsNull, BoxDynError> {
buf.extend(&self.to_be_bytes());
Ok(IsNull::No)
}
}
impl Decode<'_, Postgres> for i64 {
fn decode(value: PgValueRef<'_>) -> Result<Self, BoxDynError> {
int_decode(value)
}
}
impl PgHasArrayType for NonZeroI16 {
fn array_type_info() -> PgTypeInfo {
PgTypeInfo::INT2_ARRAY
}
}
impl PgHasArrayType for NonZeroI32 {
fn array_type_info() -> PgTypeInfo {
PgTypeInfo::INT4_ARRAY
}
}
impl PgHasArrayType for NonZeroI64 {
fn array_type_info() -> PgTypeInfo {
PgTypeInfo::INT8_ARRAY
}
}