1use std::fmt::{Result, Write};
16use std::num::FpCategory;
17
18use super::{DataType, DatumRef, ScalarRefImpl};
19use crate::dispatch_scalar_ref_variants;
20
21pub trait ToText {
40 fn write<W: Write>(&self, f: &mut W) -> Result;
44
45 fn write_with_type<W: Write>(&self, _ty: &DataType, f: &mut W) -> Result;
47
48 fn to_text_with_type(&self, ty: &DataType) -> String {
50 let mut s = String::new();
51 self.write_with_type(ty, &mut s).unwrap();
52 s
53 }
54
55 fn to_text(&self) -> String {
64 let mut s = String::new();
65 self.write(&mut s).unwrap();
66 s
67 }
68}
69
70macro_rules! implement_using_to_string {
71 ($({ $scalar_type:ty , $data_type:ident} ),*) => {
72 $(
73 impl ToText for $scalar_type {
74 fn write<W: Write>(&self, f: &mut W) -> Result {
75 write!(f, "{self}")
76 }
77 fn write_with_type<W: Write>(&self, ty: &DataType, f: &mut W) -> Result {
78 match ty {
79 DataType::$data_type => self.write(f),
80 _ => unreachable!(),
81 }
82 }
83 }
84 )*
85 };
86}
87
88macro_rules! implement_using_itoa {
89 ($({ $scalar_type:ty , $data_type:ident} ),*) => {
90 $(
91 impl ToText for $scalar_type {
92 fn write<W: Write>(&self, f: &mut W) -> Result {
93 write!(f, "{}", itoa::Buffer::new().format(*self))
94 }
95 fn write_with_type<W: Write>(&self, ty: &DataType, f: &mut W) -> Result {
96 match ty {
97 DataType::$data_type => self.write(f),
98 _ => unreachable!(),
99 }
100 }
101 }
102 )*
103 };
104}
105
106implement_using_to_string! {
107 { String ,Varchar },
108 { &str ,Varchar}
109}
110
111implement_using_itoa! {
112 { i16, Int16 },
113 { i32, Int32 },
114 { i64, Int64 }
115}
116
117macro_rules! implement_using_ryu {
118 ($({ $scalar_type:ty, $data_type:ident } ),*) => {
119 $(
120 impl ToText for $scalar_type {
121 fn write<W: Write>(&self, f: &mut W) -> Result {
122 let inner = self.0;
123 match inner.classify() {
124 FpCategory::Infinite if inner.is_sign_negative() => write!(f, "-Infinity"),
125 FpCategory::Infinite => write!(f, "Infinity"),
126 FpCategory::Zero if inner.is_sign_negative() => write!(f, "-0"),
127 FpCategory::Nan => write!(f, "NaN"),
128 _ => {
129 let mut buf = ryu::Buffer::new();
130 let mut s = buf.format_finite(self.0);
131 if let Some(trimmed) = s.strip_suffix(".0") {
132 s = trimmed;
133 }
134 if let Some(mut idx) = s.as_bytes().iter().position(|x| *x == b'e') {
135 idx += 1;
136 write!(f, "{}", &s[..idx])?;
137 if s.as_bytes()[idx] == b'-' {
138 write!(f, "-")?;
139 idx += 1;
140 } else {
141 write!(f, "+")?;
142 }
143 if idx + 1 == s.len() {
144 write!(f, "0")?;
145 }
146 write!(f, "{}", &s[idx..])?;
147 } else {
148 write!(f, "{}", s)?;
149 }
150 Ok(())
151 }
152 }
153 }
154 fn write_with_type<W: Write>(&self, ty: &DataType, f: &mut W) -> Result {
155 match ty {
156 DataType::$data_type => self.write(f),
157 _ => unreachable!(),
158 }
159 }
160 }
161 )*
162 };
163}
164
165implement_using_ryu! {
166 { crate::types::F32, Float32 },
167 { crate::types::F64, Float64 }
168}
169
170impl ToText for bool {
171 fn write<W: Write>(&self, f: &mut W) -> Result {
172 if *self {
173 write!(f, "t")
174 } else {
175 write!(f, "f")
176 }
177 }
178
179 fn write_with_type<W: Write>(&self, ty: &DataType, f: &mut W) -> Result {
180 match ty {
181 DataType::Boolean => self.write(f),
182 _ => unreachable!(),
183 }
184 }
185}
186
187impl ToText for &[u8] {
188 fn write<W: Write>(&self, f: &mut W) -> Result {
189 write!(f, "\\x{}", hex::encode(self))
190 }
191
192 fn write_with_type<W: Write>(&self, ty: &DataType, f: &mut W) -> Result {
193 match ty {
194 DataType::Bytea => self.write(f),
195 _ => unreachable!(),
196 }
197 }
198}
199
200impl ToText for ScalarRefImpl<'_> {
201 fn write<W: Write>(&self, f: &mut W) -> Result {
202 dispatch_scalar_ref_variants!(self, v, { v.write(f) })
203 }
204
205 fn write_with_type<W: Write>(&self, ty: &DataType, f: &mut W) -> Result {
206 dispatch_scalar_ref_variants!(self, v, { v.write_with_type(ty, f) })
207 }
208}
209
210impl ToText for DatumRef<'_> {
211 fn write<W: Write>(&self, f: &mut W) -> Result {
212 match self {
213 Some(data) => data.write(f),
214 None => write!(f, "NULL"),
215 }
216 }
217
218 fn write_with_type<W: Write>(&self, ty: &DataType, f: &mut W) -> Result {
219 match self {
220 Some(data) => data.write_with_type(ty, f),
221 None => write!(f, "NULL"),
222 }
223 }
224}
225
226#[cfg(test)]
227mod tests {
228 use crate::types::ToText;
229 use crate::types::ordered_float::OrderedFloat;
230
231 #[test]
232 fn test_float_to_text() {
233 let ret: OrderedFloat<f64> = OrderedFloat::<f64>::from(1.234567890123456);
235 tracing::info!("ret: {}", ret.to_text());
236 assert_eq!("1.234567890123456".to_owned(), ret.to_text());
237
238 let ret: OrderedFloat<f32> = OrderedFloat::<f32>::from(1.234567);
240 assert_eq!("1.234567".to_owned(), ret.to_text());
241 }
242}