233 lines
11 KiB
Haskell
233 lines
11 KiB
Haskell
module Feature.Query.PostGISSpec where
|
|
|
|
import Network.Wai (Application)
|
|
|
|
import Network.HTTP.Types
|
|
import Test.Hspec
|
|
import Test.Hspec.Wai
|
|
import Test.Hspec.Wai.JSON
|
|
|
|
import PostgREST.Config.PgVersion (PgVersion, pgVersion100)
|
|
|
|
import Protolude hiding (get)
|
|
import SpecHelper
|
|
|
|
spec :: PgVersion -> SpecWith ((), Application)
|
|
spec actualPgVersion = describe "PostGIS features" $
|
|
context "GeoJSON output" $ do
|
|
it "works for a table that has a geometry column" $
|
|
request methodGet "/shops"
|
|
[("Accept", "application/geo+json")] "" `shouldRespondWith`
|
|
[json| {
|
|
"type" : "FeatureCollection",
|
|
"features" : [
|
|
{"type": "Feature", "geometry": {"type":"Point","coordinates":[-71.10044,42.373695]}, "properties": {"id": 1, "address": "1369 Cambridge St"}}
|
|
, {"type": "Feature", "geometry": {"type":"Point","coordinates":[-71.10543,42.366432]}, "properties": {"id": 2, "address": "757 Massachusetts Ave"}}
|
|
, {"type": "Feature", "geometry": {"type":"Point","coordinates":[-71.081924,42.36437]}, "properties": {"id": 3, "address": "605 W Kendall St"}}
|
|
]} |]
|
|
{ matchHeaders = ["Content-Type" <:> "application/geo+json; charset=utf-8"] }
|
|
|
|
it "fails for a table that doesn't have a geometry column" $
|
|
request methodGet "/projects"
|
|
[("Accept", "application/geo+json")] "" `shouldRespondWith`
|
|
[json| {"hint":null,"details":null,"code":"22023","message":"geometry column is missing"} |]
|
|
{ matchStatus = 400
|
|
, matchHeaders = [matchContentTypeJson]
|
|
}
|
|
|
|
it "gives an empty features array on no rows" $
|
|
request methodGet "/shops?id=gt.3"
|
|
[("Accept", "application/geo+json")] "" `shouldRespondWith`
|
|
[json| {
|
|
"type" : "FeatureCollection",
|
|
"features" : []} |]
|
|
{ matchHeaders = ["Content-Type" <:> "application/geo+json; charset=utf-8"] }
|
|
|
|
it "must include the geometry column when using ?select" $
|
|
request methodGet "/shops?select=id,shop_geom&id=eq.1"
|
|
[("Accept", "application/geo+json")] "" `shouldRespondWith`
|
|
[json| {
|
|
"type" : "FeatureCollection",
|
|
"features" : [
|
|
{"type": "Feature", "geometry": {"type":"Point","coordinates":[-71.10044,42.373695]}, "properties": {"id": 1}}
|
|
] }|]
|
|
{ matchHeaders = ["Content-Type" <:> "application/geo+json; charset=utf-8"] }
|
|
|
|
it "works with resource embedding" $
|
|
request methodGet "/shops?select=*,shop_bles(*)&id=eq.1"
|
|
[("Accept", "application/geo+json")] "" `shouldRespondWith`
|
|
(if actualPgVersion >= pgVersion100
|
|
then [json| {
|
|
"type": "FeatureCollection",
|
|
"features": [
|
|
{
|
|
"type": "Feature",
|
|
"geometry": { "coordinates": [ -71.10044, 42.373695 ], "type": "Point" },
|
|
"properties": {
|
|
"address": "1369 Cambridge St", "id": 1,
|
|
"shop_bles": [
|
|
{ "id": 1, "name": "Beacon-1", "shop_id": 1 ,
|
|
"coords": { "coordinates": [ -71.10044, 42.373695 ], "crs": { "properties": { "name": "EPSG:4326" }, "type": "name" }, "type": "Point" },
|
|
"range_area": {
|
|
"coordinates": [ [ [ -71.10045254230499, 42.37387083326593 ], [ -71.10048070549963, 42.37377126199953 ], [ -71.10039688646793, 42.37375838212269 ], [ -71.10037006437777, 42.37385844878863 ], [ -71.10045254230499, 42.37387083326593 ] ] ],
|
|
"crs": { "properties": { "name": "EPSG:4326" }, "type": "name" }, "type": "Polygon" }
|
|
},
|
|
{ "coords": { "coordinates": [ -71.10044, 42.373695 ], "crs": { "properties": { "name": "EPSG:4326" }, "type": "name" }, "type": "Point" },
|
|
"id": 2, "name": "Beacon-2", "shop_id": 1,
|
|
"range_area": {
|
|
"coordinates": [ [ [ -71.10034391283989, 42.37385299961788 ], [ -71.10036939382553, 42.373756895982865 ], [ -71.1002916097641, 42.373745997623224 ], [ -71.1002641171217, 42.37384408279195 ], [ -71.10034391283989, 42.37385299961788 ] ] ],
|
|
"crs": { "properties": { "name": "EPSG:4326" }, "type": "name" }, "type": "Polygon" }
|
|
}
|
|
]
|
|
}
|
|
}
|
|
]
|
|
}|]
|
|
else [json| {
|
|
"type": "FeatureCollection",
|
|
"features": [
|
|
{
|
|
"type": "Feature",
|
|
"geometry": { "coordinates": [ -71.10044, 42.373695 ], "type": "Point" },
|
|
"properties": {
|
|
"address": "1369 Cambridge St", "id": 1,
|
|
"shop_bles": [
|
|
{ "id": 1, "name": "Beacon-1", "shop_id": 1 ,
|
|
"coords": { "coordinates": [ -71.10044, 42.373695 ], "type": "Point" },
|
|
"range_area": {
|
|
"coordinates": [ [ [ -71.10045254230499, 42.37387083326593 ], [ -71.10048070549963, 42.37377126199953 ], [ -71.10039688646793, 42.37375838212269 ], [ -71.10037006437777, 42.37385844878863 ], [ -71.10045254230499, 42.37387083326593 ] ] ],
|
|
"type": "Polygon" }
|
|
},
|
|
{ "coords": { "coordinates": [ -71.10044, 42.373695 ], "type": "Point" },
|
|
"id": 2, "name": "Beacon-2", "shop_id": 1,
|
|
"range_area": {
|
|
"coordinates": [ [ [ -71.10034391283989, 42.37385299961788 ], [ -71.10036939382553, 42.373756895982865 ], [ -71.1002916097641, 42.373745997623224 ], [ -71.1002641171217, 42.37384408279195 ], [ -71.10034391283989, 42.37385299961788 ] ] ],
|
|
"type": "Polygon" }
|
|
}
|
|
]
|
|
}
|
|
}
|
|
]
|
|
}|])
|
|
{ matchHeaders = ["Content-Type" <:> "application/geo+json; charset=utf-8"] }
|
|
|
|
it "works with RPC" $
|
|
request methodGet "/rpc/get_shop?id=1"
|
|
[("Accept", "application/geo+json")] "" `shouldRespondWith`
|
|
[json|{
|
|
"type" : "FeatureCollection",
|
|
"features" : [
|
|
{"type": "Feature", "geometry": {"type":"Point","coordinates":[-71.10044,42.373695]}, "properties": {"id": 1, "address": "1369 Cambridge St"} }
|
|
]
|
|
}|]
|
|
{ matchHeaders = ["Content-Type" <:> "application/geo+json; charset=utf-8"] }
|
|
|
|
it "works with Prefer: return=representation after POST" $
|
|
request methodPost "/shops"
|
|
[("Accept", "application/geo+json"), ("Prefer", "return=representation")] [json|
|
|
{"id": 4, "address": "1354 Massachusetts Ave", "shop_geom": "SRID=4326;POINT(-71.11834 42.373238)"}
|
|
|] `shouldRespondWith`
|
|
[json|{
|
|
"type": "FeatureCollection",
|
|
"features": [
|
|
{
|
|
"type": "Feature",
|
|
"geometry": { "coordinates": [ -71.11834, 42.373238 ], "type": "Point" },
|
|
"properties": { "address": "1354 Massachusetts Ave", "id": 4 }
|
|
}
|
|
]
|
|
}|]
|
|
{ matchStatus = 201
|
|
, matchHeaders = ["Content-Type" <:> "application/geo+json; charset=utf-8"]
|
|
}
|
|
|
|
it "works with Prefer: return=representation after PATCH" $
|
|
request methodPatch "/shops?id=eq.3"
|
|
[("Accept", "application/geo+json"), ("Prefer", "return=representation")]
|
|
[json| { "address": "1354 Massachusetts Avenue"} |]
|
|
`shouldRespondWith`
|
|
[json|{
|
|
"type": "FeatureCollection",
|
|
"features": [
|
|
{
|
|
"type": "Feature",
|
|
"geometry": { "coordinates": [-71.081924,42.36437], "type": "Point" },
|
|
"properties": { "address": "1354 Massachusetts Avenue", "id": 3 }
|
|
}
|
|
]
|
|
}|]
|
|
{ matchStatus = 200
|
|
, matchHeaders = ["Content-Type" <:> "application/geo+json; charset=utf-8"]
|
|
}
|
|
|
|
it "works with Prefer: return=representation after PUT" $
|
|
request methodPut "/shops?id=eq.3"
|
|
[("Accept", "application/geo+json"), ("Prefer", "return=representation")]
|
|
[json| { "id": 3, "address": "1354 Massachusetts Avenue"} |]
|
|
`shouldRespondWith`
|
|
[json|{
|
|
"type": "FeatureCollection",
|
|
"features": [
|
|
{
|
|
"type": "Feature",
|
|
"geometry": { "coordinates": [-71.081924,42.36437], "type": "Point" },
|
|
"properties": { "address": "1354 Massachusetts Avenue", "id": 3 }
|
|
}
|
|
]
|
|
}|]
|
|
{ matchStatus = 200
|
|
, matchHeaders = ["Content-Type" <:> "application/geo+json; charset=utf-8"]
|
|
}
|
|
|
|
it "works with Prefer: return=representation after DELETE" $
|
|
request methodDelete "/shops?id=eq.3"
|
|
[("Accept", "application/geo+json"), ("Prefer", "return=representation")] "" `shouldRespondWith`
|
|
[json|{
|
|
"type" : "FeatureCollection",
|
|
"features" : [
|
|
{"type": "Feature", "geometry": {"type":"Point","coordinates":[-71.081924,42.36437]}, "properties": {"id": 3, "address": "605 W Kendall St"}}
|
|
]
|
|
}|]
|
|
{ matchStatus = 200
|
|
, matchHeaders = ["Content-Type" <:> "application/geo+json; charset=utf-8"]
|
|
}
|
|
|
|
context "multiple geometry columns" $
|
|
it "can select the geometry column to get as a feature with ?select" $ do
|
|
request methodGet "/shop_bles?select=id,name,coords&id=eq.1"
|
|
[("Accept", "application/geo+json")] "" `shouldRespondWith`
|
|
[json|{
|
|
"type" : "FeatureCollection",
|
|
"features" : [
|
|
{"type": "Feature",
|
|
"geometry": {"type":"Point","coordinates":[-71.10044,42.373695]},
|
|
"properties": {"id": 1, "name": "Beacon-1"}
|
|
}
|
|
]}|]
|
|
{ matchHeaders = ["Content-Type" <:> "application/geo+json; charset=utf-8"] }
|
|
|
|
request methodGet "/shop_bles?select=id,name,range_area&id=eq.1"
|
|
[("Accept", "application/geo+json")] "" `shouldRespondWith`
|
|
[json|{
|
|
"type" : "FeatureCollection",
|
|
"features" : [
|
|
{"type": "Feature",
|
|
"geometry": {"type":"Polygon","coordinates":[[[-71.100452542,42.373870833],[-71.100480705,42.373771262],[-71.100396886,42.373758382],[-71.100370064,42.373858449],[-71.100452542,42.373870833]]]},
|
|
"properties": {"id": 1, "name": "Beacon-1"}}
|
|
]}|]
|
|
{ matchHeaders = ["Content-Type" <:> "application/geo+json; charset=utf-8"] }
|
|
|
|
it "gets the geojson geometry object with the regular application/json output" $
|
|
request methodGet "/shops?id=eq.1" [] "" `shouldRespondWith`
|
|
(if actualPgVersion >= pgVersion100
|
|
then [json|[{
|
|
"id":1,"address":"1369 Cambridge St",
|
|
"shop_geom":{"type":"Point","crs":{"type":"name","properties":{"name":"EPSG:4326"}},"coordinates":[-71.10044,42.373695]}
|
|
}]|]
|
|
else [json|[{
|
|
"address": "1369 Cambridge St",
|
|
"id": 1,
|
|
"shop_geom": { "coordinates": [ -71.10044, 42.373695 ], "type": "Point" }
|
|
}]|])
|
|
{ matchHeaders = [matchContentTypeJson] }
|