Scheduler
In distributed applications, it is often necessary to schedule messages for deferred execution—whether for implementing delayed workflows, retry mechanisms, or time-based business processes. Brighter provides built-in scheduling capabilities to handle these scenarios.
Overview
Brighter's scheduling support allows you to:
Defer message delivery to a specific time or after a delay
Schedule commands and events for future execution
Implement retry with delay for transient failures
Build time-based workflows and business processes
Scheduling integrates seamlessly with Brighter's messaging infrastructure, supporting both in-process handlers and external message brokers.
Use Cases
Common scenarios for message scheduling include:
Delayed Notifications: Send reminder emails after a delay (e.g., "Complete your registration")
Retry with Backoff: Retry failed operations with exponential backoff
Time-Based Workflows: Execute business processes at specific times (e.g., "End of day reports")
Rate Limiting: Spread message processing over time to avoid overwhelming downstream systems
Grace Periods: Defer actions to allow for cancellation (e.g., "Cancel order within 30 minutes")
Scheduled Jobs: Trigger recurring or one-time tasks at predetermined times
Scheduling Methods
The IAmACommandProcessor interface provides methods for scheduling delayed messages:
Method Parameters
DateTimeOffset at
Specifies the absolute time when the message should be delivered:
TimeSpan delay
Specifies a relative delay from the current time:
Return Value: Scheduler ID
All scheduling methods return a string representing the scheduler ID. This ID can be used to:
Cancel a scheduled message before it executes
Reschedule a message to a different time (cancel + schedule new)
Track scheduled messages in your application
Note: The ability to cancel/reschedule depends on your chosen scheduler implementation.
How Scheduling Works Internally
When you call a scheduling method on the Command Processor, Brighter internally creates special scheduling messages that trigger your actual message at the specified time.
Internal Message Types
Brighter uses two internal message types for scheduling:
FireSchedulerMessage: Used when scheduling messages to external brokers (with
Postmethods)FireSchedulerRequest: Used when scheduling in-process commands/events (with
Send/Publishmethods)
Scheduling Flow
Configuration
To use scheduling, you need to configure a scheduler when setting up Brighter:
Native Transport Delay Support
Some message transports have native support for message delay, eliminating the need for an external scheduler in certain scenarios.
Transports with Native Delay Support
Azure Service Bus
Yes
Unlimited
Uses ScheduledEnqueueTimeUtc property
AWS SQS
Limited
15 minutes
Native delay only up to 15 minutes
AWS SNS
No
N/A
Requires scheduler
Kafka
No
N/A
Requires scheduler
Redis
No
N/A
Requires scheduler
When to Use Native Transport Delay
Use native transport delay when:
Your transport supports it (RabbitMQ, Azure Service Bus)
Delays are within transport limits (e.g., <15 min for SQS)
You don't need to cancel/reschedule messages
You want to minimize infrastructure dependencies
Use an external scheduler when:
Your transport doesn't support native delay
You need delays beyond transport limits (e.g., >15 min on SQS)
You need to cancel or reschedule messages
You need centralized scheduling management
Requeue with Delay
Requeue with Delay is a special feature that allows handlers to defer message processing when encountering transient failures.
How It Works
When your handler encounters a transient error (e.g., database temporarily unavailable), you can throw a DeferMessageAction exception to requeue the message with a delay:
Configuration
Configure requeue delay in your subscription:
Transport-Specific Behavior
Transports with Native Delay (RabbitMQ, Azure Service Bus):
Uses native transport delay mechanism
No external scheduler required
Message requeued directly on the broker
Transports without Native Delay (Kafka, AWS SNS, etc.):
Requires an external scheduler (Quartz, Hangfire, etc.)
Message is scheduled via the configured scheduler
Falls back to immediate requeue if no scheduler configured
Choosing a Scheduler
Brighter supports multiple scheduler implementations. Your choice depends on your deployment environment and requirements.
Scheduler Comparison
Quartz.NET
Recommended
Database
Yes
Yes
No
General production use
Hangfire
Recommended
Database
Yes
Yes
No
.NET applications with dashboard
AWS Scheduler
Recommended
AWS
Limited
Limited
Yes
AWS deployments
Azure Service Bus
Recommended
Azure
No
No
Yes
Azure deployments
InMemory
Dev/Test Only
None
Yes
Yes
No
Testing, development
Scheduler Recommendations
For Production
Quartz.NET - Best for general production use:
Mature, battle-tested scheduling library
Supports persistent job stores (SQL, MongoDB, etc.)
Distributed scheduling with clustering
Full cancellation and reschedule support
Hangfire - Best for .NET applications needing a dashboard:
Easy setup and configuration
Built-in web dashboard for monitoring
Persistent job storage (SQL Server, PostgreSQL, Redis, etc.)
Background job processing beyond scheduling
Note:
Brighter.Hangfireassembly is NOT strong-named due to Hangfire limitations
For Cloud Providers
AWS Scheduler - Best for AWS deployments:
Native AWS service (EventBridge Scheduler)
No infrastructure to manage
Direct integration with SNS/SQS
IAM-based security
Limited cancellation/reschedule support
Azure Service Bus Scheduler - Best for Azure deployments:
Built into Azure Service Bus
No additional infrastructure
Native message delay support
No direct cancellation support (must use separate fire scheduler topic/queue)
For Development and Testing
InMemory Scheduler - Only for dev/test environments:
No external dependencies
Fast and simple for unit tests
Supports cancellation and reschedule
NOT durable - crashes lose all scheduled messages
NOT for production unless data loss is acceptable
Decision Guide
Code Examples
Basic Scheduling with DateTimeOffset
Schedule a command for a specific absolute time:
Basic Scheduling with TimeSpan
Schedule a command with a relative delay:
Scheduling with Post for External Bus
Schedule a message to an external broker:
Cancelling a Scheduled Message
Cancel a previously scheduled message:
Note: Cancellation support depends on your scheduler implementation. AWS Scheduler and Azure Service Bus Scheduler have limited or no cancellation support.
Retry with Exponential Backoff
Implement retry logic with increasing delays:
Using Requeue with Delay in a Handler
Configuration Examples
Configuring with Hangfire
Configuring with Quartz.NET
Configuring with InMemory (Development Only)
Best Practices
Always store scheduler IDs if you need to cancel or track scheduled messages
Use DateTimeOffset for absolute times and TimeSpan for relative delays
Consider time zones when scheduling with DateTimeOffset (use UTC when possible)
Choose the right scheduler based on your deployment environment
Use InMemory scheduler only for testing - it's not durable
Configure requeue delay appropriately - too short can overwhelm the system, too long delays recovery
Implement idempotency in handlers that process scheduled messages
Monitor scheduled jobs using your scheduler's dashboard/tools (Hangfire dashboard, AWS Console, etc.)
Set reasonable max delays for requeue to avoid indefinite retries
Use native transport delay when available to simplify infrastructure
Related Documentation
Quartz Scheduler - Quartz.NET scheduler configuration
Hangfire Scheduler - Hangfire scheduler configuration
AWS Scheduler - AWS EventBridge Scheduler configuration
Azure Scheduler - Azure Service Bus Scheduler configuration
InMemory Scheduler - InMemory scheduler for testing
Custom Scheduler - Implementing your own scheduler
Handler Failure - Error handling and retry strategies
Summary
Brighter provides comprehensive scheduling support for deferred message execution
Choose between absolute time (DateTimeOffset) or relative delay (TimeSpan)
Native transport delay available for RabbitMQ and Azure Service Bus (with limitations on SQS)
External schedulers (Quartz, Hangfire) required for transports without native delay
Requeue with Delay enables sophisticated retry strategies for transient failures
Use production schedulers (Quartz, Hangfire, AWS/Azure) for reliable scheduling
InMemory scheduler is only suitable for development and testing
All scheduling methods return a scheduler ID for cancellation/tracking
Last updated
Was this helpful?
