< Summary

Information
Class: Elsa.Workflows.IdentityGraphService
Assembly: Elsa.Workflows.Core
File(s): /home/runner/work/elsa-core/elsa-core/src/modules/Elsa.Workflows.Core/Services/IdentityGraphService.cs
Line coverage
100%
Covered lines: 51
Uncovered lines: 0
Coverable lines: 51
Total lines: 123
Line coverage: 100%
Branch coverage
100%
Covered branches: 30
Total branches: 30
Branch coverage: 100%
Method coverage

Feature is only available for sponsors

Upgrade to PRO version

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity Line coverage
.ctor(...)100%11100%
AssignIdentitiesAsync()100%11100%
AssignIdentitiesAsync()100%11100%
AssignIdentitiesAsync(...)100%11100%
AssignIdentitiesAsync()100%44100%
AssignInputOutputsAsync()100%1616100%
AssignVariables(...)100%44100%
CreateId(...)100%44100%
GetNextIndexFor(...)100%22100%

File(s)

/home/runner/work/elsa-core/elsa-core/src/modules/Elsa.Workflows.Core/Services/IdentityGraphService.cs

#LineLine coverage
 1using Elsa.Extensions;
 2using Elsa.Workflows.Activities;
 3using Elsa.Workflows.Models;
 4using Humanizer;
 5using Microsoft.Extensions.Logging;
 6
 7namespace Elsa.Workflows;
 8
 9/// <inheritdoc />
 106810public class IdentityGraphService(IActivityVisitor activityVisitor, IActivityRegistryLookupService activityRegistryLooku
 11{
 12    /// <inheritdoc />
 13    public async Task AssignIdentitiesAsync(Workflow workflow, CancellationToken cancellationToken = default)
 14    {
 279215        await AssignIdentitiesAsync((IActivity)workflow, cancellationToken);
 279216    }
 17
 18    /// <inheritdoc />
 19    public async Task AssignIdentitiesAsync(IActivity root, CancellationToken cancellationToken = default)
 20    {
 279221        var graph = await activityVisitor.VisitAsync(root, cancellationToken);
 279222        await AssignIdentitiesAsync(graph);
 279223    }
 24
 25    /// <inheritdoc />
 279226    public Task AssignIdentitiesAsync(ActivityNode root) => AssignIdentitiesAsync(root.Flatten().ToList());
 27
 28    /// <inheritdoc />
 29    public async Task AssignIdentitiesAsync(ICollection<ActivityNode> flattenedList)
 30    {
 953131        var identityCounters = new Dictionary<string, int>();
 32
 9961233        foreach (var node in flattenedList)
 34        {
 4027535            node.Activity.Id = CreateId(node, identityCounters, flattenedList);
 4027536            node.Activity.NodeId = node.NodeId;
 4027537            await AssignInputOutputsAsync(node.Activity);
 38
 4027539            if (node.Activity is IVariableContainer variableContainer)
 2026140                AssignVariables(variableContainer);
 4027541        }
 953142    }
 43
 44    /// <inheritdoc />
 45    public async Task AssignInputOutputsAsync(IActivity activity)
 46    {
 4027547        var activityDescriptor = await activityRegistryLookup.FindAsync(activity.Type, activity.Version);
 48
 4027549        if (activityDescriptor == null!)
 50        {
 74151            logger.LogWarning("Activity descriptor not found for activity type {ActivityType}. Skipping identity assignm
 74152            return;
 53        }
 54
 3953455        var inputDictionary = activityDescriptor.GetWrappedInputProperties(activity);
 56
 17910457        foreach (var (inputName, input) in inputDictionary)
 58        {
 5001859            var blockReference = input?.MemoryBlockReference();
 60
 5001861            if (blockReference == null!)
 62                continue;
 63
 3269564            if (string.IsNullOrEmpty(blockReference.Id))
 1756065                blockReference.Id = $"{activity.Id}:input-{inputName.Humanize().Kebaberize()}";
 66        }
 67
 3953468        var outputs = activity.GetOutputs();
 69
 9725270        foreach (var output in outputs)
 71        {
 909272            var blockReference = output.Value.MemoryBlockReference();
 73
 909274            if (blockReference == null!)
 75                continue;
 76
 909277            if (string.IsNullOrEmpty(blockReference.Id))
 380878                blockReference.Id = $"{activity.Id}:output-{output.Name.Humanize().Kebaberize()}";
 79        }
 4027580    }
 81
 82    /// <inheritdoc />
 83    public void AssignVariables(IVariableContainer activity)
 84    {
 2026185        var variables = activity.Variables;
 2026186        var seed = 0;
 87
 4585888        foreach (var variable in variables)
 266889            variable.Id = variable.Id != null! ? variable.Id : $"{activity.Id}:variable-{++seed}";
 2026190    }
 91
 92    private string CreateId(ActivityNode activityNode, IDictionary<string, int> identityCounters, ICollection<ActivityNo
 93    {
 4027594        if (!string.IsNullOrWhiteSpace(activityNode.Activity.Id))
 2204495            return activityNode.Activity.Id;
 96
 97        while (true)
 98        {
 1903899            var fullTypeName = activityNode.Activity.Type;
 19038100            var shortTypeName = fullTypeName.Split('.').Last();
 19038101            var index = GetNextIndexFor(shortTypeName, identityCounters);
 19038102            var name = $"{shortTypeName}{index + 1}";
 103
 110010104            if (allNodes.All(x => x.Activity.Id != name))
 18231105                return name;
 106        }
 107    }
 108
 109    private int GetNextIndexFor(string activityType, IDictionary<string, int> identityCounters)
 110    {
 19038111        if (!identityCounters.TryGetValue(activityType, out var index))
 112        {
 16891113            identityCounters[activityType] = index;
 114        }
 115        else
 116        {
 2147117            index = identityCounters[activityType] + 1;
 2147118            identityCounters[activityType] = index;
 119        }
 120
 19038121        return index;
 122    }
 123}