
Security News
npm Tooling Bug Incorrectly Marks One-Character Packages as Security Holders
npm confirmed a tooling bug incorrectly marked several one-character packages as security holders and said it was working on a rollback.
@panter/cloud-tasks
Advanced tools
A lightweight library that streamlines the use of **Google Cloud Tasks**.
A lightweight library that streamlines the use of Google Cloud Tasks.
Using Google Cloud Tasks typically requires:
This library eliminates much of this complexity by leveraging tRPC to define the API between the worker and the client.
Suppose we have one tasks worker (tasks-worker) and a backend API (api) that will be scheduling tasks.
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.
api.ts)Define a tRPC-powered task server:
// apps/tasks-worker/api.ts
import { logger } from "@repo/logger";
// yarn add @panter/cloud-tasks
import { createTasksServer } from "@panter/cloud-tasks/server";
import { z } from "zod";
logger.info("Starting tasks server...");
export const { runServer, router } = createTasksServer({
createRouter: (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 the router type for use in the client
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.
index.ts)Initialize the worker server:
// apps/tasks-worker/index.ts
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:
// catladder.ts
"tasks-worker": {
dir: "./apps/tasks-worker",
deploy: {
service: {
allowUnauthenticated: false,
executionEnvironment: "gen2",
timeout: "1800s",
},
},
}
apiNow, in the backend API (api), define a task client to send tasks:
// apps/api/src/someEndpoint.ts
import { builder } from "../builder";
import { logger } from "@repo/logger";
// NOTE: we import Router type from the tasks-worker without introducing a runtime dependency
import type { Router } from "../../tasks-worker-gitlab/api.ts";
// yarn add @panter/cloud-tasks
import { createTasksClient } from "@panter/cloud-tasks/client";
const tasks = createTasksClient<Router>({
// TASKS_WORKER_URL is the URL of the task worker service
tasksWorkerUrl: new URL(process.env.TASKS_WORKER_URL),
// special characters are not allowed in queue name
queueName: "reasonable-queue-name",
// enable emulator in local development
emulator: process.env.ENV_SHORT === "local" ? { port: 6020 } : false,
// optional: provide a custom logger
// assumes `logger.info(metadata, message)` order of arguments (pino logger)
logger,
// or provide custom adapter (e.g. for winston logger):
// logger: {
// info: (meta, message) => logger.info(message, meta),
// warn: (meta, message) => logger.warn(message, meta),
// error: (meta, message) => logger.error(message, meta),
// },
});
builder.mutationField("runJob", (t) =>
t.field({
type: "String",
resolve: async () => {
// Schedule a task with a payload
await tasks.createUser.schedule({ name: "Bilbo Baggins" });
// ^^^^^^^^^^ Try jumping to definition here!
return "ok";
},
}),
);
Pro tip: Set the
TASKS_WORKER_URLenvironment variable to the URL of the task worker service in catladder:
// catladder.ts
api: {
dir: "./apps/api",
vars: {
public: {
TASKS_WORKER_URL: "${tasks-worker:ROOT_URL_INTERNAL}",
}
},
}
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.
For local testing, use the Cloud Tasks Emulator with Docker:
# docker-compose.yml
services:
gcloud-tasks-emulator:
image: ghcr.io/aertje/cloud-tasks-emulator:latest
command: -host 0.0.0.0 -port 8123
ports:
- "6020:8123"
# NOTE: Comment out `extra_hosts` if using Podman (see: https://github.com/containers/podman/issues/21681)
extra_hosts:
- "host.containers.internal:host-gateway"
This library makes Google Cloud Tasks easy to use by:
💡 With this library, scheduling background tasks is as simple as calling a function! 🚀
FAQs
A lightweight library that streamlines the use of **Google Cloud Tasks**.
The npm package @panter/cloud-tasks receives a total of 62 weekly downloads. As such, @panter/cloud-tasks popularity was classified as not popular.
We found that @panter/cloud-tasks demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 3 open source maintainers collaborating on the project.
Did you know?

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Security News
npm confirmed a tooling bug incorrectly marked several one-character packages as security holders and said it was working on a rollback.

Research
/Security News
Newer packages in this compromise use native extensions and .pth loaders to execute JavaScript stealers in developer environments.

Research
Socket found 37 malicious PyPI wheels that abuse Python startup hooks to launch a Bun-powered credential stealer tied to Mini Shai-Hulud/Miasma.