rusqlite/src/types/url.rs

83 lines
2.5 KiB
Rust
Raw Normal View History

2020-11-22 16:34:03 +08:00
//! [`ToSql`] and [`FromSql`] implementation for [`url::Url`].
2019-03-10 11:16:37 +08:00
use crate::types::{FromSql, FromSqlError, FromSqlResult, ToSql, ToSqlOutput, ValueRef};
2019-03-20 03:45:04 +08:00
use crate::Result;
use url::Url;
2019-03-10 11:16:37 +08:00
/// Serialize `Url` to text.
impl ToSql for Url {
#[inline]
2019-03-10 11:16:37 +08:00
fn to_sql(&self) -> Result<ToSqlOutput<'_>> {
Ok(ToSqlOutput::from(self.as_str()))
}
}
/// Deserialize text to `Url`.
impl FromSql for Url {
#[inline]
2019-03-10 11:16:37 +08:00
fn column_result(value: ValueRef<'_>) -> FromSqlResult<Self> {
match value {
ValueRef::Text(s) => {
let s = std::str::from_utf8(s).map_err(|e| FromSqlError::Other(Box::new(e)))?;
Url::parse(s).map_err(|e| FromSqlError::Other(Box::new(e)))
}
_ => Err(FromSqlError::InvalidType),
2019-03-10 11:16:37 +08:00
}
}
}
#[cfg(test)]
mod test {
2019-03-20 03:45:04 +08:00
use crate::{params, Connection, Error, Result};
use url::{ParseError, Url};
2019-03-10 11:16:37 +08:00
2020-11-06 05:14:00 +08:00
fn checked_memory_handle() -> Result<Connection> {
let db = Connection::open_in_memory()?;
db.execute_batch("CREATE TABLE urls (i INTEGER, v TEXT)")?;
Ok(db)
2019-03-10 11:16:37 +08:00
}
fn get_url(db: &Connection, id: i64) -> Result<Url> {
2019-03-20 03:45:04 +08:00
db.query_row("SELECT v FROM urls WHERE i = ?", params![id], |r| r.get(0))
2019-03-10 11:16:37 +08:00
}
#[test]
2020-11-06 05:14:00 +08:00
fn test_sql_url() -> Result<()> {
let db = &checked_memory_handle()?;
2019-03-10 11:16:37 +08:00
let url0 = Url::parse("http://www.example1.com").unwrap();
let url1 = Url::parse("http://www.example1.com/👌").unwrap();
let url2 = "http://www.example2.com/👌";
db.execute(
"INSERT INTO urls (i, v) VALUES (0, ?), (1, ?), (2, ?), (3, ?)",
// also insert a non-hex encoded url (which might be present if it was
// inserted separately)
params![url0, url1, url2, "illegal"],
2020-11-06 05:14:00 +08:00
)?;
2019-03-10 11:16:37 +08:00
2020-11-06 05:14:00 +08:00
assert_eq!(get_url(db, 0)?, url0);
2019-03-10 11:16:37 +08:00
2020-11-06 05:14:00 +08:00
assert_eq!(get_url(db, 1)?, url1);
2019-03-10 11:16:37 +08:00
// Should successfully read it, even though it wasn't inserted as an
// escaped url.
2020-11-06 05:14:00 +08:00
let out_url2: Url = get_url(db, 2)?;
2019-03-10 11:16:37 +08:00
assert_eq!(out_url2, Url::parse(url2).unwrap());
// Make sure the conversion error comes through correctly.
let err = get_url(db, 3).unwrap_err();
match err {
Error::FromSqlConversionFailure(_, _, e) => {
assert_eq!(
*e.downcast::<ParseError>().unwrap(),
ParseError::RelativeUrlWithoutBase,
);
}
e => {
panic!("Expected conversion failure, got {}", e);
}
}
2020-11-06 05:14:00 +08:00
Ok(())
2019-03-10 11:16:37 +08:00
}
}