To follow this section, run Visual Studio Code (VS code) and open the folder ../smartserver-iot/apps/DspSPcontroller
of your local copy of the izot/smartserver-iot repo, from the archive you downloaded previously. You also need to confirm that your SmartServer IoT has a firewall rule to allow MQTT connections. In an SSH console, the command: sudo ufw status
is used to review the available rules, and the command sudo ufw allow 1883
is used to open the required port, if TCP port 1883 is not listed.
The SmartServer IoT has 32 internal devices. These devices are used for infrastructure support for LonTalk Routing and network interface. For example, if you have two U60 adapters connected, the SmartServer needs to create Lon Routing devices to connect the Internal IP70 channel to the pair of FT-10 channels. If you enable the IP-852 routing support, an additional router is needed. In this case, 6 internal LON devices are needed for infrastructure support.
As of release 2.51.001 of the SmartServer IoT, there is no built-in support for the SmartServer to create application devices that are assigned to the Internal Lon Devices. In the programming example we need to handle this operation. In this programming example, we work under the following assumptions:
Let's walk through the code. You should have the file dsp-sp-cntl_A.js open in VS code. There is a fair amount of code that is used from the Programming Tutorial which will will not be covered in detail here.
/* * SmartServer IoT Example application - First Porgression for DspSPcontroller application * */ "use strict"; const mqtt = require("mqtt"); /* UFPTDspSPcontroller interface (external name SpController) defined in apolloDev.typ 1.06 or higher: Inputs and CPs "SpContoller/0/nviAddRouge" SNVT_str_asc "SpContoller/0/nviRemoveRouge" SNVT_str_asc "SpContoller/0/nviShowRouge" SNVT_count "SpContoller/0/nviEnable" SNVT_switch "SpContoller/0/cpDefaultDspSP" SCPTductStaticPressureSetpoint "SpContoller/0/cpDelay" SCPTdelayTime "SpContoller/0/cpLoopInterval" SCPTmeasurementInterval "SpContoller/0/cpMaxDspSP" SCPTmaxDuctStaticPressureSetpoint "SpContoller/0/cpMinDspSP" SCPTminDuctStaticPressureSetpoint Outputs "SpContoller/0/nvoRougeReport" UNVTdevHanle (index), Device handle (string) "SpContoller/0/nvoAvgDemand" SNVT_lev_percent "SpContoller/0/nvoMinDemand" SNVT_lev_percent "SpContoller/0/nvoMaxDemand" SNTT_lev_percent "SpContoller/0/nvoDspSP" SNVT_press_p output must heartbeat and is intended as an input to the AHU controller ]; */ let myAppIf = { myAppPID : "9000010600038500", // This must match the PID for the XIF which defines this application myFbName : "SpController", myDeviceHandle : "myDev.1", // Your choice here, but must not already exist on your target SmartServer deadband : 2, gain: 1, nvoAvgDemand : -1.0, // SNVT_lev_percent 0-100 at 0.5 steps nvoMinDemand : -1.0, // SNVT_lev_percent 0-100 at 0.5 steps nvoMaxDemand : -1.0, // SNVT_lev_percent 0-100 at 0.5 steps nvoDspSP : 0, // SNVT_press_p pascals initialized : false, startup : true }; // Map object are used for keyed (device handle used for key) access to various aspects related to // monitored devices, and this application logic const myInputs = new Map(); let glpPrefix="glp/0"; // this will include the sid once determined let subscribtionsActive = false; let myAppTimer; |
npm install mqtt --save
. 'glp/0/././sid'
. The object myAppTimer is used to implement most of the control logic of the application in the function calculateDspSP().
function initializeInputs (interfaceObj) { // This function must not be called before the MQTT connection has be esstablished, and the // Internal device has been created and provision by IAP/MQ clearTimeout (myDevCreateTmo); // Cancel the auto internal device create // These key values used here must match the definiton of the profile. The developer // must recreate the keys with care for proper connection to update events myInputs.set ("nviAddRouge", interfaceObj.nviAddRouge.value); myInputs.set ("nviRemoveRouge", interfaceObj.nviRemoveRouge.value); myInputs.set ("nviShowRouge", interfaceObj.nviShowRouge.value); // The contoller is enabled by default. myInputs.set ("nviEnable", {state:1, value:100}); myInputs.set ("cpDefaultDspSP", interfaceObj.cpDefaultDspSP.value); myInputs.set ("cpDelay", interfaceObj.cpDelay.value); myInputs.set ("cpLoopInterval", interfaceObj.cpLoopInterval.value); myInputs.set ("cpMaxDspSP", interfaceObj.cpMaxDspSP.value); myInputs.set ("cpMinDspSP", interfaceObj.cpMinDspSP.value); // This application has one output to feed the Duct Static Pressure Setpoint of an AHU Controller // A 3 minute hearbeat and 5 pascal (.020 inches H20) propagation thresold will be applied // These parameters would normally be deviced a configuraton points for the FB. const pointDriverProps = { rate: 0, "lon.cfg" : { propagationHeartbeat: 180, propagationThrottle: 0, maxRcvTime: 0, propagationThreshold: 5 } }; // The following Interval timer is where the business logic for the internal application is // exectured. myAppTimer = setInterval(calculateDspSP, myInputs.get("cpDelay") * 1000); myAppIf.nvoDspSP = myInputs.get("cpDefaultDspSP"); // This IAP/MQ topic sets up the attributes for handling the primary output of this controller // The lon driver specific properties are set here for the point nvoDspSP client.publish ( `${glpPrefix}/rq/dev/lon/${myAppIf.myDeviceHandle}/if/${myAppIf.myFbName}/0/nvoDspSP/monitor`, JSON.stringify(pointDriverProps), {qos:1}, (err) => { if(err !=null) console.error ("Failed to set lon.cfg for nvoDspSP"); } ); // Initial output variable is being set client.publish( `${glpPrefix}/rq/dev/lon/${myAppIf.myDeviceHandle}/if/${myAppIf.myFbName}/0/nvoDspSP/value`, JSON.stringify(myAppIf.nvoDspSP) ) } // environ returns the value of a named environment variable, if it exists, // or returns the default value otherwise. function environ(variable, defaultValue) { return process.env.hasOwnProperty(variable) ? process.env[variable] : defaultValue; } const client = mqtt.connect(`mqtt://${environ("DEV_TARGET", "127.0.0.1")}:1883`); // If devHandle is not provided as an argument, it is assumed there is only one matching // device PID in the system // Subscribe to the segment ID topic. const sidTopic = `${glpPrefix}/././sid`; client.subscribe( sidTopic, (error) => { if (error) { console.log(error); } } ); |
../fb/dev/lon/if
topic to cache local copies of the interface values. It also configures the parameters associated with the primary output network variable to handle the heartbeat, and propagation thresholds to control network bandwidth loading. Line 104-122 connect to the IAP/MQ broker either locally (when the application runs on the SmartServer IoT) or to the DEV_TARGET environment variable that is defined by launch.json file in our VS code project. At this point you should edit the launch.json file in VS code to verify the program as shown in line 11 below, and edit line 12 to target the IP address of your SmartServer IoT as defined by the DEV_TARGET environment variable.
{ // Use IntelliSense to learn about possible attributes. // Hover to view descriptions of existing attributes. // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 "version": "0.2.0", "configurations": [ { "type": "node", "request": "launch", "name": "Launch Program", "program": "${workspaceFolder}\\dsp-sp-cntl_A.js", "env":{"DEV_TARGET":"10.1.129.7"} } ] } |
This block of code initializes SetTimeout that will execute 10 seconds into the future to automatically create an internal device according the parameters described in the createMyAppMsg object. This timer should only expire once the first time this application runs on your target SmartServer.
// This function will fire if the internal device for this application does not exist. // This is the case the very first time this application runs on the target SmartServer. // This device will exist on the SmartServer until it is deleted by user action in the CMS // or the SmartServer Apollo-reset normal... is executed. const myDevCreateTmo = setTimeout (() => { console.log("Creating the internal device for this application"); let createMyAppMsg = { action: "create", args: { unid: "auto", type: myAppIf.myAppPID, "lon.attach": "local" } } // CreateMyAppMsg {} client.publish( `${glpPrefix}/rq/dev/lon/${myAppIf.myDeviceHandle}/do`, JSON.stringify(createMyAppMsg) ) }, // IAP/MQ do {action: "create"} 10000 ); //10s timeout to determine if the Internal device exists |
The remaining code for dsp-sp-cntl-A.js is shown in the code block below. At line 146 is the message handler for the IAP/MQ topic: glp/0/././sid
. This topic is used to establish the sid for your target SmartServer. This is the first topic the application subscribes to. This handler appends the returned sid to the glpPrefix string, which forms the first part of all IAP/MQ topics from this point forward. It is typical for this message to initiate subscriptions to other IAP/MQ topics. In this example, at line 158 the client.subscribe method connects to the interface of this internal device in the fb channel. This will generate messages with an actual interface definition if the device has already been created and provisioned.
Line 160 registers this application to receive update events, which are updates to it's input network variables and CPs. Each modification to any interface element (network variable or configuration property) will generate an update on this topic '../ev/updated/dev/lon/type/9000010600038500'
This application only creates a single internal device using this program ID.
Line 175 is the handler for the updated events just mentioned. All updated events for input network variables and configuration properties are cached in the myInputs Map object. Lines 182-187 show the application acting directly on updated events to specific inputs.
Line 206 establishes the message as an Interface object for this application. Take note that the payload check for null is required because a deleted device will appear in the feedback channel after you delete the device in the CMS. If the payload is a valid interface message, the initializeInputs() function is called at line 213. This will cache the current values, and start the application logic.
function handleSid (sidMsg) { // Assuming the SID topic is a string sidMsg let nowTs = new Date(); // Seconds TS good enough if (typeof(sidMsg) === typeof("xyz")) { if (sidMsg.length > 0) { glpPrefix += `/${sidMsg}`; console.log(`${nowTs.toISOString()}- SmartServer SID: ${sidMsg}`); // Note how the "+" wild card is used to monitor device status for // any lon device. Note, once we know the SID, we need to avoid // adding multiple listeners. Subscbribe once and only once if (!subscribtionsActive) { // The following fb topic will capture the fact that the internal lon device has been created/provisioned client.subscribe (`${glpPrefix}/fb/dev/lon/${myAppIf.myDeviceHandle}/if/#`); // NVs and CP updates appear on this event channel client.subscribe (`${glpPrefix}/ev/updated/dev/lon/type/${myAppIf.myAppPID}`); client.unsubscribe (sidTopic); subscribtionsActive = true; } else { console.log(`${nowTs.toISOString()} - Redundant SID topic message`); } } else { // We are not provisioned. sid = undefined; cosole.log(`${nowTs.toISOString()} - [${s}] Problem with SID payload.`); } } else { console.error('The sid topic returned an unexpected payload type.') } } function handleMyPointUpdates (updateEvent){ // The values published in calculateDspSP() will generate updated events if (updateEvent.datapoint.search("nvo") !== -1) return; // Record value in the Interface map myInputs.set(updateEvent.datpoint, updateEvent.value); // If required, at check for nvi and CPs values that require generate a event driven response if (updateEvent.datapoint === "nviEnable") DspSP = updateEvent.datpoint.value.state == 1 ? myInputs.get("cpDefaultDspSP") : myInputs.get("cpMinDspSP"); if (updateEvent.datapoint === "cpLoopInterval") { clearInterval(myAppTimer); myAppTimer = setInterval(calculateDspSP, updateEvent.value * 1000); } console.log(`${updateEvent.datapoint}: ${JSON.stringify(updateEvent.value)}`); } // IAP/MQ. MQTT message handler. client.on( "message", (topic, message) => { try { const payload = JSON.parse(message); let devHandle; let nowTs = new Date(); // Seconds TS good enough if (topic === sidTopic) { // Assuming the SID topic is a string payload handleSid(payload); } // This topic applies to only one device in the system, this internal device if (topic.endsWith(`${myAppIf.myDeviceHandle}/if/${myAppIf.myFbName}/0`)) { // Note - The payload is null when the device is in the deleted state if (payload == null) return; // if (!myAppIf.initialized) { client.unsubscribe(`${glpPrefix}/fb/dev/lon/${myAppIf.myDeviceHandle}/if/#`); initializeInputs(payload); myAppIf.initialized = true; console.log("myDev.1 internal device interface is ready!"); } } // The IAP channel ../ev/updated will carry updates for NVs/CPs for internal Lon Devices. // note that more than one instance could exist. The Updated event has different // structure than a data event. if (topic.endsWith (`/ev/updated/dev/lon/type/${myAppIf.myAppPID}`)) { if (payload.handle == myAppIf.myDeviceHandle) { handleMyPointUpdates(payload); } } } catch(error) { console.error("MQTT Message: " + error); } } // onMessage handler ); // onMessage registration function calculateDspSP() { let nowTs = new Date(); // Seconds TS good enough if (myAppIf.startup) { clearInterval (myAppTimer); // Setup regular interval processing of algrithm myAppTimer = setInterval(calculateDspSP, myInputs.get("cpLoopInterval") * 1000); myAppIf.startup = false; } console.log (`${nowTs.toLocaleTimeString()} - SpContoller Processing`); // TODO: Implement control sequence } |
You can run this application in the VS Code debugger as long as you have port 1883 open on your SmartServer IoT target. Follow these instructions:
File.Open Folder
, and navigate to your local copy of the repo folder: ..\smartserver-iot\apps\Dsp SP Controller Example\dspSPController
View.Terminal
, in the terminal window type: npm install
. Your development computer needs an internet connection for this command to succeed..vscode\lauch.json
. You need to make adjustments as shown in the following screenshot:You have completed the walk-through of this dsp-sp_cntl_A.js internal application. This application detects whether there is an existing representation of itself defined in the SmartServer. If not found, it will create and automatically provision an internal application device. You will see this occur shortly after the application prints 'Creating the internal device for this application' to the console. When the interface is reported in the fb channel, the application will initialize local copies of the inputs, and start the interval timer to run the algorithm that will be completed in the next revision. This application also receives updated events from IAP/MQ to drive the application.
Return to the previous page, or continue to the completed dsp-sp-cntl_B.js example: Duct Static Pressure Setpoint Controller Full Example