140 lines
5.8 KiB
C#
Raw Normal View History

//-----------------------------------------------------------------------------
// Copyright (c) Microsoft Corporation. All rights reserved.
//-----------------------------------------------------------------------------
namespace System.Activities.Statements
{
using System.Activities.Runtime;
using System.Activities.Validation;
using System.Collections.Generic;
using System.ComponentModel;
public sealed class Rethrow : NativeActivity
{
public Rethrow()
{
DelegateInArgument<Rethrow> element = new DelegateInArgument<Rethrow>() { Name = "constraintArg" };
DelegateInArgument<ValidationContext> validationContext = new DelegateInArgument<ValidationContext>() { Name = "validationContext" };
base.Constraints.Add(new Constraint<Rethrow>
{
Body = new ActivityAction<Rethrow, ValidationContext>
{
Argument1 = element,
Argument2 = validationContext,
Handler = new RethrowBuildConstraint
{
ParentChain = new GetParentChain
{
ValidationContext = validationContext,
},
RethrowActivity = element
},
}
});
}
protected override void CacheMetadata(NativeActivityMetadata metadata)
{
}
protected override void Execute(NativeActivityContext context)
{
FaultContext faultContext = context.Properties.Find(TryCatch.FaultContextId) as FaultContext;
if (faultContext == null)
{
throw FxTrace.Exception.AsError(new InvalidOperationException(SR.FaultContextNotFound(this.DisplayName)));
}
context.RethrowException(faultContext);
}
class RethrowBuildConstraint : NativeActivity<bool>
{
[RequiredArgument]
[DefaultValue(null)]
public InArgument<IEnumerable<Activity>> ParentChain
{
get;
set;
}
[RequiredArgument]
[DefaultValue(null)]
public InArgument<Rethrow> RethrowActivity
{
get;
set;
}
protected override void CacheMetadata(NativeActivityMetadata metadata)
{
RuntimeArgument parentChainArgument = new RuntimeArgument("ParentChain", typeof(IEnumerable<Activity>), ArgumentDirection.In, true);
metadata.Bind(this.ParentChain, parentChainArgument);
metadata.AddArgument(parentChainArgument);
RuntimeArgument rethrowActivityArgument = new RuntimeArgument("RethrowActivity", typeof(Rethrow), ArgumentDirection.In, true);
metadata.Bind(this.RethrowActivity, rethrowActivityArgument);
metadata.AddArgument(rethrowActivityArgument);
}
protected override void Execute(NativeActivityContext context)
{
IEnumerable<Activity> parentChain = this.ParentChain.Get(context);
Rethrow rethrowActivity = this.RethrowActivity.Get(context);
Activity previousActivity = rethrowActivity;
bool privateRethrow = false;
// TryCatch with Rethrow is usually authored in the following way:
//
// TryCatch
// {
// Try = DoWork
// Catch Handler = Sequence
// {
// ProcessException,
// Rethrow
// }
// }
// Notice that the chain of Activities is TryCatch->Sequence->Rethrow
// We want to validate that Rethrow is in the catch block of TryCatch
// We walk up the parent chain until we find TryCatch. Then we check if one the catch handlers points to Sequence(the previous activity in the tree)
foreach (Activity parent in parentChain)
{
// Rethrow is only allowed under the public children of a TryCatch activity.
// If any of the activities in the tree is a private child, report a constraint violation.
if (parent.ImplementationChildren.Contains(previousActivity))
{
privateRethrow = true;
}
TryCatch tryCatch = parent as TryCatch;
if (tryCatch != null)
{
if (previousActivity != null)
{
foreach (Catch catchHandler in tryCatch.Catches)
{
ActivityDelegate catchAction = catchHandler.GetAction();
if (catchAction != null && catchAction.Handler == previousActivity)
{
if (privateRethrow)
{
Constraint.AddValidationError(context, new ValidationError(SR.RethrowMustBeAPublicChild(rethrowActivity.DisplayName), rethrowActivity));
}
return;
}
}
}
}
previousActivity = parent;
}
Constraint.AddValidationError(context, new ValidationError(SR.RethrowNotInATryCatch(rethrowActivity.DisplayName), rethrowActivity));
}
}
}
}