using System; using System.Collections.Generic; using System.IO; using System.Threading.Tasks; using Opc.Ua; using Opc.Ua.Client; using Opc.Ua.Configuration; namespace ZB.MOM.WW.LmxOpcUa.Tests.Helpers { /// /// OPC UA client helper for integration tests. Connects to a test server, /// browses, reads, and subscribes to nodes programmatically. /// internal class OpcUaTestClient : IDisposable { private Session? _session; public Session Session => _session ?? throw new InvalidOperationException("Not connected"); /// /// Resolves the namespace index for a given namespace URI (e.g., "urn:TestGalaxy:LmxOpcUa"). /// public ushort GetNamespaceIndex(string galaxyName = "TestGalaxy") { var nsUri = $"urn:{galaxyName}:LmxOpcUa"; var idx = Session.NamespaceUris.GetIndex(nsUri); if (idx < 0) throw new InvalidOperationException($"Namespace '{nsUri}' not found on server"); return (ushort)idx; } /// /// Creates a NodeId in the LmxOpcUa namespace using the server's actual namespace index. /// public NodeId MakeNodeId(string identifier, string galaxyName = "TestGalaxy") { return new NodeId(identifier, GetNamespaceIndex(galaxyName)); } public async Task ConnectAsync(string endpointUrl) { var config = new ApplicationConfiguration { ApplicationName = "OpcUaTestClient", ApplicationUri = "urn:localhost:OpcUaTestClient", ApplicationType = ApplicationType.Client, SecurityConfiguration = new SecurityConfiguration { ApplicationCertificate = new CertificateIdentifier { StoreType = CertificateStoreType.Directory, StorePath = Path.Combine(Path.GetTempPath(), "OpcUaTestClient", "pki", "own") }, TrustedIssuerCertificates = new CertificateTrustList { StoreType = CertificateStoreType.Directory, StorePath = Path.Combine(Path.GetTempPath(), "OpcUaTestClient", "pki", "issuer") }, TrustedPeerCertificates = new CertificateTrustList { StoreType = CertificateStoreType.Directory, StorePath = Path.Combine(Path.GetTempPath(), "OpcUaTestClient", "pki", "trusted") }, RejectedCertificateStore = new CertificateTrustList { StoreType = CertificateStoreType.Directory, StorePath = Path.Combine(Path.GetTempPath(), "OpcUaTestClient", "pki", "rejected") }, AutoAcceptUntrustedCertificates = true }, ClientConfiguration = new ClientConfiguration { DefaultSessionTimeout = 30000 }, TransportQuotas = new TransportQuotas() }; await config.Validate(ApplicationType.Client); config.CertificateValidator.CertificateValidation += (_, e) => e.Accept = true; var endpoint = CoreClientUtils.SelectEndpoint(config, endpointUrl, false); var endpointConfig = EndpointConfiguration.Create(config); var configuredEndpoint = new ConfiguredEndpoint(null, endpoint, endpointConfig); _session = await Session.Create( config, configuredEndpoint, false, "OpcUaTestClient", 30000, null, null); } /// /// Browse children of a node. Returns list of (DisplayName, NodeId, NodeClass). /// public async Task> BrowseAsync(NodeId nodeId) { var results = new List<(string, NodeId, NodeClass)>(); var browser = new Browser(Session) { NodeClassMask = (int)NodeClass.Object | (int)NodeClass.Variable, ReferenceTypeId = ReferenceTypeIds.HierarchicalReferences, IncludeSubtypes = true, BrowseDirection = BrowseDirection.Forward }; var refs = browser.Browse(nodeId); foreach (var rd in refs) { results.Add((rd.DisplayName.Text, ExpandedNodeId.ToNodeId(rd.NodeId, Session.NamespaceUris), rd.NodeClass)); } return results; } /// /// Read a node's value. /// public DataValue Read(NodeId nodeId) { return Session.ReadValue(nodeId); } /// /// Create a subscription with a monitored item on the given node. /// Returns the subscription and monitored item for inspection. /// public async Task<(Subscription Sub, MonitoredItem Item)> SubscribeAsync( NodeId nodeId, int intervalMs = 250) { var subscription = new Subscription(Session.DefaultSubscription) { PublishingInterval = intervalMs, DisplayName = "TestSubscription" }; var item = new MonitoredItem(subscription.DefaultItem) { StartNodeId = nodeId, DisplayName = nodeId.ToString(), SamplingInterval = intervalMs }; subscription.AddItem(item); Session.AddSubscription(subscription); await subscription.CreateAsync(); return (subscription, item); } public void Dispose() { if (_session != null) { try { _session.Close(); } catch { /* ignore */ } _session.Dispose(); } } } }