-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathExpirationManager.cs
82 lines (65 loc) · 2.89 KB
/
ExpirationManager.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
using System;
using System.Linq;
using System.Threading;
using Composite;
using Composite.Data;
using Hangfire.Common;
using Hangfire.CompositeC1.Types;
using Hangfire.Logging;
using Hangfire.Server;
namespace Hangfire.CompositeC1
{
public class ExpirationManager : IServerComponent, IBackgroundProcess
{
private static readonly TimeSpan DefaultLockTimeout = TimeSpan.FromMinutes(5);
private const int NumberOfRecordsInSinglePass = 1000;
private static readonly Type[] ExpirableTypes = (from type in typeof(CompositeC1Storage).Assembly.GetTypes()
where type.IsInterface
&& type.Namespace != null && type.Namespace.Equals("Hangfire.CompositeC1.Types")
&& typeof(IExpirable).IsAssignableFrom(type)
select type).ToArray();
private readonly ILog _logger = LogProvider.GetCurrentClassLogger();
private readonly CompositeC1Storage _storage;
private readonly TimeSpan _checkInterval;
public ExpirationManager(CompositeC1Storage storage, TimeSpan checkInterval)
{
Verify.ArgumentNotNull(storage, nameof(storage));
_storage = storage;
_checkInterval = checkInterval;
}
public void Execute(BackgroundProcessContext context)
{
Execute(context.CancellationToken);
}
public void Execute(CancellationToken cancellationToken)
{
var now = DateTime.UtcNow;
foreach (var t in ExpirableTypes)
{
_logger.Debug($"Removing outdated records from type '{t.Name}'");
_storage.UseConnection(connection =>
{
int affected;
do
{
using (connection.AcquireDistributedLock("locks:ExpirationManager", DefaultLockTimeout))
{
var table = connection.Get<IExpirable>(t);
var records = (from d in table
where d.ExpireAt < now
orderby d.ExpireAt
select d).Take(NumberOfRecordsInSinglePass).ToList();
affected = records.Count;
connection.Delete<IData>(records);
}
} while (affected == NumberOfRecordsInSinglePass);
});
}
cancellationToken.Wait(_checkInterval);
}
public override string ToString()
{
return GetType().ToString();
}
}
}