Let's cut to the chase, here is what needs to be done to track the events we want:
Subclass from TrackingParticipant and override its Track method. Within this Track method we'll look only for TrackingRecords of type WorkflowInstanceRecord, since we only want to capture
workflow level events.
NOTE: you can also attach to activity
and custom level events but that's out of the scope of this blog post. I
recommend this article for a more in depth description of Tracking: http://www.codeproject.com/KB/WF/WF4Extensions.aspx
If you want your application to
"react" to particular workflow events, we only need to inspect
the WorkflowInstanceRecord.State property. The following sample code is
a generic template you can use with all the possible states I've encountered so
far:
using System.Activities.Tracking;
public class WorkflowStateTrackingParticipant :TrackingParticipant
{
protected override void Track(TrackingRecord record, TimeSpan timeout)
{
WorkflowInstanceRecord workflowInstanceRecord
= record as WorkflowInstanceRecord;
if(workflowInstanceRecord != null)
{
switch(workflowInstanceRecord.State)
{
case "Started": break;
case "Aborted": break;
case "Canceled": break;
case "Completed": break;
case "Deleted": break;
case "Idle": break;
case "Persisted": break;
case "Resumed": break;
case "Successful": break;
case "Suspended": break;
case "Terminated": break;
case "UnhandledException": break;
case "Unloaded": break;
case "Unsuspended": break;
case "Aborted": break;
case "Canceled": break;
case "Completed": break;
case "Deleted": break;
case "Idle": break;
case "Persisted": break;
case "Resumed": break;
case "Successful": break;
case "Suspended": break;
case "Terminated": break;
case "UnhandledException": break;
case "Unloaded": break;
case "Unsuspended": break;
default: break;
}
}
}
}
To use this class, we need
to add it to the WorkflowExtensions collection of our WorkflowServiceHostFactory; we do this by subclassing from it and
overriding the CreateWorkflowServiceHost methods:
public class CustomWorkflowServiceHostFactory : WorkflowServiceHostFactory
{
protected override WorkflowServiceHost CreateWorkflowServiceHost(System.Activities.Activity activity, Uri[] baseAddresses)
{
WorkflowServiceHost host = base.CreateWorkflowServiceHost(activity, baseAddresses);
host.WorkflowExtensions.Add(new WorkflowStateTrackingParticipant());
return host;
}
protected override WorkflowServiceHost CreateWorkflowServiceHost(WorkflowService service, Uri[] baseAddresses)
{
WorkflowServiceHost host = base.CreateWorkflowServiceHost(service, baseAddresses);
host.WorkflowExtensions.Add(new WorkflowStateTrackingParticipant());
return host;
}
}
Finally, we need to
indicate this CustomWorkflowServiceHostFactory, instead of the default one, in your Web.config or .svc
files (depending on your particular implementation). In my case, I create my
.svc definition files from the .xamlx workflows, dynamically, through
Web.config via the serviceHostingEnvironment section:
<serviceHostingEnvironment multipleSiteBindingsEnabled="true">
<serviceActivations>
<add relativeAddress="MyWorkflowService.svc"
service="Workflow.xamlx"
factory="CustomWorkflowServiceHostFactory" />
</serviceActivations>
</serviceHostingEnvironment>
Verify you can hit your
.svc from a browser and you're done.
That's it, hopefully this post can help people save more than a few minutes of research.