-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathEFCoreOrchestrationSession.cs
103 lines (88 loc) · 3.41 KB
/
EFCoreOrchestrationSession.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
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using DurableTask.Core;
using LLL.DurableTask.EFCore.Entities;
using LLL.DurableTask.EFCore.Polling;
using Microsoft.EntityFrameworkCore;
namespace LLL.DurableTask.EFCore;
public class EFCoreOrchestrationSession : IOrchestrationSession
{
private readonly EFCoreOrchestrationOptions _options;
private readonly IDbContextFactory<OrchestrationDbContext> _dbContextFactory;
private readonly CancellationToken _stopCancellationToken;
public EFCoreOrchestrationSession(
EFCoreOrchestrationOptions options,
IDbContextFactory<OrchestrationDbContext> dbContextFactory,
Instance instance,
Execution execution,
OrchestrationRuntimeState runtimeState,
CancellationToken stopCancellationToken)
{
_options = options;
_dbContextFactory = dbContextFactory;
Instance = instance;
Execution = execution;
RuntimeState = runtimeState;
_stopCancellationToken = stopCancellationToken;
}
public Instance Instance { get; }
public Execution Execution { get; set; }
public OrchestrationRuntimeState RuntimeState { get; set; }
public List<OrchestrationMessage> Messages { get; } = new List<OrchestrationMessage>();
public bool Released { get; set; }
public async Task<IList<TaskMessage>> FetchNewOrchestrationMessagesAsync(
TaskOrchestrationWorkItem workItem)
{
return await BackoffPollingHelper.PollAsync(async () =>
{
using var dbContext = _dbContextFactory.CreateDbContext();
var messages = await FetchNewMessagesAsync(dbContext);
await dbContext.SaveChangesAsync();
return messages;
},
x => x == null || x.Count > 0,
_options.FetchNewMessagesPollingTimeout,
_options.PollingInterval,
_stopCancellationToken);
}
public async Task<IList<TaskMessage>> FetchNewMessagesAsync(
OrchestrationDbContext dbContext,
CancellationToken cancellationToken = default)
{
var newDbMessages = await dbContext.OrchestrationMessages
.Where(w => w.AvailableAt <= DateTime.UtcNow
&& w.InstanceId == Instance.InstanceId
&& w.Instance.LockId == Instance.LockId // Ensure we still own the lock
&& !Messages.Contains(w))
.OrderBy(w => w.AvailableAt)
.ThenBy(w => w.SequenceNumber)
.AsNoTracking()
.ToArrayAsync(cancellationToken);
var messagesToDiscard = newDbMessages
.Where(m => m.ExecutionId != null && m.ExecutionId != Instance.LastExecutionId)
.ToArray();
if (messagesToDiscard.Length > 0)
{
foreach (var message in messagesToDiscard)
{
dbContext.OrchestrationMessages.Attach(message);
dbContext.OrchestrationMessages.Remove(message);
}
newDbMessages = newDbMessages
.Except(messagesToDiscard)
.ToArray();
}
Messages.AddRange(newDbMessages);
var deserializedMessages = newDbMessages
.Select(w => _options.DataConverter.Deserialize<TaskMessage>(w.Message))
.ToList();
return deserializedMessages;
}
public void ClearMessages()
{
Messages.Clear();
}
}