Files
payloadcms/test/queues/utilities.ts
Alessio Gravili c08b2aea89 feat: scheduling jobs (#12863)
Adds a new `schedule` property to workflow and task configs that can be
used to have Payload automatically _queue_ jobs following a certain
_schedule_.

Docs:
https://payloadcms.com/docs/dynamic/jobs-queue/schedules?branch=feat/schedule-jobs

## API Example

```ts
export default buildConfig({
  // ...
  jobs: {
    // ...
    scheduler: 'manual', // Or `cron` if you're not using serverless. If `manual` is used, then user needs to set up running /api/payload-jobs/handleSchedules or payload.jobs.handleSchedules in regular intervals
    tasks: [
      {
        schedule: [
          {
            cron: '* * * * * *',
            queue: 'autorunSecond',
            // Hooks are optional
            hooks: {
              // Not an array, as providing and calling `defaultBeforeSchedule` would be more error-prone if this was an array
              beforeSchedule: async (args) => {
                // Handles verifying that there are no jobs already scheduled or processing.
                // You can override this behavior by not calling defaultBeforeSchedule, e.g. if you wanted
                // to allow a maximum of 3 scheduled jobs in the queue instead of 1, or add any additional conditions
                const result = await args.defaultBeforeSchedule(args)
                return {
                  ...result,
                  input: {
                    message: 'This task runs every second',
                  },
                }
              },
              afterSchedule: async (args) => {
                await args.defaultAfterSchedule(args) // Handles updating the payload-jobs-stats global
                args.req.payload.logger.info(
                  'EverySecond task scheduled: ' +
                  (args.status === 'success' ? args.job.id : 'skipped or failed to schedule'),
                )
              },
            },
          },
        ],
        slug: 'EverySecond',
        inputSchema: [
          {
            name: 'message',
            type: 'text',
            required: true,
          },
        ],
        handler: ({ input, req }) => {
          req.payload.logger.info(input.message)
          return {
            output: {},
          }
        },
      }
    ]
  }
})
```

---
- To see the specific tasks where the Asana app for GitHub is being
used, see below:
  - https://app.asana.com/0/0/1210495300843759
2025-07-18 06:48:27 -04:00

63 lines
1.6 KiB
TypeScript

import {
_internal_jobSystemGlobals,
countRunnableOrActiveJobsForQueue,
createLocalReq,
type Payload,
} from 'payload'
export async function waitUntilAutorunIsDone({
payload,
queue,
onlyScheduled = false,
}: {
onlyScheduled?: boolean
payload: Payload
queue: string
}): Promise<void> {
const req = await createLocalReq({}, payload)
return new Promise((resolve) => {
const interval = setInterval(async () => {
const count = await countRunnableOrActiveJobsForQueue({
queue,
req,
onlyScheduled,
})
if (count === 0) {
clearInterval(interval)
resolve()
}
}, 200)
})
}
export function timeFreeze() {
const curDate = new Date()
_internal_jobSystemGlobals.getCurrentDate = () => curDate
}
export function timeTravel(seconds: number) {
const curDate = _internal_jobSystemGlobals.getCurrentDate()
_internal_jobSystemGlobals.getCurrentDate = () => new Date(curDate.getTime() + seconds * 1000)
}
export async function withoutAutoRun<T>(fn: () => Promise<T>): Promise<T> {
const originalValue = _internal_jobSystemGlobals.shouldAutoRun
_internal_jobSystemGlobals.shouldAutoRun = false
try {
return await fn()
} finally {
_internal_jobSystemGlobals.shouldAutoRun = originalValue
}
}
export async function withoutAutoSchedule<T>(fn: () => Promise<T>): Promise<T> {
const originalValue = _internal_jobSystemGlobals.shouldAutoSchedule
_internal_jobSystemGlobals.shouldAutoSchedule = false
try {
return await fn()
} finally {
_internal_jobSystemGlobals.shouldAutoSchedule = originalValue
}
}