03/21/2018
Dear,
I have been struggling with this 'concept' a bit and therefor hope to get an answer / guideline from this forum.
I need to read a large amount of data from an OPC UA Server everytime a test on a machine ends.
This data will be made available in an 'array' of complex types.
- fpv_curve (Node Class = Object, Node Id = ns=4;.fpv_curve)
- fpv_curve[1] (Node Class = Object, Node Id = ns=4;.fpv_curve[1])
- fpv_curve[2] (Node Class = Object, Node Id = ns=4;.fpv_curve[2])
- fpv_curve[3] (Node Class = Object, Node Id = ns=4;.fpv_curve[3])
- ...
The fpv_curve[X] has a number of variables with the actual values I need.
Now I am wondering how can I can ask the OPC UA Server to give me fpv_curve[X] to fpv_curve[Y] with all the values included.
I'm thinking for getting a datastream from the OPC UA Server and start reading (and serializing) the data until a certain point.
At the moment I made a 'program' where I create a list with 'ReadValueId' for all the different values I want to read from the fpv_curves but this takes to long and is not maintainable.
So my question to you is, is there a way where I can 1) Read a 'complex' type (as in a fpv_curve struct) and 2) Can I read a array from index X till Y with the 'complex' type?
Thanks in advance!
If there are question or if you which more information, feel free.
Kind regards,
Glenn
03/21/2018
Dear Randy,
Thank you for your reply.
I just read the Appendix C, but I don't quite follow what is described there.
So what this article suggest is that I make a method on the OPC UA Server that allows me to create a file structure of an address, in my case the array of objects, to stream this over to the client?
Can this also work when we don't 'control' the way the UA Server is made? Since we only have a .NET Client API to connect to a UA Server. We don't have a API to create a OPC UA Server. (we use the UA Server from the Supplier of the PLC)
Is there a code example for this (Appendix Part 5.C) on Github?
I did some testing myself to figure out a way to read the requested data.
At this moment, I'm creating a List of <ReadValueId> that I send to the UA Server. The server responds with a list of DataValue.
The server respondsin about 1 second for 8000 DataValues, but I don't like the way that it's code (half static NodeId's) and the way the data returns to me (a list with flat values). There should be a better way to get the data (possibly streaming asynchronously) and to format/serialize the data in a useable structure.
Kind regards,
Glenn
05/30/2017
There are better ways to model the data but they all require the server to do something.
If you can't change the server then your options are limited.
That said, if you have an array to read you can read the entire array with a single ReadValueId.
It is not clear why you need 8000.
03/21/2018
Dear Randy,
What I was trying to say was that I build a List of 8000 ReadValueId's and than I call the session.Read() and pass in this list.
In the background this will be 1 request of all these Id's that will be send to the OPC UA Server.
What I essentially hoped was that I can say:
session.read(4000, ns=4;.fpv_curve, FpvCurveType)
1st value being the amount of items to read
2nd value being the NodeId that contains the array
3rd value being the 'structure' to parse it in (and this structure should than also be available on the OPC UA Server)
Kind regards,
Glenn
02/24/2014
Hello,
I am not sure if I understand what you want to achieve, but here is my bit.
If you have (maybe besides nodes that carry the individual elements) a node whose value is an array, then (assuming that the server supports it, but most do) you should be able to set the IndexRange property in the Opc.Ua.ReadValueId, and the server would return the specified sub-range of elements.
If the value of the element, or the whole array, is a "structure" that is not defined in the OPC specs and therefore does not have its encoder/decoder in the stack, you should be getting Opc.Ua.ExtensionObject in the value that you read. Is that the case?
Best regards
Zbynek
Moderators-Specifications
Moderators-Companion
Moderators-Implementation
Moderators-Certification
Moderators-COM
02/24/2014
My first question is, since the information model (data in the PLC) is controlled by the Server Vendor, do you have a more detailed description of what the Model is? What does the data look like, are there ObjectType/VariableTypes that can be provided Having that information would allow a much clearer description of what is possible from the client side. It would also be useful to know what functionality the Server contains (historical data)
Paul
Paul Hunkar - DSInteroperability
03/21/2018
Dear,
Thanks for your responses.
I will try to explain the situation as clear as possible, but keep in mind i'm new to the OPC UA way of thinking and don't have a lot of experience with it. If I have time i'm reading the documentation about OPC UA but since I'm project bound this goes slow. Also, I don't seem to be able to upload screenshots which would probably make it easier to show you the structure I have in the PLC.
We have a Beckhoff PLC with a UA Server installed on it (The Beckhoff UA Server). On the PLC there is a program that measures the force and position of a machine. Every X µs the PLC will measure the position and force and write it in a empty position in the following variable.
fpv_curve : ARRAY[1..128000] OF struct_fpv_curve_record ;
The struct_fpv_curve_record is a STRUCT with some REAL values (for the position, force and other values) in it.
At the end of the machine test the PLC will provide a int value to determine the end position of the array. (To let me know what the last index is)
If I look in the OPC UA Server and browse the items I get the following types:
- fpv_curve
Node Class : Object
- fpv_curve[1]
Node Class : Object
- fpv_curve[1].Position
Node Class : Variable
- fpv_curve[1].Force
Node Class : Variable
So at this moment the value containing the array itself is also of the type Object. (Which I will discuss with our PLC programmer)
I hope this answers your question for more information @Paul Hunkar?
@zbynekz If I try to read a NodeId with NodeClass Object I get a value of Null.
Kind regards,
Glenn
02/24/2014
It looks like that the server does not expose the array as a readable variable, and it does not expose its elements as readable variable either - only the individual properties of the individual elements. If this is the case, you won't be able to achieve what you want just by doing something on the client side.
Isn't there a setting on the server to make it expose the array as a variable? Or maybe a different way of declaring it in the PLC program?
OPC UA is capable of doing what you want, just fine - you just need to have the software (or its proper configuration) that actually implements it.
Regards
03/21/2018
Dear zbynekz,
You are a life saver!
Once I read the sentence with 'does not expose the array as a readable variable' I started looking into the PLC code.
As said before, I'm using a Beckhoff PLC and there are apparently 2 attributes you need to use to expose your variables and with the Structure.
--> {attribute 'OPC.UA.DA' := '1'}
--> {attribute 'OPC.UA.DA.StructuredType' := '1'}
Once you enable these 2 attributes you can read the NodeId's as values.
The problem I'm having now is that the PLC is not able to offer me the 128000 array. It's just to big. So I'll have to shorten it.
So I gues my problem is fixed. I can read the data in half a second. The only thing I need to know now is how to format the data i'm getting to a useable format. --> This is also fixed at the moment. I have to enable 'DecodeCustomDataTypes' in my client so that my client loads the custom structures. After that I can just read my StructureValue and map those values to a Custom Object I can use.
Thanks again for the help!
02/24/2014
Good.
The PLC may still be able to give you a range from the whole array, if you set the IndexRange in ReadValueId. Have you tried that? The IndexRange should be e.g. "1000:2000".
I am curious, which client are you talking about/are using, when mentioning the ‘DecodeCustomDataTypes’ option?
Best regards
09/07/2023
hi I think I have a similar problem but I can not understand how to solve it with javascript
I need create a structure as follow:
object:
->"ordini" (array[0..10])
->->"ordini"[1].var1 (word)
->->"ordini"[1].var2 (int)
->->"ordini"[1].var3 (array[0..10])
->->-> "ordini"[1].var3[1].var1 (string)
->->-> "ordini"[1].var3[1].var2 (word)
->->-> "ordini"[1].var3[1].var3 (float)
->->"ordini"[2].var1 (word)
->->"ordini"[2].var2 (int)
->->"ordini"[2].var3 (array[0..10])
->->-> "ordini"[2].var3[1].var1 (string)
->->-> "ordini"[2].var3[1].var2 (word)
->->-> "ordini"[2].var3[1].var3 (float)
and so on
how can create it with javascript (i'm using opcua in nodered)
I start with
function constructAlarmAddressSpace(server, addressSpace, eventObjects, done) {const opcua = coreServer.choreCompact.opcua;const LocalizedText = opcua.LocalizedText;const namespace = addressSpace.getOwnNamespace();const Variant = opcua.Variant;const DataType = opcua.DataType;const DataValue = opcua.DataValue;var flexServerInternals = this;const rootFolder = addressSpace.findNode("RootFolder");node.warn("construct new address space for OPC UA");const ordiniFolder = namespace.addFolder(rootFolder.objects, {"browseName": "ordini"});const ordine = namespace.addObject({organizedBy: ordiniFolder,browseName: "ordine"});const arrayOrdini = namespace.addObjectType({browseName: "ordini",})done();}
many many thanks in advance. I ask help for my urgent work
05/30/2017
Duplicated post from: https://opcfoundation.org/foru.....avascript/
1 Guest(s)