diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index ad5d09dd0..d9f76d2b1 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -90,11 +90,11 @@ jobs: - name: Make target directories if: github.event_name != 'pull_request' && (startsWith(github.ref, 'refs/tags/v') || github.ref == 'refs/heads/main' || github.ref == 'refs/heads/series/0.6.x') - run: mkdir -p modules/circe/.jvm/target unidocs/target modules/core/native/target modules/core/js/target modules/circe/.js/target modules/core/jvm/target modules/refined/.native/target modules/refined/.js/target modules/refined/.jvm/target modules/circe/.native/target project/target + run: mkdir -p modules/circe/.jvm/target unidocs/target modules/core/native/target modules/core/js/target modules/postgis/.native/target modules/circe/.js/target modules/postgis/.js/target modules/core/jvm/target modules/refined/.native/target modules/refined/.js/target modules/refined/.jvm/target modules/circe/.native/target modules/postgis/.jvm/target project/target - name: Compress target directories if: github.event_name != 'pull_request' && (startsWith(github.ref, 'refs/tags/v') || github.ref == 'refs/heads/main' || github.ref == 'refs/heads/series/0.6.x') - run: tar cf targets.tar modules/circe/.jvm/target unidocs/target modules/core/native/target modules/core/js/target modules/circe/.js/target modules/core/jvm/target modules/refined/.native/target modules/refined/.js/target modules/refined/.jvm/target modules/circe/.native/target project/target + run: tar cf targets.tar modules/circe/.jvm/target unidocs/target modules/core/native/target modules/core/js/target modules/postgis/.native/target modules/circe/.js/target modules/postgis/.js/target modules/core/jvm/target modules/refined/.native/target modules/refined/.js/target modules/refined/.jvm/target modules/circe/.native/target modules/postgis/.jvm/target project/target - name: Upload target directories if: github.event_name != 'pull_request' && (startsWith(github.ref, 'refs/tags/v') || github.ref == 'refs/heads/main' || github.ref == 'refs/heads/series/0.6.x') diff --git a/build.sbt b/build.sbt index ae6e1b3c0..0eae73243 100644 --- a/build.sbt +++ b/build.sbt @@ -98,7 +98,7 @@ lazy val commonSettings = Seq( lazy val skunk = tlCrossRootProject .settings(name := "skunk") - .aggregate(core, tests, circe, refined, example, unidocs) + .aggregate(core, tests, circe, refined, postgis, example, unidocs) .settings(commonSettings) lazy val core = crossProject(JVMPlatform, JSPlatform, NativePlatform) @@ -160,10 +160,23 @@ lazy val circe = crossProject(JVMPlatform, JSPlatform, NativePlatform) ) ) +lazy val postgis = crossProject(JVMPlatform, JSPlatform, NativePlatform) + .crossType(CrossType.Pure) + .in(file("modules/postgis")) + .dependsOn(core) + .enablePlugins(AutomateHeaderPlugin) + .settings(commonSettings) + .settings( + name := "skunk-postgis", + libraryDependencies ++= Seq( + "org.typelevel" %%% "cats-parse" % "1.0.0" + ) + ) + lazy val tests = crossProject(JVMPlatform, JSPlatform, NativePlatform) .crossType(CrossType.Full) .in(file("modules/tests")) - .dependsOn(core, circe, refined) + .dependsOn(core, circe, refined, postgis) .enablePlugins(AutomateHeaderPlugin, NoPublishPlugin) .settings(commonSettings) .settings( @@ -273,5 +286,4 @@ lazy val docs = project } ) -// ci - +// ci \ No newline at end of file diff --git a/docker-compose.yml b/docker-compose.yml index 6f33c21d1..2d2317334 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -2,7 +2,7 @@ version: '3' services: # main instance for testing postgres: - image: postgres:11 + image: postgis/postgis:11-3.3 volumes: - ./world/fix_perms.sh:/docker-entrypoint-initdb.d/fix_perms.sh - ./world/world.sql:/docker-entrypoint-initdb.d/world.sql @@ -65,4 +65,4 @@ services: - 4317:4317 - 4318:4318 environment: - COLLECTOR_OTLP_ENABLED: "true" + COLLECTOR_OTLP_ENABLED: "true" \ No newline at end of file diff --git a/modules/postgis/src/main/scala-2/ewkb/platform.scala b/modules/postgis/src/main/scala-2/ewkb/platform.scala new file mode 100644 index 000000000..161530105 --- /dev/null +++ b/modules/postgis/src/main/scala-2/ewkb/platform.scala @@ -0,0 +1,18 @@ +// Copyright (c) 2018-2021 by Rob Norris +// This software is licensed under the MIT License (MIT). +// For more information see LICENSE or https://opensource.org/licenses/MIT + +package skunk.postgis +package ewkb + +import scodec.Attempt +import scodec.{Codec => Scodec} +import scodec.bits.BitVector +import scodec.bits.ByteOrdering + +trait EWKBCodecPlatform extends EWKBPrimitives { + def ewkbEncode(ewkb: EWKBType, geometry: Geometry, geoEncoder: Scodec[Geometry])(implicit bo: ByteOrdering): Attempt[BitVector] = { + val encoder = byteOrdering :: ewkbType :: geoEncoder + encoder.encode(bo :: ewkb :: geometry :: shapeless.HNil) + } +} diff --git a/modules/postgis/src/main/scala-3/ewkb/platform.scala b/modules/postgis/src/main/scala-3/ewkb/platform.scala new file mode 100644 index 000000000..484d3e8d5 --- /dev/null +++ b/modules/postgis/src/main/scala-3/ewkb/platform.scala @@ -0,0 +1,18 @@ +// Copyright (c) 2018-2021 by Rob Norris +// This software is licensed under the MIT License (MIT). +// For more information see LICENSE or https://opensource.org/licenses/MIT + +package skunk.postgis +package ewkb + +import scodec.Attempt +import scodec.{Codec => Scodec} +import scodec.bits.BitVector +import scodec.bits.ByteOrdering + +trait EWKBCodecPlatform extends EWKBPrimitives { + def ewkbEncode(ewkb: EWKBType, geometry: Geometry, geoEncoder: Scodec[Geometry])(using bo: ByteOrdering): Attempt[BitVector] = { + val encoder = byteOrdering :: ewkbType :: geoEncoder + encoder.encode((bo, ewkb, geometry)) + } +} diff --git a/modules/postgis/src/main/scala/codecs.scala b/modules/postgis/src/main/scala/codecs.scala new file mode 100644 index 000000000..4dd7e9a2e --- /dev/null +++ b/modules/postgis/src/main/scala/codecs.scala @@ -0,0 +1,45 @@ +// Copyright (c) 2018-2021 by Rob Norris +// This software is licensed under the MIT License (MIT). +// For more information see LICENSE or https://opensource.org/licenses/MIT + +package skunk +package postgis +package codecs + +import scala.reflect.ClassTag + +import cats.syntax.all._ +import scodec.bits.ByteVector +import skunk.data.Type + +trait PostGISGeometryCodecs { + + val geometry: Codec[Geometry] = Codec.simple[Geometry]( + geometry => ewkb.codecs.geometry.encode(geometry).require.toHex, + str => { + ByteVector.fromHex(str).fold(s"[postgis] Bad EWKB Hex: $str".asLeft[Geometry]) { byteVector => + ewkb.codecs.geometry.decodeValue(byteVector.toBitVector).toEither.leftMap(_.message) + } + }, + Type("geometry") + ) + + val point: Codec[Point] = geometryCodec[Point] + val lineString: Codec[LineString] = geometryCodec[LineString] + val polygon: Codec[Polygon] = geometryCodec[Polygon] + val multiPoint: Codec[MultiPoint] = geometryCodec[MultiPoint] + val multiLineString: Codec[MultiLineString] = geometryCodec[MultiLineString] + val multiPolygon: Codec[MultiPolygon] = geometryCodec[MultiPolygon] + val geometryCollection: Codec[GeometryCollection] = geometryCodec[GeometryCollection] + + private def geometryCodec[A >: Null <: Geometry](implicit A: ClassTag[A]): Codec[A] = { + geometry.imap[A](geom => A.runtimeClass.cast(geom).asInstanceOf[A])(o => + o.asInstanceOf[Geometry] + ) + } + +} + +object geometry extends PostGISGeometryCodecs + +object all extends PostGISGeometryCodecs \ No newline at end of file diff --git a/modules/postgis/src/main/scala/ewkb/codecs.scala b/modules/postgis/src/main/scala/ewkb/codecs.scala new file mode 100644 index 000000000..6ff09b1df --- /dev/null +++ b/modules/postgis/src/main/scala/ewkb/codecs.scala @@ -0,0 +1,163 @@ +// Copyright (c) 2018-2021 by Rob Norris +// This software is licensed under the MIT License (MIT). +// For more information see LICENSE or https://opensource.org/licenses/MIT + +package skunk.postgis +package ewkb + +import cats.data.NonEmptyList +import cats.syntax.all._ +import scodec.Attempt +import scodec.SizeBound +import scodec.bits.BitVector +import scodec.bits.ByteOrdering +import scodec.codecs._ +import scodec.{Codec => Scodec} + +trait EWKBCodecs extends EWKBCodecPlatform { + + lazy val geometry: Scodec[Geometry] = Scodec[Geometry](encoder, decoder) + + private def decoder: scodec.Decoder[Geometry] = { + byteOrdering.flatMap { implicit byteOrdering => + ewkbType.flatMap { implicit ewkb => + ewkb.geometry match { + case EWKBGeometry.Point => point + case EWKBGeometry.LineString => lineString + case EWKBGeometry.Polygon => polygon + case EWKBGeometry.MultiPoint => multiPoint + case EWKBGeometry.MultiLineString => multiLineString + case EWKBGeometry.MultiPolygon => multiPolygon + case EWKBGeometry.GeometryCollection => geometryCollection + } + } + } + } + + def encoder: scodec.Encoder[Geometry] = new scodec.Encoder[Geometry] { + override def encode(value: Geometry): Attempt[BitVector] = { + implicit val byteOrder: ByteOrdering = ByteOrdering.LittleEndian + + implicit val ewkb = EWKBType.fromGeometry(value) + + val geoencoder = value match { + case _: Point => point.upcast[Geometry] + case _: LineString => lineString.upcast[Geometry] + case _: Polygon => polygon.upcast[Geometry] + case _: MultiPoint => multiPoint.upcast[Geometry] + case _: MultiLineString => multiLineString.upcast[Geometry] + case _: MultiPolygon => multiPolygon.upcast[Geometry] + case _: GeometryCollection => geometryCollection.upcast[Geometry] + } + + // scala 2/3 platform specific call - scodec1 encodes HLists, scodec2 encodes tuples + ewkbEncode(ewkb, value, geoencoder) + } + + override def sizeBound: SizeBound = SizeBound.unknown + } +} + +object codecs extends EWKBCodecs + +trait EWKBPrimitives { + + def geometry: Scodec[Geometry] + + def byteOrdering: Scodec[ByteOrdering] = byte.xmap( + b => b match { + case 0 => ByteOrdering.BigEndian + case 1 => ByteOrdering.LittleEndian + }, + o => o match { + case ByteOrdering.BigEndian => 0 + case ByteOrdering.LittleEndian => 1 + } + ) + + def ewkbType(implicit byteOrdering: ByteOrdering): Scodec[EWKBType] = + guint32.xmap(EWKBType.fromRaw, EWKBType.toRaw) + + def srid(implicit byteOrderering: ByteOrdering, ewkb: EWKBType): Scodec[Option[SRID]] = { + optional(provide(ewkb.srid == EWKBSRID.Present), gint32.xmap(SRID.apply, _.value)) + } + + def coordinate(implicit byteOrdering: ByteOrdering, ewkb: EWKBType): Scodec[Coordinate] = { + ewkb.coordinate match { + case Dimension.TwoD => gdouble :: gdouble :: provide(none[Double]) :: provide(none[Double]) + case Dimension.Z => gdouble :: gdouble :: gdoubleOpt :: provide(none[Double]) + case Dimension.M => gdouble :: gdouble :: provide(none[Double]) :: gdoubleOpt + case Dimension.ZM => gdouble :: gdouble :: gdoubleOpt :: gdoubleOpt + } + }.as[Coordinate] + + def point(implicit byteOrdering: ByteOrdering, ewkb: EWKBType): Scodec[Point] = { + ewkb.coordinate match { + case Dimension.TwoD => srid :: coordinate + case Dimension.Z => srid :: coordinate + case Dimension.M => srid :: coordinate + case Dimension.ZM => srid :: coordinate + } + }.as[Point] + + def lineString(implicit byteOrdering: ByteOrdering, ewkb: EWKBType): Scodec[LineString] = { + srid :: provide(ewkb.coordinate) :: listOfN(gint32, coordinate) + }.as[LineString] + + private case class PolygonRepr(srid: Option[SRID], dim: Dimension, rings: List[LinearRing]) + def polygon(implicit byteOrdering: ByteOrdering, ewkb: EWKBType): Scodec[Polygon] = { + (srid :: provide(ewkb.coordinate) :: listOfN(gint32, linearRing)).as[PolygonRepr].xmap[Polygon]( + repr => Polygon(repr.srid, repr.dim, repr.rings.headOption, repr.rings.drop(1)), + p => PolygonRepr(p.srid, p.dimension, p.shell.toList ::: p.holes) + ) + } + + def linearRing(implicit byteOrdering: ByteOrdering, ewkb: EWKBType): Scodec[LinearRing] = { + listOfN(gint32, coordinate).xmap[NonEmptyList[Coordinate]](NonEmptyList.fromListUnsafe, _.toList) + }.as[LinearRing] + + def multiPoint(implicit byteOrdering: ByteOrdering, ewkb: EWKBType): Scodec[MultiPoint] = { + srid :: provide(ewkb.coordinate) :: listOfN(gint32, geometry.downcast[Point]).xmap[List[Point]]( + pts => pts.map(_.copy(srid = None)), + identity + ) + }.as[MultiPoint] + + def multiLineString(implicit byteOrdering: ByteOrdering, ewkb: EWKBType): Scodec[MultiLineString] = { + srid.flatPrepend { srid => + provide(ewkb.coordinate) :: listOfN(gint32, geometry.downcast[LineString]).xmap[List[LineString]]( + lss => lss.map(_.copy(srid = srid)), + identity + ) + } + }.as[MultiLineString] + + def multiPolygon(implicit byteOrdering: ByteOrdering, ewkb: EWKBType): Scodec[MultiPolygon] = { + srid.flatPrepend { srid => + provide(ewkb.coordinate) :: listOfN(gint32, geometry.downcast[Polygon]).xmap[List[Polygon]]( + ps => ps.map(_.copy(srid = srid)), + identity + ) + } + }.as[MultiPolygon] + + def geometryCollection(implicit byteOrdering: ByteOrdering, ewkb: EWKBType): Scodec[GeometryCollection] = { + srid :: provide(ewkb.coordinate) :: listOfN(gint32, geometry) + }.as[GeometryCollection] + + def gint32(implicit byteOrdering: ByteOrdering) = + gcodec(int32, int32L) + def guint32(implicit byteOrdering: ByteOrdering) = + gcodec(uint32, uint32L) + def gdouble(implicit byteOrdering: ByteOrdering) = + gcodec(double, doubleL) + def gdoubleOpt(implicit byteOrdering: ByteOrdering): Scodec[Option[Double]] = + gdouble.widenOpt(x => Some(x), x => x) + + def gcodec[A](big: Scodec[A], little: Scodec[A])(implicit byteOrdering: ByteOrdering) = + byteOrdering match { + case ByteOrdering.BigEndian => big + case ByteOrdering.LittleEndian => little + } + +} diff --git a/modules/postgis/src/main/scala/ewkb/domain.scala b/modules/postgis/src/main/scala/ewkb/domain.scala new file mode 100644 index 000000000..a25d9ab58 --- /dev/null +++ b/modules/postgis/src/main/scala/ewkb/domain.scala @@ -0,0 +1,123 @@ +// Copyright (c) 2018-2021 by Rob Norris +// This software is licensed under the MIT License (MIT). +// For more information see LICENSE or https://opensource.org/licenses/MIT + +package skunk.postgis +package ewkb + +case class EWKBType( + geometry: EWKBGeometry, + coordinate: Dimension, + srid: EWKBSRID +) + +object EWKBType { + + def fromRaw(raw: Long): EWKBType = { + EWKBType( + EWKBGeometry.fromRaw(raw), + dimensionFromRaw(raw), + EWKBSRID.fromRaw(raw) + ) + } + + def toRaw(ewkb: EWKBType): Long = + EWKBGeometry.toRaw(ewkb.geometry) | dimensionToRaw(ewkb.coordinate) | EWKBSRID.toRaw(ewkb.srid) + + def fromGeometry(geometry: skunk.postgis.Geometry): EWKBType = + EWKBType( + EWKBGeometry.fromGeometry(geometry), + geometry.dimension, + EWKBSRID.fromGeometry(geometry) + ) + + private val ZMask = 0x80000000L + private val MMask = 0x40000000L + + private def dimensionFromRaw(raw: Long): Dimension = { + + val hasZ = (raw & ZMask) == ZMask || (raw & 0xffff) / 1000 == 1 || (raw & 0xffff) / 1000 == 3 + val hasM = (raw & MMask) == MMask || (raw & 0xffff) / 1000 == 2 || (raw & 0xffff) / 1000 == 3 + + (hasZ, hasM) match { + case (false, false) => Dimension.TwoD + case (true, false) => Dimension.Z + case (false, true) => Dimension.M + case (true, true) => Dimension.ZM + } + } + + private def dimensionToRaw(coordinateType: Dimension): Long = coordinateType match { + case Dimension.TwoD => 0L + case Dimension.Z => ZMask + case Dimension.M => MMask + case Dimension.ZM => ZMask | MMask + } +} + +sealed trait EWKBGeometry extends Product with Serializable + +object EWKBGeometry { + + case object Point extends EWKBGeometry + case object LineString extends EWKBGeometry + case object Polygon extends EWKBGeometry + case object MultiPoint extends EWKBGeometry + case object MultiLineString extends EWKBGeometry + case object MultiPolygon extends EWKBGeometry + case object GeometryCollection extends EWKBGeometry + + def fromRaw(id: Long): EWKBGeometry = (id & 0x000000ff) match { + case 1 => EWKBGeometry.Point + case 2 => EWKBGeometry.LineString + case 3 => EWKBGeometry.Polygon + case 4 => EWKBGeometry.MultiPoint + case 5 => EWKBGeometry.MultiLineString + case 6 => EWKBGeometry.MultiPolygon + case 7 => EWKBGeometry.GeometryCollection + case _ => throw new IllegalArgumentException(s"Invalid (or unsupported) EWKB geometry type($id), expected [1-7]") + } + + def toRaw(geometryType: EWKBGeometry): Long = geometryType match { + case Point => 1 + case LineString => 2 + case Polygon => 3 + case MultiPoint => 4 + case MultiLineString => 5 + case MultiPolygon => 6 + case GeometryCollection => 7 + } + + def fromGeometry(geometry: Geometry): EWKBGeometry = geometry match { + case _: Point => Point + case _: LineString => LineString + case _: Polygon => Polygon + case _: MultiPoint => MultiPoint + case _: MultiLineString => MultiLineString + case _: MultiPolygon => MultiPolygon + case _: GeometryCollection => GeometryCollection + } +} + +sealed trait EWKBSRID extends Product with Serializable + +object EWKBSRID { + case object Present extends EWKBSRID + case object Absent extends EWKBSRID + + val Mask = 0x20000000L + + def fromRaw(raw: Long): EWKBSRID = + if((raw & Mask) == Mask) + EWKBSRID.Present + else + EWKBSRID.Absent + + def toRaw(sridEmbedded: EWKBSRID): Long = sridEmbedded match { + case Absent => 0L + case Present => Mask + } + + def fromGeometry(geometry: skunk.postgis.Geometry): EWKBSRID = + geometry.srid.fold[EWKBSRID](EWKBSRID.Absent)(_ => EWKBSRID.Present) +} \ No newline at end of file diff --git a/modules/postgis/src/main/scala/ewkt/parser.scala b/modules/postgis/src/main/scala/ewkt/parser.scala new file mode 100644 index 000000000..ab98a53ca --- /dev/null +++ b/modules/postgis/src/main/scala/ewkt/parser.scala @@ -0,0 +1,215 @@ +// Copyright (c) 2018-2021 by Rob Norris +// This software is licensed under the MIT License (MIT). +// For more information see LICENSE or https://opensource.org/licenses/MIT + +package skunk.postgis +package ewkt + +import cats.data.NonEmptyList +import cats.parse.Numbers +import cats.parse.{Parser => P} +import cats.parse.{Parser0 => P0} + +object EWKT { + + def parse(str: String): Either[P.Error, Geometry] = + geometry.parse(str).map(_._2) + + private[this] val whitespace: P[Unit] = P.charIn(" \t\r\n").void + private[this] val whitespaces0: P[Unit] = whitespace.rep.void + private[this] val comma: P[Unit] = P.char(',').surroundedBy(whitespaces0.?) + + private[this] implicit class BetweenOps[A](parser: P[A]) { + implicit val betweenParens: P[A] = + parser.between(P.char('('), P.char(')')) + implicit val betweenParensOpt: P[A] = + parser.between(P.char('(').?, P.char(')').?) + } + + private[this] val srid: P[SRID] = + Numbers.nonNegativeIntString.map(s => SRID(Integer.parseInt(s))).between( + P.ignoreCase("SRID="), + P.char(';') + ) + + private[this] def keyword(keyword: String): P[Dimension] = + P.ignoreCase(keyword).surroundedBy(whitespaces0.?) *> + dimension + + // This is only a hint, if you leave off ZM, but provide the 4 dimensions in the coordinate, it's still ZM + // Both 'POINT( 1 2 3 4 )' and 'POINT ZM( 1 2 3 4 )' have the same dimension + private[this] val dimension: P0[Dimension] = ( + P.ignoreCase("Z").surroundedBy(whitespaces0.?).? ~ + P.ignoreCase("M").surroundedBy(whitespaces0.?).? + ).map { dimensionChars => + dimensionChars match { + case (Some(_), None) => Dimension.Z + case (None, Some(_)) => Dimension.M + case (Some(_), Some(_)) => Dimension.ZM + case _ => Dimension.TwoD + } + } + + private[this] def empty[A](value: A): P[A] = + P.ignoreCase("EMPTY").surroundedBy(whitespaces0.?).as(value) + + private[this] val double: P[Double] = + Numbers.jsonNumber.map(s => BigDecimal(s).toDouble) + + private[this] def coordinate(implicit dimensionHint: Dimension): P[Coordinate] = + double.surroundedBy(whitespaces0.?).rep(2, 4).flatMap { elements => + (elements, dimensionHint) match { + case (NonEmptyList(x, y :: Nil), Dimension.TwoD) => P.pure(Coordinate.xy(x, y)) + case (NonEmptyList(x, y :: z :: Nil), Dimension.TwoD) => P.pure(Coordinate.xyz(x, y, z)) + case (NonEmptyList(x, y :: z :: Nil), Dimension.Z) => P.pure(Coordinate.xyz(x, y, z)) + case (NonEmptyList(x, y :: m :: Nil), Dimension.M) => P.pure(Coordinate.xym(x, y, m)) + case (NonEmptyList(x, y :: z :: m :: Nil), Dimension.ZM) => P.pure(Coordinate.xyzm(x, y, z, m)) + case (NonEmptyList(x, y :: z :: m :: Nil), Dimension.TwoD) => P.pure(Coordinate.xyzm(x, y, z, m)) + case _ => P.failWith(s"""Invalid Geometry Dimensionality [$dimensionHint]: ${elements.toList.mkString(",")}""") + } + } + + private[this] def coordinateEmpty(implicit dimensionHint: Dimension): P[Coordinate] = + P.ignoreCase("EMPTY").surroundedBy(whitespaces0.?).as( + dimensionHint match { + case Dimension.TwoD => Coordinate.xy(Double.NaN, Double.NaN) + case Dimension.Z => Coordinate.xyz(Double.NaN, Double.NaN, Double.NaN) + case Dimension.M => Coordinate.xym(Double.NaN, Double.NaN, Double.NaN) + case Dimension.ZM => Coordinate.xyzm(Double.NaN, Double.NaN, Double.NaN, Double.NaN) + } + ) + + private[this] def coordinates(implicit dimensionHint: Dimension): P[NonEmptyList[Coordinate]] = + coordinate.repSep(comma).betweenParens + + ///////////////////////////////////////////////////////////////////////////// + + private[this] def nonEmptyPoint(implicit srid: Option[SRID], dimensionHint: Dimension): P[Point] = + coordinate.betweenParens.map(c => Point(srid, c)) + + private[this] def nonEmptyLineString(implicit srid: Option[SRID], dimensionHint: Dimension): P[LineString] = + coordinate.repSep(comma).betweenParens.map(points => + LineString(srid, dimensionHint, points.toList)) + + private[this] def nonEmptyPolygon(implicit srid: Option[SRID], dimensionHint: Dimension): P[Polygon] = + coordinates.map(LinearRing.apply).repSep(comma).betweenParens.map(rings => + Polygon(srid, dimensionHint, Some(rings.head), rings.tail)) + + private[this] def nonEmptyMultiPoint(implicit srid: Option[SRID], dimensionHint: Dimension): P[MultiPoint] = + coordinate.betweenParensOpt.repSep(comma) + .map(nel => nel.map(c => Point(c))).betweenParensOpt + .map(points => MultiPoint(srid, dimensionHint, points.toList)) + + private[this] def nonEmptyMultiLineString(implicit srid: Option[SRID], dimensionHint: Dimension): P[MultiLineString] = + nonEmptyLineString.repSep(comma).betweenParens.map(lineStrings => + MultiLineString(srid, dimensionHint, lineStrings.toList)) + + private[this] def nonEmptyMultiPolygon(implicit srid: Option[SRID], dimensionHint: Dimension): P[MultiPolygon] = + nonEmptyPolygon.repSep(comma).betweenParens.map(polygons => + MultiPolygon(srid, dimensionHint, polygons.toList)) + + private[this] def nonEmptyGeometryCollection(implicit srid: Option[SRID], dimensionHint: Dimension): P[GeometryCollection] = + P.oneOf( + point :: + lineString :: + polygon :: + multiPoint :: + multiLineString :: + multiPolygon :: + geometryCollection :: + Nil + ).repSep(comma).betweenParens.map(geometries => + GeometryCollection(srid, dimensionHint, geometries.toList) + ) + + ///////////////////////////////////////////////////////////////////////////// + + def point: P[Point] = + P.flatMap01(srid.?) { implicit srid => + keyword("POINT").flatMap { implicit dimensionHint => + P.oneOf( + coordinateEmpty.map(c => Point(srid, c)) :: + nonEmptyPoint :: + Nil + ) + } + } + + val lineString: P[LineString] = + P.flatMap01(srid.?) { implicit srid => + keyword("LINESTRING").flatMap { implicit dimensionHint => + P.oneOf( + empty(LineString(srid, dimensionHint, Nil)) :: + nonEmptyLineString :: + Nil + ) + } + } + + val polygon: P[Polygon] = + P.flatMap01(srid.?) { implicit srid => + keyword("POLYGON").flatMap { implicit dimensionHint => + P.oneOf( + empty(Polygon(srid, dimensionHint, None, Nil)) :: + nonEmptyPolygon :: + Nil + ) + } + } + + val multiPoint: P[MultiPoint] = + P.flatMap01(srid.?) { implicit srid => + keyword("MULTIPOINT").flatMap { implicit dimensionHint => + P.oneOf( + empty(MultiPoint(srid, dimensionHint, Nil)) :: + nonEmptyMultiPoint :: + Nil + ) + } + } + + val multiLineString: P[MultiLineString] = + P.flatMap01(srid.?) { implicit srid => + keyword("MULTILINESTRING").flatMap { implicit dimensionHint => + P.oneOf( + empty(MultiLineString(srid, dimensionHint, Nil)) :: + nonEmptyMultiLineString :: + Nil + ) + } + } + + val multiPolygon: P[MultiPolygon] = + P.flatMap01(srid.?) { implicit srid => + keyword("MULTIPOLYGON").flatMap { implicit dimensionHint => + P.oneOf( + empty(MultiPolygon(srid, dimensionHint, Nil)) :: + nonEmptyMultiPolygon :: + Nil + ) + } + } + + val geometryCollection: P[GeometryCollection] = + P.flatMap01(srid.?) { implicit srid => + keyword("GEOMETRYCOLLECTION").flatMap { implicit dimensionHint => + P.oneOf( + empty(GeometryCollection(srid, dimensionHint, Nil)) :: + nonEmptyGeometryCollection :: + Nil + ) + } + } + + val geometry: P[Geometry] = + P.oneOf( + point.backtrack :: + lineString.backtrack :: + polygon.backtrack :: + multiPoint.backtrack :: + multiLineString.backtrack :: + multiPolygon.backtrack :: + geometryCollection.backtrack :: + Nil + ) +} \ No newline at end of file diff --git a/modules/postgis/src/main/scala/geometry.scala b/modules/postgis/src/main/scala/geometry.scala new file mode 100644 index 000000000..42688dabd --- /dev/null +++ b/modules/postgis/src/main/scala/geometry.scala @@ -0,0 +1,178 @@ +// Copyright (c) 2018-2021 by Rob Norris +// This software is licensed under the MIT License (MIT). +// For more information see LICENSE or https://opensource.org/licenses/MIT + +package skunk +package postgis + +import cats.Eq +import cats.data.NonEmptyList +import cats.syntax.all._ + +final case class SRID(value: Int) + +sealed trait Dimension extends Product with Serializable + +object Dimension { + case object TwoD extends Dimension + case object Z extends Dimension + case object M extends Dimension + case object ZM extends Dimension +} + +sealed trait Geometry extends Product with Serializable { + def srid: Option[SRID] + def dimension: Dimension +} + +object Geometry { + implicit def geometryEq[A <: Geometry]: Eq[A] = new Eq[A] { + override def eqv(x: A, y: A): Boolean = x.equals(y) + } +} + +final case class Coordinate(x: Double, y: Double, z: Option[Double], m: Option[Double]) { + def hasZ: Boolean = z.isDefined + def hasM: Boolean = m.isDefined + + def dimension: Dimension = (z, m) match { + case (None, None) => Dimension.TwoD + case (Some(_), None) => Dimension.Z + case (None, Some(_)) => Dimension.M + case (Some(_), Some(_)) => Dimension.ZM + } + + def isEmpty: Boolean = + x.isNaN() && y.isNaN() && + z.map(_.isNaN()).getOrElse(true) && + m.map(_.isNaN()).getOrElse(true) + + // Added to consider NaN equal in this case + override def equals(other: Any): Boolean = + other match { + case other: Coordinate => + other.dimension == dimension && + other.isEmpty == isEmpty || ( + other.x == x && + other.y == y && + other.z == z && + other.m == m + ) + case _ => super.equals(other) + } + +} + +object Coordinate { + def xy(x: Double, y: Double): Coordinate = Coordinate(x, y, None, None) + def xyz(x: Double, y: Double, z: Double): Coordinate = Coordinate(x, y, z.some, None) + def xym(x: Double, y: Double, m: Double): Coordinate = Coordinate(x, y, None, m.some) + def xyzm(x: Double, y: Double, z: Double, m: Double): Coordinate = Coordinate(x, y, z.some, m.some) +} + +final case class Point (srid: Option[SRID], coordinate: Coordinate) extends Geometry { + override def dimension: Dimension = coordinate.dimension +} + +object Point { + def apply(coordinate: Coordinate): Point = Point(None, coordinate) + def apply(srid: SRID, coordinate: Coordinate): Point = Point(srid.some, coordinate) + + def xy(x: Double, y: Double): Point = Point(None, Coordinate.xy(x, y)) + def xyz(x: Double, y: Double, z: Double): Point = Point(None, Coordinate.xyz(x, y, z)) + def xym(x: Double, y: Double, m: Double): Point = Point(None, Coordinate.xym(x, y, m)) + def xyzm(x: Double, y: Double, z: Double, m: Double): Point = Point(None, Coordinate.xyzm(x, y, z, m)) + + def xy(srid: SRID, x: Double, y: Double): Point = Point(srid.some, Coordinate.xy(x, y)) + def xyz(srid: SRID, x: Double, y: Double, z: Double): Point = Point(srid.some, Coordinate.xyz(x, y, z)) + def xym(srid: SRID, x: Double, y: Double, m: Double): Point = Point(srid.some, Coordinate.xym(x, y, m)) + def xyzm(srid: SRID, x: Double, y: Double, z: Double, m: Double): Point = Point(srid.some, Coordinate.xyzm(x, y, z, m)) +} + +final case class LineString(srid: Option[SRID], dimensionHint: Dimension, coordinates: List[Coordinate]) extends Geometry { + override def dimension: Dimension = coordinates.headOption.fold(dimensionHint)(_.dimension) +} + +object LineString { + def apply(coordinates: Coordinate*): LineString + = apply(coordinates.toList) + def apply(coordinates: List[Coordinate]): LineString = + LineString(None, coordinates.headOption.fold[Dimension](Dimension.TwoD)(_.dimension), coordinates) + + def apply(srid: SRID, coordinates: Coordinate*): LineString = + apply(srid, coordinates.toList) + def apply(srid: SRID, coordinates: List[Coordinate]): LineString = + LineString(srid.some, coordinates.headOption.fold[Dimension](Dimension.TwoD)(_.dimension), coordinates) +} + +final case class LinearRing(coordinates: NonEmptyList[Coordinate]) + +object LinearRing { + def apply(head: Coordinate, tail: Coordinate*): LinearRing = + LinearRing(NonEmptyList(head, tail.toList)) +} + +final case class Polygon(srid: Option[SRID], dimensionHint: Dimension, shell: Option[LinearRing], holes: List[LinearRing]) extends Geometry { + override def dimension: Dimension = shell.map(_.coordinates.head.dimension).getOrElse(dimensionHint) +} + +object Polygon { + def apply(shell: LinearRing): Polygon = + Polygon(None, shell.coordinates.head.dimension, shell.some, Nil) + def apply(shell: LinearRing, holes: LinearRing*): Polygon = + Polygon(None, shell.coordinates.head.dimension, shell.some, holes.toList) + + def apply(srid: SRID, shell: LinearRing): Polygon = + Polygon(srid.some, shell.coordinates.head.dimension, shell.some, Nil) + def apply(srid: SRID, shell: LinearRing, holes: LinearRing*): Polygon = + Polygon(srid.some, shell.coordinates.head.dimension, shell.some, holes.toList) +} + +final case class MultiPoint(srid: Option[SRID], dimensionHint: Dimension, points: List[Point]) extends Geometry { + override def dimension: Dimension = points.headOption.fold(dimensionHint)(_.dimension) +} + +object MultiPoint { + def apply(points: Point*): MultiPoint = + MultiPoint(None, points.headOption.fold[Dimension](Dimension.TwoD)(_.dimension), points.toList) + + def apply(srid: SRID, points: Point*): MultiPoint = + MultiPoint(srid.some, points.headOption.fold[Dimension](Dimension.TwoD)(_.dimension), points.toList) +} + +final case class MultiLineString(srid: Option[SRID], dimensionHint: Dimension, lineStrings: List[LineString]) extends Geometry { + override def dimension: Dimension = lineStrings.headOption.fold(dimensionHint)(_.dimension) +} + +object MultiLineString { + def apply(lineStrings: LineString*): MultiLineString = + MultiLineString(None, lineStrings.headOption.fold[Dimension](Dimension.TwoD)(_.dimension), lineStrings.toList) + + def apply(srid: SRID, lineStrings: LineString*): MultiLineString = + MultiLineString(srid.some, lineStrings.headOption.fold[Dimension](Dimension.TwoD)(_.dimension), lineStrings.toList) +} + +final case class MultiPolygon(srid: Option[SRID], dimensionHint: Dimension, polygons: List[Polygon]) extends Geometry { + override def dimension: Dimension = polygons.headOption.fold(dimensionHint)(_.dimension) +} + +object MultiPolygon { + def apply(polygons: Polygon*): MultiPolygon = + MultiPolygon(None, polygons.headOption.fold[Dimension](Dimension.TwoD)(_.dimension), polygons.toList) + + def apply(srid: SRID, polygons: Polygon*): MultiPolygon = + MultiPolygon(srid.some, polygons.headOption.fold[Dimension](Dimension.TwoD)(_.dimension), polygons.toList) +} + +final case class GeometryCollection(srid: Option[SRID], dimensionHint: Dimension, geometries: List[Geometry]) extends Geometry { + override def dimension: Dimension = geometries.headOption.fold(dimensionHint)(_.dimension) +} + +object GeometryCollection { + + def apply(geometries: Geometry*): GeometryCollection = + GeometryCollection(None, geometries.headOption.fold[Dimension](Dimension.TwoD)(_.dimension), geometries.toList) + + def apply(srid: SRID, geometries: Geometry*): GeometryCollection = + GeometryCollection(srid.some, geometries.headOption.fold[Dimension](Dimension.TwoD)(_.dimension), geometries.toList) +} diff --git a/modules/tests/shared/src/test/scala/codec/PostGISCodecTest.scala b/modules/tests/shared/src/test/scala/codec/PostGISCodecTest.scala new file mode 100644 index 000000000..7a2f13db1 --- /dev/null +++ b/modules/tests/shared/src/test/scala/codec/PostGISCodecTest.scala @@ -0,0 +1,100 @@ +// Copyright (c) 2018-2021 by Rob Norris +// This software is licensed under the MIT License (MIT). +// For more information see LICENSE or https://opensource.org/licenses/MIT + +package tests +package codec + +import skunk.postgis._ +import skunk.postgis.codecs.all._ +import skunk.util.Typer + +class PostGISCodecTest extends CodecTest(strategy = Typer.Strategy.SearchPath) { + + roundtripTest(point)(Point.xy(1, 2)) + roundtripTest(point)(Point(SRID(4326), Coordinate.xy(1, 2))) + roundtripTest(point)(Point.xyz(1, 2, 3)) + roundtripTest(point)(Point.xym(1, 2, 3)) + roundtripTest(point)(Point.xyzm(1, 2, 3, 4)) + + roundtripTest(lineString)( + LineString( + Coordinate.xy(1, 2), + Coordinate.xy(3, 4), + Coordinate.xy(5, 6), + Coordinate.xy(7, 8) + ) + ) + roundtripTest(lineString)( + LineString( + Coordinate.xyz(1, 2, 1), + Coordinate.xyz(3, 4, 1), + Coordinate.xyz(5, 6, 1), + Coordinate.xyz(7, 8, 1) + ) + ) + + roundtripTest(polygon)( + Polygon( + LinearRing( + Coordinate.xy(0, 0), + Coordinate.xy(1, 0), + Coordinate.xy(1, 1), + Coordinate.xy(0, 1), + Coordinate.xy(0, 0) + ), + ) + ) + + roundtripTest(multiPoint)( + MultiPoint( + Point.xy(1, 1), + Point.xy(2, 2), + Point.xy(3, 3) + ) + ) + + roundtripTest(multiLineString)( + MultiLineString( + SRID(4326), + LineString( + SRID(4326), + Coordinate.xy(0, 0), + Coordinate.xy(1, 1) + ), + LineString( + SRID(4326), + Coordinate.xy(2, 2), + Coordinate.xy(3, 3) + ) + ) + ) + + roundtripTest(multiPolygon)( + MultiPolygon( + SRID(4326), + Polygon( + SRID(4326), + LinearRing( + Coordinate.xy(0, 0), + Coordinate.xy(10, 10), + Coordinate.xy(20, 20) + ), + ) + ) + ) + + roundtripTest(geometryCollection)( + GeometryCollection( + SRID(4326), + LineString( + Coordinate.xy(1, 2), + Coordinate.xy(3, 4), + Coordinate.xy(5, 6), + Coordinate.xy(7, 8) + ), + Point.xy(1, 2) + ) + ) + +} diff --git a/modules/tests/shared/src/test/scala/postgis/RoundtripTest.scala b/modules/tests/shared/src/test/scala/postgis/RoundtripTest.scala new file mode 100644 index 000000000..f6ecbe183 --- /dev/null +++ b/modules/tests/shared/src/test/scala/postgis/RoundtripTest.scala @@ -0,0 +1,108 @@ +// Copyright (c) 2018-2021 by Rob Norris +// This software is licensed under the MIT License (MIT). +// For more information see LICENSE or https://opensource.org/licenses/MIT + +package tests +package postgis + +import skunk._ +import skunk.implicits._ +import skunk.postgis.codecs.all._ +import skunk.postgis.ewkt.EWKT +import skunk.util.Typer + +class RoundtripTest extends SkunkTest(strategy = Typer.Strategy.SearchPath) { + + roundtripEWKT("POINT EMPTY") + roundtripEWKT("POINT Z EMPTY") + roundtripEWKT("POINT M EMPTY") + roundtripEWKT("POINTZM EMPTY") + + roundtripEWKT("POINT(0 0)") + roundtripEWKT("POINT Z (1 2 3)") + roundtripEWKT("POINT M (1 2 3)") + roundtripEWKT("POINT ZM (1 2 3 4)") + roundtripEWKT("POINT(1 2 3 4)") + + roundtripEWKT("LINESTRING EMPTY") + roundtripEWKT("LINESTRING Z EMPTY") + roundtripEWKT("LINESTRING M EMPTY") + roundtripEWKT("LINESTRING ZM EMPTY") + roundtripEWKT("LINESTRING(0 0, 1 1)") + roundtripEWKT("LINESTRING Z (0 0 2, 1 1 3)") + roundtripEWKT("LINESTRING M (0 0 2, 1 1 3)") + roundtripEWKT("LINESTRING ZM (0 0 2 3, 1 1 4 5)") + + roundtripEWKT("POLYGON EMPTY") + roundtripEWKT("POLYGON Z EMPTY") + roundtripEWKT("POLYGON M EMPTY") + roundtripEWKT("POLYGON ZM EMPTY") + roundtripEWKT("POLYGON((0 0,1 0,1 1,0 1,0 0))") + roundtripEWKT("POLYGON((0 0,10 0,10 10,0 10,0 0),(2 2,2 5,5 5,5 2,2 2))") + roundtripEWKT("POLYGON Z ((0 0 1,10 0 2 ,10 10 2,0 10 2,0 0 1),(2 2 5 ,2 5 4,5 5 3,5 2 3,2 2 5))") + roundtripEWKT("POLYGON M ((0 0 1,10 0 2 ,10 10 2,0 10 2,0 0 1),(2 2 5 ,2 5 4,5 5 3,5 2 3,2 2 5))") + roundtripEWKT("POLYGON ZM ((0 0 1 -1,10 0 2 -2,10 10 2 -2,0 10 2 -4,0 0 1 -1),(2 2 5 0,2 5 4 1,5 5 3 2,5 2 3 1,2 2 5 0))") + + roundtripEWKT("MULTIPOINT EMPTY") + roundtripEWKT("MULTIPOINT Z EMPTY") + roundtripEWKT("MULTIPOINT M EMPTY") + roundtripEWKT("MULTIPOINT ZM EMPTY") + roundtripEWKT("MULTIPOINT((0 0), (2 0))") + roundtripEWKT("MULTIPOINT Z ((0 0 0), (2 0 1))") + roundtripEWKT("MULTIPOINT M ((0 0 2), (2 0 1))") + roundtripEWKT("MULTIPOINT ZM ((0 1 2 3), (3 2 1 0))") + + roundtripEWKT("MULTILINESTRING EMPTY") + roundtripEWKT("MULTILINESTRING Z EMPTY") + roundtripEWKT("MULTILINESTRING M EMPTY") + roundtripEWKT("MULTILINESTRING ZM EMPTY") + roundtripEWKT("MULTILINESTRING((0 0, 2 0))") + roundtripEWKT("MULTILINESTRING((0 0, 2 0), (1 1, 2 2))") + roundtripEWKT("MULTILINESTRING Z ((0 0 1, 2 0 2), (1 1 3, 2 2 4))") + roundtripEWKT("MULTILINESTRING M ((0 0 1, 2 0 2), (1 1 3, 2 2 4))") + roundtripEWKT("MULTILINESTRING ZM ((0 0 1 5, 2 0 2 4), (1 1 3 3, 2 2 4 2))") + + roundtripEWKT("MULTIPOLYGON EMPTY") + roundtripEWKT("MULTIPOLYGON Z EMPTY") + roundtripEWKT("MULTIPOLYGON M EMPTY") + roundtripEWKT("MULTIPOLYGON ZM EMPTY") + roundtripEWKT("MULTIPOLYGON(((0 0,10 0,10 10,0 10,0 0),(2 2,2 5,5 5,5 2,2 2)))") + roundtripEWKT("MULTIPOLYGON Z (((0 0 3,10 0 3,10 10 3,0 10 3,0 0 3),(2 2 3,2 5 3,5 5 3,5 2 3,2 2 3)))") + roundtripEWKT("MULTIPOLYGON M (((0 0 3,10 0 3,10 10 3,0 10 3,0 0 3),(2 2 3,2 5 3,5 5 3,5 2 3,2 2 3)))") + roundtripEWKT("MULTIPOLYGON ZM (((0 0 3 2,10 0 3 2,10 10 3 2,0 10 3 2,0 0 3 2),(2 2 3 2,2 5 3 2,5 5 3 2,5 2 3 2,2 2 3 2)))") + roundtripEWKT("MULTIPOLYGON(((0 0,4 0,4 4,0 4,0 0),(1 1,2 1,2 2,1 2,1 1)), ((-1 -1,-1 -2,-2 -2,-2 -1,-1 -1)))") + + roundtripEWKT("GEOMETRYCOLLECTION EMPTY") + roundtripEWKT("GEOMETRYCOLLECTION Z EMPTY") + roundtripEWKT("GEOMETRYCOLLECTION M EMPTY") + roundtripEWKT("GEOMETRYCOLLECTION ZM EMPTY") + roundtripEWKT("GEOMETRYCOLLECTION ZM (POINT ZM (0 0 0 0),LINESTRING ZM (0 0 0 0,1 1 1 1))") + roundtripEWKT("SRID=4326;GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1))") + roundtripEWKT("GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1),GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1)))") + roundtripEWKT("GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1),POINT M EMPTY,GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1)))") + roundtripEWKT("GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1),GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1),POINT M EMPTY,GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1))),POINT M EMPTY,GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1)))") + + roundtripEWKT("SRID=4326;POINT(-44.3 60.1)") + roundtripEWKT("SRID=4326;LINESTRING(-71.160281 42.258729,-71.160837 42.259113,-71.161144 42.25932)") + roundtripEWKT("SRID=4269;POLYGON((-71.1776585052917 42.3902909739571,-71.1776820268866 42.3903701743239,-71.1776063012595 42.3903825660754,-71.1775826583081 42.3903033653531,-71.1776585052917 42.3902909739571))") + roundtripEWKT("SRID=4269;MULTILINESTRING((-71.160281 42.258729,-71.160837 42.259113,-71.161144 42.25932))") + roundtripEWKT("SRID=4269;MULTIPOLYGON(((-71.1031880899493 42.3152774590236,-71.1031627617667 42.3152960829043,-71.102923838298 42.3149156848307,-71.1023097974109 42.3151969047397,-71.1019285062273 42.3147384934248,-71.102505233663 42.3144722937587,-71.10277487471 42.3141658254797,-71.103113945163 42.3142739188902,-71.10324876416 42.31402489987,-71.1033002961013 42.3140393340215,-71.1033488797549 42.3139495090772,-71.103396240451 42.3138632439557,-71.1041521907712 42.3141153348029,-71.1041411411543 42.3141545014533,-71.1041287795912 42.3142114839058,-71.1041188134329 42.3142693656241,-71.1041112482575 42.3143272556118,-71.1041072845732 42.3143851580048,-71.1041057218871 42.3144430686681,-71.1041065602059 42.3145009876017,-71.1041097995362 42.3145589148055,-71.1041166403905 42.3146168544148,-71.1041258822717 42.3146748022936,-71.1041375307579 42.3147318674446,-71.1041492906949 42.3147711126569,-71.1041598612795 42.314808571739,-71.1042515013869 42.3151287620809,-71.1041173835118 42.3150739481917,-71.1040809891419 42.3151344119048,-71.1040438678912 42.3151191367447,-71.1040194562988 42.3151832057859,-71.1038734225584 42.3151140942995,-71.1038446938243 42.3151006300338,-71.1038315271889 42.315094347535,-71.1037393329282 42.315054824985,-71.1035447555574 42.3152608696313,-71.1033436658644 42.3151648370544,-71.1032580383161 42.3152269126061,-71.103223066939 42.3152517403219,-71.1031880899493 42.3152774590236)),((-71.1043632495873 42.315113108546,-71.1043583974082 42.3151211109857,-71.1043443253471 42.3150676015829,-71.1043850704575 42.3150793250568,-71.1043632495873 42.315113108546)))") + + // Round trip Skunk EWKT parse => PostGIS EWKT encode => Skunk EWKB decode + private def roundtripEWKT(ewkt: String) = { + sessionTest(ewkt) { s => + EWKT.parse(ewkt).fold( + err => fail(s"EWKT parse failed for '$ewkt': $err"), + parsedGeometry => { + val roundtripGeom = + s.prepare(sql"select '#$ewkt'::geometry".query(geometry)) + .flatMap(ps => ps.unique(Void)) + + roundtripGeom.map { returnedGeometry => + assertEquals(returnedGeometry, parsedGeometry) + } + } + ) + } + } +} diff --git a/modules/tests/shared/src/test/scala/postgis/ewkb/EWKBTest.scala b/modules/tests/shared/src/test/scala/postgis/ewkb/EWKBTest.scala new file mode 100644 index 000000000..3bf8de1f7 --- /dev/null +++ b/modules/tests/shared/src/test/scala/postgis/ewkb/EWKBTest.scala @@ -0,0 +1,204 @@ +// Copyright (c) 2018-2021 by Rob Norris +// This software is licensed under the MIT License (MIT). +// For more information see LICENSE or https://opensource.org/licenses/MIT + +package tests.postgis.ewkb + +import cats.syntax.all._ +import scodec.bits._ +import skunk.postgis._ + +class EWKBTest extends munit.FunSuite { + + ewkbTest("POINT(1 2)")( + hex"0101000000000000000000F03F0000000000000040", + Point.xy(1, 2) + ) + + ewkbTest("POINT(1 2 3)")( + hex"0101000080000000000000F03F00000000000000400000000000000840", + Point.xyz(1, 2, 3) + ) + + ewkbTest("POINT M(1 2 3)")( + hex"0101000040000000000000F03F00000000000000400000000000000840", + Point.xym(1, 2, 3) + ) + + ewkbTest("POINT (1 2 3 4)")( + hex"01010000C0000000000000F03F000000000000004000000000000008400000000000001040", + Point.xyzm(1, 2, 3, 4) + ) + + ewkbTest("SRID=32632;POINT(1 2)")( + hex"0101000020787F0000000000000000F03F0000000000000040", + Point(SRID(32632), Coordinate.xy(1, 2)) + ) + + ewkbTest("SRID=4326;POINT M(1 2 3)")( + hex"0101000060E6100000000000000000F03F00000000000000400000000000000840", + Point(SRID(4326), Coordinate.xym(1, 2, 3)) + ) + + ewkbTest("LINESTRING(-71.160281 42.258729,-71.160837 42.259113,-71.161144 42.25932)")( + hex"010200000003000000E44A3D0B42CA51C06EC328081E21454027BF45274BCA51C0F67B629D2A214540957CEC2E50CA51C07099D36531214540", + LineString( + Coordinate.xy(-71.160281, 42.258729), + Coordinate.xy(-71.160837, 42.259113), + Coordinate.xy(-71.161144, 42.25932) + ) + ) + + ewkbTest("SRID=4326;LINESTRING(-71.160281 42.258729,-71.160837 42.259113,-71.161144 42.25932)")( + hex"0102000020E610000003000000E44A3D0B42CA51C06EC328081E21454027BF45274BCA51C0F67B629D2A214540957CEC2E50CA51C07099D36531214540", + LineString( + SRID(4326), + Coordinate.xy(-71.160281, 42.258729), + Coordinate.xy(-71.160837, 42.259113), + Coordinate.xy(-71.161144, 42.25932) + ) + ) + + ewkbTest("POLYGON((-71.1776585052917 42.3902909739571,-71.1776820268866 42.3903701743239,-71.1776063012595 42.3903825660754,-71.1775826583081 42.3903033653531,-71.1776585052917 42.3902909739571))")( + hex"010300000001000000050000006285C7C15ECB51C0ED88FC0DF531454028A46F245FCB51C009075EA6F731454047DED1E65DCB51C0781C510EF83145404871A7835DCB51C0EBDAEE75F53145406285C7C15ECB51C0ED88FC0DF5314540", + Polygon( + None, + Dimension.TwoD, + LinearRing( + Coordinate.xy(-71.1776585052917, 42.3902909739571), + Coordinate.xy(-71.1776820268866, 42.3903701743239), + Coordinate.xy(-71.1776063012595, 42.3903825660754), + Coordinate.xy(-71.1775826583081, 42.3903033653531), + Coordinate.xy(-71.1776585052917, 42.3902909739571) + ).some, + Nil + ) + ) + + ewkbTest("SRID=4269;POLYGON((-71.1776585052917 42.3902909739571,-71.1776820268866 42.3903701743239,-71.1776063012595 42.3903825660754,-71.1775826583081 42.3903033653531,-71.1776585052917 42.3902909739571))")( + hex"0103000020AD10000001000000050000006285C7C15ECB51C0ED88FC0DF531454028A46F245FCB51C009075EA6F731454047DED1E65DCB51C0781C510EF83145404871A7835DCB51C0EBDAEE75F53145406285C7C15ECB51C0ED88FC0DF5314540", + Polygon( + SRID(4269).some, + Dimension.TwoD, + LinearRing( + Coordinate.xy(-71.1776585052917, 42.3902909739571), + Coordinate.xy(-71.1776820268866, 42.3903701743239), + Coordinate.xy(-71.1776063012595, 42.3903825660754), + Coordinate.xy(-71.1775826583081, 42.3903033653531), + Coordinate.xy(-71.1776585052917, 42.3902909739571) + ).some, + Nil + ) + ) + + ewkbTest("POLYGON((-15.66486 27.91996, -15.60610 27.91820, -15.60359 27.97169, -15.66586 27.97144,-15.66486 27.91996), (-15.65753 27.95894, -15.61610 27.95995, -15.61459 27.93157,-15.65477 27.93007,-15.65753 27.95894))")( + hex"010300000002000000050000004DD6A88768542FC0CFA0A17F82EB3B4011363CBD52362FC0EC2FBB270FEB3B40A2629CBF09352FC0A9D903ADC0F83B40DB6D179AEB542FC0B806B64AB0F83B404DD6A88768542FC0CFA0A17F82EB3B40050000001B47ACC5A7502FC084D382177DF53B4096218E75713B2FC092CB7F48BFF53B40B4E55C8AAB3A2FC04AEF1B5F7BEE3B40C8CD70033E4F2FC0A0FD481119EE3B401B47ACC5A7502FC084D382177DF53B40", + Polygon( + None, + Dimension.TwoD, + LinearRing( + Coordinate.xy(-15.66486, 27.91996), + Coordinate.xy(-15.60610, 27.91820), + Coordinate.xy(-15.60359, 27.97169), + Coordinate.xy(-15.66586, 27.97144), + Coordinate.xy(-15.66486, 27.91996) + ).some, + LinearRing( + Coordinate.xy(-15.65753, 27.95894), + Coordinate.xy(-15.61610, 27.95995), + Coordinate.xy(-15.61459, 27.93157), + Coordinate.xy(-15.65477, 27.93007), + Coordinate.xy(-15.65753, 27.95894) + ) :: Nil + ) + ) + + ewkbTest("MULTIPOINT (1 1,2 2,3 3)")( + hex"01 04000000030000000101000000000000000000F03F000000000000F03F010100000000000000000000400000000000000040010100000000000000000008400000000000000840", + MultiPoint( + Point.xy(1, 1), + Point.xy(2, 2), + Point.xy(3, 3) + ) + ) + + ewkbTest("SRID=4326;MULTIPOINT (1 1,2 2,3 3)")( + hex"0104000020E6100000030000000101000000000000000000F03F000000000000F03F010100000000000000000000400000000000000040010100000000000000000008400000000000000840", + MultiPoint( + SRID(4326), + Point.xy(1, 1), + Point.xy(2, 2), + Point.xy(3, 3) + ) + ) + + ewkbTest("MULTILINESTRING((0 0 0,1 1 0,1 2 1),(2 3 1,3 2 1,5 4 1))")( + hex"010500008002000000010200008003000000000000000000000000000000000000000000000000000000000000000000F03F000000000000F03F0000000000000000000000000000F03F0000000000000040000000000000F03F01020000800300000000000000000000400000000000000840000000000000F03F00000000000008400000000000000040000000000000F03F00000000000014400000000000001040000000000000F03F", + MultiLineString( + LineString( + Coordinate.xyz(0, 0, 0), + Coordinate.xyz(1, 1, 0), + Coordinate.xyz(1, 2, 1) + ), + LineString( + Coordinate.xyz(2, 3, 1), + Coordinate.xyz(3, 2, 1), + Coordinate.xyz(5, 4, 1) + ), + ) + ) + + ewkbTest("MULTIPOLYGON(((0 0 0,4 0 0,4 4 0,0 4 0,0 0 0),(1 1 0,2 1 0,2 2 0,1 2 0,1 1 0)),((-1 -1 0,-1 -2 0,-2 -2 0,-2 -1 0,-1 -1 0)))")( + hex"0106000080020000000103000080020000000500000000000000000000000000000000000000000000000000000000000000000010400000000000000000000000000000000000000000000010400000000000001040000000000000000000000000000000000000000000001040000000000000000000000000000000000000000000000000000000000000000005000000000000000000F03F000000000000F03F00000000000000000000000000000040000000000000F03F0000000000000000000000000000004000000000000000400000000000000000000000000000F03F00000000000000400000000000000000000000000000F03F000000000000F03F000000000000000001030000800100000005000000000000000000F0BF000000000000F0BF0000000000000000000000000000F0BF00000000000000C0000000000000000000000000000000C000000000000000C0000000000000000000000000000000C0000000000000F0BF0000000000000000000000000000F0BF000000000000F0BF0000000000000000", + MultiPolygon( + Polygon( + LinearRing( + Coordinate.xyz(0, 0, 0), + Coordinate.xyz(4, 0, 0), + Coordinate.xyz(4, 4, 0), + Coordinate.xyz(0, 4, 0), + Coordinate.xyz(0, 0, 0) + ), + LinearRing( + Coordinate.xyz(1, 1, 0), + Coordinate.xyz(2, 1, 0), + Coordinate.xyz(2, 2, 0), + Coordinate.xyz(1, 2, 0), + Coordinate.xyz(1, 1, 0) + ) + ), + Polygon( + LinearRing( + Coordinate.xyz(-1, -1, 0), + Coordinate.xyz(-1, -2, 0), + Coordinate.xyz(-2, -2, 0), + Coordinate.xyz(-2, -1, 0), + Coordinate.xyz(-1, -1, 0) + ) + ) + ) + ) + + ewkbTest("GEOMETRYCOLLECTIONM( POINTM(2 3 9), LINESTRINGM(2 3 4, 3 4 5) )")( + hex"0107000040020000000101000040000000000000004000000000000008400000000000002240010200004002000000000000000000004000000000000008400000000000001040000000000000084000000000000010400000000000001440", + GeometryCollection( + Point.xym(2, 3, 9), + LineString( + Coordinate.xym(2, 3, 4), + Coordinate.xym(3, 4, 5) + ) + ) + ) + + def ewkbTest(name: String)(expectedBytes: ByteVector, expected: Geometry) = + test(name) { + // encode + assertEquals(ewkb.codecs.geometry.encode(expected).require.toHex, expectedBytes.toHex) + + // decode + ewkb.codecs.geometry.decodeValue(expectedBytes.toBitVector).fold( + err => fail(s"[EWKB] Failed to decode geometry: $err"), + geometry => assertEquals(geometry, expected) + ) + } +} \ No newline at end of file diff --git a/modules/tests/shared/src/test/scala/postgis/ewkt/EWKTTest.scala b/modules/tests/shared/src/test/scala/postgis/ewkt/EWKTTest.scala new file mode 100644 index 000000000..4c3c04f29 --- /dev/null +++ b/modules/tests/shared/src/test/scala/postgis/ewkt/EWKTTest.scala @@ -0,0 +1,87 @@ +// Copyright (c) 2018-2021 by Rob Norris +// This software is licensed under the MIT License (MIT). +// For more information see LICENSE or https://opensource.org/licenses/MIT + +package tests.postgis.ewkt + +import skunk.postgis.ewkt.EWKT + +class EWKTTest extends munit.FunSuite { + + ewktTest("POINT EMPTY") + ewktTest("POINT Z EMPTY") + ewktTest("POINT M EMPTY") + ewktTest("POINTZM EMPTY") + ewktTest("POINT(0 0)") + ewktTest("POINT Z (1 2 3)") + ewktTest("POINT M (1 2 3)") + ewktTest("POINT ZM (1 2 3 4)") + ewktTest("POINT(1 2 3 4)") + + ewktTest("LINESTRING EMPTY") + ewktTest("LINESTRING Z EMPTY") + ewktTest("LINESTRING M EMPTY") + ewktTest("LINESTRING ZM EMPTY") + ewktTest("LINESTRING(0 0, 1 1)") + ewktTest("LINESTRING Z (0 0 2, 1 1 3)") + ewktTest("LINESTRING M (0 0 2, 1 1 3)") + ewktTest("LINESTRING ZM (0 0 2 3, 1 1 4 5)") + + ewktTest("POLYGON EMPTY") + ewktTest("POLYGON Z EMPTY") + ewktTest("POLYGON M EMPTY") + ewktTest("POLYGON ZM EMPTY") + ewktTest("POLYGON((0 0,1 0,1 1,0 1,0 0))") + ewktTest("POLYGON((0 0,10 0,10 10,0 10,0 0),(2 2,2 5,5 5,5 2,2 2))") + ewktTest("POLYGON Z ((0 0 1,10 0 2 ,10 10 2,0 10 2,0 0 1),(2 2 5 ,2 5 4,5 5 3,5 2 3,2 2 5))") + ewktTest("POLYGON M ((0 0 1,10 0 2 ,10 10 2,0 10 2,0 0 1),(2 2 5 ,2 5 4,5 5 3,5 2 3,2 2 5))") + ewktTest("POLYGON ZM ((0 0 1 -1,10 0 2 -2,10 10 2 -2,0 10 2 -4,0 0 1 -1),(2 2 5 0,2 5 4 1,5 5 3 2,5 2 3 1,2 2 5 0))") + + ewktTest("MULTIPOINT EMPTY") + ewktTest("MULTIPOINT Z EMPTY") + ewktTest("MULTIPOINT M EMPTY") + ewktTest("MULTIPOINT ZM EMPTY") + ewktTest("MULTIPOINT((0 0), 2 0)") + ewktTest("MULTIPOINT Z ((0 0 0), (2 0 1))") + ewktTest("MULTIPOINT M ((0 0 2), (2 0 1))") + ewktTest("MULTIPOINT ZM ((0 1 2 3), (3 2 1 0))") + + ewktTest("MULTILINESTRING EMPTY") + ewktTest("MULTILINESTRING Z EMPTY") + ewktTest("MULTILINESTRING M EMPTY") + ewktTest("MULTILINESTRING ZM EMPTY") + ewktTest("MULTILINESTRING((0 0, 2 0))") + ewktTest("MULTILINESTRING((0 0, 2 0), (1 1, 2 2))") + ewktTest("MULTILINESTRING Z ((0 0 1, 2 0 2), (1 1 3, 2 2 4))") + ewktTest("MULTILINESTRING M ((0 0 1, 2 0 2), (1 1 3, 2 2 4))") + ewktTest("MULTILINESTRING ZM ((0 0 1 5, 2 0 2 4), (1 1 3 3, 2 2 4 2))") + + ewktTest("MULTIPOLYGON EMPTY") + ewktTest("MULTIPOLYGON Z EMPTY") + ewktTest("MULTIPOLYGON M EMPTY") + ewktTest("MULTIPOLYGON ZM EMPTY") + ewktTest("MULTIPOLYGON(((0 0,10 0,10 10,0 10,0 0),(2 2,2 5,5 5,5 2,2 2)))") + ewktTest("MULTIPOLYGON Z (((0 0 3,10 0 3,10 10 3,0 10 3,0 0 3),(2 2 3,2 5 3,5 5 3,5 2 3,2 2 3)))") + ewktTest("MULTIPOLYGON M (((0 0 3,10 0 3,10 10 3,0 10 3,0 0 3),(2 2 3,2 5 3,5 5 3,5 2 3,2 2 3)))") + ewktTest("MULTIPOLYGON ZM (((0 0 3 2,10 0 3 2,10 10 3 2,0 10 3 2,0 0 3 2),(2 2 3 2,2 5 3 2,5 5 3 2,5 2 3 2,2 2 3 2)))") + ewktTest("MULTIPOLYGON(((0 0,4 0,4 4,0 4,0 0),(1 1,2 1,2 2,1 2,1 1)), ((-1 -1,-1 -2,-2 -2,-2 -1,-1 -1)))") + + ewktTest("GEOMETRYCOLLECTION EMPTY") + ewktTest("GEOMETRYCOLLECTION Z EMPTY") + ewktTest("GEOMETRYCOLLECTION M EMPTY") + ewktTest("GEOMETRYCOLLECTION ZM EMPTY") + ewktTest("GEOMETRYCOLLECTION ZM (POINT ZM (0 0 0 0),LINESTRING ZM (0 0 0 0,1 1 1 1))") + ewktTest("GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1))") + ewktTest("GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1),GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1)))") + ewktTest("GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1),POINT M EMPTY,GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1)))") + ewktTest("GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1),GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1),POINT M EMPTY,GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1))),POINT M EMPTY,GEOMETRYCOLLECTION M (POINT M (0 0 0),LINESTRING M (0 0 0,1 1 1)))") + + def ewktTest(ewkt: String) = + test(ewkt) { + EWKT.geometry.parse(ewkt).fold( + error => fail(s"[EWKT] Failed to parse [$ewkt]: ${error}"), + result => assert(result._1.isEmpty()) + ) + } + +} \ No newline at end of file