A framework agnostic Google Cloud Tasks library for the Push architecture.
Project description
py-cloud-task
A Framework Agnostic Client for Google Cloud Tasks.
Move from "Pull" (Workers) to "Push" (Serverless) architecture effortlessly.
py-cloud-task is a lightweight, async-first library that abstracts the complexity of Google Cloud Tasks. It provides a developer experience similar to Celery or TaskIQ but is designed specifically for Serverless environments (Cloud Run, App Engine, Cloud Functions, FastAPI).
It handles serialization, authentication (OIDC), scheduling, and—crucially—FastAPI Dependency Injection automatically.
Why use this instead of Celery/Redis?
| Feature | Celery / Redis (Pull) | py-cloud-task (Push) |
|---|---|---|
| Architecture | Workers poll Redis 24/7 ("Are there tasks?") | Google calls your API via HTTP ("Here is a task") |
| Cost | You pay for idle workers & Redis instances | Pay-per-use (Scale to Zero supported) |
| Infra | Requires Redis/RabbitMQ management | Zero Ops (Managed by Google) |
| Retries | Managed by worker code | Native (Exponential backoff managed by GCP) |
| DX | Heavy setup | Decorator-based (Just like FastAPI) |
Installation
Currently, the package is available via GitHub. You can install it using uv or pip.
Using uv (Recommended)
# Instalação Core
uv add "py-cloud-task @ git+https://github.com/uhmiller/py-cloud-task.git"
# Com suporte a FastAPI
uv add "py-cloud-task[fastapi] @ git+https://github.com/uhmiller/py-cloud-task.git"
# Para simulação local (testes)
uv add "py-cloud-task[test] @ git+https://github.com/uhmiller/py-cloud-task.git"
Quick Start
1. Configure the Client
The CloudTaskClient is the entry point. It holds the configuration for your Google Cloud project and queue.
from cloudtask import CloudTaskClient
client = CloudTaskClient(
project="my-gcp-project",
location="europe-west1",
queue="default",
url="[https://api.myapp.com/tasks/run](https://api.myapp.com/tasks/run)", # The public URL of your worker
sae="my-service-account@my-gcp-project.iam.gserviceaccount.com", # Service Account email for OIDC auth
secret="super-secret-token", # Optional: Header secret for extra security
force_to_queue=None, # Optional: Force all tasks to a specific queue (useful for Staging)
eager=None, # None = Production (Sends to Google Cloud)
)
2. Define a Task
Use the @client.task decorator. You can define tasks anywhere in your code.
@client.task(queue='high-priority', name='unique-task-name')
async def send_welcome_email(user_id: str, email: str):
print(f"Sending email to {email}...")
# ... logic to send email ...
return "sent"
3. Trigger the Task
You can trigger tasks asynchronously. This will serialize the arguments and send them to Google Cloud Tasks.
# Simple trigger
await send_welcome_email(user_id="123", email="user@example.com").push()
Advanced Usage
Scheduling (Delayed Execution)
Schedule a task to run in the future using the at parameter.
from datetime import datetime, timedelta
# Run 1 hour from now
eta = datetime.now() + timedelta(hours=1)
await send_welcome_email("123", "user@example.com").push(at=eta)
Task Deduplication (Named Tasks)
Google Cloud Tasks ensures that tasks with the same name are executed only once. You can set a custom name to prevent duplicate execution.
# Instantiate the task wrapper first
task = send_welcome_email("123", "user@example.com")
# Set a deterministic name (e.g., specific to the user and action)
task.name = "welcome-email-user-123"
# Push to cloud
await task.push()
Local Development (Eager Modes)
When developing locally, you often don't want to send tasks to Google Cloud. The eager parameter supports three modes
to help you develop and test safely.
Mode 1: Immediate Execution (eager="immediate")
Runs the function directly in the current process. Fastest option for Unit Tests.
client = CloudTaskClient(..., eager="immediate")
await send_welcome_email("123", "user@example.com").push()
# Result: Function runs instantly. No HTTP. No Serialization.
Mode 2: Remote Simulation (eager="remote")
Simulates a full HTTP request to your local worker using httpx. This is perfect for Integration Tests because it
validates serialization, headers, and dependency injection without needing Google infrastructure.
client = CloudTaskClient(..., eager="remote")
await send_welcome_email("123", "user@example.com").push()
# Result: Sends POST http://localhost:8000/tasks/run.
Mode 3: Production (eager=None)
The default behavior. Serializes the task and sends it to Google Cloud Tasks.
FastAPI Integration
py-cloud-task has first-class support for FastAPI. It leverages FastAPI's native Dependency Injection system.
1. Setup the Router
from fastapi import FastAPI
from cloudtask.fastapi import CloudTaskRouter
from app.core.tasks import client
app = FastAPI()
# Register the route that receives tasks from Google
app.include_router(CloudTaskRouter(client), prefix="/tasks")
2. Use Depends in Tasks
You can inject database sessions, services, or any other dependency directly into your tasks, just like in API endpoints.
from fastapi import Depends
from sqlalchemy.ext.asyncio import AsyncSession
from app.core.db import get_db
@client.task()
async def process_order(
order_id: int,
db: AsyncSession = Depends(get_db) # <--- Magic happens here
):
# The 'db' session is created, injected, and closed automatically!
order = await db.get(Order, order_id)
order.status = "processed"
await db.commit()
Note: When triggering the task, you only pass the data arguments. The dependencies are resolved automatically by the worker.
# Correct usage (Dependency is ignored during push)
await process_order(order_id=500).push()
Security
To ensure that only Google Cloud Tasks can call your worker endpoint, the library supports two mechanisms:
- OIDC Token (Recommended): The library automatically attaches an OIDC token identifying the Service Account. Your Cloud Run/Functions service should validate this token (Google handles this automatically for Cloud Run if you don't allow unauthenticated invocations).
- Secret Header: You can configure a shared secret.
client = CloudTaskClient(..., secret="my-secret-key")
The router will automatically validate the X-PYCT-SECRET header and reject unauthorized requests (403 Forbidden).
Contributing
Contributions are welcome! If you find a bug or want to add a feature (e.g., Flask or Django adapters), please open an issue or submit a PR.
Built with ❤️ by the engineering team at Ziett
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file py_cloud_task-0.1.0.tar.gz.
File metadata
- Download URL: py_cloud_task-0.1.0.tar.gz
- Upload date:
- Size: 11.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.9.29 {"installer":{"name":"uv","version":"0.9.29","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
952f305790d6ba6a590f3d11d5cf784690d460e40f006029ed9de13af999eb95
|
|
| MD5 |
807f7a33e1265a709fda1aea001ca781
|
|
| BLAKE2b-256 |
500bdd153f56d3f3527e578d2b22d2fb815263540ad16eb3faa6dd1689b6f3ea
|
File details
Details for the file py_cloud_task-0.1.0-py3-none-any.whl.
File metadata
- Download URL: py_cloud_task-0.1.0-py3-none-any.whl
- Upload date:
- Size: 13.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.9.29 {"installer":{"name":"uv","version":"0.9.29","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6fe8150f7ffbfd4e319ee02b4e8b06621d12e539dc3ceebbe6cca4ea2327590a
|
|
| MD5 |
74f8a05a4db6b908bd2351f76ad5db8a
|
|
| BLAKE2b-256 |
8b4c5d5b62fd04016df77ad16aac1c724ad0d550251826af1244a7db0e941760
|