ZeroAlloc.Scheduling
1.2.0
dotnet add package ZeroAlloc.Scheduling --version 1.2.0
NuGet\Install-Package ZeroAlloc.Scheduling -Version 1.2.0
<PackageReference Include="ZeroAlloc.Scheduling" Version="1.2.0" />
<PackageVersion Include="ZeroAlloc.Scheduling" Version="1.2.0" />
<PackageReference Include="ZeroAlloc.Scheduling" />
paket add ZeroAlloc.Scheduling --version 1.2.0
#r "nuget: ZeroAlloc.Scheduling, 1.2.0"
#:package ZeroAlloc.Scheduling@1.2.0
#addin nuget:?package=ZeroAlloc.Scheduling&version=1.2.0
#tool nuget:?package=ZeroAlloc.Scheduling&version=1.2.0
ZeroAlloc.Scheduling
ZeroAlloc.Scheduling is a source-generated background job scheduler for .NET 8 and .NET 10. Decorate any class with [Job] and the source generator wires up the executor, DI registration, and recurring startup automatically — no reflection, no convention scanning at runtime.
Install
The source generator is bundled into the main package — a single PackageReference is all you need:
dotnet add package ZeroAlloc.Scheduling
dotnet add package ZeroAlloc.Scheduling.InMemory # or EfCore / Redis
The standalone
ZeroAlloc.Scheduling.Generatorpackage is still published for backwards compatibility with existing direct PackageReferences, but new consumers should reference onlyZeroAlloc.Scheduling.
Example
// 1. Define a job — the generator picks it up automatically
[Job(Every = Every.Hour)]
public sealed class CleanupExpiredSessionsJob : IJob
{
private readonly ISessionRepository _repo;
public CleanupExpiredSessionsJob(ISessionRepository repo) => _repo = repo;
public async ValueTask ExecuteAsync(JobContext ctx, CancellationToken ct)
=> await _repo.DeleteExpiredAsync(ct);
}
// 2. Register — generated AddCleanupExpiredSessionsJob() wires executor + recurring startup
services.AddScheduling()
.WithInMemoryStore()
.AddCleanupExpiredSessionsJob();
// 3. Enqueue a one-off job from application code
public class OrderService(IScheduler scheduler)
{
public async Task CompleteOrderAsync(Order order, CancellationToken ct)
{
await ProcessAsync(order, ct);
await scheduler.EnqueueAsync(new SendOrderConfirmationJob(order.Id), ct);
}
}
Performance
Head-to-head vs Coravel 5.0.4 (de-facto in-process scheduler) and a hand-rolled BackgroundService + Timer baseline. .NET 10.0.7, i9-12900HK, BenchmarkDotNet v0.15.4.
| Operation | Coravel | Naive Timer | ZA.Scheduling |
|---|---|---|---|
| Job registration | 8,211 ns / 696 B per call | — (compile-time) | compile-time, 0 ns runtime |
| Single dispatch | (registration only) | 0.01 ns | 0.25 ns (parity, JIT-inlined) |
Honest reading: Coravel's Schedule() is a runtime call that allocates a queue entry; ZA.Scheduling's [Job] registration happens at compile time via source generation — no equivalent runtime cost. For dispatch overhead, ZA matches a hand-written Timer (both inlined to near-zero). The value of the abstraction is [Job] itself (declarative retries / cron / store-backed persistence), not raw dispatch speed.
Full methodology + design analysis: docs/performance.md.
Features
- Source generator —
[Job]on a class emits a typed executor, DI extension method, and optional recurring startup (IHostedService) - Recurring jobs —
[Job(Cron = "0 * * * *")]or[Job(Every = Every.Hour)]— scheduled via Cronos at startup - Retry with backoff — exponential retry up to
MaxAttempts(per-job or global); dead-letters after exhaustion - Multiple backends — InMemory (dev/test), EF Core (SQL Server / PostgreSQL / SQLite), Redis
- Dashboard — embedded HTML/JS dashboard via
app.MapJobsDashboard("/jobs") - Blazor component —
<JobsDashboard>Razor component for integration into Blazor apps - Mediator bridge —
[Job]+IRequest<Unit>auto-registersMediatorJobTypeExecutor<T>, routing execution through ZeroAlloc.Mediator pipeline behaviors - Native AOT compatible — no reflection at runtime; all dispatch resolved at compile time
Dashboard
An embedded HTML/JS dashboard — summary cards for each job state, a live-refreshing table with per-row requeue / delete actions, and fully responsive down to mobile widths.
app.MapJobsDashboard("/jobs");
// Optional: protect with auth
app.MapJobsDashboard("/jobs").RequireAuthorization("AdminPolicy");

