Skip to content

Latest commit

 

History

History
103 lines (77 loc) · 5.77 KB

Layer-Manager.md

File metadata and controls

103 lines (77 loc) · 5.77 KB

Manager (Domain Logic)

The Manager layer is primarily responsible for hosting the key business and/or workflow logic. This is also where the primary validation is performed to ensure the consistency of the request before any further processing is performed.


Usage

This layer is generally code-generated and provides options to provide a fully custom implementation, and has extension opportunities to inject additional logic into the processing pipeline.

The Operation element within the entity.xml configuration primarily drives the output. There is a generated class per Entity named {Entity}Manager.

There is also a corresonding interface named I{Entity}Manager generated so the likes of test mocking etc. can be employed.


Code-generated

An end-to-end code-generated processing pipeline generally consists of:

Step Description
ManagerInvoker The logic is wrapped by a ManagerInvoker. This enables the BusinessInvokerArgs options to be specified, including TransactionScopeOption and Exception handler. These values are generally specified in the code-generation configuration.
OperationType The ExecutionContext.OperationType is set to specify the type of operation being performed (Create, Read, Update or Delete) so other functions down the call stack can infer operation intent.
CleanUp Entity CleanUp is the process of reviewing and updating the entity properties to make sure it is in a logical / consistent state.
PreValidate* The PreValidate extension opportunity; where set this will be invoked. This enables logic to be invoked before the validation performed.
Validation The MultiValidator is used to validate all input, including an OnValidate extension opportunity, to ensure data consistency before processing.
OnBefore* The OnBefore extension opportunity; where set this will be invoked. This enables logic to be invoked before the primary Operation is performed.
DataSvc The {Entity}DataSvc layer is invoked to orchestrate the data processing.
OnAfter* The OnAfter extension opportunity; where set this will be invoked. This enables logic to be invoked after the primary Operation is performed.
CleanUp An Entity CleanUp of response before returning.

* Note: To minimize the generated code the extension opportunities are only generated where selected. This is performed by using the EntityElement and setting the ManagerExtensions attribute to true (defaults to false).

The following demonstrates the generated code (a snippet from the sample ContactManager) that does not include ManagerExtensions:

public Task<Contact> CreateAsync(Contact value)
{
    value.Validate(nameof(value)).Mandatory().Run().ThrowOnError();

    return ManagerInvoker.Current.InvokeAsync(this, async () =>
    {
        ExecutionContext.Current.OperationType = OperationType.Create;
        Cleaner.CleanUp(value);
        MultiValidator.Create()
            .Add(value.Validate(nameof(value)))
            .Run().ThrowOnError();

        return Cleaner.Clean(await _dataService.CreateAsync(value).ConfigureAwait(false));
    });
}

The following demonstrates the generated code (a snippet from the sample PersonManager) that includes ManagerExtensions:

public Task<Person> CreateAsync(Person value)
{
    value.Validate(nameof(value)).Mandatory().Run().ThrowOnError();

    return ManagerInvoker.Current.InvokeAsync(this, async () =>
    {
        ExecutionContext.Current.OperationType = OperationType.Create;
        Cleaner.CleanUp(value);
        if (_createOnPreValidateAsync != null) await _createOnPreValidateAsync(value).ConfigureAwait(false);

        MultiValidator.Create()
            .Add(value.Validate(nameof(value)).Entity(PersonValidator.Default))
            .Additional((__mv) => _createOnValidate?.Invoke(__mv, value))
            .Run().ThrowOnError();

        if (_createOnBeforeAsync != null) await _createOnBeforeAsync(value).ConfigureAwait(false);
        var __result = await _dataService.CreateAsync(value).ConfigureAwait(false);
        if (_createOnAfterAsync != null) await _createOnAfterAsync(__result).ConfigureAwait(false);
        return Cleaner.Clean(__result);
    });
}

Custom

A custom (OnImplementation) processing pipeline generally consists of:

Step Description
ManagerInvoker The logic is wrapped by a ManagerInvoker. This enables the BusinessInvokerArgs options to be specified, including TransactionScopeOption and Exception handler. These values are generally specified in the code-generation configuration.
OnImplementation Invocation of a named XxxxxOnImplementaionAsync method that must be implemented in a non-generated partial class.

The following demonstrates the generated code:

public Task AddAsync(Person? person)
{
    return ManagerInvoker.Current.InvokeAsync(this, async () =>
    {
        ExecutionContext.Current.OperationType = OperationType.Unspecified;
        await AddOnImplementationAsync(person).ConfigureAwait(false);
    });
}