The Global Task Framework (GTF) provides a unified way to manage background tasks. It handles task execution, progress tracking, cancellation, and deduplication for both synchronous and asynchronous execution. The framework uses distributed locking internally to ensure race-free operations—you don't need to worry about concurrent task creation or cancellation conflicts.
GTF is disabled by default and must be enabled via the GLOBAL_TASK_FRAMEWORK feature flag in your superset_config.py:
FEATURE_FLAGS = { "GLOBAL_TASK_FRAMEWORK": True, }
When GTF is disabled:
/api/v1/task/* endpoints return 404@task-decorated function raises GlobalTaskFrameworkDisabledError:::note Future Migration When GTF is considered stable, it will replace legacy Celery tasks for built-in features like thumbnails and alerts & reports. Enabling this flag prepares your deployment for that migration. :::
from superset_core.tasks.decorators import task, get_context @task def process_data(dataset_id: int) -> None: ctx = get_context() @ctx.on_cleanup def cleanup(): logger.info("Processing complete") data = fetch_dataset(dataset_id) process_and_cache(data)
# Async execution - schedules on Celery worker task = process_data.schedule(dataset_id=123) print(task.status) # "pending" # Sync execution - runs inline in current process task = process_data(dataset_id=123) # ... blocks until complete print(task.status) # "success"
| Method | When to Use |
|---|---|
.schedule() | Long-running operations, background processing, when you need to return immediately |
| Direct call | Short operations, when deduplication matters, when you need the result before responding |
Both execution modes provide the same task features: deduplication, progress tracking, cancellation, and visibility in the Task List UI. The difference is whether execution happens in a Celery worker (async) or inline (sync).
PENDING ──→ IN_PROGRESS ────→ SUCCESS │ │ │ ├──────────→ FAILURE │ ↓ ↑ │ ABORTING ────────────┘ │ │ │ ├──────────→ TIMED_OUT (timeout) │ │ └─────────────┴──────────→ ABORTED (user cancel)
| Status | Description |
|---|---|
PENDING | Queued, awaiting execution |
IN_PROGRESS | Executing |
ABORTING | Abort/timeout triggered, abort handlers running |
SUCCESS | Completed successfully |
FAILURE | Failed with error or abort/cleanup handler exception |
ABORTED | Cancelled by user/admin |
TIMED_OUT | Exceeded configured timeout |
Access task context via get_context() from within any @task function. The context provides methods for updating task metadata and registering handlers.
Use update_task() to report progress and store custom payload data:
@task def my_task(items: list[int]) -> None: ctx = get_context() for i, item in enumerate(items): result = process(item) ctx.update_task( progress=(i + 1, len(items)), payload={"last_result": result} )
:::tip Call update_task() once per iteration for best performance. Frequent DB writes are throttled to limit metastore load, so batching progress and payload updates together in a single call ensures both are persisted at the same time. :::
The progress parameter accepts three formats:
| Format | Example | Display |
|---|---|---|
tuple[int, int] | progress=(3, 100) | 3 of 100 (3%) with ETA |
float (0.0-1.0) | progress=0.5 | 50% with ETA |
int | progress=42 | 42 processed |
:::tip Use the tuple format (current, total) whenever possible. It provides the richest information to users: showing both the count and percentage, while still computing ETA automatically. :::
The payload parameter stores custom metadata that can help users understand what the task is doing. Each call to update_task() replaces the previous payload completely.
In the Task List UI, when a payload is defined, an info icon appears in the Details column. Users can hover over it to see the JSON content.
Register handlers to run cleanup logic or respond to abort requests:
| Handler | When it runs | Use case |
|---|---|---|
on_cleanup | Always (success, failure, abort) | Release resources, close connections |
on_abort | When task is aborted | Set stop flag, cancel external operations |
@task def my_task() -> None: ctx = get_context() @ctx.on_cleanup def cleanup(): logger.info("Task ended, cleaning up") @ctx.on_abort def handle_abort(): logger.info("Abort requested") # ... task logic
Multiple handlers of the same type execute in LIFO order (last registered runs first). Abort handlers run first when abort is detected, then cleanup handlers run when the task ends.
All registered handlers will always be attempted, even if one fails. This ensures that a failure in one handler doesn't prevent other handlers from running their cleanup logic.
For example, if you have three cleanup handlers and the second one throws an exception:
If any handler fails, the task is marked as FAILURE with combined error details showing all handler failures.
:::tip Write handlers to be independent and self-contained. Don‘t assume previous handlers succeeded, and don’t rely on shared state between handlers. :::
When users click Cancel in the Task List, the system decides whether to abort (stop) the task or unsubscribe (remove the user from a shared task). Abort occurs when:
Pending tasks can always be aborted: they simply won't start. In-progress tasks require an abort handler to be abortable:
@task def abortable_task(items: list[str]) -> None: ctx = get_context() should_stop = False @ctx.on_abort def handle_abort(): nonlocal should_stop should_stop = True logger.info("Abort signal received") @ctx.on_cleanup def cleanup(): logger.info("Task ended, cleaning up") for item in items: if should_stop: return # Exit gracefully process(item)
Key points:
on_abort marks the task as abortable and starts the abort listenerThe framework automatically skips execution if a task was aborted while pending: no manual check needed at task start.
:::tip Always implement an abort handler for long-running tasks. This allows users to cancel unneeded tasks and free up worker capacity for other operations. :::
Set a timeout to automatically abort tasks that run too long:
from superset_core.tasks.decorators import task, get_context from superset_core.tasks.types import TaskOptions # Set default timeout in decorator @task(timeout=300) # 5 minutes def process_data(dataset_id: int) -> None: ctx = get_context() should_stop = False @ctx.on_abort def handle_abort(): nonlocal should_stop should_stop = True for chunk in fetch_large_dataset(dataset_id): if should_stop: return process(chunk) # Override timeout at call time task = process_data.schedule( dataset_id=123, options=TaskOptions(timeout=600) # Override to 10 minutes )
The timeout timer starts when the task begins executing (status changes to IN_PROGRESS). When the timeout expires:
With an abort handler registered: The task transitions to ABORTING, abort handlers run, then cleanup handlers run. The final status depends on handler execution:
TIMED_OUT statusFAILURE statusWithout an abort handler: The framework cannot forcibly terminate the task. A warning is logged, and the task continues running. The Task List UI shows a warning indicator (⚠️) in the Details column to alert users that the timeout cannot be enforced.
| Source | Priority | Example |
|---|---|---|
TaskOptions.timeout | Highest | options=TaskOptions(timeout=600) |
@task(timeout=...) | Default | @task(timeout=300) |
| Not set | No timeout | Task runs indefinitely |
Call-time options always override decorator defaults, allowing tasks to have sensible defaults while permitting callers to extend or shorten the timeout for specific use cases.
:::warning Timeouts require an abort handler to be effective. Without one, the timeout triggers only a warning and the task continues running. Always implement an abort handler when using timeouts. :::
Use task_key to prevent duplicate task execution:
from superset_core.tasks.types import TaskOptions # Without key - creates new task each time (random UUID) task1 = my_task.schedule(x=1) task2 = my_task.schedule(x=1) # Different task # With key - joins existing task if active task1 = my_task.schedule(x=1, options=TaskOptions(task_key="report_123")) task2 = my_task.schedule(x=1, options=TaskOptions(task_key="report_123")) # Returns same task
When a task with matching key already exists, the user is added as a subscriber and the existing task is returned. This behavior is consistent across all scopes—private tasks naturally have only one subscriber since their deduplication key includes the user ID.
Deduplication only applies to active tasks (pending/in-progress). Once a task completes, a new task with the same key can be created.
When a sync call joins an existing task, it blocks until the task completes:
# Schedule async task task = my_task.schedule(options=TaskOptions(task_key="report_123")) # Later sync call with same key blocks until completion of the active task task2 = my_task(options=TaskOptions(task_key="report_123")) assert task.uuid == task2.uuid # True print(task2.status) # "success" (terminal status)
from superset_core.tasks.decorators import task from superset_core.tasks.types import TaskScope @task # Private by default def private_task(): ... @task(scope=TaskScope.SHARED) # Multiple users can subscribe def shared_task(): ... @task(scope=TaskScope.SYSTEM) # Admin-only visibility def system_task(): ...
| Scope | Visibility | Cancel Behavior |
|---|---|---|
PRIVATE | Creator only | Cancels immediately |
SHARED | All subscribers | Last subscriber cancels; others unsubscribe |
SYSTEM | Admins only | Admin cancels |
Completed tasks accumulate in the database over time. Configure a scheduled prune job to automatically remove old tasks:
# In your superset_config.py, add to your Celery beat schedule: CELERY_CONFIG.beat_schedule["prune_tasks"] = { "task": "prune_tasks", "schedule": crontab(minute=0, hour=0), # Run daily at midnight "kwargs": { "retention_period_days": 90, # Keep tasks for 90 days "max_rows_per_run": 10000, # Limit deletions per run }, }
The prune job only removes tasks in terminal states (SUCCESS, FAILURE, ABORTED, TIMED_OUT). Active tasks (PENDING, IN_PROGRESS, ABORTING) are never pruned.
See superset/config.py for a complete example configuration.
:::tip Distributed Coordination for Faster Notifications By default, abort detection and sync join-and-wait use database polling. Configure DISTRIBUTED_COORDINATION_CONFIG to enable Redis pub/sub for real-time notifications. See Distributed Coordination Backend for configuration details. :::
@task( name: str | None = None, scope: TaskScope = TaskScope.PRIVATE, timeout: int | None = None )
name: Task identifier (defaults to function name)scope: PRIVATE, SHARED, or SYSTEMtimeout: Default timeout in seconds (can be overridden via TaskOptions)| Method | Description |
|---|---|
update_task(progress, payload) | Update progress and/or custom payload |
on_cleanup(handler) | Register cleanup handler |
on_abort(handler) | Register abort handler (makes task abortable) |
TaskOptions( task_key: str | None = None, task_name: str | None = None, timeout: int | None = None )
task_key: Deduplication key (also used as display name if task_name is not set)task_name: Human-readable display name for the Task List UItimeout: Timeout in seconds (overrides decorator default):::tip Provide a descriptive task_name for better readability in the Task List UI. While task_key is used for deduplication and may be technical (e.g., chart_export_123), task_name can be user-friendly (e.g., "Export Sales Chart 123"). :::
Let exceptions propagate: the framework captures them automatically and sets task status to FAILURE:
@task def risky_task() -> None: # No try/catch needed - framework handles it result = operation_that_might_fail()
On failure, the framework records:
error_message: Exception messageexception_type: Exception class namestack_trace: Full traceback (visible when SHOW_STACKTRACE=True)In the Task List UI, failed tasks show error details when hovering over the status. When stack traces are enabled, a separate bug icon appears in the Details column for viewing the full traceback.
Cleanup handlers still run after an exception, so resources can be properly released as necessary.
:::tip Use descriptive exception messages. In environments where stack traces are hidden (SHOW_STACKTRACE=False), users see only the error message and exception type when hovering over failed tasks. Clear messages help users troubleshoot issues without administrator assistance. :::