diesel/pg/types/
primitives.rs1use std::io::prelude::*;
2
3#[cfg(feature = "postgres_backend")]
4use crate::deserialize::FromSqlRef;
5use crate::deserialize::{self, FromSql, Queryable};
6use crate::pg::{Pg, PgValue};
7use crate::serialize::{self, IsNull, Output, ToSql};
8use crate::sql_types;
9
10#[cfg(feature = "postgres_backend")]
11impl FromSql<sql_types::Bool, Pg> for bool {
12 fn from_sql(bytes: PgValue<'_>) -> deserialize::Result<Self> {
13 let first_byte = bytes
14 .as_bytes()
15 .first()
16 .ok_or("Received an empty response from the server")?;
17 Ok(*first_byte != 0)
18 }
19}
20
21#[cfg(feature = "postgres_backend")]
22impl ToSql<sql_types::Bool, Pg> for bool {
23 fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
24 out.write_all(&[*self as u8])
25 .map(|_| IsNull::No)
26 .map_err(Into::into)
27 }
28}
29
30#[cfg(feature = "postgres_backend")]
31impl FromSql<sql_types::CChar, Pg> for u8 {
32 fn from_sql(bytes: PgValue<'_>) -> deserialize::Result<Self> {
33 let first_byte = bytes
34 .as_bytes()
35 .first()
36 .ok_or("Received an empty response from the server")?;
37 Ok(*first_byte)
38 }
39}
40
41#[cfg(feature = "postgres_backend")]
42impl ToSql<sql_types::CChar, Pg> for u8 {
43 fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
44 out.write_all(&[*self])
45 .map(|_| IsNull::No)
46 .map_err(Into::into)
47 }
48}
49
50#[cfg(feature = "postgres_backend")]
56impl FromSql<sql_types::Text, Pg> for *const str {
57 fn from_sql(value: PgValue<'_>) -> deserialize::Result<Self> {
58 use core::str;
59 let string = str::from_utf8(value.as_bytes())?;
60 Ok(string as *const _)
61 }
62}
63
64#[cfg(feature = "postgres_backend")]
65impl<'a> FromSqlRef<'a, sql_types::Text, Pg> for &'a str {
66 fn from_sql(value: &'a mut PgValue<'_>) -> deserialize::Result<Self> {
67 Ok(core::str::from_utf8(value.as_bytes())?)
68 }
69}
70
71#[cfg(feature = "postgres_backend")]
72impl Queryable<sql_types::VarChar, Pg> for *const str {
73 type Row = Self;
74
75 fn build(row: Self::Row) -> deserialize::Result<Self> {
76 Ok(row)
77 }
78}
79
80#[cfg(feature = "postgres_backend")]
81impl ToSql<sql_types::Citext, Pg> for String {
82 fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
83 out.write_all(self.as_bytes())?;
84 Ok(IsNull::No)
85 }
86}
87
88#[cfg(feature = "postgres_backend")]
89impl ToSql<sql_types::Citext, Pg> for str {
90 fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Pg>) -> serialize::Result {
91 out.write_all(self.as_bytes())?;
92 Ok(IsNull::No)
93 }
94}
95
96#[cfg(feature = "postgres_backend")]
97impl FromSql<sql_types::Citext, Pg> for String {
98 fn from_sql(value: PgValue<'_>) -> deserialize::Result<Self> {
99 let string = String::from_utf8(value.as_bytes().to_vec())?;
100 Ok(string)
101 }
102}
103
104#[cfg(feature = "postgres_backend")]
110impl FromSql<sql_types::Binary, Pg> for *const [u8] {
111 fn from_sql(value: PgValue<'_>) -> deserialize::Result<Self> {
112 Ok(value.as_bytes() as *const _)
113 }
114}
115
116impl<'a> FromSqlRef<'a, sql_types::Binary, Pg> for &'a [u8] {
117 fn from_sql(bytes: &'a mut PgValue<'_>) -> deserialize::Result<Self> {
118 Ok(bytes.as_bytes())
119 }
120}
121
122#[cfg(feature = "postgres_backend")]
123impl Queryable<sql_types::Binary, Pg> for *const [u8] {
124 type Row = Self;
125
126 fn build(row: Self::Row) -> deserialize::Result<Self> {
127 Ok(row)
128 }
129}
130
131#[cfg(test)]
132mod tests {
133 use super::*;
134
135 #[diesel_test_helper::test]
136 fn cchar_to_sql() {
137 use crate::query_builder::bind_collector::ByteWrapper;
138
139 let mut buffer = Vec::new();
140 let mut bytes = Output::test(ByteWrapper(&mut buffer));
141 ToSql::<sql_types::CChar, Pg>::to_sql(&b'A', &mut bytes).unwrap();
142 ToSql::<sql_types::CChar, Pg>::to_sql(&b'\xc4', &mut bytes).unwrap();
143 assert_eq!(buffer, vec![65u8, 196u8]);
144 }
145
146 #[diesel_test_helper::test]
147 fn cchar_from_sql() {
148 let result = <u8 as FromSql<sql_types::CChar, Pg>>::from_nullable_sql(None);
149 assert_eq!(
150 result.unwrap_err().to_string(),
151 "Unexpected null for non-null column"
152 );
153 }
154
155 #[diesel_test_helper::test]
156 fn bool_to_sql() {
157 use crate::query_builder::bind_collector::ByteWrapper;
158
159 let mut buffer = Vec::new();
160 let mut bytes = Output::test(ByteWrapper(&mut buffer));
161 ToSql::<sql_types::Bool, Pg>::to_sql(&true, &mut bytes).unwrap();
162 ToSql::<sql_types::Bool, Pg>::to_sql(&false, &mut bytes).unwrap();
163 assert_eq!(buffer, vec![1u8, 0u8]);
164 }
165
166 #[diesel_test_helper::test]
167 fn no_bool_from_sql() {
168 let result = <bool as FromSql<sql_types::Bool, Pg>>::from_nullable_sql(None);
169 assert_eq!(
170 result.unwrap_err().to_string(),
171 "Unexpected null for non-null column"
172 );
173 }
174}