This is a small tutorial for using the OPC UA server library. It will assume you are familiar with Rust and tools such as cargo
.
- A small overview of OPC UA is here.
- Rust OPC UA's compatibility with the standard is described here.
The Rust OPC UA server API supports all of the OPC UA embedded profile services and a few of the standard profile services.
These are implemented for you so generally once you create a server configuration, set up an address space and register some callbacks you are ready to run a server.
The server API is designed to be extremely flexible, allowing you to create servers that do pretty much anything. On top of this are some defaults and tools that make assumptions about the way the server is used, which can be used to make it easier to develop servers.
- Create or load a configuration that defines the TCP address / port server runs on, the endpoints it supports, user identities etc.
- Create a server from the configuration
- Populate additional nodes into the address space and callbacks / timers that change state.
- Run the server
- Server runs forever, listening for connections.
We're going to start with a blank project.
cargo init --bin test-server
To use the server crate we need to add a dependency to the Cargo.toml
.
[dependencies]
opcua = { "0.15", features = ["server", "console-logging"] }
The server can be configured in a number of ways:
- A
ServerBuilder
is the easiest way to build a server programmatically. - A configuration file described in yaml or some other
serde
supported format that you read from.
A ServerBuilder
allows you to programmatically construct a Server
.
use std::sync::Arc;
use opcua::server::address_space::Variable;
use opcua::server::node_manager::memory::{
simple_node_manager, InMemoryNodeManager, NamespaceMetadata, SimpleNodeManager,
SimpleNodeManagerImpl,
};
use opcua::server::{ServerBuilder, SubscriptionCache};
use opcua::types::{BuildInfo, DataValue, DateTime, NodeId, UAString};
#[tokio::main]
async fn main() {
// First, for convenience, we enable console logging based on the
// `RUST_OPCUA_LOG` environment variable. You could enable this using
// any other system to interact with the `log` framework if you prefer.
opcua::console_logging::init();
// Construct the server.
let (server, handle) = ServerBuilder::new()
.application_name("Server Name")
.application_uri("urn:server_uri")
.discovery_urls(vec![endpoint_url(port_offset)])
.create_sample_keypair(true)
.pki_dir("./pki-server")
.discovery_server_url(None)
.host(hostname())
.port(1234)
.add_user_token(
sample_user_id,
ServerUserToken::new_user_pass("sample", "sample1"),
)
.add_endpoint(
"none",
(
endpoint_path,
SecurityPolicy::None,
MessageSecurityMode::None,
&user_token_ids as &[&str],
),
)
.add_endpoint(
"basic128rsa15_sign",
(
endpoint_path,
SecurityPolicy::Basic128Rsa15,
MessageSecurityMode::Sign,
&user_token_ids as &[&str],
),
)
// Add a custom node manager. The namespace index is populated after the
// server has been built.
.with_node_manager(simple_node_manager(
NamespaceMetadata {
namespace_uri: "urn:SimpleServer".to_owned(),
..Default::default()
},
"my-namespace",
))
.trust_client_certs(true)
.build().unwrap();
// Get the node manager from the server.
let node_manager = handle
.node_managers()
.get_of_type::<SimpleNodeManager>()
.unwrap();
let namespace = handle.get_namespace_index("urn:SimpleServer").unwrap();
// Add initial nodes here...
// Run the server.
server.run().await.unwrap();
}
If you prefer to construct your server from a configuration that you read from a file you can do that instead.
fn main() {
// The namespace should be 2 here, since there are two default namespaces, making
// this the third.
let ns = 2;
let node_manager = Arc::new(SimpleNodeManager::new_simple(
NamespaceMetadata {
namespace_index: ns,
namespace_uri: "urn:SimpleServer".to_owned(),
..Default::default()
},
"simple",
));
let (server, handle) = ServerBuilder::new()
.with_config_from("../server.conf")
.with_node_manager(simple_node_manager(
NamespaceMetadata {
namespace_uri: "urn:SimpleServer".to_owned(),
..Default::default()
},
"my-namespace",
))
.build()
.unwrap();
//...
}
Alternatively, let's say you use a configuration file, but how do you create it when one isn't there? Well your code logic could test if the file can load, and if it doesn't, could create the default one with a ServerBuilder
.
#[tokio::main]
async fn main() {
let server_config_path = "./myserver.conf";
let server_config = if let Ok(server_config) = ServerConfig::load(&PathBuf::from(server_config_path)) {
server_config
}
else {
let server_config = ServerBuilder::new()
.application_name("Server Name")
.application_uri("urn:server_uri")
//... Lines deleted
.config();
// Now we save it so its there next time
server_config.save(server_config_path);
server_config
}
let mut server = Server::new(server_config);
}
The default TCP config uses an address / port of localhost
and 4855
. If you intend for your server to be remotely accessible then explicitly set the address to the assigned IP address or resolvable hostname for the network adapter your server will listen on.
Also ensure that your machine has a firewall rule to allow through the port number you use.
The server configuration determines what encryption it uses on its endpoints, and also what user identity tokens it accepts.
The client and server can communicate over an insecure or a secure channel.
- An insecure channel is plaintext and is not encrypted in any way. This might be fine where trust is implicit and controlled between the client and the server, e.g. when they reside on a private network, or even the same device.
- A secure channel. The client presents a certificate to the server, the server presents a certificate to the client. Each must trust the other, at which point the session proceeds over an encrypted channel.
Once the client establishes a session with the server, the next thing it will do is present its identity for activating the session. The identity is the user's credentials which can be anonymous, user / password or X509 identity token.
Your server has an address space that contains the default OPC UA node set. The default node set describes all the standard types, server diagnostics variables and more besides.
To this you may wish to add your own objects and variables. To make this easy, you can create new nodes with a builder, e.g:
#[tokio::main]
async fn main() {
//... after server is set up
let address_space = node_manager.address_space();
{
let mut address_space = address_space.write();
// This is a convenience helper
let folder_id = NodeId::new(2, "Variables");
address_space.add_folder(&folder_id, "Variables", "Variables", &NodeId::objects_folder_id());
// Build a variable
let node_id = NodeId::new(2, "MyVar");
VariableBuilder::new(&node_id, "MyVar", "MyVar")
.organized_by(&folder_id)
.value(0u8)
.insert(&mut address_space);
}
// Make sure to not keep the address space locked, or nothing will be able to
// read from the server.
//....
}
The builder pattern allows you to set each property of your node and common relationships to other nodes before inserting it into the address space.
Clients of servers will typically read values of variables, and may do so from a subscription. The server will, by default, just get the value from the node in the address space, but there are a few ways to dynamically read values, detailed below.
In addition you may also register a write callback which is called whenever a client attempts to write a value to the variable. Your callback could ignore the change, clamp it to some range or call the physical device with the change.
For some values you may prefer to set them once when they change. How you do this is up to you - a timer, an event, a separate thread receiving messages... Whatever mechanism you use, from your handler you will call something like this:
let now = DateTime::now();
let value = 123.456f64;
let node_id = NodeId::new(2, "myvalue");
// You can set the value directly on the address space, but prefer calling this method instead,
// which will notify any listening clients.
node_manager.set_value(&handle.subscriptions(), &node_id, None, DataValue::new_at(value, now));
In this example now
is the current timestamp for when the value changed and the value is 123.456.
Alternatively you might prefer to poll values when a client actually asks for it. In this case, you can set the getter function whenever the variable is asked for and your function will be called.
This example will 123.456f
.
let node_id = NodeId::new(2, "myvalue");
node_manager.inner().add_read_callback(node_id, |_, _, _| {
Ok(DataValue::new_now(123.456f))
})
The difference with the dynamic getter is there are parameters that allow your code to conditionally decide how they return a value.
The parameters to the getter are:
NumericRange
TimestampsToReturn
f64
- the max age parameter.
This allows a getter to be broad or specific. In the example, the getter is so specific it does not require any of the parameters.
Running a server is asynchronous.
#[tokio::main]
async fn main() {
//... After server and address space are created
// Run the server. This can be terminated gracefully by calling `handle.cancel()`.
server.run().await.unwrap();
}
OPC UA for Rust provides an extensive amount of logging at error, warn, info, debug and trace levels. All this is via the standard log facade so choose which logging implementation you want to capture information. See the link for implementations that you can use.
For convenience OPC UA for Rust provides a simple opcua-console-logging
crate that wraps env_logger and writes out logging information to stdout. To use it, set an RUST_OPCUA_LOG
environment variable (not RUST_LOG
), otherwise following the documentation in env_logger
. e.g.
export RUST_OPCUA_LOG=debug
In your Cargo.toml
, ensure to add console-logging
to your opcua features:
[dependencies]
opcua = { "0.12", features = ["....", "console-logging"]}
In your main()
:
fn main() {
opcua_console_logging::init();
//...
}
The demo-server
sample demonstrates more sophisticated logging using the log4rs crate.
For advanced usage of the server, see advanced_server