306 lines
11 KiB
C#
Raw Normal View History

//-----------------------------------------------------------------------------
// Copyright (c) Microsoft Corporation. All rights reserved.
//-----------------------------------------------------------------------------
namespace System.Activities.Statements
{
using System;
using System.Activities;
using System.Runtime;
using System.ComponentModel;
using System.Activities.Persistence;
using System.Collections.Generic;
using System.Xml.Linq;
using System.Activities.Hosting;
using System.Threading;
[Fx.Tag.XamlVisible(false)]
public class DurableTimerExtension : TimerExtension, IWorkflowInstanceExtension, IDisposable, ICancelable
{
WorkflowInstanceProxy instance;
TimerTable registeredTimers;
Action<object> onTimerFiredCallback;
TimerPersistenceParticipant timerPersistenceParticipant;
static AsyncCallback onResumeBookmarkComplete = Fx.ThunkCallback(new AsyncCallback(OnResumeBookmarkComplete));
static readonly XName timerTableName = XNamespace.Get("urn:schemas-microsoft-com:System.Activities/4.0/properties").GetName("RegisteredTimers");
static readonly XName timerExpirationTimeName = XNamespace.Get("urn:schemas-microsoft-com:System.Activities/4.0/properties").GetName("TimerExpirationTime");
bool isDisposed;
[Fx.Tag.SynchronizationObject()]
object thisLock;
public DurableTimerExtension()
: base()
{
this.onTimerFiredCallback = new Action<object>(this.OnTimerFired);
this.thisLock = new object();
this.timerPersistenceParticipant = new TimerPersistenceParticipant(this);
this.isDisposed = false;
}
object ThisLock
{
get
{
return this.thisLock;
}
}
internal Action<object> OnTimerFiredCallback
{
get
{
return this.onTimerFiredCallback;
}
}
internal TimerTable RegisteredTimers
{
get
{
if (this.registeredTimers == null)
{
this.registeredTimers = new TimerTable(this);
}
return this.registeredTimers;
}
}
public virtual IEnumerable<object> GetAdditionalExtensions()
{
yield return this.timerPersistenceParticipant;
}
public virtual void SetInstance(WorkflowInstanceProxy instance)
{
if (this.instance != null && instance != null)
{
throw FxTrace.Exception.AsError(new InvalidOperationException(SR.TimerExtensionAlreadyAttached));
}
this.instance = instance;
}
protected override void OnRegisterTimer(TimeSpan timeout, Bookmark bookmark)
{
// This lock is to synchronize with the Timer callback
if (timeout < TimeSpan.MaxValue)
{
lock (this.ThisLock)
{
Fx.Assert(!this.isDisposed, "DurableTimerExtension is already disposed, it cannot be used to register a new timer.");
this.RegisteredTimers.AddTimer(timeout, bookmark);
}
}
}
protected override void OnCancelTimer(Bookmark bookmark)
{
// This lock is to synchronize with the Timer callback
lock (this.ThisLock)
{
this.RegisteredTimers.RemoveTimer(bookmark);
}
}
internal void OnSave(out IDictionary<XName, object> readWriteValues, out IDictionary<XName, object> writeOnlyValues)
{
readWriteValues = null;
writeOnlyValues = null;
// Using a lock here to prevent the timer firing back without us being ready
lock (this.ThisLock)
{
this.RegisteredTimers.MarkAsImmutable();
if (this.registeredTimers != null && this.registeredTimers.Count > 0)
{
readWriteValues = new Dictionary<XName, object>(1);
writeOnlyValues = new Dictionary<XName, object>(1);
readWriteValues.Add(timerTableName, this.registeredTimers);
writeOnlyValues.Add(timerExpirationTimeName, this.registeredTimers.GetNextDueTime());
}
}
}
internal void PersistenceDone()
{
lock (this.ThisLock)
{
this.RegisteredTimers.MarkAsMutable();
}
}
internal void OnLoad(IDictionary<XName, object> readWriteValues)
{
lock (this.ThisLock)
{
object timerTable;
if (readWriteValues != null && readWriteValues.TryGetValue(timerTableName, out timerTable))
{
this.registeredTimers = timerTable as TimerTable;
Fx.Assert(this.RegisteredTimers != null, "Timer Table cannot be null");
this.RegisteredTimers.OnLoad(this);
}
}
}
void OnTimerFired(object state)
{
Bookmark timerBookmark = state as Bookmark;
WorkflowInstanceProxy targetInstance = this.instance;
// it's possible that we've been unloaded while the timer was in the process of firing, in
// which case targetInstance will be null
if (targetInstance != null)
{
BookmarkResumptionResult resumptionResult;
IAsyncResult result = null;
bool completed = false;
result = targetInstance.BeginResumeBookmark(timerBookmark, null, TimeSpan.MaxValue,
onResumeBookmarkComplete, new BookmarkResumptionState(timerBookmark, this, targetInstance));
completed = result.CompletedSynchronously;
if (completed && result != null)
{
try
{
resumptionResult = targetInstance.EndResumeBookmark(result);
ProcessBookmarkResumptionResult(timerBookmark, resumptionResult);
}
catch (TimeoutException)
{
ProcessBookmarkResumptionResult(timerBookmark, BookmarkResumptionResult.NotReady);
}
}
}
}
static void OnResumeBookmarkComplete(IAsyncResult result)
{
if (result.CompletedSynchronously)
{
return;
}
BookmarkResumptionState state = (BookmarkResumptionState)result.AsyncState;
BookmarkResumptionResult resumptionResult = state.Instance.EndResumeBookmark(result);
state.TimerExtension.ProcessBookmarkResumptionResult(state.TimerBookmark, resumptionResult);
}
void ProcessBookmarkResumptionResult(Bookmark timerBookmark, BookmarkResumptionResult result)
{
switch (result)
{
case BookmarkResumptionResult.NotFound:
case BookmarkResumptionResult.Success:
// The bookmark is removed maybe due to WF cancel, abort or the bookmark succeeds
// no need to keep the timer around
lock (this.ThisLock)
{
if (!this.isDisposed)
{
this.RegisteredTimers.RemoveTimer(timerBookmark);
}
}
break;
case BookmarkResumptionResult.NotReady:
// The workflow maybe in one of these states: Completed, Aborted, Abandoned, unloading, Suspended
// In the first 3 cases, we will let TimerExtension.CancelTimer take care of the cleanup.
// In the 4th case, we want the timer to retry when it is loaded back, in all 4 cases we don't need to delete the timer
// In the 5th case, we want the timer to retry until it succeeds.
// Retry:
lock (this.ThisLock)
{
this.RegisteredTimers.RetryTimer(timerBookmark);
}
break;
}
}
public void Dispose()
{
if (this.registeredTimers != null)
{
lock (this.ThisLock)
{
this.isDisposed = true;
if (this.registeredTimers != null)
{
this.registeredTimers.Dispose();
}
}
}
GC.SuppressFinalize(this);
}
void ICancelable.Cancel()
{
Dispose();
}
class BookmarkResumptionState
{
public BookmarkResumptionState(Bookmark timerBookmark, DurableTimerExtension timerExtension, WorkflowInstanceProxy instance)
{
this.TimerBookmark = timerBookmark;
this.TimerExtension = timerExtension;
this.Instance = instance;
}
public Bookmark TimerBookmark
{
get;
private set;
}
public DurableTimerExtension TimerExtension
{
get;
private set;
}
public WorkflowInstanceProxy Instance
{
get;
private set;
}
}
class TimerPersistenceParticipant : PersistenceIOParticipant
{
DurableTimerExtension defaultTimerExtension;
public TimerPersistenceParticipant(DurableTimerExtension timerExtension)
: base(false, false)
{
this.defaultTimerExtension = timerExtension;
}
protected override void CollectValues(out IDictionary<XName, object> readWriteValues, out IDictionary<XName, object> writeOnlyValues)
{
this.defaultTimerExtension.OnSave(out readWriteValues, out writeOnlyValues);
}
protected override void PublishValues(IDictionary<XName, object> readWriteValues)
{
this.defaultTimerExtension.OnLoad(readWriteValues);
}
protected override IAsyncResult BeginOnSave(IDictionary<XName, object> readWriteValues, IDictionary<XName, object> writeOnlyValues, TimeSpan timeout, AsyncCallback callback, object state)
{
this.defaultTimerExtension.PersistenceDone();
return base.BeginOnSave(readWriteValues, writeOnlyValues, timeout, callback, state);
}
protected override void Abort()
{
this.defaultTimerExtension.PersistenceDone();
}
}
}
}