2018-12-13 10:26:15 +03:00
|
|
|
module Hasura.RQL.DDL.RemoteSchema
|
|
|
|
( runAddRemoteSchema
|
2019-04-17 19:29:39 +03:00
|
|
|
, addRemoteSchemaToCache
|
|
|
|
, resolveRemoteSchemas
|
2018-12-13 10:26:15 +03:00
|
|
|
, runRemoveRemoteSchema
|
2019-04-17 19:29:39 +03:00
|
|
|
, removeRemoteSchemaFromCache
|
|
|
|
, removeRemoteSchemaFromCatalog
|
2018-12-13 10:26:15 +03:00
|
|
|
, refreshGCtxMapInSchema
|
|
|
|
, fetchRemoteSchemas
|
2019-04-17 19:29:39 +03:00
|
|
|
, addRemoteSchemaP1
|
2018-12-13 10:26:15 +03:00
|
|
|
, addRemoteSchemaP2
|
|
|
|
) where
|
2018-11-23 16:02:46 +03:00
|
|
|
|
2019-03-18 19:22:21 +03:00
|
|
|
import Hasura.EncJSON
|
2018-11-23 16:02:46 +03:00
|
|
|
import Hasura.Prelude
|
|
|
|
|
|
|
|
import qualified Data.Aeson as J
|
|
|
|
import qualified Data.HashMap.Strict as Map
|
|
|
|
import qualified Database.PG.Query as Q
|
2019-04-17 19:29:39 +03:00
|
|
|
import qualified Network.HTTP.Client as HTTP
|
2018-11-23 16:02:46 +03:00
|
|
|
|
|
|
|
import Hasura.GraphQL.RemoteServer
|
|
|
|
import Hasura.RQL.Types
|
|
|
|
|
|
|
|
import qualified Hasura.GraphQL.Schema as GS
|
|
|
|
|
2018-12-13 10:26:15 +03:00
|
|
|
runAddRemoteSchema
|
2019-04-17 19:29:39 +03:00
|
|
|
:: ( QErrM m, UserInfoM m
|
|
|
|
, CacheRWM m, MonadTx m
|
|
|
|
, MonadIO m, HasHttpManager m
|
2018-12-13 10:26:15 +03:00
|
|
|
)
|
2019-03-18 19:22:21 +03:00
|
|
|
=> AddRemoteSchemaQuery -> m EncJSON
|
2018-12-13 10:26:15 +03:00
|
|
|
runAddRemoteSchema q = do
|
2019-04-17 19:29:39 +03:00
|
|
|
addRemoteSchemaP1 q >>= addRemoteSchemaP2 q
|
|
|
|
|
|
|
|
addRemoteSchemaP1
|
|
|
|
:: ( QErrM m, UserInfoM m
|
|
|
|
, MonadIO m, HasHttpManager m
|
|
|
|
)
|
|
|
|
=> AddRemoteSchemaQuery -> m RemoteSchemaInfo
|
|
|
|
addRemoteSchemaP1 q = do
|
2018-12-13 10:26:15 +03:00
|
|
|
adminOnly
|
2019-04-17 19:29:39 +03:00
|
|
|
httpMgr <- askHttpManager
|
|
|
|
rsi <- validateRemoteSchemaDef def
|
|
|
|
-- TODO:- Maintain a cache of remote schema with it's GCtx
|
|
|
|
void $ fetchRemoteSchema httpMgr name rsi
|
|
|
|
return rsi
|
|
|
|
where
|
|
|
|
AddRemoteSchemaQuery name def _ = q
|
2018-11-23 16:02:46 +03:00
|
|
|
|
|
|
|
addRemoteSchemaP2
|
|
|
|
:: ( QErrM m
|
|
|
|
, CacheRWM m
|
|
|
|
, MonadTx m
|
|
|
|
)
|
|
|
|
=> AddRemoteSchemaQuery
|
2019-04-17 19:29:39 +03:00
|
|
|
-> RemoteSchemaInfo
|
2019-03-18 19:22:21 +03:00
|
|
|
-> m EncJSON
|
2019-04-17 19:29:39 +03:00
|
|
|
addRemoteSchemaP2 q rsi = do
|
|
|
|
addRemoteSchemaToCache name rsi
|
2018-11-23 16:02:46 +03:00
|
|
|
liftTx $ addRemoteSchemaToCatalog q
|
|
|
|
return successMsg
|
2019-04-17 19:29:39 +03:00
|
|
|
where
|
|
|
|
name = _arsqName q
|
2018-11-23 16:02:46 +03:00
|
|
|
|
|
|
|
addRemoteSchemaToCache
|
|
|
|
:: CacheRWM m
|
2019-04-17 19:29:39 +03:00
|
|
|
=> RemoteSchemaName
|
2018-11-23 16:02:46 +03:00
|
|
|
-> RemoteSchemaInfo
|
|
|
|
-> m ()
|
2019-04-17 19:29:39 +03:00
|
|
|
addRemoteSchemaToCache name rmDef = do
|
2018-11-23 16:02:46 +03:00
|
|
|
sc <- askSchemaCache
|
|
|
|
let resolvers = scRemoteResolvers sc
|
2019-04-17 19:29:39 +03:00
|
|
|
writeSchemaCache sc
|
|
|
|
{scRemoteResolvers = Map.insert name rmDef resolvers}
|
2018-11-23 16:02:46 +03:00
|
|
|
|
|
|
|
refreshGCtxMapInSchema
|
|
|
|
:: (CacheRWM m, MonadIO m, MonadError QErr m, HasHttpManager m)
|
|
|
|
=> m ()
|
|
|
|
refreshGCtxMapInSchema = do
|
|
|
|
sc <- askSchemaCache
|
2019-01-25 06:31:54 +03:00
|
|
|
gCtxMap <- GS.mkGCtxMap (scTables sc) (scFunctions sc)
|
2018-11-23 16:02:46 +03:00
|
|
|
httpMgr <- askHttpManager
|
|
|
|
(mergedGCtxMap, defGCtx) <-
|
|
|
|
mergeSchemas (scRemoteResolvers sc) gCtxMap httpMgr
|
|
|
|
writeSchemaCache sc { scGCtxMap = mergedGCtxMap
|
|
|
|
, scDefaultRemoteGCtx = defGCtx }
|
|
|
|
|
2018-12-13 10:26:15 +03:00
|
|
|
runRemoveRemoteSchema
|
2019-04-17 19:29:39 +03:00
|
|
|
:: (QErrM m, UserInfoM m, CacheRWM m, MonadTx m)
|
2019-03-18 19:22:21 +03:00
|
|
|
=> RemoveRemoteSchemaQuery -> m EncJSON
|
2019-04-17 19:29:39 +03:00
|
|
|
runRemoveRemoteSchema (RemoveRemoteSchemaQuery rsn)= do
|
|
|
|
removeRemoteSchemaP1 rsn
|
|
|
|
removeRemoteSchemaP2 rsn
|
2018-11-23 16:02:46 +03:00
|
|
|
|
|
|
|
removeRemoteSchemaP1
|
2019-04-17 19:29:39 +03:00
|
|
|
:: (UserInfoM m, QErrM m, CacheRM m)
|
|
|
|
=> RemoteSchemaName -> m ()
|
|
|
|
removeRemoteSchemaP1 rsn = do
|
|
|
|
adminOnly
|
|
|
|
sc <- askSchemaCache
|
|
|
|
let resolvers = scRemoteResolvers sc
|
|
|
|
case Map.lookup rsn resolvers of
|
|
|
|
Just _ -> return ()
|
|
|
|
Nothing -> throw400 NotExists "no such remote schema"
|
2018-11-23 16:02:46 +03:00
|
|
|
|
|
|
|
removeRemoteSchemaP2
|
2019-04-17 19:29:39 +03:00
|
|
|
:: ( CacheRWM m
|
2018-11-23 16:02:46 +03:00
|
|
|
, MonadTx m
|
|
|
|
)
|
2019-04-17 19:29:39 +03:00
|
|
|
=> RemoteSchemaName
|
2019-03-18 19:22:21 +03:00
|
|
|
-> m EncJSON
|
2019-04-17 19:29:39 +03:00
|
|
|
removeRemoteSchemaP2 rsn = do
|
|
|
|
removeRemoteSchemaFromCache rsn
|
|
|
|
liftTx $ removeRemoteSchemaFromCatalog rsn
|
2018-11-23 16:02:46 +03:00
|
|
|
return successMsg
|
|
|
|
|
|
|
|
removeRemoteSchemaFromCache
|
2019-04-17 19:29:39 +03:00
|
|
|
:: CacheRWM m => RemoteSchemaName -> m ()
|
|
|
|
removeRemoteSchemaFromCache rsn = do
|
2018-11-23 16:02:46 +03:00
|
|
|
sc <- askSchemaCache
|
2019-04-17 19:29:39 +03:00
|
|
|
let resolvers = scRemoteResolvers sc
|
|
|
|
writeSchemaCache sc {scRemoteResolvers = Map.delete rsn resolvers}
|
|
|
|
|
|
|
|
resolveRemoteSchemas
|
|
|
|
:: ( MonadError QErr m
|
|
|
|
, MonadIO m
|
|
|
|
)
|
|
|
|
=> SchemaCache -> HTTP.Manager -> m SchemaCache
|
|
|
|
resolveRemoteSchemas sc httpMgr = do
|
|
|
|
(mergedGCtxMap, defGCtx) <-
|
|
|
|
mergeSchemas (scRemoteResolvers sc) gCtxMap httpMgr
|
|
|
|
return $ sc { scGCtxMap = mergedGCtxMap
|
|
|
|
, scDefaultRemoteGCtx = defGCtx
|
|
|
|
}
|
|
|
|
where
|
|
|
|
gCtxMap = scGCtxMap sc
|
2018-11-23 16:02:46 +03:00
|
|
|
|
|
|
|
addRemoteSchemaToCatalog
|
|
|
|
:: AddRemoteSchemaQuery
|
|
|
|
-> Q.TxE QErr ()
|
|
|
|
addRemoteSchemaToCatalog (AddRemoteSchemaQuery name def comment) =
|
|
|
|
Q.unitQE defaultTxErrorHandler [Q.sql|
|
|
|
|
INSERT into hdb_catalog.remote_schemas
|
|
|
|
(name, definition, comment)
|
|
|
|
VALUES ($1, $2, $3)
|
|
|
|
|] (name, Q.AltJ $ J.toJSON def, comment) True
|
|
|
|
|
2019-04-17 19:29:39 +03:00
|
|
|
removeRemoteSchemaFromCatalog :: RemoteSchemaName -> Q.TxE QErr ()
|
2018-11-23 16:02:46 +03:00
|
|
|
removeRemoteSchemaFromCatalog name =
|
|
|
|
Q.unitQE defaultTxErrorHandler [Q.sql|
|
|
|
|
DELETE FROM hdb_catalog.remote_schemas
|
|
|
|
WHERE name = $1
|
|
|
|
|] (Identity name) True
|
|
|
|
|
|
|
|
|
|
|
|
fetchRemoteSchemas :: Q.TxE QErr [AddRemoteSchemaQuery]
|
|
|
|
fetchRemoteSchemas =
|
|
|
|
map fromRow <$> Q.listQE defaultTxErrorHandler
|
|
|
|
[Q.sql|
|
|
|
|
SELECT name, definition, comment
|
|
|
|
FROM hdb_catalog.remote_schemas
|
|
|
|
|] () True
|
|
|
|
where
|
|
|
|
fromRow (n, Q.AltJ def, comm) = AddRemoteSchemaQuery n def comm
|