1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
use std::io::prelude::*;
use std::convert::From;
use diesel::deserialize::{self, FromSql};
use diesel::serialize::{self, IsNull, Output, ToSql};
use diesel::pg::Pg;
use postgis::ewkb::Point;
use crate::sql_types::*;
#[derive(Debug, Copy, Clone, PartialEq, FromSqlRow, AsExpression)]
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
#[sql_type = "Geography"]
pub struct GeogPoint {
pub x: f64,
pub y: f64,
pub srid: Option<i32>,
}
impl From<Point> for GeogPoint {
fn from(p: Point) -> Self {
let Point { x, y, srid } = p;
Self { x, y, srid }
}
}
impl From<GeogPoint> for Point {
fn from(p: GeogPoint) -> Self {
let GeogPoint { x, y, srid } = p;
Self { x, y, srid }
}
}
impl FromSql<Geography, Pg> for GeogPoint {
fn from_sql(bytes: Option<&[u8]>) -> deserialize::Result<Self> {
use std::io::Cursor;
use postgis::ewkb::EwkbRead;
let bytes = not_none!(bytes);
let mut rdr = Cursor::new(bytes);
Ok(Point::read_ewkb(&mut rdr)?.into())
}
}
impl ToSql<Geography, Pg> for GeogPoint {
fn to_sql<W: Write>(&self, out: &mut Output<W, Pg>) -> serialize::Result {
use postgis::ewkb::{AsEwkbPoint, EwkbWrite};
Point::from(*self).as_ewkb().write_ewkb(out)?;
Ok(IsNull::No)
}
}