229 lines
5.0 KiB
Go
229 lines
5.0 KiB
Go
// Copyright (c) Mainflux
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package gopcua
|
|
|
|
import (
|
|
"context"
|
|
|
|
opcuagocpua "github.com/gopcua/opcua"
|
|
"github.com/gopcua/opcua/id"
|
|
uagocpua "github.com/gopcua/opcua/ua"
|
|
"github.com/mainflux/mainflux/logger"
|
|
"github.com/mainflux/mainflux/opcua"
|
|
"github.com/mainflux/mainflux/pkg/errors"
|
|
)
|
|
|
|
const maxChildrens = 4 // max browsing node children level
|
|
|
|
// NodeDef represents the node browser responnse.
|
|
type NodeDef struct {
|
|
NodeID *uagocpua.NodeID
|
|
NodeClass uagocpua.NodeClass
|
|
BrowseName string
|
|
Description string
|
|
AccessLevel uagocpua.AccessLevelType
|
|
Path string
|
|
DataType string
|
|
Writable bool
|
|
Unit string
|
|
Scale string
|
|
Min string
|
|
Max string
|
|
}
|
|
|
|
var _ opcua.Browser = (*browser)(nil)
|
|
|
|
type browser struct {
|
|
ctx context.Context
|
|
logger logger.Logger
|
|
}
|
|
|
|
// NewBrowser returns new OPC-UA browser instance.
|
|
func NewBrowser(ctx context.Context, log logger.Logger) opcua.Browser {
|
|
return browser{
|
|
ctx: ctx,
|
|
logger: log,
|
|
}
|
|
}
|
|
|
|
func (c browser) Browse(serverURI, nodeID string) ([]opcua.BrowsedNode, error) {
|
|
opts := []opcuagocpua.Option{
|
|
opcuagocpua.SecurityMode(uagocpua.MessageSecurityModeNone),
|
|
}
|
|
|
|
oc := opcuagocpua.NewClient(serverURI, opts...)
|
|
if err := oc.Connect(c.ctx); err != nil {
|
|
return nil, errors.Wrap(errFailedConn, err)
|
|
}
|
|
defer oc.Close()
|
|
|
|
nodeList, err := browse(oc, nodeID, "", 0)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
nodes := []opcua.BrowsedNode{}
|
|
for _, s := range nodeList {
|
|
node := opcua.BrowsedNode{
|
|
NodeID: s.NodeID.String(),
|
|
DataType: s.DataType,
|
|
Description: s.Description,
|
|
Unit: s.Unit,
|
|
Scale: s.Scale,
|
|
BrowseName: s.BrowseName,
|
|
}
|
|
nodes = append(nodes, node)
|
|
}
|
|
|
|
return nodes, nil
|
|
}
|
|
|
|
func browse(oc *opcuagocpua.Client, nodeID, path string, level int) ([]NodeDef, error) {
|
|
if level > maxChildrens {
|
|
return nil, nil
|
|
}
|
|
|
|
nid, err := uagocpua.ParseNodeID(nodeID)
|
|
if err != nil {
|
|
return []NodeDef{}, err
|
|
}
|
|
n := oc.Node(nid)
|
|
|
|
attrs, err := n.Attributes(
|
|
uagocpua.AttributeIDNodeClass,
|
|
uagocpua.AttributeIDBrowseName,
|
|
uagocpua.AttributeIDDescription,
|
|
uagocpua.AttributeIDAccessLevel,
|
|
uagocpua.AttributeIDDataType,
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
def := NodeDef{
|
|
NodeID: n.ID,
|
|
}
|
|
|
|
switch err := attrs[0].Status; err {
|
|
case uagocpua.StatusOK:
|
|
def.NodeClass = uagocpua.NodeClass(attrs[0].Value.Int())
|
|
default:
|
|
return nil, err
|
|
}
|
|
|
|
switch err := attrs[1].Status; err {
|
|
case uagocpua.StatusOK:
|
|
def.BrowseName = attrs[1].Value.String()
|
|
default:
|
|
return nil, err
|
|
}
|
|
|
|
switch err := attrs[2].Status; err {
|
|
case uagocpua.StatusOK:
|
|
def.Description = attrs[2].Value.String()
|
|
case uagocpua.StatusBadAttributeIDInvalid:
|
|
// ignore
|
|
default:
|
|
return nil, err
|
|
}
|
|
|
|
switch err := attrs[3].Status; err {
|
|
case uagocpua.StatusOK:
|
|
def.AccessLevel = uagocpua.AccessLevelType(attrs[3].Value.Int())
|
|
def.Writable = def.AccessLevel&uagocpua.AccessLevelTypeCurrentWrite == uagocpua.AccessLevelTypeCurrentWrite
|
|
case uagocpua.StatusBadAttributeIDInvalid:
|
|
// ignore
|
|
default:
|
|
return nil, err
|
|
}
|
|
|
|
switch err := attrs[4].Status; err {
|
|
case uagocpua.StatusOK:
|
|
switch v := attrs[4].Value.NodeID().IntID(); v {
|
|
case id.DateTime:
|
|
def.DataType = "time.Time"
|
|
case id.Boolean:
|
|
def.DataType = "bool"
|
|
case id.SByte:
|
|
def.DataType = "int8"
|
|
case id.Int16:
|
|
def.DataType = "int16"
|
|
case id.Int32:
|
|
def.DataType = "int32"
|
|
case id.Byte:
|
|
def.DataType = "byte"
|
|
case id.UInt16:
|
|
def.DataType = "uint16"
|
|
case id.UInt32:
|
|
def.DataType = "uint32"
|
|
case id.UtcTime:
|
|
def.DataType = "time.Time"
|
|
case id.String:
|
|
def.DataType = "string"
|
|
case id.Float:
|
|
def.DataType = "float32"
|
|
case id.Double:
|
|
def.DataType = "float64"
|
|
default:
|
|
def.DataType = attrs[4].Value.NodeID().String()
|
|
}
|
|
case uagocpua.StatusBadAttributeIDInvalid:
|
|
// ignore
|
|
default:
|
|
return nil, err
|
|
}
|
|
|
|
def.Path = join(path, def.BrowseName)
|
|
|
|
var nodes []NodeDef
|
|
if def.NodeClass == uagocpua.NodeClassVariable {
|
|
nodes = append(nodes, def)
|
|
}
|
|
|
|
bc, err := browseChildren(oc, n, def.Path, level, id.HasComponent)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
nodes = append(nodes, bc...)
|
|
|
|
bc, err = browseChildren(oc, n, def.Path, level, id.Organizes)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
nodes = append(nodes, bc...)
|
|
|
|
bc, err = browseChildren(oc, n, def.Path, level, id.HasProperty)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
nodes = append(nodes, bc...)
|
|
|
|
return nodes, nil
|
|
}
|
|
|
|
func browseChildren(c *opcuagocpua.Client, n *opcuagocpua.Node, path string, level int, typeDef uint32) ([]NodeDef, error) {
|
|
nodes := []NodeDef{}
|
|
refs, err := n.ReferencedNodes(typeDef, uagocpua.BrowseDirectionForward, uagocpua.NodeClassAll, true)
|
|
if err != nil {
|
|
return []NodeDef{}, err
|
|
}
|
|
|
|
for _, ref := range refs {
|
|
children, err := browse(c, ref.ID.String(), path, level+1)
|
|
if err != nil {
|
|
return []NodeDef{}, err
|
|
}
|
|
nodes = append(nodes, children...)
|
|
}
|
|
|
|
return nodes, nil
|
|
}
|
|
|
|
func join(a, b string) string {
|
|
if a == "" {
|
|
return b
|
|
}
|
|
return a + "." + b
|
|
}
|