use std::{
self,
convert::TryFrom,
fmt,
io::{Read, Write},
str::FromStr,
sync::atomic::{AtomicUsize, Ordering},
u16, u32,
};
use serde::{de, Deserialize, Deserializer, Serialize, Serializer};
use serde_json::{self, json};
use crate::types::{
byte_string::ByteString,
encoding::*,
guid::Guid,
node_ids::{ObjectId, ReferenceTypeId},
status_codes::StatusCode,
string::*,
};
#[derive(Eq, PartialEq, Clone, Debug, Hash)]
pub enum Identifier {
Numeric(u32),
String(UAString),
Guid(Guid),
ByteString(ByteString),
}
impl fmt::Display for Identifier {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match self {
Identifier::Numeric(v) => write!(f, "i={}", *v),
Identifier::String(v) => write!(f, "s={}", v),
Identifier::Guid(v) => write!(f, "g={:?}", v),
Identifier::ByteString(v) => write!(f, "b={}", v.as_base64()),
}
}
}
impl FromStr for Identifier {
type Err = ();
fn from_str(s: &str) -> Result<Self, Self::Err> {
if s.len() < 2 {
Err(())
} else {
let k = &s[..2];
let v = &s[2..];
match k {
"i=" => v.parse::<u32>().map(|v| v.into()).map_err(|_| ()),
"s=" => Ok(UAString::from(v).into()),
"g=" => Guid::from_str(v).map(|v| v.into()).map_err(|_| ()),
"b=" => ByteString::from_base64(v).map(|v| v.into()).ok_or(()),
_ => Err(()),
}
}
}
}
impl From<i32> for Identifier {
fn from(v: i32) -> Self {
Identifier::Numeric(v as u32)
}
}
impl From<u32> for Identifier {
fn from(v: u32) -> Self {
Identifier::Numeric(v as u32)
}
}
impl<'a> From<&'a str> for Identifier {
fn from(v: &'a str) -> Self {
Identifier::from(UAString::from(v))
}
}
impl From<&String> for Identifier {
fn from(v: &String) -> Self {
Identifier::from(UAString::from(v))
}
}
impl From<String> for Identifier {
fn from(v: String) -> Self {
Identifier::from(UAString::from(v))
}
}
impl From<UAString> for Identifier {
fn from(v: UAString) -> Self {
Identifier::String(v)
}
}
impl From<Guid> for Identifier {
fn from(v: Guid) -> Self {
Identifier::Guid(v)
}
}
impl From<ByteString> for Identifier {
fn from(v: ByteString) -> Self {
Identifier::ByteString(v)
}
}
#[derive(Debug)]
pub struct NodeIdError;
impl fmt::Display for NodeIdError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "NodeIdError")
}
}
impl std::error::Error for NodeIdError {}
#[derive(PartialEq, Eq, Clone, Debug, Hash)]
pub struct NodeId {
pub namespace: u16,
pub identifier: Identifier,
}
impl fmt::Display for NodeId {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
if self.namespace != 0 {
write!(f, "ns={};{}", self.namespace, self.identifier)
} else {
write!(f, "{}", self.identifier)
}
}
}
#[derive(Serialize, Deserialize)]
struct JsonNodeId {
#[serde(skip_serializing_if = "Option::is_none")]
#[serde(rename = "Type")]
id_type: Option<u32>,
#[serde(rename = "Id")]
id: serde_json::Value,
#[serde(skip_serializing_if = "Option::is_none")]
#[serde(rename = "Namespace")]
namespace: Option<serde_json::Value>,
}
impl Serialize for NodeId {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
let (id_type, id) = match &self.identifier {
Identifier::Numeric(id) => (None, json!(id)),
Identifier::String(id) => (Some(1), json!(id.as_ref())),
Identifier::Guid(id) => (Some(2), json!(id.to_string())),
Identifier::ByteString(id) => (Some(3), json!(id.as_base64())),
};
let namespace = if self.namespace == 0 {
None
} else {
Some(json!(self.namespace))
};
let json = JsonNodeId {
id_type,
id,
namespace,
};
json.serialize(serializer)
}
}
impl<'de> Deserialize<'de> for NodeId {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: Deserializer<'de>,
{
let v = JsonNodeId::deserialize(deserializer)?;
let namespace = if let Some(namespace) = v.namespace {
let namespace = namespace
.as_u64()
.ok_or_else(|| de::Error::custom("Expected numeric namespace index"))?;
if namespace > u16::MAX as u64 {
return Err(de::Error::custom("Numeric namespace index is out of range"));
}
namespace as u16
} else {
0
};
let id_type = v.id_type.unwrap_or(0);
match id_type {
0 => {
let v =
v.id.as_u64()
.ok_or_else(|| de::Error::custom("Expected Numeric identifier"))?;
Ok(NodeId::new(namespace, v as u32))
}
1 => {
let v =
v.id.as_str()
.ok_or_else(|| de::Error::custom("Expected String identifier"))?;
if v.is_empty() {
Err(de::Error::custom("String identifier is empty"))
} else {
Ok(NodeId::new(namespace, String::from(v)))
}
}
2 => {
let v =
v.id.as_str()
.ok_or_else(|| de::Error::custom("Expected Guid identifier"))?;
if v.is_empty() {
Err(de::Error::custom("Guid identifier is empty"))
} else {
let v = Guid::from_str(v)
.map_err(|_| de::Error::custom("Error parsing Guid identifier"))?;
Ok(NodeId::new(namespace, v))
}
}
3 => {
let v =
v.id.as_str()
.ok_or_else(|| de::Error::custom("Expected ByteString identifier"))?;
if v.is_empty() {
Err(de::Error::custom("ByteString identifier is empty"))
} else {
let v = ByteString::from_base64(v)
.ok_or_else(|| de::Error::custom("Error parsing ByteString identifier"))?;
Ok(NodeId::new(namespace, v))
}
}
_ => Err(de::Error::custom("Invalid IdType")),
}
}
}
impl BinaryEncoder<NodeId> for NodeId {
fn byte_len(&self) -> usize {
let size: usize = match self.identifier {
Identifier::Numeric(value) => {
if self.namespace == 0 && value <= 255 {
2
} else if self.namespace <= 255 && value <= 65535 {
4
} else {
7
}
}
Identifier::String(ref value) => 3 + value.byte_len(),
Identifier::Guid(ref value) => 3 + value.byte_len(),
Identifier::ByteString(ref value) => 3 + value.byte_len(),
};
size
}
fn encode<S: Write>(&self, stream: &mut S) -> EncodingResult<usize> {
let mut size: usize = 0;
match &self.identifier {
Identifier::Numeric(value) => {
if self.namespace == 0 && *value <= 255 {
size += write_u8(stream, 0x0)?;
size += write_u8(stream, *value as u8)?;
} else if self.namespace <= 255 && *value <= 65535 {
size += write_u8(stream, 0x1)?;
size += write_u8(stream, self.namespace as u8)?;
size += write_u16(stream, *value as u16)?;
} else {
size += write_u8(stream, 0x2)?;
size += write_u16(stream, self.namespace)?;
size += write_u32(stream, *value)?;
}
}
Identifier::String(value) => {
size += write_u8(stream, 0x3)?;
size += write_u16(stream, self.namespace)?;
size += value.encode(stream)?;
}
Identifier::Guid(value) => {
size += write_u8(stream, 0x4)?;
size += write_u16(stream, self.namespace)?;
size += value.encode(stream)?;
}
Identifier::ByteString(value) => {
size += write_u8(stream, 0x5)?;
size += write_u16(stream, self.namespace)?;
size += value.encode(stream)?;
}
}
assert_eq!(size, self.byte_len());
Ok(size)
}
fn decode<S: Read>(stream: &mut S, decoding_options: &DecodingOptions) -> EncodingResult<Self> {
let identifier = read_u8(stream)?;
let node_id = match identifier {
0x0 => {
let namespace = 0;
let value = read_u8(stream)?;
NodeId::new(namespace, u32::from(value))
}
0x1 => {
let namespace = read_u8(stream)?;
let value = read_u16(stream)?;
NodeId::new(u16::from(namespace), u32::from(value))
}
0x2 => {
let namespace = read_u16(stream)?;
let value = read_u32(stream)?;
NodeId::new(namespace, value)
}
0x3 => {
let namespace = read_u16(stream)?;
let value = UAString::decode(stream, decoding_options)?;
NodeId::new(namespace, value)
}
0x4 => {
let namespace = read_u16(stream)?;
let value = Guid::decode(stream, decoding_options)?;
NodeId::new(namespace, value)
}
0x5 => {
let namespace = read_u16(stream)?;
let value = ByteString::decode(stream, decoding_options)?;
NodeId::new(namespace, value)
}
_ => {
error!("Unrecognized node id type {}", identifier);
return Err(StatusCode::BadDecodingError);
}
};
Ok(node_id)
}
}
impl FromStr for NodeId {
type Err = StatusCode;
fn from_str(s: &str) -> std::result::Result<Self, Self::Err> {
use regex::Regex;
lazy_static! {
static ref RE: Regex = Regex::new(r"^(ns=(?P<ns>[0-9]+);)?(?P<t>[isgb]=.+)$").unwrap();
}
let captures = RE.captures(s).ok_or(StatusCode::BadNodeIdInvalid)?;
let namespace = if let Some(ns) = captures.name("ns") {
ns.as_str()
.parse::<u16>()
.map_err(|_| StatusCode::BadNodeIdInvalid)?
} else {
0
};
let t = captures.name("t").unwrap();
Identifier::from_str(t.as_str())
.map(|t| NodeId::new(namespace, t))
.map_err(|_| StatusCode::BadNodeIdInvalid)
}
}
impl From<&NodeId> for NodeId {
fn from(v: &NodeId) -> Self {
v.clone()
}
}
impl Into<String> for NodeId {
fn into(self) -> String {
self.to_string()
}
}
impl<'a> From<(u16, &'a str)> for NodeId {
fn from(v: (u16, &'a str)) -> Self {
Self::new(v.0, UAString::from(v.1))
}
}
impl From<(u16, UAString)> for NodeId {
fn from(v: (u16, UAString)) -> Self {
Self::new(v.0, v.1)
}
}
impl From<(u16, u32)> for NodeId {
fn from(v: (u16, u32)) -> Self {
Self::new(v.0, v.1)
}
}
impl From<(u16, Guid)> for NodeId {
fn from(v: (u16, Guid)) -> Self {
Self::new(v.0, v.1)
}
}
impl From<(u16, ByteString)> for NodeId {
fn from(v: (u16, ByteString)) -> Self {
Self::new(v.0, v.1)
}
}
static NEXT_NODE_ID_NUMERIC: AtomicUsize = AtomicUsize::new(0);
impl Default for NodeId {
fn default() -> Self {
NodeId::null()
}
}
impl NodeId {
pub fn new<T>(namespace: u16, value: T) -> NodeId
where
T: 'static + Into<Identifier>,
{
NodeId {
namespace,
identifier: value.into(),
}
}
pub fn root_folder_id() -> NodeId {
ObjectId::RootFolder.into()
}
pub fn objects_folder_id() -> NodeId {
ObjectId::ObjectsFolder.into()
}
pub fn types_folder_id() -> NodeId {
ObjectId::TypesFolder.into()
}
pub fn views_folder_id() -> NodeId {
ObjectId::ViewsFolder.into()
}
pub fn is_null(&self) -> bool {
self.namespace == 0 && self.identifier == Identifier::Numeric(0)
}
pub fn null() -> NodeId {
NodeId::new(0, 0u32)
}
pub fn next_numeric(namespace: u16) -> NodeId {
NodeId::new(
namespace,
NEXT_NODE_ID_NUMERIC.fetch_add(1, Ordering::SeqCst) as u32,
)
}
pub fn as_object_id(&self) -> std::result::Result<ObjectId, NodeIdError> {
match self.identifier {
Identifier::Numeric(id) if self.namespace == 0 => {
ObjectId::try_from(id).map_err(|_| NodeIdError)
}
_ => Err(NodeIdError),
}
}
pub fn as_reference_type_id(&self) -> std::result::Result<ReferenceTypeId, NodeIdError> {
if self.is_null() {
Err(NodeIdError)
} else {
match self.identifier {
Identifier::Numeric(id) if self.namespace == 0 => {
ReferenceTypeId::try_from(id).map_err(|_| NodeIdError)
}
_ => Err(NodeIdError),
}
}
}
pub fn is_numeric(&self) -> bool {
matches!(self.identifier, Identifier::Numeric(_))
}
pub fn is_string(&self) -> bool {
matches!(self.identifier, Identifier::String(_))
}
pub fn is_guid(&self) -> bool {
matches!(self.identifier, Identifier::Guid(_))
}
pub fn is_byte_string(&self) -> bool {
matches!(self.identifier, Identifier::ByteString(_))
}
}