Files
Mapo-IOB-WIN/IOB-OPC-UA/Libraries/Opc.Ua.Client/DataDictionary.cs
T
2021-03-25 18:25:25 +01:00

326 lines
11 KiB
C#

/* ========================================================================
* Copyright (c) 2005-2020 The OPC Foundation, Inc. All rights reserved.
*
* OPC Foundation MIT License 1.00
*
* Permission is hereby granted, free of charge, to any person
* obtaining a copy of this software and associated documentation
* files (the "Software"), to deal in the Software without
* restriction, including without limitation the rights to use,
* copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following
* conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
* OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE.
*
* The complete license agreement can be found here:
* http://opcfoundation.org/License/MIT/1.00/
* ======================================================================*/
using Opc.Ua.Schema;
using Opc.Ua.Schema.Binary;
using System;
using System.Collections.Generic;
using System.IO;
using System.Runtime.Serialization;
using System.Threading.Tasks;
namespace Opc.Ua.Client
{
/// <summary>
/// A class that holds the configuration for a UA service.
/// </summary>
[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Naming", "CA1711:IdentifiersShouldNotHaveIncorrectSuffix"), DataContract(Namespace = Namespaces.OpcUaXsd)]
public class DataDictionary
{
#region Constructors
/// <summary>
/// The default constructor.
/// </summary>
public DataDictionary(Session session)
{
Initialize();
m_session = session;
}
/// <summary>
/// Sets private members to default values.
/// </summary>
private void Initialize()
{
m_session = null;
DataTypes = new Dictionary<NodeId, ReferenceDescription>();
m_validator = null;
TypeSystemId = null;
TypeSystemName = null;
DictionaryId = null;
Name = null;
}
#endregion
#region Public Interface
/// <summary>
/// The node id for the dictionary.
/// </summary>
public NodeId DictionaryId { get; private set; }
/// <summary>
/// The display name for the dictionary.
/// </summary>
public string Name { get; private set; }
/// <summary>
/// The node id for the type system.
/// </summary>
public NodeId TypeSystemId { get; private set; }
/// <summary>
/// The display name for the type system.
/// </summary>
public string TypeSystemName { get; private set; }
/// <summary>
/// The type dictionary.
/// </summary>
public TypeDictionary TypeDictionary { get; private set; }
/// <summary>
/// The data type dictionary.
/// </summary>
public Dictionary<NodeId, ReferenceDescription> DataTypes { get; private set; }
/// <summary>
/// Loads the dictionary identified by the node id.
/// </summary>
public Task Load(ReferenceDescription dictionary)
{
if (dictionary == null)
{
throw new ArgumentNullException(nameof(dictionary));
}
NodeId dictionaryId = ExpandedNodeId.ToNodeId(dictionary.NodeId, m_session.NamespaceUris);
return Load(dictionaryId, dictionary.ToString());
}
/// <summary>
/// Loads the dictionary identified by the node id.
/// </summary>
public async Task Load(NodeId dictionaryId, string name)
{
if (dictionaryId == null)
{
throw new ArgumentNullException(nameof(dictionaryId));
}
GetTypeSystem(dictionaryId);
byte[] schema = ReadDictionary(dictionaryId);
if (schema == null || schema.Length == 0)
{
throw ServiceResultException.Create(StatusCodes.BadUnexpectedError, "Cannot parse empty data dictionary.");
}
// Interoperability: some server may return a null terminated dictionary string, adjust length
int zeroTerminator = Array.IndexOf<byte>(schema, (byte)0);
if (zeroTerminator >= 0)
{
Array.Resize(ref schema, zeroTerminator);
}
await Validate(schema);
ReadDataTypes(dictionaryId);
DictionaryId = dictionaryId;
Name = name;
}
/// <summary>
/// Returns true if the dictionary contains the data type description;
/// </summary>
public bool Contains(NodeId descriptionId)
{
return DataTypes.ContainsKey(descriptionId);
}
/// <summary>
/// Returns the schema for the specified type (returns the entire dictionary if null).
/// </summary>
public string GetSchema(NodeId descriptionId)
{
ReferenceDescription description = null;
if (descriptionId != null)
{
if (!DataTypes.TryGetValue(descriptionId, out description))
{
return null;
}
return m_validator.GetSchema(description.BrowseName.Name);
}
return m_validator.GetSchema(null);
}
#endregion
#region Private Members
/// <summary>
/// Retrieves the type system for the dictionary.
/// </summary>
private void GetTypeSystem(NodeId dictionaryId)
{
Browser browser = new Browser(m_session);
browser.BrowseDirection = BrowseDirection.Inverse;
browser.ReferenceTypeId = ReferenceTypeIds.HasComponent;
browser.IncludeSubtypes = false;
browser.NodeClassMask = 0;
ReferenceDescriptionCollection references = browser.Browse(dictionaryId);
if (references.Count > 0)
{
TypeSystemId = ExpandedNodeId.ToNodeId(references[0].NodeId, m_session.NamespaceUris);
TypeSystemName = references[0].ToString();
}
}
/// <summary>
/// Retrieves the data types in the dictionary.
/// </summary>
/// <remarks>
/// In order to allow for fast Linq matching of dictionary
/// QNames with the data type nodes, the BrowseName of
/// the DataType node is replaced with Value string.
/// </remarks>
private void ReadDataTypes(NodeId dictionaryId)
{
Browser browser = new Browser(m_session);
browser.BrowseDirection = BrowseDirection.Forward;
browser.ReferenceTypeId = ReferenceTypeIds.HasComponent;
browser.IncludeSubtypes = false;
browser.NodeClassMask = 0;
ReferenceDescriptionCollection references = browser.Browse(dictionaryId);
foreach (ReferenceDescription reference in references)
{
NodeId datatypeId = ExpandedNodeId.ToNodeId(reference.NodeId, m_session.NamespaceUris);
if (datatypeId != null)
{
// read the value to get the name that is used in the dictionary
var value = m_session.ReadValue(datatypeId);
var dictName = (String)value.Value;
// replace the BrowseName with type name used in the dictionary
reference.BrowseName = new QualifiedName(dictName, datatypeId.NamespaceIndex);
DataTypes[datatypeId] = reference;
}
}
}
/// <summary>
/// Reads the contents of a data dictionary.
/// </summary>
private byte[] ReadDictionary(NodeId dictionaryId)
{
// create item to read.
ReadValueId itemToRead = new ReadValueId();
itemToRead.NodeId = dictionaryId;
itemToRead.AttributeId = Attributes.Value;
itemToRead.IndexRange = null;
itemToRead.DataEncoding = null;
ReadValueIdCollection itemsToRead = new ReadValueIdCollection();
itemsToRead.Add(itemToRead);
// read value.
DataValueCollection values;
DiagnosticInfoCollection diagnosticInfos;
ResponseHeader responseHeader = m_session.Read(
null,
0,
TimestampsToReturn.Neither,
itemsToRead,
out values,
out diagnosticInfos);
ClientBase.ValidateResponse(values, itemsToRead);
ClientBase.ValidateDiagnosticInfos(diagnosticInfos, itemsToRead);
// check for error.
if (StatusCode.IsBad(values[0].StatusCode))
{
ServiceResult result = ClientBase.GetResult(values[0].StatusCode, 0, diagnosticInfos, responseHeader);
throw new ServiceResultException(result);
}
// return as a byte array.
return values[0].Value as byte[];
}
/// <summary>
/// Validates the type dictionary.
/// </summary>
/// <param name="dictionary"></param>
private async Task Validate(byte[] dictionary)
{
MemoryStream istrm = new MemoryStream(dictionary);
if (TypeSystemId == Objects.XmlSchema_TypeSystem)
{
Schema.Xml.XmlSchemaValidator validator = new Schema.Xml.XmlSchemaValidator();
try
{
validator.Validate(istrm);
}
catch (Exception e)
{
Utils.Trace(e, "Could not validate schema.");
}
m_validator = validator;
}
if (TypeSystemId == Objects.OPCBinarySchema_TypeSystem)
{
Schema.Binary.BinarySchemaValidator validator = new Schema.Binary.BinarySchemaValidator();
try
{
await validator.Validate(istrm);
}
catch (Exception e)
{
Utils.Trace(e, $"Could not validate schema. {e.Message}");
}
m_validator = validator;
TypeDictionary = validator.Dictionary;
}
}
#endregion
#region Private Members
private Session m_session;
private SchemaValidator m_validator;
#endregion
}
}