graphql-engine/server/src-lib/Hasura/Backends/Postgres/Translate/Select/Internal/JoinTree.hs
Samir Talwar dd46aa6715 server: Preserve ordering when possible, and sort when it's not.
When upgrading to GHC v9.4, we noticed a number of failures because the sort order of HashMaps has changed. With this changeset, I am endeavoring to mitigate this now and in the future.

This makes one of two changes in a few areas where we depend on the sort order of elements in a `HashMap`:

  1. the ordering of the request is preserved with `InsOrdHashMap`, or
  2. we sort the data after retrieving it.

Fortunately, we do not do this anywhere where we _must_ preserve order; it's "just" descriptions, error messages, and OpenAPI metadata. The main problem is that tests are likely to fail each time we upgrade GHC (or whatever is providing the hash seed).

[NDAT-705]: https://hasurahq.atlassian.net/browse/NDAT-705?atlOrigin=eyJpIjoiNWRkNTljNzYxNjVmNDY3MDlhMDU5Y2ZhYzA5YTRkZjUiLCJwIjoiZ2l0aHViLWNvbS1KU1cifQ

PR-URL: https://github.com/hasura/graphql-engine-mono/pull/9390
GitOrigin-RevId: 84503e029b44094edbbc298651744bc2843c15f3
2023-06-02 09:31:26 +00:00

116 lines
3.8 KiB
Haskell

-- | Stuff gutted from Translate.Select
module Hasura.Backends.Postgres.Translate.Select.Internal.JoinTree
( withWriteJoinTree,
withWriteObjectRelation,
withWriteArrayRelation,
withWriteArrayConnection,
withWriteComputedFieldTableSet,
)
where
import Control.Monad.Writer.Strict
import Data.HashMap.Strict qualified as HashMap
import Hasura.Backends.Postgres.SQL.DML qualified as S
import Hasura.Backends.Postgres.Translate.Types
import Hasura.Prelude
-- | This is the lowest level function which deals with @MonadWriter SelectWriter@, which contains @JoinTree@ whose
-- purpose is to essentially create the selection tree across relationships.
--
-- Each type of relationship uses a different kind of update function; see
-- 'withWriteObjectRelation', 'withWriteArrayRelation', 'withWriteArrayConnection',
-- and 'withWriteComputedFieldTableSet'.
--
-- See the definition of 'JoinTree' for details before diving further
-- (particularly its components and Monoid instance).
withWriteJoinTree ::
(MonadWriter SelectWriter m) =>
(JoinTree -> b -> JoinTree) ->
m (a, b) ->
m a
withWriteJoinTree joinTreeUpdater action =
pass $ do
(out, result) <- action
let fromSelectWriter =
mapJoinTree (`joinTreeUpdater` result)
pure (out, fromSelectWriter)
-- | change the `JoinTree` inside a `SelectWriter`
mapJoinTree :: (JoinTree -> JoinTree) -> SelectWriter -> SelectWriter
mapJoinTree f sw = sw {_swJoinTree = f (_swJoinTree sw)}
withWriteObjectRelation ::
(MonadWriter SelectWriter m) =>
m
( ObjectRelationSource,
InsOrdHashMap S.ColumnAlias S.SQLExp,
a
) ->
m a
withWriteObjectRelation action =
withWriteJoinTree updateJoinTree $ do
(source, nodeExtractors, out) <- action
pure (out, (source, nodeExtractors))
where
updateJoinTree joinTree (source, nodeExtractors) =
let selectNode = SelectNode nodeExtractors joinTree
in mempty {_jtObjectRelations = HashMap.singleton source selectNode}
withWriteArrayRelation ::
(MonadWriter SelectWriter m) =>
m
( ArrayRelationSource,
S.Extractor,
InsOrdHashMap S.ColumnAlias S.SQLExp,
a
) ->
m a
withWriteArrayRelation action =
withWriteJoinTree updateJoinTree $ do
(source, topExtractor, nodeExtractors, out) <- action
pure (out, (source, topExtractor, nodeExtractors))
where
updateJoinTree joinTree (source, topExtractor, nodeExtractors) =
let arraySelectNode =
MultiRowSelectNode [topExtractor]
$ SelectNode nodeExtractors joinTree
in mempty {_jtArrayRelations = HashMap.singleton source arraySelectNode}
withWriteArrayConnection ::
(MonadWriter SelectWriter m) =>
m
( ArrayConnectionSource,
S.Extractor,
InsOrdHashMap S.ColumnAlias S.SQLExp,
a
) ->
m a
withWriteArrayConnection action =
withWriteJoinTree updateJoinTree $ do
(source, topExtractor, nodeExtractors, out) <- action
pure (out, (source, topExtractor, nodeExtractors))
where
updateJoinTree joinTree (source, topExtractor, nodeExtractors) =
let arraySelectNode =
MultiRowSelectNode [topExtractor]
$ SelectNode nodeExtractors joinTree
in mempty {_jtArrayConnections = HashMap.singleton source arraySelectNode}
withWriteComputedFieldTableSet ::
(MonadWriter SelectWriter m) =>
m
( ComputedFieldTableSetSource,
S.Extractor,
InsOrdHashMap S.ColumnAlias S.SQLExp,
a
) ->
m a
withWriteComputedFieldTableSet action =
withWriteJoinTree updateJoinTree $ do
(source, topExtractor, nodeExtractors, out) <- action
pure (out, (source, topExtractor, nodeExtractors))
where
updateJoinTree joinTree (source, topExtractor, nodeExtractors) =
let selectNode = MultiRowSelectNode [topExtractor] $ SelectNode nodeExtractors joinTree
in mempty {_jtComputedFieldTableSets = HashMap.singleton source selectNode}