mirror of
https://github.com/hasura/graphql-engine.git
synced 2024-12-16 18:42:30 +03:00
e32f5a1fb1
1. Haskel library `pg-client-hs` has been updated to expose a function that helps listen to `postgres` notifications over a `channel` in this [PR](https://github.com/hasura/pg-client-hs/pull/5) 2. The server records an event in a table `hdb_catalog.hdb_cache_update_event` whenever any `/v1/query` (that changes metadata) is requested. A trigger notifies a `cache update` event via `hasura_cache_update` channel 3. The server runs two concurrent threads namely `listener` and `processor`. The `listener` thread listens to events on `hasura_cache_update` channel and pushed into a `Queue`. The `processor` thread fetches events from that `Queue` and processes it. Thus server rebuilds schema cache from database and updates.
211 lines
6.3 KiB
Haskell
211 lines
6.3 KiB
Haskell
module Hasura.Server.SchemaUpdate
|
|
(startSchemaSync)
|
|
where
|
|
|
|
import Hasura.Prelude
|
|
|
|
import Hasura.Logging
|
|
import Hasura.RQL.DDL.Schema.Table (buildSchemaCache)
|
|
import Hasura.RQL.Types
|
|
import Hasura.Server.App (SchemaCacheRef (..), withSCUpdate)
|
|
import Hasura.Server.Init (InstanceId (..))
|
|
import Hasura.Server.Logging
|
|
import Hasura.Server.Query
|
|
|
|
import Data.Aeson
|
|
import Data.Aeson.Casing
|
|
import Data.Aeson.TH
|
|
|
|
import qualified Control.Concurrent as C
|
|
import qualified Control.Concurrent.STM as STM
|
|
import qualified Data.Text as T
|
|
import qualified Data.Time as UTC
|
|
import qualified Database.PG.Query as PG
|
|
import qualified Database.PostgreSQL.LibPQ as PQ
|
|
import qualified Network.HTTP.Client as HTTP
|
|
|
|
pgChannel :: PG.PGChannel
|
|
pgChannel = "hasura_schema_update"
|
|
|
|
data ThreadType
|
|
= TTListener
|
|
| TTProcessor
|
|
deriving (Eq)
|
|
|
|
instance Show ThreadType where
|
|
show TTListener = "listener"
|
|
show TTProcessor = "processor"
|
|
|
|
|
|
data SchemaSyncThreadLog
|
|
= SchemaSyncThreadLog
|
|
{ suelLogLevel :: !LogLevel
|
|
, suelThreadType :: !ThreadType
|
|
, suelInfo :: !Value
|
|
} deriving (Show, Eq)
|
|
|
|
instance ToJSON SchemaSyncThreadLog where
|
|
toJSON (SchemaSyncThreadLog _ t info) =
|
|
object [ "thread_type" .= show t
|
|
, "info" .= info
|
|
]
|
|
|
|
instance ToEngineLog SchemaSyncThreadLog where
|
|
toEngineLog threadLog =
|
|
(suelLogLevel threadLog, "schema_sync_thread", toJSON threadLog)
|
|
|
|
data EventPayload
|
|
= EventPayload
|
|
{ _epInstanceId :: !InstanceId
|
|
, _epOccurredAt :: !UTC.UTCTime
|
|
} deriving (Show, Eq)
|
|
$(deriveJSON (aesonDrop 3 snakeCase) ''EventPayload)
|
|
|
|
data ThreadError
|
|
= TEJsonParse !T.Text
|
|
| TEQueryError !QErr
|
|
$(deriveToJSON
|
|
defaultOptions { constructorTagModifier = snakeCase . drop 2
|
|
, sumEncoding = TaggedObject "type" "info"
|
|
}
|
|
''ThreadError)
|
|
|
|
-- | An IO action that enables metadata syncing
|
|
startSchemaSync
|
|
:: Bool
|
|
-> PG.PGPool
|
|
-> Logger
|
|
-> HTTP.Manager
|
|
-> SchemaCacheRef
|
|
-> InstanceId
|
|
-> Maybe UTC.UTCTime -> IO ()
|
|
startSchemaSync strfyNum pool logger httpMgr cacheRef instanceId cacheInitTime = do
|
|
-- Init events queue
|
|
eventsQueue <- STM.newTQueueIO
|
|
|
|
-- Start listener thread
|
|
lTId <- C.forkIO $ listener strfyNum pool
|
|
logger httpMgr eventsQueue cacheRef instanceId cacheInitTime
|
|
logThreadStarted TTListener lTId
|
|
|
|
-- Start processor thread
|
|
pTId <- C.forkIO $ processor strfyNum pool
|
|
logger httpMgr eventsQueue cacheRef instanceId
|
|
logThreadStarted TTProcessor pTId
|
|
|
|
where
|
|
logThreadStarted threadType threadId =
|
|
let msg = T.pack (show threadType) <> " thread started"
|
|
in unLogger logger $
|
|
StartupLog LevelInfo "threads" $
|
|
object [ "instance_id" .= getInstanceId instanceId
|
|
, "thread_id" .= show threadId
|
|
, "message" .= msg
|
|
]
|
|
|
|
-- | An IO action that listens to postgres for events and pushes them to a Queue
|
|
listener
|
|
:: Bool
|
|
-> PG.PGPool
|
|
-> Logger
|
|
-> HTTP.Manager
|
|
-> STM.TQueue EventPayload
|
|
-> SchemaCacheRef
|
|
-> InstanceId
|
|
-> Maybe UTC.UTCTime -> IO ()
|
|
listener strfyNum pool logger httpMgr eventsQueue
|
|
cacheRef instanceId cacheInitTime =
|
|
-- Never exits
|
|
forever $ do
|
|
listenResE <-
|
|
liftIO $ runExceptT $ PG.listen pool pgChannel notifyHandler
|
|
either onError return listenResE
|
|
logWarn
|
|
C.threadDelay $ 1 * 1000 * 1000 -- 1 second
|
|
where
|
|
threadType = TTListener
|
|
|
|
shouldRefresh dbInstId accrdAt =
|
|
case cacheInitTime of
|
|
Nothing -> True
|
|
Just time -> (dbInstId /= instanceId) && accrdAt > time
|
|
|
|
refreshCache Nothing = return ()
|
|
refreshCache (Just (dbInstId, accrdAt)) =
|
|
when (shouldRefresh dbInstId accrdAt) $
|
|
refreshSchemaCache strfyNum pool logger httpMgr cacheRef
|
|
threadType "schema cache reloaded after postgres listen init"
|
|
|
|
notifyHandler = \case
|
|
PG.PNEOnStart -> do
|
|
eRes <- runExceptT $ PG.runTx pool
|
|
(PG.Serializable, Nothing) fetchLastUpdate
|
|
case eRes of
|
|
Left e -> onError e
|
|
Right mLastUpd -> refreshCache mLastUpd
|
|
|
|
PG.PNEPQNotify notif ->
|
|
case eitherDecodeStrict $ PQ.notifyExtra notif of
|
|
Left e -> logError logger threadType $ TEJsonParse $ T.pack e
|
|
Right payload -> do
|
|
logInfo logger threadType $ object ["received_event" .= payload]
|
|
-- Push a notify event to Queue
|
|
STM.atomically $ STM.writeTQueue eventsQueue payload
|
|
|
|
onError = logError logger threadType . TEQueryError
|
|
logWarn = unLogger logger $
|
|
SchemaSyncThreadLog LevelWarn TTListener $ String
|
|
"error occurred, retrying postgres listen after 1 second"
|
|
|
|
|
|
-- | An IO action that processes events from Queue
|
|
processor
|
|
:: Bool
|
|
-> PG.PGPool
|
|
-> Logger
|
|
-> HTTP.Manager
|
|
-> STM.TQueue EventPayload
|
|
-> SchemaCacheRef
|
|
-> InstanceId -> IO ()
|
|
processor strfyNum pool logger httpMgr eventsQueue
|
|
cacheRef instanceId =
|
|
-- Never exits
|
|
forever $ do
|
|
event <- STM.atomically $ STM.readTQueue eventsQueue
|
|
logInfo logger threadType $ object ["processed_event" .= event]
|
|
when (shouldReload event) $
|
|
refreshSchemaCache strfyNum pool logger httpMgr cacheRef
|
|
threadType "schema cache reloaded"
|
|
where
|
|
threadType = TTProcessor
|
|
|
|
-- If event is from another server
|
|
shouldReload payload = _epInstanceId payload /= instanceId
|
|
|
|
refreshSchemaCache
|
|
:: Bool
|
|
-> PG.PGPool
|
|
-> Logger
|
|
-> HTTP.Manager
|
|
-> SchemaCacheRef
|
|
-> ThreadType
|
|
-> T.Text -> IO ()
|
|
refreshSchemaCache strfyNum pool logger httpManager cacheRef threadType msg = do
|
|
-- Reload schema cache from catalog
|
|
resE <- liftIO $ runExceptT $ withSCUpdate cacheRef $
|
|
peelRun emptySchemaCache adminUserInfo
|
|
httpManager strfyNum pool PG.Serializable buildSchemaCache
|
|
case resE of
|
|
Left e -> logError logger threadType $ TEQueryError e
|
|
Right _ ->
|
|
logInfo logger threadType $ object ["message" .= msg]
|
|
|
|
logInfo :: Logger -> ThreadType -> Value -> IO ()
|
|
logInfo logger threadType val = unLogger logger $
|
|
SchemaSyncThreadLog LevelInfo threadType val
|
|
|
|
logError :: ToJSON a => Logger -> ThreadType -> a -> IO ()
|
|
logError logger threadType err =
|
|
unLogger logger $ SchemaSyncThreadLog LevelError threadType $
|
|
object ["error" .= toJSON err]
|