2020-05-13 15:33:16 +03:00
|
|
|
module Hasura.RQL.DDL.ScheduledTrigger
|
|
|
|
( runCreateCronTrigger
|
|
|
|
, runDeleteCronTrigger
|
2020-12-08 17:22:31 +03:00
|
|
|
, dropCronTriggerInMetadata
|
2020-05-13 15:33:16 +03:00
|
|
|
, resolveCronTrigger
|
|
|
|
, runCreateScheduledEvent
|
2021-01-07 12:04:22 +03:00
|
|
|
, runDeleteScheduledEvent
|
|
|
|
, runGetScheduledEvents
|
|
|
|
, runGetEventInvocations
|
2021-05-26 19:19:26 +03:00
|
|
|
, populateInitialCronTriggerEvents
|
2020-05-13 15:33:16 +03:00
|
|
|
) where
|
|
|
|
|
2021-09-09 14:54:19 +03:00
|
|
|
import System.Cron.Types (CronSchedule)
|
2021-05-26 19:19:26 +03:00
|
|
|
|
2020-05-13 15:33:16 +03:00
|
|
|
import Hasura.Prelude
|
|
|
|
|
2021-09-09 14:54:19 +03:00
|
|
|
import qualified Data.Aeson as J
|
|
|
|
import qualified Data.Environment as Env
|
|
|
|
import qualified Data.HashMap.Strict as Map
|
|
|
|
import qualified Data.HashMap.Strict.InsOrd as OMap
|
|
|
|
import qualified Data.Time.Clock as C
|
2020-05-13 15:33:16 +03:00
|
|
|
|
2021-05-11 18:18:31 +03:00
|
|
|
import Hasura.Base.Error
|
|
|
|
import Hasura.EncJSON
|
|
|
|
import Hasura.Eventing.ScheduledTrigger
|
|
|
|
import Hasura.Metadata.Class
|
2021-09-09 14:54:19 +03:00
|
|
|
import Hasura.RQL.DDL.EventTrigger (getHeaderInfosFromConf)
|
2021-05-11 18:18:31 +03:00
|
|
|
import Hasura.RQL.Types
|
|
|
|
|
2021-05-26 19:19:26 +03:00
|
|
|
populateInitialCronTriggerEvents
|
|
|
|
:: ( MonadIO m
|
|
|
|
, MonadMetadataStorageQueryAPI m
|
|
|
|
)
|
|
|
|
=> CronSchedule
|
|
|
|
-> TriggerName
|
|
|
|
-> m ()
|
|
|
|
populateInitialCronTriggerEvents schedule triggerName = do
|
|
|
|
currentTime <- liftIO C.getCurrentTime
|
|
|
|
let scheduleTimes = generateScheduleTimes currentTime 100 schedule
|
|
|
|
createScheduledEvent $ SESCron $ map (CronEventSeed triggerName) scheduleTimes
|
|
|
|
pure ()
|
2021-05-11 18:18:31 +03:00
|
|
|
|
2020-05-13 15:33:16 +03:00
|
|
|
-- | runCreateCronTrigger will update a existing cron trigger when the 'replace'
|
|
|
|
-- value is set to @true@ and when replace is @false@ a new cron trigger will
|
|
|
|
-- be created
|
2020-12-08 17:22:31 +03:00
|
|
|
runCreateCronTrigger
|
2020-12-14 07:30:19 +03:00
|
|
|
:: ( CacheRWM m, MonadIO m
|
2020-12-28 15:56:00 +03:00
|
|
|
, MetadataM m, MonadMetadataStorageQueryAPI m
|
2020-12-14 07:30:19 +03:00
|
|
|
)
|
|
|
|
=> CreateCronTrigger -> m EncJSON
|
2020-05-13 15:33:16 +03:00
|
|
|
runCreateCronTrigger CreateCronTrigger {..} = do
|
2020-10-28 19:40:33 +03:00
|
|
|
let q = CronTriggerMetadata cctName
|
2020-05-13 15:33:16 +03:00
|
|
|
cctWebhook
|
|
|
|
cctCronSchedule
|
|
|
|
cctPayload
|
|
|
|
cctRetryConf
|
|
|
|
cctHeaders
|
|
|
|
cctIncludeInMetadata
|
2020-10-28 19:40:33 +03:00
|
|
|
cctComment
|
2020-05-13 15:33:16 +03:00
|
|
|
case cctReplace of
|
|
|
|
True -> updateCronTrigger q
|
|
|
|
False -> do
|
|
|
|
cronTriggersMap <- scCronTriggers <$> askSchemaCache
|
|
|
|
case Map.lookup (ctName q) cronTriggersMap of
|
|
|
|
Nothing -> pure ()
|
|
|
|
Just _ -> throw400 AlreadyExists $
|
|
|
|
"cron trigger with name: "
|
2020-10-28 19:40:33 +03:00
|
|
|
<> triggerNameToTxt (ctName q)
|
2020-05-13 15:33:16 +03:00
|
|
|
<> " already exists"
|
|
|
|
|
2020-12-08 17:22:31 +03:00
|
|
|
let metadataObj = MOCronTrigger cctName
|
|
|
|
metadata = CronTriggerMetadata cctName cctWebhook cctCronSchedule
|
|
|
|
cctPayload cctRetryConf cctHeaders cctIncludeInMetadata
|
|
|
|
cctComment
|
|
|
|
buildSchemaCacheFor metadataObj
|
|
|
|
$ MetadataModifier
|
|
|
|
$ metaCronTriggers %~ OMap.insert cctName metadata
|
2021-05-26 19:19:26 +03:00
|
|
|
populateInitialCronTriggerEvents cctCronSchedule cctName
|
2020-05-13 15:33:16 +03:00
|
|
|
return successMsg
|
|
|
|
|
|
|
|
resolveCronTrigger
|
2020-07-14 22:00:58 +03:00
|
|
|
:: (QErrM m)
|
|
|
|
=> Env.Environment
|
2020-12-08 17:22:31 +03:00
|
|
|
-> CronTriggerMetadata
|
2020-07-14 22:00:58 +03:00
|
|
|
-> m CronTriggerInfo
|
2020-12-08 17:22:31 +03:00
|
|
|
resolveCronTrigger env CronTriggerMetadata{..} = do
|
|
|
|
webhookInfo <- resolveWebhook env ctWebhook
|
|
|
|
headerInfo <- getHeaderInfosFromConf env ctHeaders
|
2020-05-13 15:33:16 +03:00
|
|
|
pure $
|
2020-12-08 17:22:31 +03:00
|
|
|
CronTriggerInfo ctName
|
|
|
|
ctSchedule
|
|
|
|
ctPayload
|
|
|
|
ctRetryConf
|
2020-05-13 15:33:16 +03:00
|
|
|
webhookInfo
|
|
|
|
headerInfo
|
2020-12-08 17:22:31 +03:00
|
|
|
ctComment
|
2020-05-13 15:33:16 +03:00
|
|
|
|
2020-12-08 17:22:31 +03:00
|
|
|
updateCronTrigger
|
|
|
|
:: ( CacheRWM m
|
|
|
|
, MonadIO m
|
|
|
|
, MetadataM m
|
2020-12-28 15:56:00 +03:00
|
|
|
, MonadMetadataStorageQueryAPI m
|
2020-12-08 17:22:31 +03:00
|
|
|
)
|
|
|
|
=> CronTriggerMetadata -> m EncJSON
|
2020-05-13 15:33:16 +03:00
|
|
|
updateCronTrigger cronTriggerMetadata = do
|
2020-12-08 17:22:31 +03:00
|
|
|
let triggerName = ctName cronTriggerMetadata
|
|
|
|
checkExists triggerName
|
|
|
|
buildSchemaCacheFor (MOCronTrigger triggerName)
|
|
|
|
$ MetadataModifier
|
|
|
|
$ metaCronTriggers %~ OMap.insert triggerName cronTriggerMetadata
|
2021-05-26 19:19:26 +03:00
|
|
|
dropFutureCronEvents $ SingleCronTrigger triggerName
|
2020-06-23 18:21:34 +03:00
|
|
|
currentTime <- liftIO C.getCurrentTime
|
2020-12-08 17:22:31 +03:00
|
|
|
let scheduleTimes = generateScheduleTimes currentTime 100 $ ctSchedule cronTriggerMetadata
|
2020-12-14 07:30:19 +03:00
|
|
|
createScheduledEvent $ SESCron $ map (CronEventSeed triggerName) scheduleTimes
|
2020-12-08 17:22:31 +03:00
|
|
|
pure successMsg
|
2020-05-13 15:33:16 +03:00
|
|
|
|
2020-12-08 17:22:31 +03:00
|
|
|
runDeleteCronTrigger
|
|
|
|
:: ( CacheRWM m
|
|
|
|
, MetadataM m
|
2020-12-28 15:56:00 +03:00
|
|
|
, MonadMetadataStorageQueryAPI m
|
2020-12-08 17:22:31 +03:00
|
|
|
)
|
|
|
|
=> ScheduledTriggerName -> m EncJSON
|
2020-05-13 15:33:16 +03:00
|
|
|
runDeleteCronTrigger (ScheduledTriggerName stName) = do
|
|
|
|
checkExists stName
|
2020-12-08 17:22:31 +03:00
|
|
|
withNewInconsistentObjsCheck
|
|
|
|
$ buildSchemaCache
|
|
|
|
$ dropCronTriggerInMetadata stName
|
2021-05-26 19:19:26 +03:00
|
|
|
dropFutureCronEvents $ SingleCronTrigger stName
|
2020-05-13 15:33:16 +03:00
|
|
|
return successMsg
|
|
|
|
|
2020-12-08 17:22:31 +03:00
|
|
|
dropCronTriggerInMetadata :: TriggerName -> MetadataModifier
|
|
|
|
dropCronTriggerInMetadata name =
|
|
|
|
MetadataModifier $ metaCronTriggers %~ OMap.delete name
|
2020-05-13 15:33:16 +03:00
|
|
|
|
2020-12-14 07:30:19 +03:00
|
|
|
runCreateScheduledEvent
|
2021-02-18 19:46:14 +03:00
|
|
|
:: ( MonadMetadataStorageQueryAPI m )
|
|
|
|
=> CreateScheduledEvent -> m EncJSON
|
|
|
|
runCreateScheduledEvent scheduledEvent = do
|
|
|
|
createScheduledEvent $ SESOneOff scheduledEvent
|
|
|
|
pure successMsg
|
2020-05-13 15:33:16 +03:00
|
|
|
|
|
|
|
checkExists :: (CacheRM m, MonadError QErr m) => TriggerName -> m ()
|
|
|
|
checkExists name = do
|
|
|
|
cronTriggersMap <- scCronTriggers <$> askSchemaCache
|
|
|
|
void $ onNothing (Map.lookup name cronTriggersMap) $
|
|
|
|
throw400 NotExists $
|
2020-10-28 19:40:33 +03:00
|
|
|
"cron trigger with name: " <> triggerNameToTxt name <> " does not exist"
|
2021-01-07 12:04:22 +03:00
|
|
|
|
|
|
|
runDeleteScheduledEvent
|
2021-02-18 19:46:14 +03:00
|
|
|
:: ( MonadMetadataStorageQueryAPI m ) => DeleteScheduledEvent -> m EncJSON
|
2021-01-07 12:04:22 +03:00
|
|
|
runDeleteScheduledEvent DeleteScheduledEvent{..} = do
|
|
|
|
dropEvent _dseEventId _dseType
|
|
|
|
pure successMsg
|
|
|
|
|
|
|
|
runGetScheduledEvents
|
|
|
|
:: ( CacheRM m
|
|
|
|
, MonadMetadataStorageQueryAPI m
|
|
|
|
)
|
|
|
|
=> GetScheduledEvents -> m EncJSON
|
|
|
|
runGetScheduledEvents gse = do
|
|
|
|
case _gseScheduledEvent gse of
|
|
|
|
SEOneOff -> pure ()
|
|
|
|
SECron name -> checkExists name
|
|
|
|
encJFromJValue <$> fetchScheduledEvents gse
|
|
|
|
|
|
|
|
runGetEventInvocations
|
|
|
|
:: ( CacheRM m
|
|
|
|
, MonadMetadataStorageQueryAPI m
|
|
|
|
)
|
|
|
|
=> GetEventInvocations -> m EncJSON
|
|
|
|
runGetEventInvocations GetEventInvocations{..} = do
|
|
|
|
case _geiInvocationsBy of
|
|
|
|
GIBEventId _ _ -> pure ()
|
|
|
|
GIBEvent event -> case event of
|
|
|
|
SEOneOff -> pure ()
|
|
|
|
SECron name -> checkExists name
|
|
|
|
WithTotalCount count invocations <- fetchInvocations _geiInvocationsBy _geiPagination
|
|
|
|
pure $ encJFromJValue $ J.object [ "invocations" J..= invocations
|
|
|
|
, "count" J..= count
|
|
|
|
]
|