Dependency Injection
June 20, 2026 · View on GitHub
This document describes the unified Microsoft.Extensions.DependencyInjection
surface for the OPC UA .NET Standard libraries. The surface is rooted in
a single services.AddOpcUa() call that returns an IOpcUaBuilder on
which every feature library hangs its own fluent .AddXxx(...) extension.
The dependency injection surface is consistent across:
- The OPC UA Core stack (
Stack/Opc.Ua.Core) - Application configuration (
Libraries/Opc.Ua.Configuration) - The client (
Libraries/Opc.Ua.Client) - The complex types client (
Libraries/Opc.Ua.Client.ComplexTypes) - Alarms and conditions client (
Libraries/Opc.Ua.Client.Alarms) - The server (
Libraries/Opc.Ua.Server) - The GDS client (
Libraries/Opc.Ua.Gds.Client.Common) - The GDS server (
Libraries/Opc.Ua.Gds.Server.Common) - The LDS server (
Libraries/Opc.Ua.Lds.Server) - The WoT Connectivity server (
Libraries/Opc.Ua.WotCon.Server) - The WoT Connectivity client (
Libraries/Opc.Ua.WotCon.Client)
PubSub is not part of the dependency injection surface.
The non-dependency-injection public constructors and factories of every library
(new ApplicationInstance(telemetry), new StandardServer(telemetry),
new LdsServer(telemetry), new ManagedSession(...) etc.) remain
unchanged. Use dependency injection when you want the .NET Generic Host to own application
lifetime, logging, and configuration; use the manual constructors when
you need finer control.
Quick reference
| Feature library | Method on IOpcUaBuilder | Returns | Hosted? | Section |
|---|---|---|---|---|
Opc.Ua.Core (root) | services.AddOpcUa() | IOpcUaBuilder | — | — |
Opc.Ua.Configuration | builder.AddApplicationInstance() | IOpcUaBuilder | — | — |
Opc.Ua.Client | builder.AddClient(opt => …) | IOpcUaClientBuilder | — | OpcUa:Client |
Opc.Ua.Client.ComplexTypes | builder.AddComplexTypes() | IOpcUaBuilder | — | — |
Opc.Ua.Client.Alarms (within Opc.Ua.Client) | builder.AddAlarms() | IOpcUaBuilder | — | — |
Opc.Ua.Server | builder.AddServer(opt => …) | IOpcUaServerBuilder | yes | OpcUa:Server |
Opc.Ua.Gds.Client.Common | builder.AddGdsClient(opt => …) | IGdsClientBuilder | — | OpcUa:Gds:Client |
Opc.Ua.Gds.Server.Common | builder.AddGdsServer(opt => …) | IGdsServerBuilder | yes | OpcUa:Gds:Server |
Opc.Ua.Lds.Server | builder.AddLdsServer(opt => …) | ILdsServerBuilder | yes | OpcUa:Lds |
Opc.Ua.WotCon.Server | builder.AddWotConServer(opt => …) | IWotConServerBuilder | yes (via AddServer) | OpcUa:WotCon:Server |
Opc.Ua.WotCon.Client | builder.AddWotConClient(opt => …) | IOpcUaBuilder | — | OpcUa:WotCon:Client |
Identity-provider extensions hang off IOpcUaServerBuilder,
IOpcUaClientBuilder, and IGdsServerBuilder:
| Extension | Builder | Section |
|---|---|---|
AddDefaultIdentityAuthenticators(...) | server, gds | OpcUa:Server:Identity:Defaults |
AddIdentityAuthenticator<T>() | server, gds | — |
AddIdentityAugmenter<T>() | server, gds | — |
AddGdsApplicationSelfAdminProvider() | gds | — |
AddJwtIssuer(...) | server, gds | OpcUa:Server:Identity:Issuers[] |
WithAuthorizationService(...) / <TIssuer>() | gds | — |
WithKeyCredentialPush(...) | server | — |
ConfigureRoles(...) | server, gds | OpcUa:Server:Roles |
AddIdentityProvider(...) / <T>() | client | OpcUa:Client:Identity |
AddAccessTokenProvider(...) / <T>() | client | — |
See the Identity (server), Identity (client), and Identity (GDS server) sections below and the full Identity Providers guide.
Server features marked Hosted? = yes register an IHostedService so
the .NET Generic Host (Host.CreateApplicationBuilder(args)) owns their
lifetime, certificate setup, and Ctrl+C / SIGTERM handling.
Root: services.AddOpcUa()
services.AddOpcUa() is the only entry point. It does two things:
- Registers
ITelemetryContextas aServiceProviderTelemetryContextsingleton (viaTryAddSingleton, so any prior user registration wins). - Returns an
IOpcUaBuilderwhose.Servicesproperty exposes the underlyingIServiceCollectionfor advanced scenarios.
using Microsoft.Extensions.DependencyInjection;
services.AddOpcUa();
AddOpcUa is idempotent. The returned IOpcUaBuilder derives from the
older IDependencyInjectionBuilder so existing
builder.AddLogging() / builder.AddMetrics() extensions still compile
unchanged. New AddLogging(this IOpcUaBuilder) /
AddMetrics(this IOpcUaBuilder) overloads return IOpcUaBuilder for
fluent chaining into feature methods:
services.AddOpcUa()
.AddLogging(b => b.AddConsole())
.AddMetrics()
.AddServer(o => /* … */)
.AddNodeManager<MyNodeManagerFactory>();
Options binding
Every feature .AddXxx(...) has three overloads:
// 1. Action — AOT-safe, recommended for all consumers.
builder.AddServer(o => { o.ApplicationName = "MyServer"; /* … */ });
// 2. IConfiguration — bind from a configuration root's default section.
builder.AddServer(builder.Configuration);
// 3. IConfigurationSection — bind from an explicit section.
builder.AddServer(builder.Configuration.GetSection("MyApp:Server"));
The default section names are:
| Feature | Section |
|---|---|
| Server | OpcUa:Server |
| Client | OpcUa:Client |
| GDS Client | OpcUa:Gds:Client |
| GDS Server | OpcUa:Gds:Server |
| LDS Server | OpcUa:Lds |
| WoT Connectivity Server | OpcUa:WotCon:Server |
| WoT Connectivity Client | OpcUa:WotCon:Client |
The IConfiguration / IConfigurationSection overloads are
AOT-safe on every library. Each dependency-injection-emitting library opts into the
.NET 8+ Configuration Binding Source Generator
(<EnableConfigurationBindingGenerator>true</EnableConfigurationBindingGenerator>),
which replaces the reflection-based binder with statically-generated
C# 12 interceptors.
The Tests/Opc.Ua.Aot.Tests project verifies that dotnet publish
under PublishAot=true produces zero IL2026 / IL3050 warnings
from the dependency injection surface.
Server feature
builder.AddServer(o => …) registers an OPC UA StandardServer as an
IHostedService via a private OpcUaServerHostedService. Endpoints,
PKI root, security policies, and the application instance certificate
are all set up on host startup.
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;
HostApplicationBuilder builder = Host.CreateApplicationBuilder(args);
builder.Logging.AddConsole();
builder.Services
.AddOpcUa()
.AddServer(o =>
{
o.ApplicationName = "MyServer";
o.ApplicationUri = "urn:localhost:MyOrg:MyServer";
o.ProductUri = "uri:myorg:myserver";
o.AutoAcceptUntrustedCertificates = true;
o.EndpointUrls.Add("opc.tcp://localhost:51210/MyServer");
})
.AddNodeManager<MyNodeManagerFactory>() // IAsyncNodeManagerFactory
.AddSyncNodeManager<LegacyNodeManagerFactory>(); // INodeManagerFactory
await builder.Build().RunAsync();
.AddNodeManager<T>() and .AddSyncNodeManager<T>() register the
factory under an OpcUaServerNodeManagerRegistration wrapper that is
scoped to the regular server feature. Node managers registered this
way are not visible to the GDS / LDS hosted services running in the
same container. See Combined hosts below.
.AddServer(...) throws InvalidOperationException on a second call:
at most one regular server may be registered per service collection.
For advanced configuration (custom security policies, custom security
stores), set OpcUaServerOptions.ConfigureBuilder — it receives the
underlying IApplicationConfigurationBuilderServerSelected between the
default policy/quota steps and CreateAsync.
First-class server options
In addition to the basic application-identity / endpoint / PKI knobs
covered above, OpcUaServerOptions exposes the following first-class
properties (bindable from IConfiguration or set via the
Action<OpcUaServerOptions> overload). Anything not listed here remains
reachable through ConfigureBuilder.
| Property | Underlying builder call | Purpose |
|---|---|---|
IncludeEccPolicies | AddEccSignAndEncryptPolicies() | Add ECC sign-and-encrypt security policies. Off by default. |
UserTokenPolicies | AddUserTokenPolicy(UserTokenType) | List of user-token policies advertised on every endpoint. Defaults to Anonymous when empty. |
MaxMessageSize | SetMaxMessageSize(int) | Transport quota in bytes; null keeps the stack default. |
OperationTimeoutMs | SetOperationTimeout(int) | Transport operation timeout in ms; null keeps the stack default. |
RejectSHA1Certificates | SetRejectSHA1SignedCertificates(bool) | Security hardening — defaults to true. |
MinCertificateKeySize | SetMinimumCertificateKeySize(ushort) | Security hardening — defaults to 2048. Set to 0 to keep the stack default. |
RegistrationEndpointUrl | SetRegistrationEndpoint(EndpointDescription) | LDS/GDS endpoint URL the server registers itself with on startup. |
ReverseConnect | SetReverseConnect(ReverseConnectServerConfiguration) | Server-side reverse-connect clients (see below). |
OperationLimits | SetOperationLimits(OperationLimits) | Per-service node limits (max nodes per read/write/browse/...). |
Server-side reverse connect
A server can dial back to clients via reverse-hello using
OpcUaServerOptions.ReverseConnect. The data binds directly from
OpcUa:Server:ReverseConnect:
{
"OpcUa": {
"Server": {
"ReverseConnect": {
"ConnectIntervalMs": 15000,
"ConnectTimeoutMs": 30000,
"RejectTimeoutMs": 60000,
"Clients": [
{
"EndpointUrl": "opc.tcp://client.example.com:4841",
"Timeout": 30000,
"MaxSessionCount": 1,
"Enabled": true
}
]
}
}
}
}
Equivalent code-only configuration:
services.AddOpcUa().AddServer(o =>
{
o.ReverseConnect = new ServerReverseConnectOptions();
o.ReverseConnect.Clients.Add(new ServerReverseConnectClientOptions
{
EndpointUrl = "opc.tcp://client.example.com:4841",
Enabled = true
});
});
Operation limits
services.AddOpcUa().AddServer(o =>
{
o.OperationLimits = new OperationLimitsOptions
{
MaxNodesPerRead = 1000,
MaxNodesPerWrite = 1000,
MaxNodesPerBrowse = 1000,
MaxMonitoredItemsPerCall = 5000
};
});
Bindable from OpcUa:Server:OperationLimits. Any value left at zero is
treated as "unlimited" by the OPC UA server stack.
User token policies
services.AddOpcUa().AddServer(o =>
{
o.UserTokenPolicies.Add(new OpcUaUserTokenPolicy
{
TokenType = UserTokenType.UserName
});
o.UserTokenPolicies.Add(new OpcUaUserTokenPolicy
{
TokenType = UserTokenType.Certificate
});
});
Bindable from OpcUa:Server:UserTokenPolicies. When the list is empty
the hosted service falls back to a single Anonymous policy.
Identity (server)
Full identity surface is documented in Identity Providers. This section covers only the dependency injection bindings.
builder.AddServer(...) exposes identity-related extension methods on
IOpcUaServerBuilder:
| Extension | Purpose |
|---|---|
ConfigureRoles(Action<RoleConfigurationOptions>) / (IConfiguration) | Registers RoleConfigurationOptions for future role-related tuning. The DTO currently has no configurable members; the extension exists as a stable expansion point. |
AddIdentityAuthenticator<TAuth>() | Registers a single custom IUserTokenAuthenticator implementation. The hosted service adds it to IServerInternal.IdentityRegistry on startup. |
AddIdentityAugmenter<TAugmenter>() | Registers a single custom IIdentityAugmenter implementation. The hosted service runs it after accepted authentications. |
AddDefaultIdentityAuthenticators(Action<DefaultAuthenticatorOptions>) / (IConfiguration) | Registers the four in-box authenticators (Anonymous, UserNamePassword, X509, Jwt) with toggles per type plus the JWT audience / clock-skew settings. |
AddJwtIssuer(Action<JwtIssuerOptions>) / (IConfiguration) | Registers a trusted JWT issuer. Multiple calls coexist; each contributes a StaticIssuerKeyResolver and / or JwksIssuerKeyResolver keyed by IssuerUri. |
WithKeyCredentialPush(Action<KeyCredentialPushOptions>?) | Enables the Part 12 §8 resource-server Push binding and registers an IKeyCredentialStore if none is supplied. |
Configuration binding under OpcUa:Server:Identity:
{
"OpcUa": {
"Server": {
"Identity": {
"Defaults": {
"EnableAnonymous": true,
"EnableUserNamePassword": true,
"EnableX509": true,
"EnableJwt": true,
"ExpectedAudience": "urn:my-server",
"ClockSkewTolerance": "00:01:00",
"UserCertificateTrustList": "Users"
},
"Issuers": [
{
"IssuerUri": "https://login.microsoftonline.com/{tenant}/v2.0",
"JwksUri": "https://login.microsoftonline.com/{tenant}/discovery/v2.0/keys",
"Algorithms": [ "RS256" ]
}
]
}
}
}
}
The AddServer(IConfiguration) overload walks this layout and (in
addition to binding OpcUaServerOptions.Identity) registers each
Issuers[] entry through AddJwtIssuer(...), binds the Roles
sub-section into RoleConfigurationOptions, and enables the bridge
that creates the four default authenticators from the bound
Identity:Defaults flags. The AddServer(Action<OpcUaServerOptions>)
code-only overload does NOT auto-wire any of these — call the fluent
extensions explicitly when configuring from code.
The IUserDatabase / IUserManagement services needed by the
UserNamePassword authenticator must be registered separately — the
hosted service skips that authenticator if neither is in dependency injection. See
Role-Based Security for the role-mapping
layer.
Client feature
builder.AddClient(opt => …) registers a lazy ManagedSession factory
delegate. The factory caches the connected session — subsequent awaits
return the same instance.
using Microsoft.Extensions.DependencyInjection;
using Opc.Ua.Client;
services
.AddOpcUa()
.AddClient(opt =>
{
opt.Configuration = applicationConfiguration;
opt.Session = new ManagedSessionOptions
{
Endpoint = endpoint,
ReconnectPolicy = new ReconnectPolicyOptions
{
Strategy = BackoffStrategy.Exponential
}
};
});
// Resolve and connect on first use:
var sessionFactory = sp.GetRequiredService<Func<CancellationToken, Task<ManagedSession>>>();
ManagedSession session = await sessionFactory(ct);
AddClient also registers ITelemetryContext, ISessionFactory (a
DefaultSessionFactory configured with the V2 subscription engine),
ManagedSessionFactory, and the top-level OpcUaClientOptions.
Identity (client)
Full identity surface is documented in Identity Providers. This section covers only the dependency injection bindings.
builder.AddClient(...) exposes:
| Extension | Purpose |
|---|---|
AddIdentityProvider<TProvider>() | Registers a custom IClientIdentityProvider type. All registered providers are composed into a single resolver by the session builder. |
AddIdentityProvider(Action<CompositeClientIdentityProviderBuilder>) | Fluent composite-builder with shortcuts: .AddAnonymous(), .AddUserName(...), .AddX509(...), .AddIssuedToken(...). |
AddIdentityProvider(IConfiguration section) | Binds OpcUaClientIdentityOptions and rolls a CompositeClientIdentityProvider from the bound entries. |
AddAccessTokenProvider<T>() / (instance) / (factory) | Registers an IAccessTokenProvider keyed by AuthorityUri. Issued-token identity providers select the matching provider at runtime via the URI. |
Configuration binding under OpcUa:Client:Identity:
{
"OpcUa": {
"Client": {
"Identity": {
"EnableAnonymous": true,
"UserName": {
"UserName": "alice",
"SecretName": "alice-password",
"SecretStoreType": "InMemory"
},
"X509": {
"StoreType": "X509Store",
"StorePath": "CurrentUser/My",
"SubjectName": "CN=Alice"
},
"IssuedToken": {
"ProfileUri": "http://opcfoundation.org/UA/UserToken#JWT",
"AuthorityUri": "https://issuer.example"
},
"Order": [ "IssuedToken", "UserName", "X509", "Anonymous" ]
}
}
}
}
When Order is non-empty the composite presents providers in that
preference; otherwise registration order is preserved. The IssuedToken
entry resolves the registered IAccessTokenProvider whose
AuthorityUri matches.
How the section is consumed.
AddClient(IConfiguration)bindsOpcUa:Client:IdentityintoOpcUaClientOptions.Identity. At session-factory resolution time, if noIClientIdentityProviderservice is explicitly registered, the factory builds a composite from the bound options provided at least one non-default field is set (any ofUserName,X509,IssuedToken,Order, orEnableAnonymous = false). Setting onlyEnableAnonymous = true(the default) does NOT register any provider — call.AddIdentityProvider(configuration.GetSection("OpcUa:Client:Identity"))if you want an explicit eager registration regardless of the option values, or use theAddIdentityProvider(Action<CompositeClientIdentityProviderBuilder>)overload for code-only configuration. The composite-builder shortcuts (.AddUserName(configure, registry),.AddX509(configure, provider, passwords), and.AddIssuedToken(configure, provider)) each take the supporting service (anISecretRegistry,ICertificateProvider+ICertificatePasswordProvider, orIAccessTokenProvider) as a second positional argument.
ManagedSessionOptions.Identity (eager) is [Obsolete] — set
ManagedSessionOptions.IdentityProvider instead for lazy / refreshable
identities. The ManagedSession proactive-refresh scheduler keys off
IClientIdentityProvider.ExpiresAt.
Complex types
services.AddOpcUa().AddClient(/* … */).AddComplexTypes();
Registers a ComplexTypeSystemFactory (transient) that can be resolved
and used to build a ComplexTypeSystem for a connected session:
var factory = sp.GetRequiredService<ComplexTypeSystemFactory>();
ComplexTypeSystem cts = factory.Create(session);
await cts.LoadAsync(...);
Alarms and conditions
services.AddOpcUa().AddClient(/* … */).AddAlarms();
Registers a singleton AlarmClientFactory so dependency-injection-hosted client
applications can obtain a Part 9 AlarmClient per connected session
without new-ing one manually. Source-generated event records
(ConditionTypeRecord, AlarmConditionTypeRecord,
DialogConditionTypeRecord, all subtypes including vendor extensions)
and the streaming extensions (SubscribeAlarmsAsync,
SubscribeConditionsAsync, SubscribeDialogsAsync) require no
registration — they compose naturally with the
ManagedSession.DefaultStreaming surface that AddClient(...)
already wires up.
ManagedSession session = await sessionFactory(ct);
var factory = sp.GetRequiredService<AlarmClientFactory>();
AlarmClient alarms = factory.Create(session);
// Acknowledge an alarm:
await alarms.AcknowledgeAsync(conditionId, eventId,
new LocalizedText("en", "Acknowledged"), ct);
// Stream typed records via the session's default streaming subscription:
await foreach (ConditionTypeRecord record in session.DefaultStreaming
.SubscribeAlarmsAsync(ObjectIds.Server, ct: ct))
{
/* … */
}
The non-dependency-injection path (session.GetAlarmClient() extension and the public
AlarmClient constructor) remains available for callers that do not
use the dependency injection infrastructure. See
Alarms and Conditions for the full
developer guide.
Client-side reverse connect
When OpcUaClientOptions.ReverseConnect is set, the dependency injection container
registers a singleton ReverseConnectManager that opens the configured
listener endpoints on first resolution. Inbound reverse-hello messages
are surfaced via
ReverseConnectManager.WaitForConnectionAsync(endpointUrl, serverUri, ct),
and the values are also mirrored into
ApplicationConfiguration.ClientConfiguration.ReverseConnect.
{
"OpcUa": {
"Client": {
"ReverseConnect": {
"HoldTimeMs": 15000,
"WaitTimeoutMs": 20000,
"ClientEndpointUrls": [
"opc.tcp://0.0.0.0:4841"
]
}
}
}
}
Equivalent code-only:
services.AddOpcUa().AddClient(opt =>
{
opt.Configuration = applicationConfiguration;
opt.ReverseConnect = new ClientReverseConnectOptions();
opt.ReverseConnect.ClientEndpointUrls.Add("opc.tcp://0.0.0.0:4841");
});
// Resolve the manager and await an inbound reverse-hello connection:
var reverseConnect = sp.GetRequiredService<ReverseConnectManager>();
ITransportWaitingConnection connection =
await reverseConnect.WaitForConnectionAsync(endpointUrl, serverUri: null, ct);
// pass `connection` to Session.Create / DefaultSessionFactory.RecreateAsync
Note: the Func<CancellationToken, Task<ManagedSession>> delegate
registered by AddClient does not automatically consume the
reverse-connect manager for initial connection — it still dials the
configured endpoint outbound. Use the ReverseConnectManager directly
when the server initiates the session. The
ReverseConnect configuration is also consumed by Session /
SessionReconnectHandler for reconnect-via-reverse-hello scenarios.
Channel manager
AddClient(...) automatically registers an IClientChannelManager
singleton (ClientChannelManager implementation) built from
OpcUaClientOptions.Configuration. The
Func<CancellationToken, Task<ManagedSession>> delegate wires this
manager into every new ManagedSession it creates, so multiple
ManagedSession instances resolved from the same DI container will
share underlying transport channels per
ConfiguredEndpoint. Reconnect is coalesced and notified to attached
sessions transparently. On .NET 8 and later, HTTPS transport channels
also get the named IHttpClientFactory client (Opc.Ua.Client) and its
standard resilience handler through AddClient(...). See
Sessions and reconnect § 4
for details on identity, retry policy, HTTPS request resilience, and the
shared retry budget between channel-manager reconnect and
ManagedSession's outer IReconnectPolicy.
Security trade-off — OPC UA TLS validation vs the DI HttpClient pipeline. When an OPC UA
CertificateValidatoris configured for a channel (the normal case for any non-SecurityPolicies.NoneHTTPS profile),HttpsTransportChannelalways takes the direct construction path so the OPC UA trust list, OPC UA mutual TLS, the redirect lock, and the message-size quotas are guaranteed. The namedOpc.Ua.ClientHttpClient pipeline (Polly resilience handler, etc.) is not applied to those channels and the channel emits a one-timeLogWarningto make the bypass visible. To use both the Polly resilience handler AND OPC UA cert validation on the same channel, register the named client with aConfigurePrimaryHttpMessageHandlerthat wires the OPC UAServerCertificateCustomValidationCallbackand the OPC UA application instance certificate yourself. See Sessions and reconnect § HTTPS factory + OPC UA cert validation: secure-by-default fallback.
To override the default channel manager (e.g. to inject a custom
IChannelReconnectPolicy):
services.AddOpcUa().AddClient(opt => { ... });
services.Replace(ServiceDescriptor.Singleton<IClientChannelManager>(sp =>
new ClientChannelManager(
sp.GetRequiredService<OpcUaClientOptions>().Configuration!,
sp.GetRequiredService<ITelemetryContext>(),
reconnectPolicy: new ExponentialBackoffChannelReconnectPolicy
{
MaxAttempts = 5
})));
Application instance (advanced)
services.AddOpcUa().AddApplicationInstance();
Registers an IApplicationInstanceFactory singleton. Hosted server
services (regular, GDS, LDS, WotCon) call the factory to create their
own per-host IApplicationInstance instead of sharing a process-wide
singleton — this avoids configuration overwrite and certificate-lifecycle
conflicts when multiple servers coexist in one process. AddServer /
AddGdsServer / AddLdsServer register the factory automatically, so
calling AddApplicationInstance() explicitly is only needed when you
want the factory available outside the hosted-server context.
GDS Client
services.AddOpcUa().AddGdsClient(opt =>
{
opt.SessionTimeout = TimeSpan.FromMinutes(2);
opt.MaxConnectAttempts = 10;
});
Requires an ApplicationConfiguration registered in the container.
Resolves GlobalDiscoveryServerClient and ServerPushConfigurationClient
singletons.
GDS Server
services
.AddOpcUa()
.AddGdsServer(o =>
{
o.ApplicationName = "MyGds";
o.ApplicationUri = "urn:localhost:MyOrg:MyGds";
o.ProductUri = "uri:myorg:mygds";
o.AutoAcceptUntrustedCertificates = true;
o.EndpointUrls.Add("opc.tcp://localhost:58810/GlobalDiscoveryServer");
o.AuthoritiesStorePath = "%LocalApplicationData%/OPC Foundation/pki/CA";
})
.AddApplicationsDatabase<MyApplicationsDatabase>()
.AddCertificateGroup<MyCertificateGroup>()
.AddCertificateRequest<MyCertificateRequest>()
.AddUserDatabase<MyUserDatabase>();
// Optionally:
// .WithAuthorizationService<MyTokenIssuer>(o => o.IssuerUri = "urn:my-gds")
// .AddAccessTokenProvider<MyAccessTokenProvider>()
// .AddKeyCredentialRequestStore<MyKeyCredentialRequestStore>()
// .AddConfigurationDataStore<MyConfigurationDataStore>();
Throws on a second .AddGdsServer(...). Auto-registers an internal
GdsHostedServer : GlobalDiscoverySampleServer and its
ApplicationsNodeManager. The pluggable services (IApplicationsDatabase,
ICertificateGroup, ICertificateRequest, etc.) are resolved from the
container at startup.
Identity (GDS server)
IGdsServerBuilder forwards every identity-related extension to the
underlying server builder so a GDS host configures identity the same
way as a regular server:
services
.AddOpcUa()
.AddGdsServer(opt => opt.ApplicationName = "MyGds")
.AddDefaultIdentityAuthenticators(opt =>
{
opt.EnableAnonymous = false;
opt.EnableUserNamePassword = true;
})
.AddJwtIssuer(opt =>
{
opt.IssuerUri = "https://login.microsoftonline.com/{tenant}/v2.0";
opt.JwksUri = "https://login.microsoftonline.com/{tenant}/discovery/v2.0/keys";
});
Action<> and IConfiguration overloads are available for
ConfigureRoles, AddDefaultIdentityAuthenticators, and AddJwtIssuer.
AddIdentityAugmenter<T>() registers post-authentication identity
augmenters; AddGdsApplicationSelfAdminProvider() registers the built-in
OPC 10000-12 §7.2 SelfAdmin provider and is also wired by the GDS
AddDefaultIdentityAuthenticators(...) helper.
WithAuthorizationService(...) enables the default CertificateJwtIssuer;
WithAuthorizationService<TIssuer>(...) registers a custom ITokenIssuer
for cloud KMS, HSM, or external token-service signing.
GdsServerHostedService consumes these forwarded registrations during
startup and adds them to the same identity registry used by regular OPC
UA hosted servers.
See Identity Providers for the full reference.
LDS Server
services
.AddOpcUa()
.AddLdsServer(o =>
{
o.ApplicationName = "MyLds";
o.ApplicationUri = "urn:localhost:MyOrg:MyLds";
o.ProductUri = "uri:myorg:mylds";
o.EndpointUrls.Add("opc.tcp://localhost:4840/UADiscovery");
o.EnableMulticast = true; // LDS-ME multicast advertisement
o.MulticastLoopbackOnly = false; // set true for in-process tests
o.ServerCapabilities.Add("DA"); // extra capabilities (LDS / LDS-ME are always implicit)
});
Throws on a second .AddLdsServer(...). The server is advertised as
ApplicationType.DiscoveryServer (the LDS class promotes the type after
ApplicationConfiguration.CreateAsync).
MulticastLoopbackOnly restricts the mDNS announcer to the loopback NIC
— intended for in-process tests where LDS-ME traffic must stay local.
ServerCapabilities is additive — LDS is always included, plus
LDS-ME when EnableMulticast is true.
WoT Connectivity Server
The WoT Connectivity server lives inside a regular StandardServer, so
AddWotConServer must be combined with AddServer:
services
.AddOpcUa()
.AddServer(o => /* regular server options */)
.AddNodeManager<MyNodeManagerFactory>() // your domain node managers
.Services
.AddOpcUa() // returns the same builder
.AddWotConServer(o =>
{
o.AssetNamespaceUri = "http://myorg/UA/WoT-Con/Assets/";
o.ThingDescriptionStorageFolder = "tds";
})
.AddAssetProvider<MyAssetProviderFactory>()
.AddDiscoveryProvider<MyDiscoveryProvider>();
AddWotConServer registers the WoT WotConnectivityNodeManagerFactory
as an OpcUaServerNodeManagerRegistration so it attaches to the regular
server feature. IWotAssetProviderFactory and IWotAssetDiscoveryProvider
services registered in dependency injection are picked up automatically.
WoT Connectivity Client
services.AddOpcUa().AddClient(/* … */).AddWotConClient();
// Resolve and connect on first use:
var wotClient = sp.GetRequiredService<Func<CancellationToken, Task<WotConnectivityClient>>>();
WotConnectivityClient client = await wotClient(ct);
The WoT client reuses the connected ManagedSession registered by
AddClient(...).
Combined hosts
You can run a regular server, a GDS server, and an LDS server inside a single Generic Host. Each owns its own:
OpcUaServerOptions/GdsServerOptions/LdsServerOptions- PKI root and certificate
- Endpoint URLs and ports
IApplicationInstance(created from the sharedIApplicationInstanceFactory)BackgroundService(the registrations don't collide)
builder.Services
.AddOpcUa()
.AddServer(o => /* regular server on 51210 */)
.AddNodeManager<MyNodeManagerFactory>()
.Services
.AddOpcUa()
.AddLdsServer(o => /* LDS on 4840 */)
.Services
.AddOpcUa()
.AddGdsServer(o => /* GDS on 58810 */)
.AddApplicationsDatabase<MyApplicationsDatabase>()
.AddCertificateGroup<MyCertificateGroup>()
.AddCertificateRequest<MyCertificateRequest>()
.AddUserDatabase<MyUserDatabase>();
Node managers registered under a feature wrapper (e.g.
OpcUaServerNodeManagerRegistration for the regular server) are
isolated: the GDS / LDS hosted services do not see them.
Native AOT
Every .AddXxx(...) overload — both the Action<TOptions> shape and
the IConfiguration / IConfigurationSection shapes — is AOT-safe.
The .NET 8+ Configuration Binding Source Generator is enabled on every
library that performs configuration binding
(<EnableConfigurationBindingGenerator>true</EnableConfigurationBindingGenerator>),
so the reflection-based binder is replaced by statically-generated
C# 12 interceptors
at compile time.
services.AddOpcUa()
.AddServer(builder.Configuration.GetSection("OpcUa:Server")) // bind from appsettings.json — AOT-safe
.AddNodeManager<MyAotNodeManagerFactory>();
Action<TOptions> (code-only) and IConfiguration overloads can be
mixed freely; consumers no longer need to choose between them for AOT
compatibility.
Notes:
- The source generator targets net8.0+. On older TFMs (net48 / netstandard2.0 / netstandard2.1) the generator is a no-op and the reflection-based binder is used — those TFMs don't support PublishAot anyway.
- Options properties whose type is an interface or a non-default-
constructible class (e.g.
ApplicationConfiguration,IUserIdentity,ISubscriptionEngineFactory) are silently skipped by the generator with an informationalSYSLIB1100/SYSLIB1101diagnostic. Those properties are runtime-only — set them in code, not inappsettings.json. The affected libraries suppress those diagnostics in their csproj. - The
Tests/Opc.Ua.Aot.Testsproject verifies the end-to-end AOT path: build + AOT publish produce zeroIL2026/IL3050warnings from any dependency injection extension.
For AOT consumers, configure options through code or configuration — both are supported:
services.AddOpcUa()
.AddServer(o =>
{
o.ApplicationName = "AotServer";
o.ApplicationUri = "urn:host:AotServer";
o.EndpointUrls.Add("opc.tcp://localhost:51210/AotServer");
o.AutoAcceptUntrustedCertificates = true;
})
.AddNodeManager<MyAotNodeManagerFactory>();
Telemetry
AddOpcUa() registers an ITelemetryContext that resolves the host's
ILoggerFactory on first use. To override:
services.AddSingleton<ITelemetryContext>(myCustomTelemetry);
services.AddOpcUa(); // TryAddSingleton — custom one wins
To configure logging fluently:
services.AddOpcUa()
.AddLogging(b => b.AddConsole().SetMinimumLevel(LogLevel.Information))
.AddServer(/* … */);
See also
- Sessions —
ManagedSession, reconnect, subscription engines. - Source Generated NodeManagers —
IAsyncNodeManagerFactoryfrom a model design XML. - Native AOT — AOT testing setup.
- GDS Developer Guide — GDS service interfaces and provider patterns.
- WoT Connectivity — OPC 10100-1 information model.
- Diagnostics —
ITelemetryContextend-to-end.