e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
667 lines
23 KiB
C#
667 lines
23 KiB
C#
//-----------------------------------------------------------------------------
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
//-----------------------------------------------------------------------------
|
|
|
|
namespace System.Activities.Runtime
|
|
{
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics.CodeAnalysis;
|
|
using System.Runtime;
|
|
using System.Runtime.Serialization;
|
|
|
|
[DataContract]
|
|
class ExecutionPropertyManager
|
|
{
|
|
ActivityInstance owningInstance;
|
|
|
|
Dictionary<string, ExecutionProperty> properties;
|
|
|
|
// Since the ExecutionProperty objects in this list
|
|
// could exist in several places we need to make sure
|
|
// that we clean up any booleans before more work items run
|
|
List<ExecutionProperty> threadProperties;
|
|
bool ownsThreadPropertiesList;
|
|
|
|
string lastPropertyName;
|
|
object lastProperty;
|
|
IdSpace lastPropertyVisibility;
|
|
|
|
// used by the root activity instance to chain parents correctly
|
|
ExecutionPropertyManager rootPropertyManager;
|
|
|
|
int exclusiveHandleCount;
|
|
|
|
public ExecutionPropertyManager(ActivityInstance owningInstance)
|
|
{
|
|
Fx.Assert(owningInstance != null, "null instance should be using the internal host-based ctor");
|
|
this.owningInstance = owningInstance;
|
|
|
|
// This object is only constructed if we know we have properties to add to it
|
|
this.properties = new Dictionary<string, ExecutionProperty>();
|
|
|
|
if (owningInstance.HasChildren)
|
|
{
|
|
ActivityInstance previousOwner = owningInstance.PropertyManager != null ? owningInstance.PropertyManager.owningInstance : null;
|
|
|
|
// we're setting a handle property. Walk the children and associate the new property manager
|
|
// then walk our instance list, fixup parent references, and perform basic validation
|
|
ActivityUtilities.ProcessActivityInstanceTree(owningInstance, null, (instance, executor) => AttachPropertyManager(instance, previousOwner));
|
|
}
|
|
else
|
|
{
|
|
owningInstance.PropertyManager = this;
|
|
}
|
|
}
|
|
|
|
public ExecutionPropertyManager(ActivityInstance owningInstance, ExecutionPropertyManager parentPropertyManager)
|
|
: this(owningInstance)
|
|
{
|
|
Fx.Assert(parentPropertyManager != null, "caller must verify");
|
|
this.threadProperties = parentPropertyManager.threadProperties;
|
|
|
|
// if our parent is null, capture any root properties
|
|
if (owningInstance.Parent == null)
|
|
{
|
|
this.rootPropertyManager = parentPropertyManager.rootPropertyManager;
|
|
}
|
|
}
|
|
|
|
internal ExecutionPropertyManager(ActivityInstance owningInstance, Dictionary<string, ExecutionProperty> properties)
|
|
{
|
|
Fx.Assert(properties != null, "properties should never be null");
|
|
this.owningInstance = owningInstance;
|
|
this.properties = properties;
|
|
|
|
// owningInstance can be null (for host-provided root properties)
|
|
if (owningInstance == null)
|
|
{
|
|
this.rootPropertyManager = this;
|
|
}
|
|
}
|
|
|
|
[DataMember(EmitDefaultValue = false, Name = "properties")]
|
|
internal Dictionary<string, ExecutionProperty> SerializedProperties
|
|
{
|
|
get { return this.properties; }
|
|
set { this.properties = value; }
|
|
}
|
|
|
|
[DataMember(EmitDefaultValue = false, Name = "exclusiveHandleCount")]
|
|
internal int SerializedExclusiveHandleCount
|
|
{
|
|
get { return this.exclusiveHandleCount; }
|
|
set { this.exclusiveHandleCount = value; }
|
|
}
|
|
|
|
internal Dictionary<string, ExecutionProperty> Properties
|
|
{
|
|
get
|
|
{
|
|
return this.properties;
|
|
}
|
|
}
|
|
|
|
internal bool HasExclusiveHandlesInScope
|
|
{
|
|
get
|
|
{
|
|
return this.exclusiveHandleCount > 0;
|
|
}
|
|
}
|
|
|
|
bool AttachPropertyManager(ActivityInstance instance, ActivityInstance previousOwner)
|
|
{
|
|
if (instance.PropertyManager == null || instance.PropertyManager.owningInstance == previousOwner)
|
|
{
|
|
instance.PropertyManager = this;
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public object GetProperty(string name, IdSpace currentIdSpace)
|
|
{
|
|
Fx.Assert(!string.IsNullOrEmpty(name), "The name should be validated by the caller.");
|
|
|
|
if (lastPropertyName == name && (this.lastPropertyVisibility == null || this.lastPropertyVisibility == currentIdSpace))
|
|
{
|
|
return lastProperty;
|
|
}
|
|
|
|
ExecutionPropertyManager currentManager = this;
|
|
|
|
while (currentManager != null)
|
|
{
|
|
ExecutionProperty property;
|
|
if (currentManager.properties.TryGetValue(name, out property))
|
|
{
|
|
if (!property.IsRemoved && (!property.HasRestrictedVisibility || property.Visibility == currentIdSpace))
|
|
{
|
|
this.lastPropertyName = name;
|
|
this.lastProperty = property.Property;
|
|
this.lastPropertyVisibility = property.Visibility;
|
|
|
|
return this.lastProperty;
|
|
}
|
|
}
|
|
|
|
currentManager = GetParent(currentManager);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
void AddProperties(IDictionary<string, ExecutionProperty> properties, IDictionary<string, object> flattenedProperties, IdSpace currentIdSpace)
|
|
{
|
|
foreach (KeyValuePair<string, ExecutionProperty> item in properties)
|
|
{
|
|
if (!item.Value.IsRemoved && !flattenedProperties.ContainsKey(item.Key) && (!item.Value.HasRestrictedVisibility || item.Value.Visibility == currentIdSpace))
|
|
{
|
|
flattenedProperties.Add(item.Key, item.Value.Property);
|
|
}
|
|
}
|
|
}
|
|
|
|
public IEnumerable<KeyValuePair<string, object>> GetFlattenedProperties(IdSpace currentIdSpace)
|
|
{
|
|
ExecutionPropertyManager currentManager = this;
|
|
Dictionary<string, object> flattenedProperties = new Dictionary<string, object>();
|
|
while (currentManager != null)
|
|
{
|
|
AddProperties(currentManager.Properties, flattenedProperties, currentIdSpace);
|
|
currentManager = GetParent(currentManager);
|
|
}
|
|
return flattenedProperties;
|
|
}
|
|
|
|
//Currently this is only used for the exclusive scope processing
|
|
internal List<T> FindAll<T>() where T : class
|
|
{
|
|
ExecutionPropertyManager currentManager = this;
|
|
List<T> list = null;
|
|
|
|
while (currentManager != null)
|
|
{
|
|
foreach (ExecutionProperty property in currentManager.Properties.Values)
|
|
{
|
|
if (property.Property is T)
|
|
{
|
|
if (list == null)
|
|
{
|
|
list = new List<T>();
|
|
}
|
|
list.Add((T)property.Property);
|
|
}
|
|
}
|
|
|
|
currentManager = GetParent(currentManager);
|
|
}
|
|
|
|
return list;
|
|
}
|
|
|
|
static ExecutionPropertyManager GetParent(ExecutionPropertyManager currentManager)
|
|
{
|
|
if (currentManager.owningInstance != null)
|
|
{
|
|
if (currentManager.owningInstance.Parent != null)
|
|
{
|
|
return currentManager.owningInstance.Parent.PropertyManager;
|
|
}
|
|
else
|
|
{
|
|
return currentManager.rootPropertyManager;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public void Add(string name, object property, IdSpace visibility)
|
|
{
|
|
Fx.Assert(!string.IsNullOrEmpty(name), "The name should be validated before calling this collection.");
|
|
Fx.Assert(property != null, "The property should be validated before caling this collection.");
|
|
|
|
ExecutionProperty executionProperty = new ExecutionProperty(name, property, visibility);
|
|
this.properties.Add(name, executionProperty);
|
|
|
|
if (this.lastPropertyName == name)
|
|
{
|
|
this.lastProperty = property;
|
|
}
|
|
|
|
if (property is ExclusiveHandle)
|
|
{
|
|
this.exclusiveHandleCount++;
|
|
|
|
UpdateChildExclusiveHandleCounts(1);
|
|
}
|
|
|
|
if (property is IExecutionProperty)
|
|
{
|
|
AddIExecutionProperty(executionProperty, false);
|
|
}
|
|
}
|
|
|
|
void UpdateChildExclusiveHandleCounts(int amountToUpdate)
|
|
{
|
|
Queue<HybridCollection<ActivityInstance>> toProcess = null;
|
|
|
|
HybridCollection<ActivityInstance> children = this.owningInstance.GetRawChildren();
|
|
|
|
if (children != null && children.Count > 0)
|
|
{
|
|
ProcessChildrenForExclusiveHandles(children, amountToUpdate, ref toProcess);
|
|
|
|
if (toProcess != null)
|
|
{
|
|
while (toProcess.Count > 0)
|
|
{
|
|
children = toProcess.Dequeue();
|
|
ProcessChildrenForExclusiveHandles(children, amountToUpdate, ref toProcess);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void ProcessChildrenForExclusiveHandles(HybridCollection<ActivityInstance> children, int amountToUpdate, ref Queue<HybridCollection<ActivityInstance>> toProcess)
|
|
{
|
|
for (int i = 0; i < children.Count; i++)
|
|
{
|
|
ActivityInstance child = children[i];
|
|
|
|
ExecutionPropertyManager childManager = child.PropertyManager;
|
|
|
|
if (childManager.IsOwner(child))
|
|
{
|
|
childManager.exclusiveHandleCount += amountToUpdate;
|
|
}
|
|
|
|
HybridCollection<ActivityInstance> tempChildren = child.GetRawChildren();
|
|
|
|
if (tempChildren != null && tempChildren.Count > 0)
|
|
{
|
|
if (toProcess == null)
|
|
{
|
|
toProcess = new Queue<HybridCollection<ActivityInstance>>();
|
|
}
|
|
|
|
toProcess.Enqueue(tempChildren);
|
|
}
|
|
}
|
|
}
|
|
|
|
void AddIExecutionProperty(ExecutionProperty property, bool isDeserializationFixup)
|
|
{
|
|
bool willCleanupBeCalled = !isDeserializationFixup;
|
|
|
|
if (this.threadProperties == null)
|
|
{
|
|
this.threadProperties = new List<ExecutionProperty>(1);
|
|
this.ownsThreadPropertiesList = true;
|
|
}
|
|
else if (!this.ownsThreadPropertiesList)
|
|
{
|
|
List<ExecutionProperty> updatedProperties = new List<ExecutionProperty>(this.threadProperties.Count);
|
|
|
|
// We need to copy all properties to our new list and we
|
|
// need to mark hidden properties as "to be removed" (or just
|
|
// not copy them on the deserialization path)
|
|
for (int i = 0; i < this.threadProperties.Count; i++)
|
|
{
|
|
ExecutionProperty currentProperty = this.threadProperties[i];
|
|
|
|
if (currentProperty.Name == property.Name)
|
|
{
|
|
if (willCleanupBeCalled)
|
|
{
|
|
currentProperty.ShouldBeRemovedAfterCleanup = true;
|
|
updatedProperties.Add(currentProperty);
|
|
}
|
|
|
|
// If cleanup won't be called then we are on the
|
|
// deserialization path and shouldn't copy this
|
|
// property over to our new list
|
|
}
|
|
else
|
|
{
|
|
updatedProperties.Add(currentProperty);
|
|
}
|
|
}
|
|
|
|
this.threadProperties = updatedProperties;
|
|
this.ownsThreadPropertiesList = true;
|
|
}
|
|
else
|
|
{
|
|
for (int i = this.threadProperties.Count - 1; i >= 0; i--)
|
|
{
|
|
ExecutionProperty currentProperty = this.threadProperties[i];
|
|
|
|
if (currentProperty.Name == property.Name)
|
|
{
|
|
if (willCleanupBeCalled)
|
|
{
|
|
currentProperty.ShouldBeRemovedAfterCleanup = true;
|
|
}
|
|
else
|
|
{
|
|
this.threadProperties.RemoveAt(i);
|
|
}
|
|
|
|
// There will only be at most one property in this list that
|
|
// matches the name
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
property.ShouldSkipNextCleanup = willCleanupBeCalled;
|
|
this.threadProperties.Add(property);
|
|
}
|
|
|
|
public void Remove(string name)
|
|
{
|
|
Fx.Assert(!string.IsNullOrEmpty(name), "This should have been validated by the caller.");
|
|
|
|
ExecutionProperty executionProperty = this.properties[name];
|
|
|
|
Fx.Assert(executionProperty != null, "This should only be called if we know the property exists");
|
|
|
|
if (executionProperty.Property is IExecutionProperty)
|
|
{
|
|
Fx.Assert(this.ownsThreadPropertiesList && this.threadProperties != null, "We should definitely be the list owner if we have an IExecutionProperty");
|
|
|
|
if (!this.threadProperties.Remove(executionProperty))
|
|
{
|
|
Fx.Assert("We should have had this property in the list.");
|
|
}
|
|
}
|
|
|
|
this.properties.Remove(name);
|
|
|
|
if (executionProperty.Property is ExclusiveHandle)
|
|
{
|
|
this.exclusiveHandleCount--;
|
|
|
|
UpdateChildExclusiveHandleCounts(-1);
|
|
}
|
|
|
|
if (this.lastPropertyName == name)
|
|
{
|
|
this.lastPropertyName = null;
|
|
this.lastProperty = null;
|
|
}
|
|
}
|
|
|
|
public object GetPropertyAtCurrentScope(string name)
|
|
{
|
|
Fx.Assert(!string.IsNullOrEmpty(name), "This should be validated elsewhere");
|
|
|
|
ExecutionProperty property;
|
|
if (this.properties.TryGetValue(name, out property))
|
|
{
|
|
return property.Property;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public bool IsOwner(ActivityInstance instance)
|
|
{
|
|
return this.owningInstance == instance;
|
|
}
|
|
|
|
[SuppressMessage(FxCop.Category.Performance, FxCop.Rule.AvoidUncalledPrivateCode, Justification = "Called from Serialization")]
|
|
internal bool ShouldSerialize(ActivityInstance instance)
|
|
{
|
|
return IsOwner(instance) && this.properties.Count > 0;
|
|
}
|
|
|
|
public void SetupWorkflowThread()
|
|
{
|
|
if (this.threadProperties != null)
|
|
{
|
|
for (int i = 0; i < this.threadProperties.Count; i++)
|
|
{
|
|
ExecutionProperty executionProperty = this.threadProperties[i];
|
|
executionProperty.ShouldSkipNextCleanup = false;
|
|
IExecutionProperty property = (IExecutionProperty)executionProperty.Property;
|
|
|
|
property.SetupWorkflowThread();
|
|
}
|
|
}
|
|
}
|
|
|
|
// This method only throws fatal exceptions
|
|
public void CleanupWorkflowThread(ref Exception abortException)
|
|
{
|
|
if (this.threadProperties != null)
|
|
{
|
|
for (int i = this.threadProperties.Count - 1; i >= 0; i--)
|
|
{
|
|
ExecutionProperty current = this.threadProperties[i];
|
|
|
|
if (current.ShouldSkipNextCleanup)
|
|
{
|
|
current.ShouldSkipNextCleanup = false;
|
|
}
|
|
else
|
|
{
|
|
IExecutionProperty property = (IExecutionProperty)current.Property;
|
|
|
|
try
|
|
{
|
|
property.CleanupWorkflowThread();
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e))
|
|
{
|
|
throw;
|
|
}
|
|
|
|
abortException = e;
|
|
}
|
|
}
|
|
|
|
if (current.ShouldBeRemovedAfterCleanup)
|
|
{
|
|
this.threadProperties.RemoveAt(i);
|
|
current.ShouldBeRemovedAfterCleanup = false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public void UnregisterProperties(ActivityInstance completedInstance, IdSpace currentIdSpace)
|
|
{
|
|
UnregisterProperties(completedInstance, currentIdSpace, false);
|
|
}
|
|
|
|
public void UnregisterProperties(ActivityInstance completedInstance, IdSpace currentIdSpace, bool ignoreExceptions)
|
|
{
|
|
if (IsOwner(completedInstance))
|
|
{
|
|
RegistrationContext registrationContext = new RegistrationContext(this, currentIdSpace);
|
|
|
|
foreach (ExecutionProperty property in this.properties.Values)
|
|
{
|
|
// We do a soft removal because we're about to throw away this dictionary
|
|
// and we don't want to mess up our enumerator
|
|
property.IsRemoved = true;
|
|
|
|
IPropertyRegistrationCallback registrationCallback = property.Property as IPropertyRegistrationCallback;
|
|
|
|
if (registrationCallback != null)
|
|
{
|
|
try
|
|
{
|
|
registrationCallback.Unregister(registrationContext);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e) || !ignoreExceptions)
|
|
{
|
|
throw;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Fx.Assert(completedInstance == null || completedInstance.GetRawChildren() == null || completedInstance.GetRawChildren().Count == 0, "There must not be any children at this point otherwise our exclusive handle count would be incorrect.");
|
|
|
|
// We still need to clear this list in case any non-serializable
|
|
// properties were being used in a no persist zone
|
|
this.properties.Clear();
|
|
}
|
|
}
|
|
|
|
public void ThrowIfAlreadyDefined(string name, ActivityInstance executingInstance)
|
|
{
|
|
if (executingInstance == this.owningInstance)
|
|
{
|
|
if (this.properties.ContainsKey(name))
|
|
{
|
|
throw FxTrace.Exception.Argument("name", SR.ExecutionPropertyAlreadyDefined(name));
|
|
}
|
|
}
|
|
}
|
|
|
|
public void OnDeserialized(ActivityInstance owner, ActivityInstance parent, IdSpace visibility, ActivityExecutor executor)
|
|
{
|
|
this.owningInstance = owner;
|
|
|
|
if (parent != null)
|
|
{
|
|
if (parent.PropertyManager != null)
|
|
{
|
|
this.threadProperties = parent.PropertyManager.threadProperties;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
this.rootPropertyManager = executor.RootPropertyManager;
|
|
}
|
|
|
|
foreach (ExecutionProperty property in this.properties.Values)
|
|
{
|
|
if (property.Property is IExecutionProperty)
|
|
{
|
|
AddIExecutionProperty(property, true);
|
|
}
|
|
|
|
if (property.HasRestrictedVisibility)
|
|
{
|
|
property.Visibility = visibility;
|
|
}
|
|
}
|
|
}
|
|
|
|
[DataContract]
|
|
internal class ExecutionProperty
|
|
{
|
|
string name;
|
|
object property;
|
|
bool hasRestrictedVisibility;
|
|
|
|
public ExecutionProperty(string name, object property, IdSpace visibility)
|
|
{
|
|
this.Name = name;
|
|
this.Property = property;
|
|
|
|
if (visibility != null)
|
|
{
|
|
this.Visibility = visibility;
|
|
this.HasRestrictedVisibility = true;
|
|
}
|
|
}
|
|
|
|
public string Name
|
|
{
|
|
get
|
|
{
|
|
return this.name;
|
|
}
|
|
private set
|
|
{
|
|
this.name = value;
|
|
}
|
|
}
|
|
|
|
public object Property
|
|
{
|
|
get
|
|
{
|
|
return this.property;
|
|
}
|
|
private set
|
|
{
|
|
this.property = value;
|
|
}
|
|
}
|
|
|
|
public bool HasRestrictedVisibility
|
|
{
|
|
get
|
|
{
|
|
return this.hasRestrictedVisibility;
|
|
}
|
|
private set
|
|
{
|
|
this.hasRestrictedVisibility = value;
|
|
}
|
|
}
|
|
|
|
// This property is fixed up at deserialization time
|
|
public IdSpace Visibility
|
|
{
|
|
get;
|
|
set;
|
|
}
|
|
|
|
// This is always false at persistence because
|
|
// a removed property belongs to an activity which
|
|
// has completed and is therefore not part of the
|
|
// instance map anymore
|
|
public bool IsRemoved { get; set; }
|
|
|
|
// These don't need to be serialized because they are only
|
|
// ever false at persistence time. We potentially set
|
|
// them to true when a property is added but we always
|
|
// reset them to false after cleaning up the thread
|
|
public bool ShouldBeRemovedAfterCleanup { get; set; }
|
|
public bool ShouldSkipNextCleanup { get; set; }
|
|
|
|
[DataMember(Name = "Name")]
|
|
internal string SerializedName
|
|
{
|
|
get { return this.Name; }
|
|
set { this.Name = value; }
|
|
}
|
|
|
|
[DataMember(Name = "Property")]
|
|
internal object SerializedProperty
|
|
{
|
|
get { return this.Property; }
|
|
set { this.Property = value; }
|
|
}
|
|
|
|
[DataMember(EmitDefaultValue = false, Name = "HasRestrictedVisibility")]
|
|
internal bool SerializedHasRestrictedVisibility
|
|
{
|
|
get { return this.HasRestrictedVisibility; }
|
|
set { this.HasRestrictedVisibility = value; }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|