From 37a3f818ddc2380e5930489cc00f74f70b9c6113 Mon Sep 17 00:00:00 2001 From: Laurence Isla Date: Sat, 16 Sep 2023 15:02:59 -0500 Subject: [PATCH] fix: error when requesting "Prefer: count=" with null filters on embedded resources --- src/PostgREST/Query/QueryBuilder.hs | 14 ++- src/PostgREST/Query/SqlFragment.hs | 1 + test/spec/Feature/Query/RelatedQueriesSpec.hs | 103 ++++++++++++++++++ 3 files changed, 117 insertions(+), 1 deletion(-) diff --git a/src/PostgREST/Query/QueryBuilder.hs b/src/PostgREST/Query/QueryBuilder.hs index fb6408a2a4..02a4a8926f 100644 --- a/src/PostgREST/Query/QueryBuilder.hs +++ b/src/PostgREST/Query/QueryBuilder.hs @@ -203,7 +203,7 @@ readPlanToCountQuery (Node ReadPlan{from=mainQi, fromAlias=tblAlias, where_=logi then mempty else " WHERE " ) <> intercalateSnippet " AND " ( - map (pgFmtLogicTree qi) logicForest ++ + map (pgFmtLogicTreeCount qi) logicForest ++ map pgFmtJoinCondition relJoinConds ++ subQueries ) @@ -216,6 +216,18 @@ readPlanToCountQuery (Node ReadPlan{from=mainQi, fromAlias=tblAlias, where_=logi if joinType == Just JTInner then ("EXISTS (" <> readPlanToCountQuery readReq <> " )"):rest else rest + findNullEmbedRel fld = find (\(Node ReadPlan{relAggAlias} _) -> fld == relAggAlias) forest + + -- https://github.com/PostgREST/postgrest/pull/2930#discussion_r1325293698 + pgFmtLogicTreeCount :: QualifiedIdentifier -> CoercibleLogicTree -> SQL.Snippet + pgFmtLogicTreeCount qiCount (CoercibleExpr hasNot op frst) = SQL.sql notOp <> " (" <> intercalateSnippet (opSql op) (pgFmtLogicTreeCount qiCount <$> frst) <> ")" + where + notOp = if hasNot then "NOT" else mempty + opSql And = " AND " + opSql Or = " OR " + pgFmtLogicTreeCount _ (CoercibleStmnt (CoercibleFilterNullEmbed hasNot fld)) = + maybe mempty (\x -> (if not hasNot then "NOT " else mempty) <> "EXISTS (" <> readPlanToCountQuery x <> ")") (findNullEmbedRel fld) + pgFmtLogicTreeCount qiCount (CoercibleStmnt flt) = pgFmtFilter qiCount flt limitedQuery :: SQL.Snippet -> Maybe Integer -> SQL.Snippet limitedQuery query maxRows = query <> SQL.sql (maybe mempty (\x -> " LIMIT " <> BS.pack (show x)) maxRows) diff --git a/src/PostgREST/Query/SqlFragment.hs b/src/PostgREST/Query/SqlFragment.hs index a56abdce35..f2d21efce6 100644 --- a/src/PostgREST/Query/SqlFragment.hs +++ b/src/PostgREST/Query/SqlFragment.hs @@ -15,6 +15,7 @@ module PostgREST.Query.SqlFragment , mutRangeF , orderF , pgFmtColumn + , pgFmtFilter , pgFmtIdent , pgFmtJoinCondition , pgFmtLogicTree diff --git a/test/spec/Feature/Query/RelatedQueriesSpec.hs b/test/spec/Feature/Query/RelatedQueriesSpec.hs index 0b83cbb680..de0dd8f2d0 100644 --- a/test/spec/Feature/Query/RelatedQueriesSpec.hs +++ b/test/spec/Feature/Query/RelatedQueriesSpec.hs @@ -2,6 +2,7 @@ module Feature.Query.RelatedQueriesSpec where import Network.Wai (Application) +import Network.HTTP.Types import Test.Hspec import Test.Hspec.Wai import Test.Hspec.Wai.JSON @@ -275,3 +276,105 @@ spec = describe "related queries" $ do { matchStatus = 200 , matchHeaders = [matchContentTypeJson] } + + it "works with count=exact" $ do + request methodGet "/projects?select=name,clients(name)&clients=not.is.null" + [("Prefer", "count=exact")] "" + `shouldRespondWith` + [json|[ + {"name":"Windows 7", "clients":{"name":"Microsoft"}}, + {"name":"Windows 10", "clients":{"name":"Microsoft"}}, + {"name":"IOS", "clients":{"name":"Apple"}}, + {"name":"OSX", "clients":{"name":"Apple"}} + ]|] + { matchStatus = 200 + , matchHeaders = [ matchContentTypeJson + , "Content-Range" <:> "0-3/4" ] + } + request methodGet "/projects?select=name,clients()&clients=is.null" + [("Prefer", "count=exact")] "" + `shouldRespondWith` + [json|[{"name":"Orphan"}]|] + { matchStatus = 200 + , matchHeaders = [ matchContentTypeJson + , "Content-Range" <:> "0-0/1" ] + } + request methodGet "/client?select=*,clientinfo(),contact()&clientinfo.other=ilike.*main*&contact.name=ilike.*tabby*&or=(clientinfo.not.is.null,contact.not.is.null)" + [("Prefer", "count=exact")] "" + `shouldRespondWith` + [json|[ + {"id":1,"name":"Walmart"}, + {"id":2,"name":"Target"} + ]|] + { matchStatus = 200 + , matchHeaders = [ matchContentTypeJson + , "Content-Range" <:> "0-1/2" ] + } + + it "works with count=planned" $ do + request methodGet "/projects?select=name,clients(name)&clients=not.is.null" + [("Prefer", "count=planned")] "" + `shouldRespondWith` + [json|[ + {"name":"Windows 7", "clients":{"name":"Microsoft"}}, + {"name":"Windows 10", "clients":{"name":"Microsoft"}}, + {"name":"IOS", "clients":{"name":"Apple"}}, + {"name":"OSX", "clients":{"name":"Apple"}} + ]|] + { matchStatus = 206 + , matchHeaders = [ matchContentTypeJson + , "Content-Range" <:> "0-3/1200" ] + } + request methodGet "/projects?select=name,clients()&clients=is.null" + [("Prefer", "count=planned")] "" + `shouldRespondWith` + [json|[{"name":"Orphan"}]|] + { matchStatus = 200 + , matchHeaders = [ matchContentTypeJson + , "Content-Range" <:> "0-0/1" ] + } + request methodGet "/client?select=*,clientinfo(),contact()&clientinfo.other=ilike.*main*&contact.name=ilike.*tabby*&or=(clientinfo.not.is.null,contact.not.is.null)" + [("Prefer", "count=planned")] "" + `shouldRespondWith` + [json|[ + {"id":1,"name":"Walmart"}, + {"id":2,"name":"Target"} + ]|] + { matchStatus = 206 + , matchHeaders = [ matchContentTypeJson + , "Content-Range" <:> "0-1/952" ] + } + + it "works with count=estimated" $ do + request methodGet "/projects?select=name,clients(name)&clients=not.is.null" + [("Prefer", "count=estimated")] "" + `shouldRespondWith` + [json|[ + {"name":"Windows 7", "clients":{"name":"Microsoft"}}, + {"name":"Windows 10", "clients":{"name":"Microsoft"}}, + {"name":"IOS", "clients":{"name":"Apple"}}, + {"name":"OSX", "clients":{"name":"Apple"}} + ]|] + { matchStatus = 206 + , matchHeaders = [ matchContentTypeJson + , "Content-Range" <:> "0-3/1200" ] + } + request methodGet "/projects?select=name,clients()&clients=is.null" + [("Prefer", "count=estimated")] "" + `shouldRespondWith` + [json|[{"name":"Orphan"}]|] + { matchStatus = 200 + , matchHeaders = [ matchContentTypeJson + , "Content-Range" <:> "0-0/1" ] + } + request methodGet "/client?select=*,clientinfo(),contact()&clientinfo.other=ilike.*main*&contact.name=ilike.*tabby*&or=(clientinfo.not.is.null,contact.not.is.null)" + [("Prefer", "count=estimated")] "" + `shouldRespondWith` + [json|[ + {"id":1,"name":"Walmart"}, + {"id":2,"name":"Target"} + ]|] + { matchStatus = 206 + , matchHeaders = [ matchContentTypeJson + , "Content-Range" <:> "0-1/952" ] + }