OPC UA Complex Types
June 20, 2026 · View on GitHub
Overview
The Opc.Ua.Client and Opc.Ua.Client.ComplexTypes libraries provide support for handling custom data types (complex types) in OPC UA client applications. Complex types include:
- Custom Structures: User-defined structured data types with multiple fields
- Custom Enumerations: User-defined enumeration types with custom values
The library allows OPC UA clients to automatically discover, load, and work with server-specific custom types, enabling seamless reading and writing of structured data without manual type definitions.
Key Concepts
What are Complex Types?
In OPC UA, complex types are custom data types defined by the server that extend beyond the built-in OPC UA data types. These types are commonly used to represent structured data such as:
- Configuration structures with multiple parameters
- Device status information with multiple fields
- Custom enumerations specific to a domain or device
Type Discovery and Loading
The ComplexTypeSystem class manages the discovery and loading of custom types from an OPC UA server. It:
- Browses the server's type system to discover custom types
- Loads type definitions (using DataTypeDefinition attribute or binary/XML dictionaries)
- Registers types complying to the type definitions in the session's type factory for encoding/decoding
Supported Type Systems
The library supports multiple type definition mechanisms:
- OPC UA 1.04+ DataTypeDefinition Attribute: Modern structured type definitions
- OPC UA 1.03 Binary Schema Dictionaries: Legacy binary type dictionaries
- OPC UA 1.03 XML Schema Dictionaries: Legacy XML type dictionaries
The library automatically uses the most appropriate mechanism available on the server.
Getting Started
Type builders
Default type builder
A type builder builds the types that are registered in the EncodeableFactory by the ComplexTypeSystem class. The default type builder registers in memory IEncodeable "adapter" classes that wrap discovered DataTypeDefinition and provide "state" (a list of Variants for the properties) clone, compare, and encode/decode behavior. The default type builder is part of the Opc.Ua.Client library and used when no type builder is provided in the constructor of the ComplexTypeSystem class.
Reflection.Emit based type builder
The OPCFoundation.NetStandard.Opc.Ua.Client.ComplexTypes nuget package extends the Opc.Ua.Client library by adding a type builder implementation that supports dynamically generating .NET types via Reflection.Emit at runtime. This has the added benefit over the default approach that the types can be reflected over by other tools, e.g. serializers. However, this is not NativeAOT compliant, and therefore building against a NativeAOT runtime will always fall back to the default behavior.
To use the Reflection.Emit type builder add the NuGet package to your project:
dotnet add package OPCFoundation.NetStandard.Opc.Ua.Client.ComplexTypes
Or via Package Manager:
Install-Package OPCFoundation.NetStandard.Opc.Ua.Client.ComplexTypes
Basic Usage
1. Loading All Custom Types from a Server
The most common approach is to load all custom types after establishing a session:
using Opc.Ua;
using Opc.Ua.Client;
using Opc.Ua.Client.ComplexTypes;
// Create and connect session
var session = await Session.Create(...);
// Create and load the complex type system
ComplexTypeSystem complexTypeSystem;
if (!useReflectionEmitTypeBuilder)
{
// Uses the default type builder
complexTypeSystem = new ComplexTypeSystem(session);
}
else
{
// Uses the Reflection.Emit type builder
// Only works if the OPCFoundation.NetStandard.Opc.Ua.Client.ComplexTypes
// nuget is referenced
complexTypeSystem = ComplexTypeSystem.Create(session);
}
await complexTypeSystem.LoadAsync();
Console.WriteLine($"Loaded {complexTypeSystem.GetDefinedTypes().Count} custom types");
After loading, the session can automatically encode and decode custom types when reading or writing values.
2. Reading Values with Complex Types
Once the type system is loaded, reading complex type values is straightforward:
// Read a value that contains a complex type
NodeId nodeId = new NodeId("ns=2;s=MyCustomStructVariable");
DataValue dataValue = await session.ReadValueAsync(nodeId);
// The value is automatically decoded to a .NET type
// Note: DataValue.Value is obsolete. Use WrappedValue (Variant) instead.
if (dataValue.WrappedValue.TryGetValue(out ExtensionObject extensionObject))
{
// Access the structured data
if (extensionObject.Body is IStructure complexType)
{
// Access properties by name
Variant temperatureValue = complexType["Temperature"];
Variant pressureValue = complexType["Pressure"];
Console.WriteLine($"Temperature: {temperatureValue}");
Console.WriteLine($"Pressure: {pressureValue}");
// Enumerate all properties
foreach (IStructureField property in complexType.GetFields())
{
Console.WriteLine($"{property.Name}: {complexType[property.Name]}");
}
}
}
3. Writing Values with Complex Types
To write complex type values, modify the properties and write back:
// Read current value
DataValue dataValue = await session.ReadValueAsync(nodeId);
dataValue.WrappedValue.TryGetValue(out ExtensionObject extensionObject);
var complexType = (IStructure)extensionObject.Body;
// Modify properties
complexType["Temperature"] = 25.5;
complexType["Pressure"] = 101.3;
// Write the modified value back
var writeValue = new WriteValue
{
NodeId = nodeId,
AttributeId = Attributes.Value,
Value = new DataValue
{
WrappedValue = extensionObject,
SourceTimestamp = DateTime.UtcNow
}
};
var writeValues = new WriteValueCollection { writeValue };
var response = await session.WriteAsync(null, writeValues, CancellationToken.None);
if (StatusCode.IsGood(response.Results[0]))
{
Console.WriteLine("Value written successfully");
}
Approaches for Working with Custom Types
There are three ways to work with custom data types from an OPC UA server. The right approach depends on your use case.
Approach 1: Hand-Written IEncodeable + EncodeableFactory Registration
If you have already implemented (or generated) a class that implements IEncodeable, you can register it with the session's EncodeableFactory. The stack will then automatically decode incoming ExtensionObject values into instances of your type.
Step 1 – Implement IEncodeable
// Required namespaces:
// using Opc.Ua;
// using System.Runtime.Serialization;
[DataContract(Namespace = "http://www.siemens.com/simatic-s7-opcua")]
public class UDT_SemVer : IEncodeable
{
public UDT_SemVer() { Initialize(); }
[OnDeserializing]
private void Initialize(StreamingContext context = default) { }
[DataMember(Name = "Major", IsRequired = true, Order = 1)]
public byte Major { get; set; }
[DataMember(Name = "Minor", IsRequired = true, Order = 2)]
public byte Minor { get; set; }
[DataMember(Name = "Patch", IsRequired = true, Order = 3)]
public byte Patch { get; set; }
// TypeId must match the DataType NodeId on the server.
// The "DT_" prefix is a Siemens S7 OPC UA convention for data type nodes;
// the quoted name is the Siemens-specific identifier format.
public ExpandedNodeId TypeId =>
new ExpandedNodeId(
"nsu=http://www.siemens.com/simatic-s7-opcua;s=DT_\"UDT_SemVer\"");
// BinaryEncodingId must match the Binary Encoding NodeId on the server.
// The "TE_" prefix is the Siemens S7 OPC UA convention for type encoding nodes.
public ExpandedNodeId BinaryEncodingId =>
new ExpandedNodeId(
"nsu=http://www.siemens.com/simatic-s7-opcua;s=TE_\"UDT_SemVer\"");
public ExpandedNodeId XmlEncodingId => ExpandedNodeId.Null;
public void Encode(IEncoder encoder)
{
encoder.WriteByte("Major", Major);
encoder.WriteByte("Minor", Minor);
encoder.WriteByte("Patch", Patch);
}
public void Decode(IDecoder decoder)
{
Major = decoder.ReadByte("Major");
Minor = decoder.ReadByte("Minor");
Patch = decoder.ReadByte("Patch");
}
public bool IsEqual(IEncodeable encodeable) =>
encodeable is UDT_SemVer other &&
Major == other.Major && Minor == other.Minor && Patch == other.Patch;
public void EncodeJson(IJsonEncoder encoder)
{
encoder.WriteByte("Major", Major);
encoder.WriteByte("Minor", Minor);
encoder.WriteByte("Patch", Patch);
}
public void DecodeJson(IJsonDecoder decoder)
{
Major = decoder.ReadByte("Major");
Minor = decoder.ReadByte("Minor");
Patch = decoder.ReadByte("Patch");
}
public object Clone() => MemberwiseClone();
}
Important:
BinaryEncodingIdmust match the Binary Encoding NodeId returned by the server (visible in the OPC UA address space as theDataTypeEncodingTypechild node named"Default Binary"). This is what the decoder uses to look up the registered type.
Step 2 – Register the Type Before Reading
Register the type with the session's factory before reading any values that use it. The simplest way is to add the Type directly:
// Create and connect the session
var session = await Session.Create(...);
// Register the custom type with the session's encodeable factory
session.Factory.Builder
.AddEncodeableType(typeof(UDT_SemVer))
.Commit();
// Now read – the stack decodes the ExtensionObject automatically
DataValue result = await session.ReadValueAsync(
"ns=3;s=\"H35dispenser\".\"ID\".\"ElectricalVersion\"");
if (result.WrappedValue.TryGetValue(out ExtensionObject extObject) &&
extObject.Body is UDT_SemVer semVer)
{
Console.WriteLine($"Major={semVer.Major} Minor={semVer.Minor} Patch={semVer.Patch}");
}
You can also register all types from an entire assembly at once:
session.Factory.Builder
.AddEncodeableTypes(typeof(UDT_SemVer).Assembly)
.Commit();
Or register multiple types individually in a single builder chain:
session.Factory.Builder
.AddEncodeableType(typeof(UDT_SemVer))
.AddEncodeableType(typeof(UDT_AnotherType))
.Commit();
Note:
ComplexTypeSystem.LoadAsync()is not required when using this approach. If you callLoadAsync()after registering your own types, the ComplexTypeSystem will skip types that are already registered in the factory.
Approach 2: Source-Generated IEncodeable (Recommended for New Code)
Starting with version 2.0, you can annotate a POCO class with [DataType] and
optional [DataTypeField] attributes and the OPC UA source generator will
automatically generate the IEncodeable implementation for you. This eliminates
the need to hand-write Encode, Decode, IsEqual, and Clone methods.
See Source-Generated Data Types for full details.
using Opc.Ua;
[DataType(Namespace = "http://www.siemens.com/simatic-s7-opcua",
BinaryEncodingId = "s=TE_\"UDT_SemVer\"")]
public partial class UDT_SemVer
{
[DataTypeField(Order = 1)]
public byte Major { get; set; }
[DataTypeField(Order = 2)]
public byte Minor { get; set; }
[DataTypeField(Order = 3)]
public byte Patch { get; set; }
}
After adding the source generator NuGet package, the generated registration extension method can be used:
session.Factory.Builder
.AddMyNamespaceDataTypes()
.Commit();
Approach 3: Runtime IStructure (No Pre-defined Types Required)
If you do not know the type structure at compile time, or prefer not to create
.NET types for every server type, use ComplexTypeSystem to load the type
definitions at runtime and access fields via the IStructure interface. This
requires no hand-written types.
See the Basic Usage and Advanced Usage sections below for examples.
Advanced Usage
Loading Specific Types
Instead of loading all types, you can load specific types or namespaces:
Load a Specific Type
var complexTypeSystem = new ComplexTypeSystem(session);
// Load a specific type by NodeId
ExpandedNodeId typeNodeId = new ExpandedNodeId("ns=2;i=3001");
IType? systemType = await complexTypeSystem.LoadTypeAsync(typeNodeId);
if (systemType != null)
{
Console.WriteLine($"Loaded type: {systemType.XmlName}");
}
Load a Specific Type with Subtypes
// Load a type and all its subtypes
ExpandedNodeId typeNodeId = new ExpandedNodeId("ns=2;i=3001");
IType? systemType = await complexTypeSystem.LoadTypeAsync(typeNodeId, subTypes: true);
Load All Types from a Namespace
var complexTypeSystem = new ComplexTypeSystem(session);
// Load all custom types from a specific namespace
string namespaceUri = "http://mycompany.com/MyCustomTypes";
bool success = await complexTypeSystem.LoadNamespaceAsync(namespaceUri);
if (success)
{
Console.WriteLine($"Successfully loaded types from namespace: {namespaceUri}");
}
Working with Complex Type Properties
The IStructure interface provides flexible access to complex type fields:
Access by Name
if (extensionObject.Body is IStructure complexType)
{
// Get property value by name
Variant value = complexType["PropertyName"];
// Set property value by name
complexType["PropertyName"] = newValue;
}
Access by Index
if (extensionObject.Body is IStructure complexType)
{
// Get property value by index
Variant value = complexType[0];
// Set property value by index
complexType[0] = newValue;
}
Enumerate Properties
if (extensionObject.Body is IStructure complexType)
{
// Enumerate with detailed information
foreach (IStructureField property in complexType.GetFields())
{
Console.WriteLine($"Property: {property.Name}");
Console.WriteLine($" Type: {property.TypeInfo}");
Console.WriteLine($" IsOptional: {property.IsOptional}");
Console.WriteLine($" Value: {complexType[property.Name]}");
}
}
Handling Enumeration Types
Custom enumerations are also supported:
// After loading the type system, enum values are automatically decoded
DataValue dataValue = await session.ReadValueAsync(enumNodeId);
if (dataValue.WrappedValue.TryGetValue(out EnumValue enumValue))
{
// The value is the numeric representation
Console.WriteLine($"Enum value: {enumValue}");
// You can also get the enum type from the factory
ExpandedNodeId enumTypeId = ...;
if (session.Factory.TryGetEnumeratedType(enumTypeId, out IEnumeratedType enumType))
{
Console.WriteLine($"Enum type: {enumType.XmlName}");
}
}
Type Information and Introspection
Get All Loaded Types
var complexTypeSystem = new ComplexTypeSystem(session);
await complexTypeSystem.LoadAsync();
// Get all types that were dynamically created
IReadOnlyList<XmlQualifiedName> definedTypes = complexTypeSystem.GetDefinedTypes();
foreach (XmlQualifiedName type in definedTypes)
{
Console.WriteLine($"Type: {type.Namespace}.{type.Name}");
}
Get Type Definitions
// Get the DataTypeDefinition for a specific type
ExpandedNodeId dataTypeId = new ExpandedNodeId("ns=2;i=3001");
var definitions = complexTypeSystem.GetDataTypeDefinitionsForDataType(dataTypeId);
foreach (var kvp in definitions)
{
Console.WriteLine($"NodeId: {kvp.Key}");
Console.WriteLine($"Definition: {kvp.Value}");
}
Get Loaded Data Type IDs
// Get all NodeIds for loaded data types
IEnumerable<ExpandedNodeId> dataTypeIds = complexTypeSystem.GetDefinedDataTypeIds();
foreach (var dataTypeId in dataTypeIds)
{
Console.WriteLine($"Loaded type: {dataTypeId}");
}
Using Custom Type Factories
You can provide a custom type factory for advanced scenarios:
// Create a custom factory (implement IComplexTypeFactory)
IComplexTypeFactory customFactory = new MyCustomComplexTypeFactory();
// Use it with the ComplexTypeSystem
var complexTypeSystem = new ComplexTypeSystem(session, customFactory);
await complexTypeSystem.LoadAsync();
Working with Telemetry and Logging
The ComplexTypeSystem supports the OPC UA telemetry context for observability:
// Use the session's telemetry context (default)
var complexTypeSystem = new ComplexTypeSystem(session);
// Or provide a custom telemetry context
ITelemetryContext telemetry = myCustomTelemetryContext;
var complexTypeSystem = new ComplexTypeSystem(session, telemetry);
// The system will log type loading information
await complexTypeSystem.LoadAsync();
Common Patterns
Pattern 1: One-Time Type Loading at Session Start
public async Task<ISession> CreateSessionWithTypes(string endpointUrl)
{
var session = await Session.Create(
configuration,
new ConfiguredEndpoint(null, new EndpointDescription(endpointUrl)),
false,
"MyClient",
60000,
null,
null
);
// Load all custom types immediately after connection
var complexTypeSystem = new ComplexTypeSystem(session);
await complexTypeSystem.LoadAsync();
return session;
}
Pattern 2: Lazy Loading on Demand
public class OpcUaClient
{
private ISession _session;
private ComplexTypeSystem _complexTypeSystem;
private bool _typesLoaded;
public async Task EnsureTypesLoadedAsync()
{
if (!_typesLoaded)
{
_complexTypeSystem = new ComplexTypeSystem(_session);
await _complexTypeSystem.LoadAsync();
_typesLoaded = true;
}
}
public async Task<DataValue> ReadComplexValueAsync(NodeId nodeId)
{
await EnsureTypesLoadedAsync();
return await _session.ReadValueAsync(nodeId);
}
}
Pattern 3: Reading Multiple Complex Values
public async Task ReadMultipleComplexValuesAsync(IList<NodeId> nodeIds)
{
// Load types once
var complexTypeSystem = new ComplexTypeSystem(session);
await complexTypeSystem.LoadAsync();
// Read all values
var nodesToRead = nodeIds.Select(id => new ReadValueId
{
NodeId = id,
AttributeId = Attributes.Value
}).ToList();
var response = await session.ReadAsync(
null,
0,
TimestampsToReturn.Both,
new ReadValueIdCollection(nodesToRead),
CancellationToken.None
);
// Process results
for (int i = 0; i < response.Results.Count; i++)
{
var dataValue = response.Results[i];
if (dataValue.WrappedValue.TryGetValue(out ExtensionObject extensionObject) &&
extensionObject.Body is IStructure complexType)
{
Console.WriteLine($"NodeId: {nodeIds[i]}");
foreach (IStructureField property in complexType.GetFields())
{
Console.WriteLine($" {property.Name}: {complexType[property.Name]}");
}
}
}
}
Error Handling
Handling Type Loading Failures
try
{
var complexTypeSystem = new ComplexTypeSystem(session);
bool success = await complexTypeSystem.LoadAsync(throwOnError: true);
if (success)
{
Console.WriteLine("All types loaded successfully");
}
else
{
Console.WriteLine("Some types could not be loaded");
}
}
catch (ServiceResultException ex)
{
Console.WriteLine($"Failed to load types: {ex.Message}");
// Handle error appropriately
}
Handling Missing Type Definitions
// Try to load a specific type
var complexTypeSystem = new ComplexTypeSystem(session);
IType? systemType = await complexTypeSystem.LoadTypeAsync(typeNodeId, throwOnError: false);
if (systemType == null)
{
Console.WriteLine($"Type {typeNodeId} could not be loaded");
// Fall back to reading as ExtensionObject with opaque body
}
Performance Considerations
Type System Caching
- The
ComplexTypeSystemloads types once and caches them in the session's factory - Subsequent reads/writes use the cached types automatically
- Types remain available for the lifetime of the session
Minimizing Load Time
// Load only enumerations (faster)
var complexTypeSystem = new ComplexTypeSystem(session);
await complexTypeSystem.LoadAsync(onlyEnumTypes: true);
// Or load only specific namespaces
await complexTypeSystem.LoadNamespaceAsync("http://mycompany.com/MyTypes");
Batch Operations
When reading multiple values with complex types, read them in batches to minimize round-trips:
// Read multiple values in one call
var response = await session.ReadAsync(
null,
0,
TimestampsToReturn.Both,
new ReadValueIdCollection(nodesToRead),
CancellationToken.None
);
Troubleshooting
Types Not Loading
Problem: LoadAsync() completes but types are not available
Solutions:
- Verify the server supports DataTypeDefinition or provides type dictionaries
- Check server logs for encoding/dictionary availability
- Use verbose logging to see what types are discovered
// Enable detailed logging through telemetry context
var telemetry = session.MessageContext.Telemetry;
var complexTypeSystem = new ComplexTypeSystem(session, telemetry);
// The ComplexTypeSystem will log detailed information during type loading
await complexTypeSystem.LoadAsync();
Values Still Encoded as ExtensionObject
Problem: Values are read as ExtensionObject with opaque body instead of structured types
Solutions:
- If using
ComplexTypeSystem: ensureLoadAsync()was called before reading the value, verify the type is actually loaded withcomplexTypeSystem.GetDefinedTypes(), and check if the server's type definition is complete and valid. - If using a hand-written
IEncodeable(Approach 1): ensure the type is registered withsession.Factory.Builder.AddEncodeableType(typeof(YourType)).Commit()before reading. TheBinaryEncodingIdon your type must exactly match the Binary Encoding NodeId reported by the server — use a generic OPC UA client (e.g. UA Expert) to inspect the correct NodeId. - If using source-generated types (Approach 2): ensure the generated
Add...DataTypes()extension method is called onsession.Factory.Builderbefore reading.
Performance Issues
Problem: Type loading takes too long
Solutions:
- Load only required namespaces instead of all types
- Load types once at session start rather than on-demand
- Consider caching type information across sessions if reconnecting frequently
API Reference
ComplexTypeSystem Class
The main class for managing complex types.
Constructors
// Create with session (uses DefaultComplexTypeFactory and session's telemetry)
ComplexTypeSystem(ISession session)
// Create with session and custom telemetry
ComplexTypeSystem(ISession session, ITelemetryContext telemetry)
// Create with session and custom type builder factory
ComplexTypeSystem(ISession session, IComplexTypeFactory complexTypeBuilderFactory)
// Create with session, custom factory and telemetry
ComplexTypeSystem(ISession session, IComplexTypeFactory complexTypeBuilderFactory, ITelemetryContext telemetry)
// Create with resolver and telemetry
ComplexTypeSystem(IComplexTypeResolver complexTypeResolver, ITelemetryContext telemetry)
// Create with resolver, custom factory and telemetry
ComplexTypeSystem(IComplexTypeResolver complexTypeResolver, IComplexTypeFactory complexTypeBuilderFactory, ITelemetryContext telemetry)
// Create with Reflection.Emit type builder (requires OPCFoundation.NetStandard.Opc.Ua.Client.ComplexTypes)
static ComplexTypeSystem ComplexTypeSystem.Create(ISession session, ITelemetryContext telemetry)
static ComplexTypeSystem ComplexTypeSystem.Create(IComplexTypeResolver complexTypeResolver, ITelemetryContext telemetry)
Methods
// Load all custom types from the server
ValueTask<bool> LoadAsync(bool onlyEnumTypes = false, bool throwOnError = false, CancellationToken ct = default)
// Load a specific type with optional subtypes
Task<IType?> LoadTypeAsync(ExpandedNodeId nodeId, bool subTypes = false, bool throwOnError = false, CancellationToken ct = default)
// Load all types from a namespace
Task<bool> LoadNamespaceAsync(string ns, bool throwOnError = false, CancellationToken ct = default)
// Get all dynamically created type names
IReadOnlyList<XmlQualifiedName> GetDefinedTypes()
// Get all loaded data type NodeIds
IEnumerable<ExpandedNodeId> GetDefinedDataTypeIds()
// Get data type definitions for a type
NodeIdDictionary<DataTypeDefinition> GetDataTypeDefinitionsForDataType(ExpandedNodeId dataTypeId)
// Clear the data type cache
void ClearDataTypeCache()
Properties
// Get the loaded data type dictionaries
NodeIdDictionary<DataDictionary> DataTypeSystem { get; }
IStructure Interface
Interface for accessing properties of complex types.
// Access property by zero-based index
Variant this[int index] { get; set; }
// Access property by name
Variant this[string name] { get; set; }
// Get property names
IReadOnlyList<IStructureField> GetFields()
IStructureField Interface
Provides metadata about a property in a complex type.
string Name { get; } // Property name
bool IsOptional { get; } // Whether field is optional
TypeInfo TypeInfo { get; } // Type info of the field
Known Limitations
- OptionSet Support: Concrete Structure-backed sub-types of the abstract
OptionSetDataType (i=12755, e.g.AccessRights,CarExtras) are automatically registered by the defaultComplexTypeSystembuilder asOpc.Ua.Encoders.OptionSetruntime instances, driven by either theEnumDefinitioncarried inDataTypeDefinitionor a fallback synthesized from theOptionSetValuesproperty. The runtime class exposes the two canonicalValue/ValidBitsByteStrings plus bit accessors keyed by field name or bit index. Per Part 3 §8.40 / §3.2.8, the overall ByteString length is fixed by the sub-type's declared bits (exposed asByteLength); setting a bit outside that range throwsArgumentOutOfRangeException. Remaining limitations:- UInteger-backed OptionSet DataTypes (DataTypes deriving from an unsigned integer with
IsOptionSet=true) continue to be represented as their underlying unsigned integer in aVariant— no per-bit metadata is surfaced. - The legacy Reflection.Emit builder in
Opc.Ua.Client.ComplexTypesthrowsNotSupportedExceptionfor OptionSet sub-types; switch to the default builder (new ComplexTypeSystem(session)) for OptionSet support.
- UInteger-backed OptionSet DataTypes (DataTypes deriving from an unsigned integer with
- Legacy Dictionary Support: Some OPC UA 1.03 structured types that cannot be mapped to OPC UA 1.04 definitions are ignored
- Type Modifications: Once loaded, types cannot be dynamically updated during a session. Reconnect to reload modified types.
Additional Resources
- OPC UA Specification Part 3 - Address Space Model
- OPC UA Specification Part 5 - Information Model
- OPC UA Specification Part 6 - Mappings
- OPC Foundation .NET Standard Samples
- NuGet Package: OPCFoundation.NetStandard.Opc.Ua.Client.ComplexTypes
See Also
- Source-Generated Data Types - Auto-generate IEncodeable implementations from annotated POCO classes
- Platform Build Documentation - Information about building and versioning
- Diagnostics Documentation - Logging, telemetry, audit events, server diagnostics, and packet capture
- Console Reference Client - Example client implementation