mirror of
https://github.com/hasura/graphql-engine.git
synced 2024-12-17 12:31:52 +03:00
9bd5826020
* allow customizing GraphQL root field names, close #981 * document v2 track_table API in reference * support customising column field names in GraphQL schema * [docs] add custom column fields doc in API reference * add tests * rename 'ColField' to 'ColumnField' * embed column's graphql field in 'PGColumnInfo' -> Value constructor of 'PGCol' is not exposed -> Using 'parseJSON' to construct 'PGCol' in 'FromJSON' instances * avoid using 'Maybe TableConfig' * refactors & 'custom_column_fields' -> 'custom_column_names' * cli-test: add configuration field in metadata export test * update expected keys in `FromJSON` instance of `TableMeta` * use `buildSchemaCacheFor` to update configuration in v2 track_table * remove 'GraphQLName' type and use 'isValidName' exposed from parser lib * point graphql-parser-hs library git repo to hasura * support 'set_table_custom_fields' query API & added docs and tests
119 lines
3.9 KiB
Haskell
119 lines
3.9 KiB
Haskell
module Hasura.RQL.DML.Mutation
|
|
( Mutation(..)
|
|
, runMutation
|
|
, mutateAndFetchCols
|
|
, mkSelCTEFromColVals
|
|
)
|
|
where
|
|
|
|
import Hasura.Prelude
|
|
|
|
import qualified Data.HashMap.Strict as Map
|
|
import qualified Data.Sequence as DS
|
|
import qualified Database.PG.Query as Q
|
|
|
|
import qualified Hasura.SQL.DML as S
|
|
|
|
import Hasura.EncJSON
|
|
import Hasura.RQL.DML.Internal
|
|
import Hasura.RQL.DML.Returning
|
|
import Hasura.RQL.DML.Select
|
|
import Hasura.RQL.Instances ()
|
|
import Hasura.RQL.Types
|
|
import Hasura.SQL.Types
|
|
import Hasura.SQL.Value
|
|
|
|
data Mutation
|
|
= Mutation
|
|
{ _mTable :: !QualifiedTable
|
|
, _mQuery :: !(S.CTE, DS.Seq Q.PrepArg)
|
|
, _mFields :: !MutFlds
|
|
, _mCols :: ![PGColumnInfo]
|
|
, _mStrfyNum :: !Bool
|
|
} deriving (Show, Eq)
|
|
|
|
runMutation :: Mutation -> Q.TxE QErr EncJSON
|
|
runMutation mut =
|
|
bool (mutateAndReturn mut) (mutateAndSel mut) $
|
|
hasNestedFld $ _mFields mut
|
|
|
|
mutateAndReturn :: Mutation -> Q.TxE QErr EncJSON
|
|
mutateAndReturn (Mutation qt (cte, p) mutFlds _ strfyNum) =
|
|
encJFromBS . runIdentity . Q.getRow
|
|
<$> Q.rawQE dmlTxErrorHandler (Q.fromBuilder $ toSQL selWith)
|
|
(toList p) True
|
|
where
|
|
selWith = mkSelWith qt cte mutFlds False strfyNum
|
|
|
|
mutateAndSel :: Mutation -> Q.TxE QErr EncJSON
|
|
mutateAndSel (Mutation qt q mutFlds allCols strfyNum) = do
|
|
-- Perform mutation and fetch unique columns
|
|
MutateResp _ colVals <- mutateAndFetchCols qt allCols q strfyNum
|
|
selCTE <- mkSelCTEFromColVals qt allCols colVals
|
|
let selWith = mkSelWith qt selCTE mutFlds False strfyNum
|
|
-- Perform select query and fetch returning fields
|
|
encJFromBS . runIdentity . Q.getRow
|
|
<$> Q.rawQE dmlTxErrorHandler (Q.fromBuilder $ toSQL selWith) [] True
|
|
|
|
|
|
mutateAndFetchCols
|
|
:: QualifiedTable
|
|
-> [PGColumnInfo]
|
|
-> (S.CTE, DS.Seq Q.PrepArg)
|
|
-> Bool
|
|
-> Q.TxE QErr MutateResp
|
|
mutateAndFetchCols qt cols (cte, p) strfyNum =
|
|
Q.getAltJ . runIdentity . Q.getRow
|
|
<$> Q.rawQE dmlTxErrorHandler (Q.fromBuilder sql) (toList p) True
|
|
where
|
|
aliasIden = Iden $ qualObjectToText qt <> "__mutation_result"
|
|
tabFrom = TableFrom qt $ Just aliasIden
|
|
tabPerm = TablePerm annBoolExpTrue Nothing
|
|
selFlds = flip map cols $
|
|
\ci -> (fromPGCol $ pgiColumn ci, FCol ci Nothing)
|
|
|
|
sql = toSQL selectWith
|
|
selectWith = S.SelectWith [(S.Alias aliasIden, cte)] select
|
|
select = S.mkSelect {S.selExtr = [S.Extractor extrExp Nothing]}
|
|
extrExp = S.applyJsonBuildObj
|
|
[ S.SELit "affected_rows", affRowsSel
|
|
, S.SELit "returning_columns", colSel
|
|
]
|
|
|
|
affRowsSel = S.SESelect $
|
|
S.mkSelect
|
|
{ S.selExtr = [S.Extractor S.countStar Nothing]
|
|
, S.selFrom = Just $ S.FromExp [S.FIIden aliasIden]
|
|
}
|
|
colSel = S.SESelect $ mkSQLSelect False $
|
|
AnnSelG selFlds tabFrom tabPerm noTableArgs strfyNum
|
|
|
|
mkSelCTEFromColVals
|
|
:: (MonadError QErr m)
|
|
=> QualifiedTable -> [PGColumnInfo] -> [ColVals] -> m S.CTE
|
|
mkSelCTEFromColVals qt allCols colVals =
|
|
S.CTESelect <$> case colVals of
|
|
[] -> return selNoRows
|
|
_ -> do
|
|
tuples <- mapM mkTupsFromColVal colVals
|
|
let fromItem = S.FIValues (S.ValuesExp tuples) tableAls $ Just colNames
|
|
return S.mkSelect
|
|
{ S.selExtr = [S.selectStar]
|
|
, S.selFrom = Just $ S.FromExp [fromItem]
|
|
}
|
|
where
|
|
tableAls = S.Alias $ Iden $ snakeCaseQualObject qt
|
|
colNames = map pgiColumn allCols
|
|
mkTupsFromColVal colVal =
|
|
fmap S.TupleExp $ forM allCols $ \ci -> do
|
|
let pgCol = pgiColumn ci
|
|
val <- onNothing (Map.lookup pgCol colVal) $
|
|
throw500 $ "column " <> pgCol <<> " not found in returning values"
|
|
toTxtValue <$> parsePGScalarValue (pgiType ci) val
|
|
|
|
selNoRows =
|
|
S.mkSelect { S.selExtr = [S.selectStar]
|
|
, S.selFrom = Just $ S.mkSimpleFromExp qt
|
|
, S.selWhere = Just $ S.WhereFrag $ S.BELit False
|
|
}
|