Introducción
En este artículo, aprenderemos cómo crear un servicio en segundo plano en ASP.NET Core que se ejecutará según un intervalo de tiempo definido mediante una expresión CRON, similar a las tareas programadas en sistemas Linux.
Nota 💡: Puedes encontrar el código fuente completo en GitHub.
El formato CRON es ampliamente utilizado para expresar horarios de tareas programadas. Se compone de 5 o 6 campos que representan distintas unidades de tiempo, como se muestra a continuación:
Allowed values Allowed special characters Comment
┌───────────── second (optional) 0-59 * , - /
│ ┌───────────── minute 0-59 * , - /
│ │ ┌───────────── hour 0-23 * , - /
│ │ │ ┌───────────── day of month 1-31 * , - / L W ?
│ │ │ │ ┌───────────── month 1-12 or JAN-DEC * , - /
│ │ │ │ │ ┌───────────── day of week 0-6 or SUN-SAT * , - / # L ? Both 0 and 7 means SUN
│ │ │ │ │ │
* * * * * *
Por ejemplo, las siguientes expresiones CRON programan tareas en diferentes intervalos:
Expresión | Descripción |
---|---|
* * * * * | Cada minuto |
0 0 1 * * | A media noche, en día primero de cada mes |
0 0 * * MON-FRI | A las 0:00, de Lunes a Viernes |
Nota 💡: Para más detalles sobre las expresiones CRON, puedes consultar Cronos en GitHub.
Implementación en ASP.NET Core y Hosted Services
Si bien existen soluciones robustas como HangFire y Azure Functions que soportan tareas programadas basadas en CRON, a veces es preferible mantener las cosas simples. En este artículo, exploraremos cómo implementar una solución personalizada utilizando Hosted Services en ASP.NET Core.
Creación del Proyecto
Primero, creamos un nuevo proyecto web vacío o de consola, dependiendo de nuestras necesidades:
dotnet new web -o BackgroundJob.Cron
A continuación, instalamos la librería Cronos para poder interpretar y manejar expresiones CRON:
dotnet add package Cronos
CronBackgroundJob
El core de nuestra implementación es la clase CronBackgroundJob
, una clase abstracta que ejecuta un proceso en segundo plano según un intervalo definido por una expresión CRON.
using Cronos;
namespace BackgroundJob.Cron.Jobs;
public abstract class CronBackgroundJob : BackgroundService
{
private readonly CronExpression _cronExpression;
private readonly TimeZoneInfo _timeZone;
public CronBackgroundJob(string rawCronExpression, TimeZoneInfo timeZone)
{
_cronExpression = CronExpression.Parse(rawCronExpression);
_timeZone = timeZone;
}
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
{
while (!stoppingToken.IsCancellationRequested)
{
DateTimeOffset? nextOccurrence = _cronExpression.GetNextOccurrence(DateTimeOffset.UtcNow, _timeZone);
if (!nextOccurrence.HasValue)
return;
var delay = nextOccurrence.Value - DateTimeOffset.UtcNow;
if (delay.TotalMilliseconds > 0)
{
try
{
await Task.Delay(delay, stoppingToken);
}
catch (TaskCanceledException)
{
// Handle cancellation if needed
return;
}
}
try
{
await DoWork(stoppingToken);
}
catch (Exception ex)
{
// Handle or log the exception as needed
}
}
}
protected abstract Task DoWork(CancellationToken stoppingToken);
}
El método ExecuteAsync
es el núcleo del servicio en segundo plano. Este método se ejecuta en un bucle hasta que se solicite la cancelación del servicio (a través del CancellationToken
).
-
Cálculo de la Próxima Ejecución:
-
nextOccurrence
se calcula utilizando la expresión CRON y la zona horaria, determinando cuándo debe ejecutarse la tarea a continuación. - Si no hay una próxima ocurrencia (
!nextOccurrence.HasValue
), el método simplemente retorna y detiene la ejecución.
-
-
Espera hasta la Próxima Ejecución:
- Si hay una próxima ocurrencia, se calcula el
delay
entre el momento actual y el momento de la próxima ejecución. - Si el
delay
es mayor que cero, el servicio espera (Task.Delay
) hasta ese momento. Esta espera se puede cancelar si se solicita la cancelación a través delstoppingToken
.
- Si hay una próxima ocurrencia, se calcula el
-
Ejecución de la Tarea:
- Una vez transcurrido el
delay
, se ejecuta el método abstractoDoWork
, que debe ser implementado por cualquier clase que herede deCronBackgroundJob
. Este método es donde se define la lógica que se desea ejecutar en el intervalo programado.
- Una vez transcurrido el
-
Manejo de Errores:
- Se captura cualquier excepción que ocurra durante la ejecución de
DoWork
para que el servicio pueda manejar o registrar errores sin detener el servicio completo.
- Se captura cualquier excepción que ocurra durante la ejecución de
Configuración del Job
Para correr un job basado en CronBackgroundJob
, necesitamos configurar la expresión CRON y el huso horario a utilizar. Esto lo hacemos mediante la clase CronSettings
:
namespace BackgroundJob.Cron.Jobs;
public class CronSettings<T>
{
public string CronExpression { get; set; } = default!;
public TimeZoneInfo TimeZone { get; set; } = default!;
}
Registro de Servicios
Para facilitar la integración de nuestros jobs con la configuración, creamos un método de extensión que registra las dependencias necesarias:
namespace BackgroundJob.Cron.Jobs;
public static class CronBackgroundJobExtensions
{
public static IServiceCollection AddCronJob<T>(this IServiceCollection services, Action<CronSettings<T>> options)
where T: CronBackgroundJob
{
if (options == null)
{
throw new ArgumentNullException(nameof(options));
}
var config = new CronSettings<T>();
options.Invoke(config);
if (string.IsNullOrWhiteSpace(config.CronExpression))
{
throw new ArgumentNullException(nameof(CronSettings<T>.CronExpression));
}
services.AddSingleton<CronSettings<T>>(config);
services.AddHostedService<T>();
return services;
}
}
Usamos el Options Pattern muy común en ASP.NET para registrar cada background job que necesitemos.
Es obligatorio que se indique una configuración por medio de CronSettings<T>
y también es obligatorio tener una expresión cron.
Ejemplo de Job: MySchedulerJob
A continuación, creamos un job simple que hereda de CronBackgroundJob
:
namespace BackgroundJob.Cron.Jobs;
public class MySchedulerJob : CronBackgroundJob
{
private readonly ILogger<MySchedulerJob> _log;
public MySchedulerJob(CronSettings<MySchedulerJob> settings, ILogger<MySchedulerJob> log)
:base(settings.CronExpression, settings.TimeZone)
{
_log = log;
}
protected override Task DoWork(CancellationToken stoppingToken)
{
_log.LogInformation("Running... at {0}", DateTime.UtcNow);
return Task.CompletedTask;
}
}
Este job simplemente registra la fecha y hora en que se ejecuta, lo que nos permite verificar su correcto funcionamiento.
Configuración Final en Program.cs
Por último, registramos nuestro job en el contenedor de dependencias:
using BackgroundJob.Cron.Jobs;
var builder = WebApplication.CreateBuilder(args);
builder.Services.AddCronJob<MySchedulerJob>(options =>
{
// Corre cada minuto
options.CronExpression = "* * * * *";
options.TimeZone = TimeZoneInfo.Local;
});
var app = builder.Build();
app.Run();
Al ejecutar la aplicación, verás que el job se ejecuta cada minuto, tal como se especifica en la expresión CRON.
Conclusión
Aunque existen soluciones maduras y robustas para manejar tareas en segundo plano, como HangFire y Azure Functions, una implementación sencilla basada en Hosted Services puede ser la opción ideal cuando necesitas algo ligero y fácil de mantener.
Si tus necesidades evolucionan hacia una mayor escalabilidad o resiliencia, considera migrar a Azure Functions o Hangfire, dependiendo de tus requisitos específicos.
Referencias
- Schedule Cron Jobs using HostedService in ASP.NET Core | by Changhui Xu | codeburst
- PeriodicTimer: Temporizadores asíncronos en .NET 6 | Variable not found
- HangfireIO/Cronos: Fully-featured .NET library for working with Cron expressions. Built with time zones in mind and intuitively handles daylight saving time transitions (github.com)
- PeriodicTimer Class (System.Threading) | Microsoft Learn
Top comments (1)
Actualización ⚠️
Tuve que actualizar este post por que al ver el código me di cuenta que la recursividad no es buena idea en tareas / procesos que pueden durar mucho tiempo vivos.