Ships Stream B.1-B.6 — the data-plane authorization engine Phase 6.2 runs on.
Integration into OPC UA dispatch (Stream C — Read / Write / HistoryRead /
Subscribe / Browse / Call etc.) is the next PR on this branch.
New Core.Abstractions:
- OpcUaOperation enum enumerates every OPC UA surface the evaluator gates:
Browse, Read, WriteOperate/Tune/Configure (split by SecurityClassification),
HistoryRead, HistoryUpdate, CreateMonitoredItems, TransferSubscriptions,
Call, AlarmAcknowledge/Confirm/Shelve. Stream C maps each one back to its
dispatch call site.
New Core.Authorization namespace:
- NodeScope record + NodeHierarchyKind — 6-level scope addressing for
Equipment-kind (UNS) namespaces, folder-segment walk for SystemPlatform-kind
(Galaxy). NodeScope carries a Kind selector so the evaluator knows which
hierarchy to descend.
- AuthorizationDecision { Verdict, Provenance } + AuthorizationVerdict
{Allow, NotGranted, Denied} + MatchedGrant. Tri-state per decision #149;
Phase 6.2 only produces Allow + NotGranted, Denied stays reserved for v2.1
Explicit Deny without API break.
- IPermissionEvaluator.Authorize(session, operation, scope).
- PermissionTrie + PermissionTrieNode + TrieGrant. In-memory trie keyed on
the ACL scope hierarchy. CollectMatches walks Cluster → Namespace →
UnsArea → UnsLine → Equipment → Tag (or → FolderSegment(s) → Tag on
Galaxy). Pure additive union — matches that share an LDAP group with the
session contribute flags; OR across levels.
- PermissionTrieBuilder static factory. Build(clusterId, generationId, rows,
scopePaths?) returns a trie for one generation. Cross-cluster rows are
filtered out so the trie is cluster-coherent. Stream C follow-up wires a
real scopePaths lookup from the live DB; tests supply hand-built paths.
- PermissionTrieCache — process-singleton, keyed on (ClusterId, GenerationId).
Install(trie) adds a generation + promotes to "current" when the id is
highest-known (handles out-of-order installs gracefully). Prior generations
retained so an in-flight request against a prior trie still succeeds; GC
via Prune(cluster, keepLatest).
- UserAuthorizationState — per-session cache of resolved LDAP groups +
AuthGenerationId + MembershipVersion + MembershipResolvedUtc. Bounded by
MembershipFreshnessInterval (default 15 min per decision #151) +
AuthCacheMaxStaleness (default 5 min per decision #152).
- TriePermissionEvaluator — default IPermissionEvaluator. Fails closed on
stale sessions (IsStale check short-circuits to NotGranted), on cross-
cluster requests, on empty trie cache. Maps OpcUaOperation → NodePermissions
via MapOperationToPermission (total — every enum value has a mapping; tested).
Tests (27 new, all pass):
- PermissionTrieTests (7): cluster-level grant cascades to every tag;
equipment-level grant doesn't leak to sibling equipment; multi-group union
ORs flags; no-matching-group returns empty; Galaxy folder-segment grant
doesn't leak to sibling folder; cross-cluster rows don't land in this
cluster's trie; build is idempotent (B.6 invariants).
- TriePermissionEvaluatorTests (8): allow when flag matches; NotGranted when
no matching group; NotGranted when flags insufficient; HistoryRead requires
its own bit (decision-level requirement); cross-cluster session denied;
stale session fails closed; no cached trie denied; MapOperationToPermission
is total across every OpcUaOperation.
- PermissionTrieCacheTests (8): empty cache returns null; install-then-get
round-trips; new generation becomes current; out-of-order install doesn't
downgrade current; invalidate drops one cluster; prune retains most recent;
prune no-op when fewer than keep; cluster isolation.
- UserAuthorizationStateTests (4): fresh is not stale; IsStale after 5 min
default; NeedsRefresh true between freshness + staleness windows.
Full solution dotnet test: 1078 passing (baseline 906, Phase 6.1 = 1042,
Phase 6.2 Stream A = +9, Stream B = +27 = 1078). Pre-existing Client.CLI
Subscribe flake unchanged.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
155 lines
5.3 KiB
C#
155 lines
5.3 KiB
C#
using Shouldly;
|
|
using Xunit;
|
|
using ZB.MOM.WW.OtOpcUa.Configuration.Entities;
|
|
using ZB.MOM.WW.OtOpcUa.Configuration.Enums;
|
|
using ZB.MOM.WW.OtOpcUa.Core.Abstractions;
|
|
using ZB.MOM.WW.OtOpcUa.Core.Authorization;
|
|
|
|
namespace ZB.MOM.WW.OtOpcUa.Core.Tests.Authorization;
|
|
|
|
[Trait("Category", "Unit")]
|
|
public sealed class TriePermissionEvaluatorTests
|
|
{
|
|
private static readonly DateTime Now = new(2026, 4, 19, 12, 0, 0, DateTimeKind.Utc);
|
|
private readonly FakeTimeProvider _time = new();
|
|
|
|
private sealed class FakeTimeProvider : TimeProvider
|
|
{
|
|
public DateTime Utc { get; set; } = Now;
|
|
public override DateTimeOffset GetUtcNow() => new(Utc, TimeSpan.Zero);
|
|
}
|
|
|
|
private static NodeAcl Row(string group, NodeAclScopeKind scope, string? scopeId, NodePermissions flags) =>
|
|
new()
|
|
{
|
|
NodeAclRowId = Guid.NewGuid(),
|
|
NodeAclId = $"acl-{Guid.NewGuid():N}",
|
|
GenerationId = 1,
|
|
ClusterId = "c1",
|
|
LdapGroup = group,
|
|
ScopeKind = scope,
|
|
ScopeId = scopeId,
|
|
PermissionFlags = flags,
|
|
};
|
|
|
|
private static UserAuthorizationState Session(string[] groups, DateTime? resolvedUtc = null, string clusterId = "c1") =>
|
|
new()
|
|
{
|
|
SessionId = "sess",
|
|
ClusterId = clusterId,
|
|
LdapGroups = groups,
|
|
MembershipResolvedUtc = resolvedUtc ?? Now,
|
|
AuthGenerationId = 1,
|
|
MembershipVersion = 1,
|
|
};
|
|
|
|
private static NodeScope Scope(string cluster = "c1") =>
|
|
new()
|
|
{
|
|
ClusterId = cluster,
|
|
NamespaceId = "ns",
|
|
UnsAreaId = "area",
|
|
UnsLineId = "line",
|
|
EquipmentId = "eq",
|
|
TagId = "tag",
|
|
Kind = NodeHierarchyKind.Equipment,
|
|
};
|
|
|
|
private TriePermissionEvaluator MakeEvaluator(NodeAcl[] rows)
|
|
{
|
|
var cache = new PermissionTrieCache();
|
|
cache.Install(PermissionTrieBuilder.Build("c1", 1, rows));
|
|
return new TriePermissionEvaluator(cache, _time);
|
|
}
|
|
|
|
[Fact]
|
|
public void Allow_When_RequiredFlag_Matched()
|
|
{
|
|
var evaluator = MakeEvaluator([Row("cn=ops", NodeAclScopeKind.Cluster, null, NodePermissions.Read)]);
|
|
|
|
var decision = evaluator.Authorize(Session(["cn=ops"]), OpcUaOperation.Read, Scope());
|
|
|
|
decision.Verdict.ShouldBe(AuthorizationVerdict.Allow);
|
|
decision.Provenance.Count.ShouldBe(1);
|
|
}
|
|
|
|
[Fact]
|
|
public void NotGranted_When_NoMatchingGroup()
|
|
{
|
|
var evaluator = MakeEvaluator([Row("cn=ops", NodeAclScopeKind.Cluster, null, NodePermissions.Read)]);
|
|
|
|
var decision = evaluator.Authorize(Session(["cn=unrelated"]), OpcUaOperation.Read, Scope());
|
|
|
|
decision.Verdict.ShouldBe(AuthorizationVerdict.NotGranted);
|
|
decision.Provenance.ShouldBeEmpty();
|
|
}
|
|
|
|
[Fact]
|
|
public void NotGranted_When_FlagsInsufficient()
|
|
{
|
|
var evaluator = MakeEvaluator([Row("cn=ops", NodeAclScopeKind.Cluster, null, NodePermissions.Read)]);
|
|
|
|
var decision = evaluator.Authorize(Session(["cn=ops"]), OpcUaOperation.WriteOperate, Scope());
|
|
|
|
decision.Verdict.ShouldBe(AuthorizationVerdict.NotGranted);
|
|
}
|
|
|
|
[Fact]
|
|
public void HistoryRead_Requires_Its_Own_Bit()
|
|
{
|
|
// User has Read but not HistoryRead
|
|
var evaluator = MakeEvaluator([Row("cn=ops", NodeAclScopeKind.Cluster, null, NodePermissions.Read)]);
|
|
|
|
var liveRead = evaluator.Authorize(Session(["cn=ops"]), OpcUaOperation.Read, Scope());
|
|
var historyRead = evaluator.Authorize(Session(["cn=ops"]), OpcUaOperation.HistoryRead, Scope());
|
|
|
|
liveRead.IsAllowed.ShouldBeTrue();
|
|
historyRead.IsAllowed.ShouldBeFalse("HistoryRead uses its own NodePermissions flag, not Read");
|
|
}
|
|
|
|
[Fact]
|
|
public void CrossCluster_Session_Denied()
|
|
{
|
|
var evaluator = MakeEvaluator([Row("cn=ops", NodeAclScopeKind.Cluster, null, NodePermissions.Read)]);
|
|
var otherSession = Session(["cn=ops"], clusterId: "c-other");
|
|
|
|
var decision = evaluator.Authorize(otherSession, OpcUaOperation.Read, Scope(cluster: "c1"));
|
|
|
|
decision.Verdict.ShouldBe(AuthorizationVerdict.NotGranted);
|
|
}
|
|
|
|
[Fact]
|
|
public void StaleSession_FailsClosed()
|
|
{
|
|
var evaluator = MakeEvaluator([Row("cn=ops", NodeAclScopeKind.Cluster, null, NodePermissions.Read)]);
|
|
var session = Session(["cn=ops"], resolvedUtc: Now);
|
|
_time.Utc = Now.AddMinutes(10); // well past the 5-min AuthCacheMaxStaleness default
|
|
|
|
var decision = evaluator.Authorize(session, OpcUaOperation.Read, Scope());
|
|
|
|
decision.Verdict.ShouldBe(AuthorizationVerdict.NotGranted);
|
|
}
|
|
|
|
[Fact]
|
|
public void NoCachedTrie_ForCluster_Denied()
|
|
{
|
|
var cache = new PermissionTrieCache(); // empty cache
|
|
var evaluator = new TriePermissionEvaluator(cache, _time);
|
|
|
|
var decision = evaluator.Authorize(Session(["cn=ops"]), OpcUaOperation.Read, Scope());
|
|
|
|
decision.Verdict.ShouldBe(AuthorizationVerdict.NotGranted);
|
|
}
|
|
|
|
[Fact]
|
|
public void OperationToPermission_Mapping_IsTotal()
|
|
{
|
|
foreach (var op in Enum.GetValues<OpcUaOperation>())
|
|
{
|
|
// Must not throw — every OpcUaOperation needs a mapping or the compliance-check
|
|
// "every operation wired" fails.
|
|
TriePermissionEvaluator.MapOperationToPermission(op);
|
|
}
|
|
}
|
|
}
|