e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
241 lines
9.9 KiB
C#
241 lines
9.9 KiB
C#
//----------------------------------------------------------------
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
//----------------------------------------------------------------
|
|
namespace System.Activities.Statements
|
|
{
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.ComponentModel;
|
|
using System.Runtime;
|
|
using System.Activities.Validation;
|
|
using System.Collections.ObjectModel;
|
|
using System.Linq;
|
|
using System.Activities.Expressions;
|
|
|
|
public sealed class Confirm : NativeActivity
|
|
{
|
|
static Constraint confirmWithNoTarget = Confirm.ConfirmWithNoTarget();
|
|
|
|
InternalConfirm internalConfirm;
|
|
DefaultConfirmation defaultConfirmation;
|
|
|
|
Variable<CompensationToken> currentCompensationToken;
|
|
|
|
public Confirm()
|
|
: base()
|
|
{
|
|
this.currentCompensationToken = new Variable<CompensationToken>();
|
|
}
|
|
|
|
public InArgument<CompensationToken> Target
|
|
{
|
|
get;
|
|
set;
|
|
}
|
|
|
|
DefaultConfirmation DefaultConfirmation
|
|
{
|
|
get
|
|
{
|
|
if (this.defaultConfirmation == null)
|
|
{
|
|
this.defaultConfirmation = new DefaultConfirmation()
|
|
{
|
|
Target = new InArgument<CompensationToken>(this.currentCompensationToken),
|
|
};
|
|
}
|
|
|
|
return this.defaultConfirmation;
|
|
}
|
|
}
|
|
|
|
InternalConfirm InternalConfirm
|
|
{
|
|
get
|
|
{
|
|
if (this.internalConfirm == null)
|
|
{
|
|
this.internalConfirm = new InternalConfirm()
|
|
{
|
|
Target = new InArgument<CompensationToken>(new ArgumentValue<CompensationToken> { ArgumentName = "Target" }),
|
|
};
|
|
}
|
|
|
|
return this.internalConfirm;
|
|
}
|
|
}
|
|
|
|
protected override void CacheMetadata(NativeActivityMetadata metadata)
|
|
{
|
|
RuntimeArgument targetArgument = new RuntimeArgument("Target", typeof(CompensationToken), ArgumentDirection.In);
|
|
metadata.Bind(this.Target, targetArgument);
|
|
|
|
metadata.SetArgumentsCollection(
|
|
new Collection<RuntimeArgument>
|
|
{
|
|
targetArgument
|
|
});
|
|
|
|
metadata.SetImplementationVariablesCollection(
|
|
new Collection<Variable>
|
|
{
|
|
this.currentCompensationToken
|
|
});
|
|
|
|
Fx.Assert(DefaultConfirmation != null, "DefaultConfirmation must be valid");
|
|
Fx.Assert(InternalConfirm != null, "InternalConfirm must be valid");
|
|
metadata.SetImplementationChildrenCollection(
|
|
new Collection<Activity>
|
|
{
|
|
DefaultConfirmation,
|
|
InternalConfirm
|
|
});
|
|
}
|
|
|
|
internal override IList<Constraint> InternalGetConstraints()
|
|
{
|
|
return new List<Constraint>(1) { confirmWithNoTarget };
|
|
}
|
|
|
|
static Constraint ConfirmWithNoTarget()
|
|
{
|
|
DelegateInArgument<Confirm> element = new DelegateInArgument<Confirm> { Name = "element" };
|
|
DelegateInArgument<ValidationContext> validationContext = new DelegateInArgument<ValidationContext> { Name = "validationContext" };
|
|
Variable<bool> assertFlag = new Variable<bool> { Name = "assertFlag" };
|
|
Variable<IEnumerable<Activity>> elements = new Variable<IEnumerable<Activity>>() { Name = "elements" };
|
|
Variable<int> index = new Variable<int>() { Name = "index" };
|
|
|
|
return new Constraint<Confirm>
|
|
{
|
|
Body = new ActivityAction<Confirm, ValidationContext>
|
|
{
|
|
Argument1 = element,
|
|
Argument2 = validationContext,
|
|
Handler = new Sequence
|
|
{
|
|
Variables =
|
|
{
|
|
assertFlag,
|
|
elements,
|
|
index
|
|
},
|
|
Activities =
|
|
{
|
|
new If
|
|
{
|
|
Condition = new InArgument<bool>((env) => element.Get(env).Target != null),
|
|
Then = new Assign<bool>
|
|
{
|
|
To = assertFlag,
|
|
Value = true
|
|
},
|
|
Else = new Sequence
|
|
{
|
|
Activities =
|
|
{
|
|
new Assign<IEnumerable<Activity>>
|
|
{
|
|
To = elements,
|
|
Value = new GetParentChain
|
|
{
|
|
ValidationContext = validationContext,
|
|
},
|
|
},
|
|
new While(env => (assertFlag.Get(env) != true) &&
|
|
index.Get(env) < elements.Get(env).Count())
|
|
{
|
|
Body = new Sequence
|
|
{
|
|
Activities =
|
|
{
|
|
new If(env => (elements.Get(env).ElementAt(index.Get(env))).GetType() == typeof(CompensationParticipant))
|
|
{
|
|
Then = new Assign<bool>
|
|
{
|
|
To = assertFlag,
|
|
Value = true
|
|
},
|
|
},
|
|
new Assign<int>
|
|
{
|
|
To = index,
|
|
Value = new InArgument<int>(env => index.Get(env) + 1)
|
|
},
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
new AssertValidation
|
|
{
|
|
Assertion = new InArgument<bool>(assertFlag),
|
|
Message = new InArgument<string>(SR.ConfirmWithNoTargetConstraint)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
protected override void Execute(NativeActivityContext context)
|
|
{
|
|
CompensationExtension compensationExtension = context.GetExtension<CompensationExtension>();
|
|
if (compensationExtension == null)
|
|
{
|
|
throw FxTrace.Exception.AsError(new InvalidOperationException(SR.ConfirmWithoutCompensableActivity(this.DisplayName)));
|
|
}
|
|
|
|
if (Target.IsEmpty)
|
|
{
|
|
CompensationToken ambientCompensationToken = (CompensationToken)context.Properties.Find(CompensationToken.PropertyName);
|
|
CompensationTokenData ambientTokenData = ambientCompensationToken == null ? null : compensationExtension.Get(ambientCompensationToken.CompensationId);
|
|
|
|
if (ambientTokenData != null && ambientTokenData.IsTokenValidInSecondaryRoot)
|
|
{
|
|
this.currentCompensationToken.Set(context, ambientCompensationToken);
|
|
if (ambientTokenData.ExecutionTracker.Count > 0)
|
|
{
|
|
context.ScheduleActivity(DefaultConfirmation);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
throw FxTrace.Exception.AsError(new InvalidOperationException(SR.InvalidConfirmActivityUsage(this.DisplayName)));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
CompensationToken compensationToken = Target.Get(context);
|
|
CompensationTokenData tokenData = compensationToken == null ? null : compensationExtension.Get(compensationToken.CompensationId);
|
|
|
|
if (compensationToken == null)
|
|
{
|
|
throw FxTrace.Exception.Argument("Target", SR.InvalidCompensationToken(this.DisplayName));
|
|
}
|
|
|
|
if (compensationToken.ConfirmCalled)
|
|
{
|
|
// No-Op
|
|
return;
|
|
}
|
|
|
|
if (tokenData == null || tokenData.CompensationState != CompensationState.Completed)
|
|
{
|
|
throw FxTrace.Exception.AsError(new InvalidOperationException(SR.CompensableActivityAlreadyConfirmedOrCompensated));
|
|
}
|
|
|
|
// A valid in-arg was passed...
|
|
tokenData.CompensationState = CompensationState.Confirming;
|
|
compensationToken.ConfirmCalled = true;
|
|
context.ScheduleActivity(InternalConfirm);
|
|
}
|
|
}
|
|
|
|
protected override void Cancel(NativeActivityContext context)
|
|
{
|
|
// Suppress Cancel
|
|
}
|
|
}
|
|
}
|