Working Server Example #749
Replies: 37 comments 5 replies
-
I ended up splitting the address space up by namespace and made a namespace interface so different back-ends can be added at the namespace level. I've ported the browse and read/write attribute handlers back into the main server and it is working for regular nodes and my map-based "nodes". The pub/sub stuff is next, but I'll have to think about the best way to do that for a bit before I start moving it back in. |
Beta Was this translation helpful? Give feedback.
-
Wow 🥇 Thank you @danomagnum !!! That might be the push to get us off the ground with the server. I'll have a look. |
Beta Was this translation helpful? Give feedback.
-
CC: @kung-foo |
Beta Was this translation helpful? Give feedback.
-
So I've got quite a bit of it working now. I've re-integrated it back into the server module of the library - nothing in the example folder except for an actual example using the library to host a server now. Assuming the simplest use case is the most common, I think it's pretty usable at this point. I tested it now against ignition's UA client, pyopc's client, and UaExpert. They're all reading/writing/subscribing to data and working correctly. I'm sure there are still bugs to work out of course. Once you get a chance to look at it let me know and I'll get it merged back up to the latest master client revision and I can do a pull request. Here's the current status of all the services in the ua spec:
|
Beta Was this translation helpful? Give feedback.
-
The thing I got stuck at was to get the basic node structure in the prosys browser. The Root, Objects, Types, Views and so forth. Maybe you can get this to work and then I'm all for merging this back to gopcua server standard opcua server |
Beta Was this translation helpful? Give feedback.
-
I think you need to implement browsing for this. |
Beta Was this translation helpful? Give feedback.
-
So browsing (edit: partially - no continuation/browsenext) works, but you've got to have all the node references defined for it to know what to return as the browse results. If I manually set them up, they start showing up.
I only did a handful of the references to see if it worked or not. There is an absolute ton of references that are needed to get everything filled in properly. I'll have to take a look at how nodes_gen is being generated to see if we can build those references automatically. |
Beta Was this translation helpful? Give feedback.
-
I made an adjustment to the generation file and while it's not perfect yet it's browsing the pre-defined nodes automatically now. Need to do some more investigation on why they are showing up as X's instead of folder icons. There are some loops in the references too that need to be sorted out. |
Beta Was this translation helpful? Give feedback.
-
How about merging your changes on top of our |
Beta Was this translation helpful? Give feedback.
-
To get the full nodes generated we should probably import a NodeSpec2.xml file and generate the code from that. Sooner or later we need to add support for importing a node set anyway. |
Beta Was this translation helpful? Give feedback.
-
@danomagnum I've sent you an invite. You should use the second one since I've canceled the first one. |
Beta Was this translation helpful? Give feedback.
-
Invite accepted. I'll try and get the changes merged properly onto the server branch in the next couple days and I'll let you know. I will probably bring in the changes from the main branch too, just so the server branch gets caught back up. I've been away from this project for the holidays but intend to pick it back up in the next week or two. Before the break I was able to get a lot (but still not all) of the auto-import from xml working using the PredefinedNodes.xml file (which looks like it is generated by the opc-ua dotnet project from the nodeset2.xml file?). At one point the generated .go file it created was too large and the compiler choked on it so I had to refactor how it was generating the cross-references to re-use references instead of generating them anew for each node. The strangest thing going on right now is that the FolderType reference is showing up as a child item when browsing, even though as far as I can tell there isn't any difference between the packets gopcua is sending and the packets other similar opc servers send. So far getting the predefined nodes more correct has fixed this kind of thing so that's where I'm betting the problem is. As you suggest, It may probably be easier to get everything correct if we start with the nodeset2.xml file directly instead of going through the generated xml file. |
Beta Was this translation helpful? Give feedback.
-
Maybe we can embed the Nodeset2.xml file and parse it on startup. |
Beta Was this translation helpful? Give feedback.
-
Switching to the nodeset2 xml file was a good move since there is a schema definition file available and I was able to auto-generate the go structs for it using xgen which saved a lot of time parsing it. I feel much more confident the references are correct using this method. It is embedded in the schema sub package and is now loading on server startup instead of using the generated file. Unfortunately, the strange behavior with the FolderType showing up under every folder during browse with some clients remains. I merged everything into the server branch of the main repo. So far I had been testing without encryption so I could easily wireshark the packets. I tried using encryption today and that was not an immediate success so there may be some work to do there also. |
Beta Was this translation helpful? Give feedback.
-
Awesome. I'll have a look |
Beta Was this translation helpful? Give feedback.
-
I haven't worked on it in a little bit because it's (the server branch) mostly implemented enough for my needs. I've had it running in a non-critical production application for a couple months now with no issues. I didn't ever comment about it in this issue, but I did fix the FolderType bug and everything looks pretty good now when browsing from a client. Read, Write, Subscribe, and Browse all work, so it's pretty functional. The two things I was working on when I got pulled in a different direction were:
I was able to change a couple lines in the code and get specific encryption/signing methods to work, but I wasn't having much luck picking up on the correct method programmatically. If you need a server, you should be able to specifically pull the server branch - it's up to date with master except for some readme changes. I'm open to issuing a pull request to merge it into the main branch if everyone else is good with it. |
Beta Was this translation helpful? Give feedback.
-
I would love for that branch to be merged. If you know of some issues with it, then why not just merge it to main and immediately create the issues so they can be tracked? And if the server is already working in production for you, then IMO it's good to go as a first version. Better than not having it at all. Just put a usage example to the examples folda and if the crypto is not fully working, it can just be marked with |
Beta Was this translation helpful? Give feedback.
-
@magiconair is this something that can be done? I am trying to advocate for use of this library over the python one because of the performance, but its hard without a working server implementation |
Beta Was this translation helpful? Give feedback.
-
Would love to have this merged as well! Would also make it so that more people are aware of the server itself which might just help with solving the open issues |
Beta Was this translation helpful? Give feedback.
-
Just FYI, I messed something up when I put the pull request together. The version in my repo is working correctly, but I need to re-merge it into the server branch of this repo and re-issue a pull request. |
Beta Was this translation helpful? Give feedback.
-
Pull request is back live. Now #737 |
Beta Was this translation helpful? Give feedback.
-
Dropped the ball here a bit. I suggest we keep the We need help with this feature and someone who drives this. Who is interested? If I don't respond here try pinging me on Twitter. Same handle. |
Beta Was this translation helpful? Give feedback.
-
I don't mind to continue driving the server portion of the code. |
Beta Was this translation helpful? Give feedback.
-
We could provide a separate tag for the server branch so that people can use it for testing, e.g. 0.5.3-server. |
Beta Was this translation helpful? Give feedback.
-
@danomagnum lets try that. You are a dev in the org so you should be able to push to the server branch. Let me check if we can just rebase it. |
Beta Was this translation helpful? Give feedback.
-
Seems to work. Let me push it. |
Beta Was this translation helpful? Give feedback.
-
Please add a PR for the
|
Beta Was this translation helpful? Give feedback.
-
I've created a |
Beta Was this translation helpful? Give feedback.
-
@kung-foo @danomagnum Shall we merge the server to main and call it v0.6.0? |
Beta Was this translation helpful? Give feedback.
-
At long last we have a server. It took several people a couple of years and a number of attempts but we are finally there. Thank you @danomagnum for giving it the final push! |
Beta Was this translation helpful? Give feedback.
-
I've been working on getting the server working and I've got a version of it working on the server branch of my fork here.
There is still quite a bit to implement to get it production ready, but as a proof-of-concept it is working. I've been (manually) testing it using ignition and pyopcua.
I tried to keep as much of the new code in /examples/server/ as I could.
What (mostly) works:
I would like to contribute this back to the project, but I've got to do quite a bit of refactoring first and would like some recommendations on what the refactored code should look like before I start on that so that I don't have to refactor it again to match what you all are looking for in the server code.
As an example, one thing that will have to change is I am ignoring the concept as nodes as much as possible - I've got a
map[string]any
providing the backed and only using string node IDs as the keys to the map. This is a lot more convenient for the actual use case I've got that prompted me to start working on this so I'd like to keep that functionality in the end, but I think I should be able to make a pseudo-node that would hold the map and provide the references, etc... to make it invisible to the server. Maybe.Anyway, like I said I'm really looking for feedback, suggestions, and recommendations on what you'd like to see. Obviously my use case is really what I need to achieve but if I can make it work for merging back in that would be a win-win.
Beta Was this translation helpful? Give feedback.
All reactions