|
1 | 1 | //! Convert most of the [Time Strings](http://sqlite.org/lang_datefunc.html) to chrono types. |
2 | 2 |
|
3 | | -use chrono::{DateTime, Local, NaiveDate, NaiveDateTime, NaiveTime, TimeZone, Utc}; |
| 3 | +use chrono::{DateTime, Duration, Local, NaiveDate, NaiveDateTime, NaiveTime, TimeZone, Utc}; |
| 4 | +use num_integer::Integer; |
4 | 5 |
|
5 | 6 | use crate::{ |
6 | 7 | types::{FromSql, FromSqlError, FromSqlResult, TimeUnit, ToSql, ToSqlOutput, ValueRef}, |
7 | 8 | Result, |
8 | 9 | }; |
9 | 10 |
|
| 11 | +use super::Value; |
| 12 | + |
10 | 13 | /// ISO 8601 calendar date without timezone => "YYYY-MM-DD" |
11 | 14 | impl ToSql for NaiveDate { |
12 | 15 | #[inline] |
@@ -126,13 +129,55 @@ impl FromSql for DateTime<Local> { |
126 | 129 | } |
127 | 130 | } |
128 | 131 |
|
| 132 | +impl FromSql for Duration { |
| 133 | + fn column_result(value: ValueRef<'_>) -> FromSqlResult<Self> { |
| 134 | + match value { |
| 135 | + ValueRef::Interval { months, days, nanos } => { |
| 136 | + let days = days + (months * 30); |
| 137 | + let (additional_seconds, nanos) = nanos.div_mod_floor(&NANOS_PER_SECOND); |
| 138 | + let seconds = additional_seconds + (i64::from(days) * 24 * 3600); |
| 139 | + |
| 140 | + match nanos.try_into() { |
| 141 | + Ok(nanos) => { |
| 142 | + if let Some(duration) = Duration::new(seconds, nanos) { |
| 143 | + Ok(duration) |
| 144 | + } else { |
| 145 | + Err(FromSqlError::Other("Invalid duration".into())) |
| 146 | + } |
| 147 | + } |
| 148 | + Err(err) => Err(FromSqlError::Other(format!("Invalid duration: {err}").into())), |
| 149 | + } |
| 150 | + } |
| 151 | + _ => Err(FromSqlError::InvalidType), |
| 152 | + } |
| 153 | + } |
| 154 | +} |
| 155 | + |
| 156 | +const DAYS_PER_MONTH: i64 = 30; |
| 157 | +const SECONDS_PER_DAY: i64 = 24 * 3600; |
| 158 | +const NANOS_PER_SECOND: i64 = 1_000_000_000; |
| 159 | +const NANOS_PER_DAY: i64 = SECONDS_PER_DAY * NANOS_PER_SECOND; |
| 160 | + |
| 161 | +impl ToSql for Duration { |
| 162 | + fn to_sql(&self) -> Result<ToSqlOutput<'_>> { |
| 163 | + let nanos = self.num_nanoseconds().unwrap(); |
| 164 | + let (days, nanos) = nanos.div_mod_floor(&NANOS_PER_DAY); |
| 165 | + let (months, days) = days.div_mod_floor(&DAYS_PER_MONTH); |
| 166 | + Ok(ToSqlOutput::Owned(Value::Interval { |
| 167 | + months: months.try_into().unwrap(), |
| 168 | + days: days.try_into().unwrap(), |
| 169 | + nanos, |
| 170 | + })) |
| 171 | + } |
| 172 | +} |
| 173 | + |
129 | 174 | #[cfg(test)] |
130 | 175 | mod test { |
131 | 176 | use crate::{ |
132 | | - types::{FromSql, ValueRef}, |
| 177 | + types::{FromSql, ToSql, ToSqlOutput, ValueRef}, |
133 | 178 | Connection, Result, |
134 | 179 | }; |
135 | | - use chrono::{DateTime, Duration, Local, NaiveDate, NaiveDateTime, NaiveTime, TimeZone, Utc}; |
| 180 | + use chrono::{DateTime, Duration, Local, NaiveDate, NaiveDateTime, NaiveTime, TimeDelta, TimeZone, Utc}; |
136 | 181 |
|
137 | 182 | fn checked_memory_handle() -> Result<Connection> { |
138 | 183 | let db = Connection::open_in_memory()?; |
@@ -216,6 +261,35 @@ mod test { |
216 | 261 | Ok(()) |
217 | 262 | } |
218 | 263 |
|
| 264 | + #[test] |
| 265 | + fn test_time_delta_roundtrip() { |
| 266 | + roundtrip_type(TimeDelta::new(3600, 0).unwrap()); |
| 267 | + roundtrip_type(TimeDelta::new(3600, 1000).unwrap()); |
| 268 | + } |
| 269 | + |
| 270 | + #[test] |
| 271 | + fn test_time_delta() -> Result<()> { |
| 272 | + let db = checked_memory_handle()?; |
| 273 | + let td = TimeDelta::new(3600, 0).unwrap(); |
| 274 | + |
| 275 | + let row: Result<TimeDelta> = db.query_row("SELECT ?", [td], |row| Ok(row.get(0)))?; |
| 276 | + |
| 277 | + assert_eq!(row.unwrap(), td); |
| 278 | + |
| 279 | + Ok(()) |
| 280 | + } |
| 281 | + |
| 282 | + fn roundtrip_type<T: FromSql + ToSql + Eq + std::fmt::Debug>(td: T) { |
| 283 | + let sqled = td.to_sql().unwrap(); |
| 284 | + let value = match sqled { |
| 285 | + ToSqlOutput::Borrowed(v) => v, |
| 286 | + ToSqlOutput::Owned(ref v) => ValueRef::from(v), |
| 287 | + }; |
| 288 | + let reversed = FromSql::column_result(value).unwrap(); |
| 289 | + |
| 290 | + assert_eq!(td, reversed); |
| 291 | + } |
| 292 | + |
219 | 293 | #[test] |
220 | 294 | fn test_date_time_local() -> Result<()> { |
221 | 295 | let db = checked_memory_handle()?; |
|
0 commit comments