web/docs/advanced/jobs.md
import { CardLink } from '@site/src/components/CardLink' import { Required } from '@site/src/components/Tag' import { ShowForTs, ShowForJs } from '@site/src/components/TsJsHelpers' import ReferencingCodeFromSrcNote from '../_referencing-code-from-src-note.md'
In most web apps, users send requests to the server and receive responses with some data. When the server responds quickly, the app feels responsive and smooth.
What if the server needs extra time to fully process the request? This might mean sending an email or making a slow HTTP request to an external API. In that case, it's a good idea to respond to the user as soon as possible and do the remaining work in the background.
Wasp supports background jobs that can help you with this:
Let's write an example Job that will print a message to the console and return a list of tasks from the database.
import { app, job } from "@wasp.sh/spec"
import { mySpecialJob } from "./src/workers/bar" with { type: "ref" }
export default app({
// ...
spec: [
job(mySpecialJob, {
executor: "PgBoss",
entities: ["Task"],
}),
],
})
type Input = { name: string; }
type Output = { tasks: Task[]; }
export const mySpecialJob: MySpecialJob<Input, Output> = async ({ name }, context) => {
console.log(`Hello ${name}!`)
const tasks = await context.entities.Task.findMany({})
return { tasks }
}
```
:::info The worker function
The worker function must be an async function. The function's return value represents the Job's result.
The worker function accepts two arguments:
args: The data passed into the job when it's submitted.context: { entities }: The context object containing entities you put in the Job spec.
:::const submittedJob = await mySpecialJob.submit({ name: "Johnny" })
// Or, if you'd prefer it to execute in the future, just add a .delay().
// It takes a number of seconds, Date, or ISO date string.
await mySpecialJob
.delay(10)
.submit({ name: "Johnny" })
```
const submittedJob = await mySpecialJob.submit({ name: "Johnny" })
// Or, if you'd prefer it to execute in the future, just add a .delay().
// It takes a number of seconds, Date, or ISO date string.
await mySpecialJob
.delay(10)
.submit({ name: "Johnny" })
```
And that's it. Your job will be executed by PgBoss as if you called mySpecialJob({ name: "Johnny" }).
In our example, mySpecialJob takes an argument, but passing arguments to jobs is not a requirement. It depends on how you've implemented your worker function.
If you have work that needs to be done on some recurring basis, you can add a schedule to your job spec:
import { app, job } from "@wasp.sh/spec"
import { mySpecialJob } from "./src/workers/bar" with { type: "ref" }
export default app({
// ...
spec: [
job(mySpecialJob, {
executor: "PgBoss",
schedule: {
cron: "0 * * * *",
args: { name: "Johnny" }, // optional
},
}),
],
})
In this example, you don't need to invoke anything in <ShowForJs>JavaScript</ShowForJs><ShowForTs>TypeScript</ShowForTs>. You can imagine mySpecialJob({ name: "Johnny" }) getting automatically scheduled and invoked for you every hour.
Wasp supports Jobs through the use of job executors. A job executor is responsible for handling the scheduling, monitoring, and execution of jobs.
Currently, Wasp only has support for one job executor, PgBoss.
PgBoss {#pgboss}PgBoss is a lightweight job queue built on top of PostgreSQL. It is suitable for low-volume production use cases and does not require any additional infrastructure or complex management. By using PostgreSQL (and SKIP LOCKED) as its storage and synchronization mechanism, you get many benefits of a traditional job queue, on top of your existing Postgres database.
PgBoss requires that your database provider is set to "postgresql" in your schema.prisma file. Read more about setting the provider here.
PgBoss runs together with your web server, whenever it is up. This means that it is not a separate process or service, but rather a part of your web server's application. As such, it is not suitable for CPU-heavy workloads, as it shares the CPU with your web server's application logic.
The PgBoss executor in Wasp does not (yet) support independent, horizontal scaling of pg-boss-only applications, nor starting them as separate workers/processes/threads. This means that your server must be running whenever you want to process jobs. If you need to scale your job processing, you will need to run multiple instances of your web server, each with its own PgBoss instance.
If you need to customize the creation of the PgBoss instance, you can set an environment variable called PG_BOSS_NEW_OPTIONS to a stringified JSON object containing the initialization parameters. See the pg-boss documentation.
Please note that setting PG_BOSS_NEW_OPTIONS environment variable overwrites all Wasp defaults, so you must include the connectionString parameter inside it as well.
For example, to set the connection string and change the job archival and deletion settings, you can set the environment variable like this:
# In an .env file
PG_BOSS_NEW_OPTIONS={"connectionString":"postgresql://user:password@server:5432/database","archiveCompletedAfterSeconds":86400,"deleteAfterDays":30,"maintenanceIntervalMinutes":5}
# In the shell
PG_BOSS_NEW_OPTIONS='{"connectionString":"postgresql://user:password@server:5432/database","archiveCompletedAfterSeconds":86400,"deleteAfterDays":30,"maintenanceIntervalMinutes":5}'
You can read more about escaping JSON in environment variables in the JSON Env Vars documentation.
:::tip You don't need to set up the database manually
When using PgBoss, the database setup is automatically taken care of by the Wasp server, and doesn't need to be reflected in your schemas or migrations. The following information is given for your reference, and is explained in more detail in the PgBoss documentation.
:::
All job data will be stored in a separate database schema called pgboss. It has some internal tracking tables, such as job, archive, and schedule. PgBoss tables have a name column in most tables that will correspond to your Job identifier. Additionally, these tables maintain arguments, states, return values, retry information, start and expiration times, and other metadata required by PgBoss.
Renaming scheduled jobs
Wasp derives the Job's name from the worker function you pass to job. For example, job(emailReminder, ...) creates a Job named emailReminder, and Wasp uses that name in the name column of pgboss tables. If you change a name that had a schedule associated with it, pg-boss will continue scheduling those jobs but they will have no handlers associated, and will thus become stale and expire. To resolve this, you can remove the applicable row from the pgboss.schedule table.
For example, if you renamed a job from emailReminder to sendEmailReminder, you would need to remove the old scheduled job with the following SQL query:
BEGIN;
DELETE FROM pgboss.schedule WHERE name = 'emailReminder';
COMMIT;
Important: Only modify the database directly if you're comfortable with SQL operations. If you're unsure, consider keeping the old job name or restarting with a fresh database in development.
By default, PgBoss keeps job data for 12 hours after completion or failure. After that, it moves the data to an archive table, where it is kept for 7 days before being deleted. If you want to change this behavior, you can configure the PG_BOSS_NEW_OPTIONS environment variable to set custom values for job archival (archivedCompletedAfterSeconds/archiveFailedAfterSeconds) and removal (deleteAfterSeconds/deleteAfterMinutes/etc).
PG_BOSS_NEW_OPTIONS={"connectionString":"...your postgress connection url...","archiveCompletedAfterSeconds":86400,"deleteAfterDays":30,"maintenanceIntervalMinutes":5}
job specification<CardLink to="../api/@wasp.sh/spec/functions/job" kind="api" title="job" description="All the options for defining a job in the Wasp spec." />
An async function that performs the Job's work. Since Wasp executes Jobs on the server, its import path must lead to a NodeJS file. It receives two arguments:
args: Input: The data passed to the job when it's submitted.context: { entities: Entities }: The context object containing the entities you put in the Job spec.Here's an example worker function:
<Tabs groupId="js-ts"> <TabItem value="js" label="JavaScript"> ```js title="src/workers/bar.js" export const mySpecialJob = async ({ name }, context) => { console.log(`Hello ${name}!`) const tasks = await context.entities.Task.findMany({}) return { tasks } } ``` </TabItem> <TabItem value="ts" label="TypeScript"> ```ts title="src/workers/bar.ts" import { type MySpecialJob } from "wasp/server/jobs"type Input = { name: string; }
type Output = { tasks: Task[]; }
export const mySpecialJob: MySpecialJob<Input, Output> = async ({ name }, context) => {
console.log(`Hello ${name}!`)
const tasks = await context.entities.Task.findMany({})
return { tasks }
}
```
Read more about type-safe jobs in the [JavaScript API section](#javascript-api).
:::info Type-safe jobs
Wasp generates a generic type for each Job, which you can use to type your worker function. The type is named after the worker function you pass to `job`, converted to PascalCase, and is available in the `wasp/server/jobs` module. In the example above, the type is `MySpecialJob`.
The type takes two type arguments:
- `Input`: The type of the `args` argument of the worker function.
- `Output`: The type of the return value of the worker function.
:::
submit(jobArgs, executorOptions)jobArgs: InputexecutorOptions: objectSubmits a Job to be executed by an executor, optionally passing in a JSON job argument your job handler function receives, and executor-specific submit options.
<Tabs groupId="js-ts"> <TabItem value="js" label="JavaScript"> ```js title="someAction.js" const submittedJob = await mySpecialJob.submit({ name: "Johnny" }) ``` </TabItem> <TabItem value="ts" label="TypeScript"> ```ts title="someAction.ts" const submittedJob = await mySpecialJob.submit({ name: "Johnny" }) ``` </TabItem> </Tabs>delay(startAfter)startAfter: int | string | Date <Required />Delaying the invocation of the job handler. The delay can be one of:
The return value of submit() is an instance of SubmittedJob, which has the following fields:
jobId: The ID for the job in that executor.jobName: The Job name Wasp derived from the worker function you passed to job.executorName: The Symbol of the name of the job executor.There are also some namespaced, job executor-specific objects.