Skip to content

Commit 7ef16af

Browse files
committed
perf: optimize count=exact when there's no limits, offsets or db-max-rows
1 parent a482524 commit 7ef16af

File tree

3 files changed

+24
-21
lines changed

3 files changed

+24
-21
lines changed

src/PostgREST/Query.hs

Lines changed: 3 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -43,16 +43,16 @@ data MainQuery = MainQuery
4343

4444
mainQuery :: ActionPlan -> AppConfig -> ApiRequest -> AuthResult -> Maybe QualifiedIdentifier -> MainQuery
4545
mainQuery (NoDb _) _ _ _ _ = MainQuery mempty Nothing mempty (mempty, mempty, mempty) mempty
46-
mainQuery (Db plan) conf@AppConfig{..} apiReq@ApiRequest{iPreferences=Preferences{..}} authRes preReq =
46+
mainQuery (Db plan) conf@AppConfig{..} apiReq@ApiRequest{iTopLevelRange=range, iPreferences=Preferences{..}} authRes preReq =
4747
let genQ = MainQuery (PreQuery.txVarQuery plan conf authRes apiReq) (PreQuery.preReqQuery <$> preReq) in
4848
case plan of
4949
DbCrud _ WrappedReadPlan{..} ->
5050
let countQuery = QueryBuilder.readPlanToCountQuery wrReadPlan in
51-
genQ (Statements.mainRead wrReadPlan countQuery preferCount configDbMaxRows pMedia wrHandler) (mempty, mempty, mempty)
51+
genQ (Statements.mainRead wrReadPlan countQuery preferCount configDbMaxRows range pMedia wrHandler) (mempty, mempty, mempty)
5252
(if shouldExplainCount preferCount then Just (Statements.postExplain countQuery) else Nothing)
5353
DbCrud _ MutateReadPlan{..} ->
5454
genQ (Statements.mainWrite mrReadPlan mrMutatePlan pMedia mrHandler preferRepresentation preferResolution) (mempty, mempty, mempty) mempty
5555
DbCrud _ CallReadPlan{..} ->
56-
genQ (Statements.mainCall crProc crCallPlan crReadPlan preferCount pMedia crHandler) (mempty, mempty, mempty) mempty
56+
genQ (Statements.mainCall crProc crCallPlan crReadPlan preferCount configDbMaxRows range pMedia crHandler) (mempty, mempty, mempty) mempty
5757
MayUseDb InspectPlan{ipSchema=tSchema} ->
5858
genQ mempty (SqlFragment.accessibleTables tSchema, SqlFragment.accessibleFuncs tSchema, SqlFragment.schemaDescription tSchema) mempty

src/PostgREST/Query/SqlFragment.hs

Lines changed: 9 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -487,15 +487,15 @@ pgFmtGroup _ CoercibleSelectField{csAggFunction=Just _} = Nothing
487487
pgFmtGroup _ CoercibleSelectField{csAlias=Just alias, csAggFunction=Nothing} = Just $ pgFmtIdent alias
488488
pgFmtGroup qi CoercibleSelectField{csField=fld, csAlias=Nothing, csAggFunction=Nothing} = Just $ pgFmtField qi fld
489489

490-
countF :: SQL.Snippet -> Bool -> (SQL.Snippet, SQL.Snippet)
491-
countF countQuery shouldCount =
492-
if shouldCount
493-
then (
494-
", pgrst_source_count AS (" <> countQuery <> ")"
495-
, "(SELECT pg_catalog.count(*) FROM pgrst_source_count)" )
496-
else (
497-
mempty
498-
, "null::bigint")
490+
countF :: SQL.Snippet -> SQL.Snippet -> Bool -> Maybe Integer -> NonnegRange -> (SQL.Snippet, SQL.Snippet)
491+
countF countQuery pageCountSelect shouldCount maxRows range
492+
| shouldCount = if isJust maxRows || range /= allRange
493+
then ( ", pgrst_source_count AS (" <> countQuery <> ")"
494+
, "(SELECT pg_catalog.count(*) FROM pgrst_source_count)" )
495+
-- When there are no db-max-rows and limits/offsets, the total count will be the same as the page count,
496+
-- so we use the same page count here to avoid doing a separate aggregated count.
497+
else ( mempty, pageCountSelect )
498+
| otherwise = ( mempty, "null::bigint" )
499499

