You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			306 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			306 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //----------------------------------------------------------------------------- | ||
|  | // 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(); | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | } |