296 lines
9.5 KiB
C#
296 lines
9.5 KiB
C#
|
#pragma warning disable 1634, 1691
|
||
|
using System;
|
||
|
using System.Text;
|
||
|
using System.CodeDom;
|
||
|
using System.Reflection;
|
||
|
using System.Globalization;
|
||
|
using System.Collections.Generic;
|
||
|
using System.Workflow.ComponentModel;
|
||
|
using System.Workflow.ComponentModel.Compiler;
|
||
|
using System.Workflow.ComponentModel.Serialization;
|
||
|
using System.Workflow.Activities.Common;
|
||
|
|
||
|
namespace System.Workflow.Activities.Rules
|
||
|
{
|
||
|
[Serializable]
|
||
|
public abstract class RuleAction
|
||
|
{
|
||
|
public abstract bool Validate(RuleValidation validator);
|
||
|
public abstract void Execute(RuleExecution context);
|
||
|
public abstract ICollection<string> GetSideEffects(RuleValidation validation);
|
||
|
public abstract RuleAction Clone();
|
||
|
}
|
||
|
|
||
|
[Serializable]
|
||
|
public class RuleHaltAction : RuleAction
|
||
|
{
|
||
|
public override bool Validate(RuleValidation validator)
|
||
|
{
|
||
|
// Trivial... nothing to validate.
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
public override void Execute(RuleExecution context)
|
||
|
{
|
||
|
if (context == null)
|
||
|
throw new ArgumentNullException("context");
|
||
|
context.Halted = true;
|
||
|
}
|
||
|
|
||
|
public override ICollection<string> GetSideEffects(RuleValidation validation)
|
||
|
{
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
public override RuleAction Clone()
|
||
|
{
|
||
|
return (RuleAction)this.MemberwiseClone();
|
||
|
}
|
||
|
|
||
|
public override string ToString()
|
||
|
{
|
||
|
return "Halt";
|
||
|
}
|
||
|
|
||
|
public override bool Equals(object obj)
|
||
|
{
|
||
|
return (obj is RuleHaltAction);
|
||
|
}
|
||
|
|
||
|
public override int GetHashCode()
|
||
|
{
|
||
|
return base.GetHashCode();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
[Serializable]
|
||
|
public class RuleUpdateAction : RuleAction
|
||
|
{
|
||
|
private string path;
|
||
|
|
||
|
public RuleUpdateAction(string path)
|
||
|
{
|
||
|
this.path = path;
|
||
|
}
|
||
|
|
||
|
public RuleUpdateAction()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
public string Path
|
||
|
{
|
||
|
get { return path; }
|
||
|
set { path = value; }
|
||
|
}
|
||
|
|
||
|
public override bool Validate(RuleValidation validator)
|
||
|
{
|
||
|
if (validator == null)
|
||
|
throw new ArgumentNullException("validator");
|
||
|
|
||
|
bool success = true;
|
||
|
|
||
|
if (path == null)
|
||
|
{
|
||
|
ValidationError error = new ValidationError(Messages.NullUpdate, ErrorNumbers.Error_ParameterNotSet);
|
||
|
error.UserData[RuleUserDataKeys.ErrorObject] = this;
|
||
|
validator.AddError(error);
|
||
|
success = false;
|
||
|
}
|
||
|
|
||
|
// now make sure that the path is valid
|
||
|
string[] parts = path.Split('/');
|
||
|
if (parts[0] == "this")
|
||
|
{
|
||
|
Type currentType = validator.ThisType;
|
||
|
for (int i = 1; i < parts.Length; ++i)
|
||
|
{
|
||
|
if (parts[i] == "*")
|
||
|
{
|
||
|
if (i < parts.Length - 1)
|
||
|
{
|
||
|
// The "*" occurred in the middle of the path, which is a no-no.
|
||
|
ValidationError error = new ValidationError(Messages.InvalidWildCardInPathQualifier, ErrorNumbers.Error_InvalidWildCardInPathQualifier);
|
||
|
error.UserData[RuleUserDataKeys.ErrorObject] = this;
|
||
|
validator.AddError(error);
|
||
|
success = false;
|
||
|
break;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// It occurred at the end, which is okay.
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
else if (string.IsNullOrEmpty(parts[i]) && i == parts.Length - 1)
|
||
|
{
|
||
|
// It's okay to end with a "/".
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
while (currentType.IsArray)
|
||
|
currentType = currentType.GetElementType();
|
||
|
|
||
|
BindingFlags bindingFlags = BindingFlags.Public | BindingFlags.Instance | BindingFlags.Static | BindingFlags.FlattenHierarchy;
|
||
|
if (validator.AllowInternalMembers(currentType))
|
||
|
bindingFlags |= BindingFlags.NonPublic;
|
||
|
FieldInfo field = currentType.GetField(parts[i], bindingFlags);
|
||
|
if (field != null)
|
||
|
{
|
||
|
currentType = field.FieldType;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
PropertyInfo property = currentType.GetProperty(parts[i], bindingFlags);
|
||
|
if (property != null)
|
||
|
{
|
||
|
currentType = property.PropertyType;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
string message = string.Format(CultureInfo.CurrentCulture, Messages.UpdateUnknownFieldOrProperty, parts[i]);
|
||
|
ValidationError error = new ValidationError(message, ErrorNumbers.Error_InvalidUpdate);
|
||
|
error.UserData[RuleUserDataKeys.ErrorObject] = this;
|
||
|
validator.AddError(error);
|
||
|
success = false;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ValidationError error = new ValidationError(Messages.UpdateNotThis, ErrorNumbers.Error_InvalidUpdate);
|
||
|
error.UserData[RuleUserDataKeys.ErrorObject] = this;
|
||
|
validator.AddError(error);
|
||
|
success = false;
|
||
|
}
|
||
|
|
||
|
return success;
|
||
|
}
|
||
|
|
||
|
public override void Execute(RuleExecution context)
|
||
|
{
|
||
|
// This action has no execution behaviour.
|
||
|
}
|
||
|
|
||
|
public override ICollection<string> GetSideEffects(RuleValidation validation)
|
||
|
{
|
||
|
return new string[] { this.path };
|
||
|
}
|
||
|
|
||
|
public override RuleAction Clone()
|
||
|
{
|
||
|
return (RuleAction)this.MemberwiseClone();
|
||
|
}
|
||
|
|
||
|
public override string ToString()
|
||
|
{
|
||
|
return "Update(\"" + this.path + "\")";
|
||
|
}
|
||
|
|
||
|
public override bool Equals(object obj)
|
||
|
{
|
||
|
#pragma warning disable 56506
|
||
|
RuleUpdateAction other = obj as RuleUpdateAction;
|
||
|
return ((other != null) && (string.Equals(this.Path, other.Path, StringComparison.Ordinal)));
|
||
|
#pragma warning restore 56506
|
||
|
}
|
||
|
|
||
|
public override int GetHashCode()
|
||
|
{
|
||
|
return base.GetHashCode();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[Serializable]
|
||
|
public class RuleStatementAction : RuleAction
|
||
|
{
|
||
|
private CodeStatement codeDomStatement;
|
||
|
|
||
|
public RuleStatementAction(CodeStatement codeDomStatement)
|
||
|
{
|
||
|
this.codeDomStatement = codeDomStatement;
|
||
|
}
|
||
|
|
||
|
public RuleStatementAction(CodeExpression codeDomExpression)
|
||
|
{
|
||
|
this.codeDomStatement = new CodeExpressionStatement(codeDomExpression);
|
||
|
}
|
||
|
|
||
|
public RuleStatementAction()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
public CodeStatement CodeDomStatement
|
||
|
{
|
||
|
get { return codeDomStatement; }
|
||
|
set { codeDomStatement = value; }
|
||
|
}
|
||
|
|
||
|
public override bool Validate(RuleValidation validator)
|
||
|
{
|
||
|
if (validator == null)
|
||
|
throw new ArgumentNullException("validator");
|
||
|
|
||
|
if (codeDomStatement == null)
|
||
|
{
|
||
|
ValidationError error = new ValidationError(Messages.NullStatement, ErrorNumbers.Error_ParameterNotSet);
|
||
|
error.UserData[RuleUserDataKeys.ErrorObject] = this;
|
||
|
validator.AddError(error);
|
||
|
return false;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return CodeDomStatementWalker.Validate(validator, codeDomStatement);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public override void Execute(RuleExecution context)
|
||
|
{
|
||
|
if (codeDomStatement == null)
|
||
|
throw new InvalidOperationException(Messages.NullStatement);
|
||
|
CodeDomStatementWalker.Execute(context, codeDomStatement);
|
||
|
}
|
||
|
|
||
|
public override ICollection<string> GetSideEffects(RuleValidation validation)
|
||
|
{
|
||
|
RuleAnalysis analysis = new RuleAnalysis(validation, true);
|
||
|
if (codeDomStatement != null)
|
||
|
CodeDomStatementWalker.AnalyzeUsage(analysis, codeDomStatement);
|
||
|
return analysis.GetSymbols();
|
||
|
}
|
||
|
|
||
|
public override RuleAction Clone()
|
||
|
{
|
||
|
RuleStatementAction newAction = (RuleStatementAction)this.MemberwiseClone();
|
||
|
newAction.codeDomStatement = CodeDomStatementWalker.Clone(codeDomStatement);
|
||
|
return newAction;
|
||
|
}
|
||
|
|
||
|
public override string ToString()
|
||
|
{
|
||
|
if (codeDomStatement == null)
|
||
|
return "";
|
||
|
|
||
|
StringBuilder decompilation = new StringBuilder();
|
||
|
CodeDomStatementWalker.Decompile(decompilation, codeDomStatement);
|
||
|
return decompilation.ToString();
|
||
|
}
|
||
|
|
||
|
public override bool Equals(object obj)
|
||
|
{
|
||
|
#pragma warning disable 56506
|
||
|
RuleStatementAction other = obj as RuleStatementAction;
|
||
|
return ((other != null) && (CodeDomStatementWalker.Match(CodeDomStatement, other.CodeDomStatement)));
|
||
|
#pragma warning restore 56506
|
||
|
}
|
||
|
|
||
|
public override int GetHashCode()
|
||
|
{
|
||
|
return base.GetHashCode();
|
||
|
}
|
||
|
}
|
||
|
}
|