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,
|
2022-11-03 13:21:56 +03:00
|
|
|
runGetScheduledEventInvocations,
|
2021-05-26 19:19:26 +03:00
|
|
|
populateInitialCronTriggerEvents,
|
2022-01-27 09:43:39 +03:00
|
|
|
runGetCronTriggers,
|
2020-05-13 15:33:16 +03:00
|
|
|
)
|
|
|
|
where
|
2021-09-24 01:56:37 +03:00
|
|
|
|
2021-09-09 14:54:19 +03:00
|
|
|
import Data.Aeson qualified as J
|
|
|
|
import Data.Environment qualified as Env
|
|
|
|
import Data.HashMap.Strict qualified as Map
|
|
|
|
import Data.HashMap.Strict.InsOrd qualified as OMap
|
|
|
|
import Data.Time.Clock qualified as C
|
2022-06-05 23:27:09 +03:00
|
|
|
import Data.URL.Template (printURLTemplate)
|
2021-05-11 18:18:31 +03:00
|
|
|
import Hasura.Base.Error
|
|
|
|
import Hasura.EncJSON
|
|
|
|
import Hasura.Eventing.ScheduledTrigger
|
|
|
|
import Hasura.Metadata.Class
|
2020-05-13 15:33:16 +03:00
|
|
|
import Hasura.Prelude
|
2021-09-09 14:54:19 +03:00
|
|
|
import Hasura.RQL.DDL.EventTrigger (getHeaderInfosFromConf)
|
2022-04-27 16:57:28 +03:00
|
|
|
import Hasura.RQL.Types.Common
|
|
|
|
import Hasura.RQL.Types.EventTrigger
|
|
|
|
import Hasura.RQL.Types.Metadata
|
|
|
|
import Hasura.RQL.Types.Metadata.Object
|
|
|
|
import Hasura.RQL.Types.ScheduledTrigger
|
|
|
|
import Hasura.RQL.Types.SchemaCache
|
|
|
|
import Hasura.RQL.Types.SchemaCache.Build
|
2021-09-09 14:54:19 +03:00
|
|
|
import System.Cron.Types (CronSchedule)
|
2021-09-24 01:56:37 +03:00
|
|
|
|
2021-05-26 19:19:26 +03:00
|
|
|
populateInitialCronTriggerEvents ::
|
|
|
|
( MonadIO m,
|
2023-02-03 04:03:23 +03:00
|
|
|
MonadError QErr m,
|
2023-04-03 16:35:15 +03:00
|
|
|
MonadMetadataStorage m
|
2021-05-26 19:19:26 +03:00
|
|
|
) =>
|
|
|
|
CronSchedule ->
|
|
|
|
TriggerName ->
|
|
|
|
m ()
|
|
|
|
populateInitialCronTriggerEvents schedule triggerName = do
|
|
|
|
currentTime <- liftIO C.getCurrentTime
|
|
|
|
let scheduleTimes = generateScheduleTimes currentTime 100 schedule
|
2023-02-03 04:03:23 +03:00
|
|
|
liftEitherM $ insertCronEvents $ map (CronEventSeed triggerName) scheduleTimes
|
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 ::
|
2023-02-03 04:03:23 +03:00
|
|
|
( MonadError QErr m,
|
|
|
|
CacheRWM m,
|
2020-12-14 07:30:19 +03:00
|
|
|
MonadIO m,
|
2020-12-28 15:56:00 +03:00
|
|
|
MetadataM m,
|
2023-04-03 16:35:15 +03:00
|
|
|
MonadMetadataStorage 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
|
2022-03-11 02:22:54 +03:00
|
|
|
_cctName
|
|
|
|
_cctWebhook
|
|
|
|
_cctCronSchedule
|
|
|
|
_cctPayload
|
|
|
|
_cctRetryConf
|
|
|
|
_cctHeaders
|
|
|
|
_cctIncludeInMetadata
|
|
|
|
_cctComment
|
|
|
|
_cctRequestTransform
|
|
|
|
_cctResponseTransform
|
|
|
|
case _cctReplace of
|
2020-05-13 15:33:16 +03:00
|
|
|
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"
|
2021-09-24 01:56:37 +03:00
|
|
|
|
2022-03-11 02:22:54 +03:00
|
|
|
let metadataObj = MOCronTrigger _cctName
|
2020-12-08 17:22:31 +03:00
|
|
|
metadata =
|
|
|
|
CronTriggerMetadata
|
2022-03-11 02:22:54 +03:00
|
|
|
_cctName
|
|
|
|
_cctWebhook
|
|
|
|
_cctCronSchedule
|
|
|
|
_cctPayload
|
|
|
|
_cctRetryConf
|
|
|
|
_cctHeaders
|
|
|
|
_cctIncludeInMetadata
|
|
|
|
_cctComment
|
|
|
|
_cctRequestTransform
|
|
|
|
_cctResponseTransform
|
2020-12-08 17:22:31 +03:00
|
|
|
buildSchemaCacheFor metadataObj $
|
|
|
|
MetadataModifier $
|
2022-03-11 02:22:54 +03:00
|
|
|
metaCronTriggers %~ OMap.insert _cctName metadata
|
|
|
|
populateInitialCronTriggerEvents _cctCronSchedule _cctName
|
2020-05-13 15:33:16 +03:00
|
|
|
return successMsg
|
2021-09-24 01:56:37 +03:00
|
|
|
|
2020-05-13 15:33:16 +03:00
|
|
|
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
|
2022-06-05 23:27:09 +03:00
|
|
|
let urlTemplate = printURLTemplate $ unInputWebhook ctWebhook
|
2020-05-13 15:33:16 +03:00
|
|
|
pure $
|
2020-12-08 17:22:31 +03:00
|
|
|
CronTriggerInfo
|
|
|
|
ctName
|
|
|
|
ctSchedule
|
|
|
|
ctPayload
|
|
|
|
ctRetryConf
|
2022-06-05 23:27:09 +03:00
|
|
|
(EnvRecord urlTemplate webhookInfo)
|
2020-05-13 15:33:16 +03:00
|
|
|
headerInfo
|
2020-12-08 17:22:31 +03:00
|
|
|
ctComment
|
2022-01-19 07:46:42 +03:00
|
|
|
ctRequestTransform
|
|
|
|
ctResponseTransform
|
2021-09-24 01:56:37 +03:00
|
|
|
|
2020-12-08 17:22:31 +03:00
|
|
|
updateCronTrigger ::
|
2023-02-03 04:03:23 +03:00
|
|
|
( MonadError QErr m,
|
|
|
|
CacheRWM m,
|
2020-12-08 17:22:31 +03:00
|
|
|
MonadIO m,
|
|
|
|
MetadataM m,
|
2023-04-03 16:35:15 +03:00
|
|
|
MonadMetadataStorage 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
|
2023-02-03 04:03:23 +03:00
|
|
|
liftEitherM $ 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
|
2023-02-03 04:03:23 +03:00
|
|
|
liftEitherM $ insertCronEvents $ 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 ::
|
2023-02-03 04:03:23 +03:00
|
|
|
( MonadError QErr m,
|
|
|
|
CacheRWM m,
|
2020-12-08 17:22:31 +03:00
|
|
|
MetadataM m,
|
2023-04-03 16:35:15 +03:00
|
|
|
MonadMetadataStorage 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
|
2023-02-03 04:03:23 +03:00
|
|
|
liftEitherM $ 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 ::
|
2023-04-03 16:35:15 +03:00
|
|
|
(MonadError QErr m, MonadMetadataStorage m) =>
|
2021-02-18 19:46:14 +03:00
|
|
|
CreateScheduledEvent ->
|
|
|
|
m EncJSON
|
|
|
|
runCreateScheduledEvent scheduledEvent = do
|
2023-02-03 04:03:23 +03:00
|
|
|
eid <- liftEitherM $ createOneOffScheduledEvent scheduledEvent
|
2021-09-13 21:00:53 +03:00
|
|
|
pure $ encJFromJValue $ J.object ["message" J..= J.String "success", "event_id" J..= eid]
|
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-09-24 01:56:37 +03:00
|
|
|
|
2021-01-07 12:04:22 +03:00
|
|
|
runDeleteScheduledEvent ::
|
2023-04-03 16:35:15 +03:00
|
|
|
(MonadMetadataStorage m, MonadError QErr m) => DeleteScheduledEvent -> m EncJSON
|
2021-01-07 12:04:22 +03:00
|
|
|
runDeleteScheduledEvent DeleteScheduledEvent {..} = do
|
2023-02-03 04:03:23 +03:00
|
|
|
liftEitherM $ dropEvent _dseEventId _dseType
|
2021-01-07 12:04:22 +03:00
|
|
|
pure successMsg
|
|
|
|
|
|
|
|
runGetScheduledEvents ::
|
2023-02-03 04:03:23 +03:00
|
|
|
( MonadError QErr m,
|
|
|
|
CacheRM m,
|
2023-04-03 16:35:15 +03:00
|
|
|
MonadMetadataStorage m
|
2021-01-07 12:04:22 +03:00
|
|
|
) =>
|
|
|
|
GetScheduledEvents ->
|
|
|
|
m EncJSON
|
|
|
|
runGetScheduledEvents gse = do
|
|
|
|
case _gseScheduledEvent gse of
|
|
|
|
SEOneOff -> pure ()
|
|
|
|
SECron name -> checkExists name
|
2023-02-03 04:03:23 +03:00
|
|
|
encJFromJValue <$> liftEitherM (fetchScheduledEvents gse)
|
2021-01-07 12:04:22 +03:00
|
|
|
|
2022-11-03 13:21:56 +03:00
|
|
|
runGetScheduledEventInvocations ::
|
2023-02-03 04:03:23 +03:00
|
|
|
( MonadError QErr m,
|
|
|
|
CacheRM m,
|
2023-04-03 16:35:15 +03:00
|
|
|
MonadMetadataStorage m
|
2021-01-07 12:04:22 +03:00
|
|
|
) =>
|
2022-11-03 13:21:56 +03:00
|
|
|
GetScheduledEventInvocations ->
|
2021-01-07 12:04:22 +03:00
|
|
|
m EncJSON
|
2022-11-03 13:21:56 +03:00
|
|
|
runGetScheduledEventInvocations getEventInvocations@GetScheduledEventInvocations {..} = do
|
2021-01-07 12:04:22 +03:00
|
|
|
case _geiInvocationsBy of
|
|
|
|
GIBEventId _ _ -> pure ()
|
|
|
|
GIBEvent event -> case event of
|
|
|
|
SEOneOff -> pure ()
|
|
|
|
SECron name -> checkExists name
|
2023-02-03 04:03:23 +03:00
|
|
|
WithOptionalTotalCount countMaybe invocations <- liftEitherM $ fetchScheduledEventInvocations getEventInvocations
|
2021-01-07 12:04:22 +03:00
|
|
|
pure $
|
|
|
|
encJFromJValue $
|
2022-09-15 22:10:53 +03:00
|
|
|
J.object $
|
|
|
|
("invocations" J..= invocations) : (maybe mempty (\count -> ["count" J..= count]) countMaybe)
|
2022-01-27 09:43:39 +03:00
|
|
|
|
|
|
|
-- | Metadata API handler to retrieve all the cron triggers from the metadata
|
|
|
|
runGetCronTriggers :: MetadataM m => m EncJSON
|
|
|
|
runGetCronTriggers = do
|
|
|
|
cronTriggers <- toList . _metaCronTriggers <$> getMetadata
|
|
|
|
pure $
|
|
|
|
encJFromJValue $
|
|
|
|
J.object
|
|
|
|
["cron_triggers" J..= cronTriggers]
|