Skip to content

Commit

Permalink
Infer geometry type and dimension from WKT
Browse files Browse the repository at this point in the history
  • Loading branch information
kylebarron committed Oct 6, 2024
1 parent c3088cd commit ec7410e
Show file tree
Hide file tree
Showing 5 changed files with 150 additions and 2 deletions.
131 changes: 131 additions & 0 deletions src/infer_type.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,131 @@
use crate::types::{Dimension, GeometryType};

const POINT: &str = "POINT";
const LINESTRING: &str = "LINESTRING";
const POLYGON: &str = "POLYGON";
const MULTIPOINT: &str = "MULTIPOINT";
const MULTILINESTRING: &str = "MULTILINESTRING";
const MULTIPOLYGON: &str = "MULTIPOLYGON";
const GEOMETRYCOLLECTION: &str = "GEOMETRYCOLLECTION";

/// Infer the geometry type and dimension from an input WKT string slice.
///
/// An `EMPTY` WKT object will return `None` in place of the dimension.
///
/// ```
/// use wkt::infer_type;
/// use wkt::types::{Dimension, GeometryType};
///
/// assert_eq!(
/// infer_type("POINT (10 20.1)").unwrap(),
/// (GeometryType::Point, Some(Dimension::XY))
/// );
///
/// assert_eq!(
/// infer_type("POINT EMPTY").unwrap(),
/// (GeometryType::Point, None)
/// );
/// ```
pub fn infer_type(input: &str) -> Result<(GeometryType, Option<Dimension>), String> {
if let Some((prefix, _suffix)) = input.split_once("(") {
let prefix = prefix.to_uppercase();

let (geom_type, dim_str) = if let Some(dim_str) = prefix.strip_prefix(POINT) {
(GeometryType::Point, dim_str)
} else if let Some(dim_str) = prefix.strip_prefix(LINESTRING) {
(GeometryType::LineString, dim_str)
} else if let Some(dim_str) = prefix.strip_prefix(POLYGON) {
(GeometryType::Polygon, dim_str)
} else if let Some(dim_str) = prefix.strip_prefix(MULTIPOINT) {
(GeometryType::MultiPoint, dim_str)
} else if let Some(dim_str) = prefix.strip_prefix(MULTILINESTRING) {
(GeometryType::MultiLineString, dim_str)
} else if let Some(dim_str) = prefix.strip_prefix(MULTIPOLYGON) {
(GeometryType::MultiPolygon, dim_str)
} else if let Some(dim_str) = prefix.strip_prefix(GEOMETRYCOLLECTION) {
(GeometryType::GeometryCollection, dim_str)
} else {
return Err(format!("Unsupported WKT prefix {}", prefix));
};

let dim = if dim_str.contains("ZM") {
Dimension::XYZM
} else if dim_str.contains("Z") {
Dimension::XYZ
} else if dim_str.contains("M") {
Dimension::XYM
} else {
Dimension::XY
};

Ok((geom_type, Some(dim)))
} else {
let input = input.to_uppercase();
if !input.contains("EMPTY") {
return Err("Invalid WKT; no '(' character and not EMPTY".to_string());
}

if input.starts_with(POINT) {
Ok((GeometryType::Point, None))
} else if input.starts_with(LINESTRING) {
Ok((GeometryType::LineString, None))
} else if input.starts_with(POLYGON) {
Ok((GeometryType::Polygon, None))
} else if input.starts_with(MULTIPOINT) {
Ok((GeometryType::MultiPoint, None))
} else if input.starts_with(MULTILINESTRING) {
Ok((GeometryType::MultiLineString, None))
} else if input.starts_with(MULTIPOLYGON) {
Ok((GeometryType::MultiPolygon, None))
} else if input.starts_with(GEOMETRYCOLLECTION) {
Ok((GeometryType::GeometryCollection, None))
} else {
return Err(format!("Unsupported WKT prefix {}", input));
}
}
}

#[cfg(test)]
mod test {
use super::*;

#[test]
fn test_points() {
assert_eq!(
infer_type("POINT (10 20.1)").unwrap(),
(GeometryType::Point, Some(Dimension::XY))
);
assert_eq!(
infer_type("POINT Z (10 20.1 5)").unwrap(),
(GeometryType::Point, Some(Dimension::XYZ))
);
assert_eq!(
infer_type("POINT M (10 20.1 80)").unwrap(),
(GeometryType::Point, Some(Dimension::XYM))
);
assert_eq!(
infer_type("POINT ZM (10 20.1 5 80)").unwrap(),
(GeometryType::Point, Some(Dimension::XYZM))
);
}

#[test]
fn lowercase_point() {
assert_eq!(
infer_type("point EMPTY").unwrap(),
(GeometryType::Point, None)
);
}

#[test]
fn test_empty() {
assert_eq!(
infer_type("POINT EMPTY").unwrap(),
(GeometryType::Point, None)
);
assert_eq!(
infer_type("MULTIPOLYGON EMPTY").unwrap(),
(GeometryType::MultiPolygon, None)
);
}
}
4 changes: 4 additions & 0 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,10 @@ mod tokenizer;
/// `WKT` primitive types and collections
pub mod types;

mod infer_type;

pub use infer_type::infer_type;

#[cfg(feature = "geo-types")]
extern crate geo_types;

Expand Down
2 changes: 1 addition & 1 deletion src/types/dimension.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
/// The dimension of geometry that we're parsing.
#[allow(clippy::upper_case_acronyms)]
#[derive(Clone, Copy, Debug, Default)]
#[derive(Clone, Copy, Debug, Default, PartialEq, Eq, Hash)]
pub enum Dimension {
#[default]
XY,
Expand Down
11 changes: 11 additions & 0 deletions src/types/geometry_type.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
/// The geometry type of the WKT object
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
pub enum GeometryType {
Point,
LineString,
Polygon,
MultiPoint,
MultiLineString,
MultiPolygon,
GeometryCollection,
}
4 changes: 3 additions & 1 deletion src/types/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,8 @@
// limitations under the License.

pub use self::coord::Coord;
pub(crate) use self::dimension::Dimension;
pub use self::dimension::Dimension;
pub use self::geometry_type::GeometryType;
pub use self::geometrycollection::GeometryCollection;
pub use self::linestring::LineString;
pub use self::multilinestring::MultiLineString;
Expand All @@ -24,6 +25,7 @@ pub use self::polygon::Polygon;

mod coord;
mod dimension;
mod geometry_type;
mod geometrycollection;
mod linestring;
mod multilinestring;
Expand Down

0 comments on commit ec7410e

Please sign in to comment.