Tablet (768 × 1024) and mobile (375 × 812) captures live in docs/screenshots/.
See Dashboard for the full endpoint reference and the Blazor component.
Packages
| Package | Description |
|---|---|
ZeroAlloc.Scheduling |
Core interfaces, worker service, scheduler |
ZeroAlloc.Scheduling.Generator |
Source generator (analyzer reference) |
ZeroAlloc.Scheduling.InMemory |
In-process store for development and testing |
ZeroAlloc.Scheduling.EfCore |
EF Core store (SQL Server, PostgreSQL, SQLite) |
ZeroAlloc.Scheduling.Redis |
Redis store for distributed deployments |
ZeroAlloc.Scheduling.Dashboard |
Embedded HTML dashboard (MapJobsDashboard) |
ZeroAlloc.Scheduling.Dashboard.Blazor |
Blazor component library |
ZeroAlloc.Scheduling.Mediator |
ZeroAlloc.Mediator bridge |
Documentation
| Page | Description |
|---|---|
| Getting Started | Install and schedule your first job in five minutes |
| Source Generator | [Job], Every, Cron, generated extension methods |
| Backends | InMemory, EF Core, and Redis store configuration |
| Dashboard | Embedded HTML dashboard and Blazor component |
| Mediator Bridge | Route job execution through ZeroAlloc.Mediator |
| Diagnostics | ZASCH001 compiler warning reference |
| Performance | Throughput, allocation profile, and tuning guide |
License
MIT
| Product | Versions Compatible and additional computed target framework versions. |
|---|---|
| .NET | net8.0 is compatible. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 is compatible. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. net10.0 is compatible. net10.0-android was computed. net10.0-browser was computed. net10.0-ios was computed. net10.0-maccatalyst was computed. net10.0-macos was computed. net10.0-tvos was computed. net10.0-windows was computed. |
-
net10.0
- Cronos (>= 0.9.0)
- Microsoft.Extensions.DependencyInjection.Abstractions (>= 10.0.0)
- Microsoft.Extensions.Hosting.Abstractions (>= 8.0.1)
- Microsoft.Extensions.Logging.Abstractions (>= 8.0.2)
- Microsoft.Extensions.Options (>= 8.0.2)
- ZeroAlloc.Resilience (>= 1.0.0)
- ZeroAlloc.Serialisation (>= 2.0.0)
- ZeroAlloc.StateMachine (>= 1.0.0)
- ZeroAlloc.ValueObjects (>= 1.2.0)
-
net8.0
- Cronos (>= 0.9.0)
- Microsoft.Extensions.DependencyInjection.Abstractions (>= 10.0.0)
- Microsoft.Extensions.Hosting.Abstractions (>= 8.0.1)
- Microsoft.Extensions.Logging.Abstractions (>= 8.0.2)
- Microsoft.Extensions.Options (>= 8.0.2)
- ZeroAlloc.Resilience (>= 1.0.0)
- ZeroAlloc.Serialisation (>= 2.0.0)
- ZeroAlloc.StateMachine (>= 1.0.0)
- ZeroAlloc.ValueObjects (>= 1.2.0)
-
net9.0
- Cronos (>= 0.9.0)
- Microsoft.Extensions.DependencyInjection.Abstractions (>= 10.0.0)
- Microsoft.Extensions.Hosting.Abstractions (>= 8.0.1)
- Microsoft.Extensions.Logging.Abstractions (>= 8.0.2)
- Microsoft.Extensions.Options (>= 8.0.2)
- ZeroAlloc.Resilience (>= 1.0.0)
- ZeroAlloc.Serialisation (>= 2.0.0)
- ZeroAlloc.StateMachine (>= 1.0.0)
- ZeroAlloc.ValueObjects (>= 1.2.0)
NuGet packages (8)
Showing the top 5 NuGet packages that depend on ZeroAlloc.Scheduling:
| Package | Downloads |
|---|---|
|
ZeroAlloc.Scheduling.EfCore
EF Core job store adapter for ZeroAlloc.Scheduling. |
|
|
ZeroAlloc.Scheduling.Dashboard.Blazor
Blazor component library for ZeroAlloc.Scheduling dashboard. |
|
|
ZeroAlloc.Scheduling.InMemory
In-memory job store for ZeroAlloc.Scheduling — use in tests. |
|
|
ZeroAlloc.Scheduling.Dashboard
Minimal API dashboard for ZeroAlloc.Scheduling — JSON endpoints + embedded HTML/JS UI. |
|
|
ZeroAlloc.Scheduling.Mediator
ZeroAlloc.Mediator bridge for ZeroAlloc.Scheduling — dispatches jobs via IMediator.Send. |
GitHub repositories
This package is not used by any popular GitHub repositories.
| Version | Downloads | Last Updated |
|---|---|---|
| 1.2.0 | 172 | 5/14/2026 |
| 1.1.4 | 249 | 5/12/2026 |
| 1.1.3 | 243 | 5/7/2026 |
| 1.1.2 | 240 | 5/6/2026 |
| 1.1.1 | 256 | 5/3/2026 |
| 1.1.0 | 232 | 5/1/2026 |
| 1.0.4 | 198 | 5/1/2026 |
| 1.0.3 | 222 | 4/30/2026 |
| 1.0.2 | 230 | 4/29/2026 |
| 1.0.1 | 224 | 4/29/2026 |
| 1.0.0 | 223 | 4/28/2026 |
| 0.3.1 | 200 | 4/28/2026 |
| 0.3.0 | 189 | 4/28/2026 |
| 0.2.2 | 196 | 4/28/2026 |
| 0.2.1 | 189 | 4/26/2026 |
| 0.2.0 | 202 | 4/26/2026 |
| 0.1.6 | 200 | 4/25/2026 |
| 0.1.5 | 197 | 4/25/2026 |
| 0.1.4 | 190 | 4/24/2026 |
| 0.1.3 | 190 | 4/23/2026 |