• Recent
    • Tags
    • Popular
    • Register
    • Login

    Please Note This forum exists for community support for the Mango product family and the Radix IoT Platform. Although Radix IoT employees participate in this forum from time to time, there is no guarantee of a response to anything posted here, nor can Radix IoT, LLC guarantee the accuracy of any information expressed or conveyed. Specific project questions from customers with active support contracts are asked to send requests to support@radixiot.com.

    Radix IoT Website Mango 4 Documentation Website Mango 5 Documentation Website Radix IoT LinkedIn

    Best way to structure for multiple instances of device

    Scheduled Pinned Locked Moved User help
    20 Posts 3 Posters 5.8k Views 3 Watching
    Loading More Posts
    • Oldest to Newest
    • Newest to Oldest
    • Most Votes
    Reply
    • Reply as topic
    Log in to reply
    This topic has been deleted. Only users with topic management privileges can see it.
    • P Offline
      pyeager
      last edited by

      @phildunlap said in Best way to structure for multiple instances of device:

      If simple point links, why not just set to the Modbus I/P points directly? Is this for the extra "variables not connected to a remote I/O point?"

      Yes. The virtual data source exists as a place for variables that don't map directly to Modbus I/P points.

      I suspect that I can simply copy both the virtual and Modbus I/P data sources, and then try to make global scripts "glue" things together.

      1 Reply Last reply Reply Quote 0
      • phildunlapP Offline
        phildunlap
        last edited by

        I would recommend using points on the Scripting data sources you're using, possibly. They're pretty much the same as no-change virtual points that are already in the script's context. But, again, I do not feel I understand the objectives well enough to make clear advisements.

        1 Reply Last reply Reply Quote 0
        • P Offline
          pyeager
          last edited by pyeager

          Phil:
          Thanks for taking the time to discuss my challenge.

          Our initial implementation will probably include a dozen machines under control of Mango. A full implementation will include somewhere in the range of 100 machines.

          The Modbus I/P data sources are sufficient for controlling the machine, however there are modes, settings, and state variables per machine that must live somewhere. There are also process variables that are shared by groups of machines. A virtual data source per machine seems to be the logical place to put those as I understand the available options. Point links and scripts tie the virtual data source to physical machine (via the Modbus I/P data source) and shared process variables.

          Please let me know if I am missing something.

          I have been working on a script to iterate over a set of data sources, which I hope is the beginning of a scripted data source to implement control logic and sequencing. While I have been able to use DataSourceQuery to obtain a list of data sources, I am having trouble accessing the associated list of data points.

          This code finds the desired set of data sources:

          var sources = DataSourceQuery.query('like(name,polecat*VIRT)&limit(10)');
          for(var i=0; i<sources.size(); i++ ) {
              source = sources.get(i);
              print( source.name );
          }
          

          However, I have been unable to modify it to access the data points, which documentation indicates should be available through source.points.

          What am I missing?

          Edit: In the documentation at [https://help.infiniteautomation.com/about-mango-java-script], I find this:

          *Data Source Queries
          Data sources can be queried in much the same way. There is one method, DataSourceQuery.query(stringRql); A returned object will contain:

          name - String
          xid - String
          enabled - boolean
          type - String
          points - List<DataPointWrapper>*

          I find that I can reference name, xid, and enabled in my script, but type and points are undefined.

          1 Reply Last reply Reply Quote 0
          • P Offline
            pyeager
            last edited by

            After further investigation, I find that type doesn't exist, but typeName does. Perhaps a documentation bug?

            While dumping one of the objects returned from DataSourceQuery, I find that points actually does exist, but when I try to reference it, I get "undefined".

            1 Reply Last reply Reply Quote 0
            • MattFoxM Offline
              MattFox
              last edited by

              Note: I'm using 3.5.6 until we're ready to upgrade..
              Your query will likely work better if you iterate rather than using get:

              var sources = DataSourceQuery.query('like(name,polecat*VIRT)&limit(10)');
              //I'm not confident with the like(name,polecat*VIRT), but hey if it works then good
              for(var i=0; i<sources.length; i++ ) {
                  source = sources[ i ];
                  print( source);
              }
              

              You should be able to print(source) and see all the associated datapoints in that datasource, regardless if the datasource is enabled or not. P.s I also found it to be typeName for the type of datasource. Most likely not been updated.

              My advice here that I can give you would be to look at tagging all of the datapoints instead and then using an RQL query to pull them in. Not only will it enable you to better index your point information but will also give you plenty of control over what points you want to pull.

              Do not follow where the path may lead; go instead where there is no path.
              And leave a trail - Muriel Strode

              P 2 Replies Last reply Reply Quote 0
              • phildunlapP Offline
                phildunlap
                last edited by

                Thanks for bringing that to our attention! It looks like everything should be working, but there is no getter in the DataSourceWrapper for the points, so its scope is private! It appears to have been this way since it was originally added, but I have fixed it for next release.

                For now, I think the easiest way to work around that would be a function in a global script to do the necessary reflecting to get out the points list. Like...

                function reflectPointsList(dataSourceWrapper) {
                    var pointsListField = dataSourceWrapper.getClass().getDeclaredField("points");
                    pointsListField.setAccessible(true);
                    return pointsListField.get(dataSourceWrapper);
                }
                
                var dsw = DataSourceQuery.byXid("DS_XID");
                var points = reflectPointsList(dsw);
                print(points)
                
                1 Reply Last reply Reply Quote 1
                • phildunlapP Offline
                  phildunlap
                  last edited by

                  @pyeager said in Best way to structure for multiple instances of device:

                  After further investigation, I find that type doesn't exist, but typeName does. Perhaps a documentation bug?

                  Yes, thanks for bringing this to our attention, as well!

                  1 Reply Last reply Reply Quote 0
                  • P Offline
                    pyeager
                    last edited by

                    Thanks, Phil! That should get things moving.

                    One more thing for your attention....

                    Here is some example code from [https://help.infiniteautomation.com/about-mango-java-script]

                    var points = DataPointQuery.query('like(name, *boiler*)&limit(10)');
                    var average = 0.0;
                    
                    LOG.info('Found ' + points.size() + ' points to compute average.');
                    
                    for(var i=0; i<points.size(); i++){
                        if(points.get(i).runtime !== null){
                          LOG.info('Adding ' + points.get(i).runtime.value + ' to the average.');
                          average = average + points.get(i).runtime.value;
                        }
                    }
                    
                    return average/points.size();
                    

                    It might be considered a "nitpick", but if any of the data points are disabled, the computation will be incorrect.

                    Why? Because the last line divides by the number of points, not the number of enabled points.

                    1 Reply Last reply Reply Quote 0
                    • P Offline
                      pyeager @MattFox
                      last edited by

                      @mattfox thanks for the suggestion regarding iteration. It looks much cleaner your way. I just followed the method used in the Mango documentation. Haven't done much javascript for quite a while!

                      1 Reply Last reply Reply Quote 0
                      • phildunlapP Offline
                        phildunlap
                        last edited by

                        Great!

                        Yeah, that may be nitpicky ;). You could so easily count the number of enabled points when you're checking the runtime not being null...

                        1 Reply Last reply Reply Quote 0
                        • MattFoxM Offline
                          MattFox
                          last edited by

                          Happy to help, my javascript ain't too bad IMO. I've got nothing on Jared or Phil mind you but always here if needed.

                          Do not follow where the path may lead; go instead where there is no path.
                          And leave a trail - Muriel Strode

                          1 Reply Last reply Reply Quote 0
                          • P Offline
                            pyeager @MattFox
                            last edited by

                            @mattfox said in Best way to structure for multiple instances of device:

                            My advice here that I can give you would be to look at tagging all of the datapoints instead and then using an RQL query to pull them in. Not only will it enable you to better index your point information but will also give you plenty of control over what points you want to pull.

                            Thanks for the suggestion, Matt!

                            However, I have a question.

                            If querying the data source gets me all the points I need, where is the advantage?

                            MattFoxM 1 Reply Last reply Reply Quote 0
                            • MattFoxM Offline
                              MattFox @pyeager
                              last edited by MattFox

                              @pyeager

                              @pyeager said in Best way to structure for multiple instances of device:

                              If querying the data source gets me all the points I need, where is the advantage?

                              In this case you may have additional info regarding location or logger brand etc etc.
                              Which if in the case of all being in the data sources then that's fine.

                              My other reasoning is prior to tags being introduced, I used to do the same thing and link by name or data sources etc Which is fine for single site installs. But what if you find you suddenly need only some particular devices, or a sensor or something is installed which feeds back in different units or magnitude? Or even you're using a mango unit on multiple locations and need different level access? User permissions can get you so far but you cannot always rely the name of the device being consistently named.

                              My other point is being able to also create generic dashboards that pull data by query rather than individually mapping them. You may be able to individually name them differently, but what if you have the same issues as above? This is where tags are handy.

                              This is my 10 cents. (2 cents plus inflation and tax etc etc..) And in my case I have dozens of sites and users to take into account with their own locations (some shared) and not necessarily using the same hardware we have.

                              Do not follow where the path may lead; go instead where there is no path.
                              And leave a trail - Muriel Strode

                              P 1 Reply Last reply Reply Quote 0
                              • P Offline
                                pyeager @MattFox
                                last edited by

                                @mattfox @phildunlap

                                Where is DataSourceQuery documented? I haven't yet found full documentation for that or much else. I've been figuring things out from the examples.

                                1 Reply Last reply Reply Quote 0
                                • MattFoxM Offline
                                  MattFox
                                  last edited by MattFox

                                  Get yourself into the mangoUI and go to Administration -> Watchlist builder then select query as the watchlist type. It will allow you to put together a query and teach you what parameters are available. That was my go to for the first month or so when I started learning RQL.

                                  EDIT: As for the datasource query info, I've not got much. It honestly looks like you use it just to pull datasource property information in order to pull points by it's XID / id.

                                  Do not follow where the path may lead; go instead where there is no path.
                                  And leave a trail - Muriel Strode

                                  1 Reply Last reply Reply Quote 1
                                  • P Offline
                                    pyeager
                                    last edited by pyeager

                                    @phildunlap

                                    function reflectPointsList(dataSourceWrapper)
                                    

                                    works perfectly.

                                    Do I understand correctly that it is a workaround until the next release?

                                    1 Reply Last reply Reply Quote 0
                                    • phildunlapP Offline
                                      phildunlap
                                      last edited by

                                      Correct, although it should continue to work even in subsequent releases. But, once 3.6.2 is released the concerns raised in this thread will have been fixed,

                                      1. points list not available via .points
                                      2. JavaScript help says type not typeName while .typeName is correct
                                      1 Reply Last reply Reply Quote 1
                                      • P Offline
                                        pyeager
                                        last edited by

                                        This post is deleted!
                                        1 Reply Last reply Reply Quote 0
                                        • First post
                                          Last post