500500
pageCountSelectF :: Maybe Routine -> SQL.Snippet
501501
pageCountSelectF rout =

src/PostgREST/Query/Statements.hs

Lines changed: 12 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@ import PostgREST.Plan.MutatePlan as MTPlan
2020
import PostgREST.Plan.ReadPlan
2121
import PostgREST.Query.QueryBuilder
2222
import PostgREST.Query.SqlFragment
23+
import PostgREST.RangeQuery (NonnegRange)
2324
import PostgREST.SchemaCache.Routine (MediaHandler (..), Routine)
2425

2526
import Protolude
@@ -63,50 +64,52 @@ mainWrite rPlan mtplan mt handler rep resolution = mtSnippet mt snippet
6364
_ -> (False,False, mempty);
6465

6566
mainRead :: ReadPlanTree -> SQL.Snippet -> Maybe PreferCount -> Maybe Integer ->
66-
MediaType -> MediaHandler -> SQL.Snippet
67-
mainRead rPlan countQuery pCount maxRows mt handler = mtSnippet mt snippet
67+
NonnegRange -> MediaType -> MediaHandler -> SQL.Snippet
68+
mainRead rPlan countQuery pCount maxRows range mt handler = mtSnippet mt snippet
6869
where
6970
snippet =
7071
"WITH " <> sourceCTE <> " AS ( " <> selectQuery <> " ) " <>
7172
countCTEF <> " " <>
7273
"SELECT " <>
7374
countResultF <> " AS total_result_set, " <>
74-
pageCountSelectF Nothing <> " AS page_total, " <>
75+
pageCountSelect <> " AS page_total, " <>
7576
handlerF Nothing handler <> " AS body, " <>
7677
responseHeadersF <> " AS response_headers, " <>
7778
responseStatusF <> " AS response_status, " <>
7879
"''" <> " AS response_inserted " <>
7980
"FROM ( SELECT * FROM " <> sourceCTE <> " ) _postgrest_t"
8081

81-
(countCTEF, countResultF) = countF countQ $ shouldCount pCount
82+
(countCTEF, countResultF) = countF countQ pageCountSelect (shouldCount pCount) maxRows range
8283
selectQuery = readPlanToQuery rPlan
84+
pageCountSelect = pageCountSelectF Nothing
8385
countQ =
8486
if pCount == Just EstimatedCount then
8587
-- LIMIT maxRows + 1 so we can determine below that maxRows was surpassed
8688
limitedQuery countQuery ((+ 1) <$> maxRows)
8789
else
8890
countQuery
8991

90-
mainCall :: Routine -> CallPlan -> ReadPlanTree -> Maybe PreferCount ->
91-
MediaType -> MediaHandler -> SQL.Snippet
92-
mainCall rout cPlan rPlan pCount mt handler = mtSnippet mt snippet
92+
mainCall :: Routine -> CallPlan -> ReadPlanTree -> Maybe PreferCount -> Maybe Integer ->
93+
NonnegRange-> MediaType -> MediaHandler -> SQL.Snippet
94+
mainCall rout cPlan rPlan pCount maxRows range mt handler = mtSnippet mt snippet
9395
where
9496
snippet =
9597
"WITH " <> sourceCTE <> " AS (" <> callProcQuery <> ") " <>
9698
countCTEF <>
9799
"SELECT " <>
98100
countResultF <> " AS total_result_set, " <>
99-
pageCountSelectF (Just rout) <> " AS page_total, " <>
101+
pageCountSelect <> " AS page_total, " <>
100102
handlerF (Just rout) handler <> " AS body, " <>
101103
responseHeadersF <> " AS response_headers, " <>
102104
responseStatusF <> " AS response_status, " <>
103105
"''" <> " AS response_inserted " <>
104106
"FROM (" <> selectQuery <> ") _postgrest_t"
105107

106-
(countCTEF, countResultF) = countF countQuery $ shouldCount pCount
108+
(countCTEF, countResultF) = countF countQuery pageCountSelect (shouldCount pCount) maxRows range
107109
selectQuery = readPlanToQuery rPlan
108110
callProcQuery = callPlanToQuery cPlan
109111
countQuery = readPlanToCountQuery rPlan
112+
pageCountSelect = pageCountSelectF (Just rout)
110113

111114
-- This occurs after the main query runs, that's why it's prefixed with "post"
112115
postExplain :: SQL.Snippet -> SQL.Snippet

0 commit comments

Comments
 (0)