π Google Cloud Tasks, Simplified
A lightweight library that streamlines the use of Google Cloud Tasks.
πΉ Why Use This Library?
Using Google Cloud Tasks typically requires:
- Implementing a worker service.
- Defining and managing the API between the worker and client.
- Handling task queue management and scheduling.
- Setting up a local development environment.
- Configuring Google Cloud Tasks to send HTTP requests to the worker.
This library eliminates much of this complexity by leveraging tRPC to define the API between the worker and the client.
β
Key Features
- Seamless API communication between workers and clients.
- Simple task scheduling with an intuitive API.
- Type safety and jump-to-definition support via tRPC.
- Built-in support for local development with a Cloud Tasks emulator.
π Example Usage
Suppose we have one tasks worker (tasks-worker) and a backend API (api) that will be scheduling tasks.
Project Structure
An example project structure with Turborepo might look like this:
.
βββ apps/
βββ tasks-worker/
β βββ api.ts
β βββ index.ts
βββ api/
βββ src/
βββ someEndpoint.ts
Pro tip: If you have multiple task workers, use naming convention tasks-worker-<name>.
Two apps that are deployed as two services: api and tasks-worker.
π Step 1: Create the Task Worker (api.ts)
Define a tRPC-powered task server:
import { logger } from "@repo/logger";
import { createTasksServer } from "@panter/cloud-tasks/server";
import { z } from "zod";
logger.info("Starting tasks server...");
export const { runServer, router } = createTasksServer((t) =>
t.router({
createUser: t.procedure
.input(z.object({ name: z.string().min(5) }))
.mutation(async (opts) => {
logger.info(`creating user ${opts.input.name}`);
}),
doNothing: t.procedure.mutation(() => {
logger.info("doing nothing");
}),
sendEmail: t.procedure.mutation(() => {
logger.info("sending email");
}),
}),
);
export type Router = typeof router;
This creates a task server with three tRPC mutations: createUser, doNothing, and sendEmail.
Notice that mutations return nothing, as they are called by Google Cloud Tasks and their response is ignored.
π Step 2: Start the Task Worker (index.ts)
Initialize the worker server:
import { logger } from "@repo/logger";
import { runServer } from "./api";
const port = parseInt(process.env.PORT);
runServer(port);
logger.info(`π Task worker tRPC server running at http://localhost:${port}/`);
Pro tip: Set execution environment for tasks workers to "gen2", deny unauthenticated requests and set higher timeout in catladder:
"tasks-worker": {
dir: "./apps/tasks-worker",
deploy: {
service: {
allowUnauthenticated: false,
executionEnvironment: "gen2",
timeout: "1800s",
},
},
}
π Step 3: Create the Task Client in api
Now, in the backend API (api), define a task client to send tasks:
import { builder } from "../builder";
import { logger } from "@repo/logger";
import type { Router } from "../../tasks-worker-gitlab/api.ts";
import { createTasksClient } from "@panter/cloud-tasks/client";
const tasks = createTasksClient<Router>({
tasksWorkerUrl: new URL(process.env.TASKS_WORKER_URL),
queueName: "reasonable-queue-name",
emulator: process.env.ENV_SHORT === "local" ? { port: 6020 } : false,
logger,
});
builder.mutationField("runJob", (t) =>
t.field({
type: "String",
resolve: async () => {
await tasks.createUser.schedule({ name: "Bilbo Baggins" });
return "ok";
},
}),
);
Pro tip: Set the TASKS_WORKER_URL environment variable to the URL of the task worker service in catladder:
api: {
dir: "./apps/api",
vars: {
public: {
TASKS_WORKER_URL: "${tasks-worker:ROOT_URL_INTERNAL}",
}
},
}
π Important: Avoid Circular Dependencies
Notice that Router is imported as a type and by relative path. This makes sure there won't be a circular dependency when e.g. taks-worker needs to import api to use some business logic.
π Local Development: Using Cloud Tasks Emulator
For local testing, use the Cloud Tasks Emulator with Docker:
services:
gcloud-tasks-emulator:
image: ghcr.io/aertje/cloud-tasks-emulator:latest
command: -host 0.0.0.0 -port 8123
ports:
- "6020:8123"
extra_hosts:
- "host.containers.internal:host-gateway"
π― Summary
This library makes Google Cloud Tasks easy to use by:
- Removing the need for manual HTTP request handling.
- Providing a type-safe, tRPC-powered API.
- Enabling seamless communication between workers and clients.
- Offering built-in support for local development.
π‘ With this library, scheduling background tasks is as simple as calling a function! π