DR-JSON Manual
DR-JSON Manual
December 9, 2019
Niagara JSON Toolkit Guide
Tridium, Inc.
3951 Westerre Parkway, Suite 350
Richmond, Virginia 23233
U.S.A.
Confidentiality
The information contained in this document is confidential information of Tridium, Inc., a Delaware
corporation (“Tridium”). Such information and the software described herein, is furnished under a license
agreement and may be used only in accordance with that agreement.
The information contained in this document is provided solely for use by Tridium employees, licensees, and
system owners; and, except as permitted under the below copyright notice, is not to be released to, or
reproduced for, anyone else.
While every effort has been made to assure the accuracy of this document, Tridium is not responsible for
damages of any kind, including without limitation consequential damages, arising from the application of the
information contained herein. Information and specifications published here are current as of the date of this
publication and are subject to change without notice. The latest product specifications can be found by
contacting our corporate headquarters, Richmond, Virginia.
Trademark notice
BACnet and ASHRAE are registered trademarks of American Society of Heating, Refrigerating and Air-
Conditioning Engineers. Microsoft, Excel, Internet Explorer, Windows, Windows Vista, Windows Server, and
SQL Server are registered trademarks of Microsoft Corporation. Oracle and Java are registered trademarks
of Oracle and/or its affiliates. Mozilla and Firefox are trademarks of the Mozilla Foundation. Echelon, LON,
LonMark, LonTalk, and LonWorks are registered trademarks of Echelon Corporation. Tridium, JACE,
Niagara Framework, and Sedona Framework are registered trademarks, and Workbench are trademarks of
Tridium Inc. All other product names and services mentioned in this publication that are known to be
trademarks, registered trademarks, or service marks are the property of their respective owners.
December 9, 2019 3
Contents Niagara JSON Toolkit Guide
Chapter 5 Components...................................................................................65
JsonSchema (Json Schema).......................................................................66
Config (Json Schema Config Folder)..........................................................67
Tuning Policy (Json Schema Tuning Policy).......................................69
Overrides (Json Schema Overrides Folder) ......................................70
Debug (Json Schema Debug Folder) .........................................................70
Schema Output History Debug (Schema History Debug) ..................71
Metrics (Json Schema Metrics) ........................................................71
Queries (Json Schema Query Folder).........................................................73
Query (Json Schema Query)............................................................73
RelativeHistoryQuery (Relative History Query).................................74
BoundQueryResult (Json Schema Bound Query Result) ....................74
Base Query (Base Query) ................................................................75
RelativeJsonSchema (Relative Json Schema) ..............................................75
JsonSchemaService (Json Schema Service) ................................................76
S M A Expiration Monitor (S M A Expiration Monitor).......................77
Global Cov Slot Filter (Subscription Slot Blacklist) ............................78
Object (Json Schema Object) ....................................................................78
BoundObject (Json Schema Bound Object) ...............................................79
Array (Json Schema Array) ........................................................................80
BoundArray (Json Schema Bound Array.....................................................80
FixedString (Json Schema String Property) ................................................81
FixedNumeric (Json Schema Numeric Property) ........................................81
FixedBoolean (Json Schema Boolean Property)..........................................82
Count (Json Schema Count Property) ........................................................82
CurrentTime (Json Schema Current Time Property)....................................82
UnixTime (Json Schema Unix Time Property) .............................................83
BoundProperty (Json Schema Bound Property)..........................................83
BoundCSVProperty (Json Schema Bound Csv Property) .............................84
Facet (Json Schema Facet Property)..........................................................85
FacetList (Json Schema Facet List).............................................................85
Tag (Json Schema Tag Property) ...............................................................86
TagList (Json Schema Tag List)..................................................................87
Query (Json Schema Query)......................................................................87
RelativeHistoryQuery (Relative History Query)...........................................88
BoundQueryResult (Json Schema Bound Query Result) ..............................88
JsonAlarmRecipient (Json Alarm Recipient) ...............................................89
4 December 9, 2019
Niagara JSON Toolkit Guide Contents
Index...............................................................................................................111
Glossary ..........................................................................................................113
December 9, 2019 5
Contents Niagara JSON Toolkit Guide
6 December 9, 2019
About this guide
The JSON Toolkit provides a way to easily extract data from a station as well as a way to input information
to control connected devices.
The beginning chapters introduce data output and input. The Developer Guide chapter explains how to ex-
tend toolkit features.
December 9, 2019
Initial document release
Related documentation
These documents provide additional information about how to construct data models using the Niagara
Framework.
In te r n al resou rce s
• Niagara Developer Guide
• Niagara Drivers Guide
• Niagara Graphics Guide
• Bajadoc (accessed through the Workbench Help system)
Ex t er n al re sourc es
• Java Platform Standard Edition 7 Documentation: https://docs.oracle.com/javase/7/docs/
• Unix time: https://en.wikipedia.org/wiki/Unix_time
• Chart.js, JavaScript charting for designers and developers: https://www.chartjs.org/
• Google Chart: https://developers.google.com/chart
Chapter 1 Introd ucti on
Topics covered in this chapter
♦ Quick JSON example
♦ JSON Toolkit use cases
♦ Transport protocols
♦ Feature summary
♦ Comparison to alternatives
♦ License requirements
♦ JSON schema service
♦ Supervisor
The JSON Toolkit module adds functionality to the Niagara Framework®, enabling you to export JSON
data (payloads) from a station, or, when importing data, to influence the station in some way. A schema
generates a payload for export, whilst a handler processes imported JSON. The Toolkit is intended to give
you the power to adapt as needed.
JSON (JavaScript Object Notation) is a simple, lightweight, data encoded string. Used for data interchange
since 2002 to communicate between a web browser and a server for the Javascript language, it has gained
popularity and is used in many scenarios beyond those implemented in 2002. Many IoT devices can easily
receive a JSON payload.
December 9, 2019 9
Chapter 1 Introduction Niagara JSON Toolkit Guide
On the left is the universe of data available to a Niagara Station. The station database provides some data;
other data can come from outside the station. The Schema contains configuration properties, which set up
its functions. A schema updates when a CoV triggers a generation action from a bound entity or a person
invokes the generate action on a schema Property Sheet. This causes any linked properties of the JSON
Object payload to create the Output string, which retrieves and routes the data onward through a Transport
Point and Handler to an alarm recipient, the cloud or other destination, and back to the device for asset
control, such as to control the lighting in a home or acknowledge an alarm.
The format of the JSON output string is relatively simple, organised into a list of key:value pairs, with
support for data types: Numeric, Boolean, Enum and String much like Niagara points. JSON messages can
use any sequence of objects, arrays and key/value pairs. The JSON Toolkit is flexible. You, as a developer
can extend the Toolkit or use APIs to access station data. You can drag schema elements around and change
the order of the messages.
JSON supports two data structures: objects and arrays. Complexity emerges from these simple constructs
mainly due to the variation in expected payload between different pieces of software, and also their
expected encoding of non-primitive types, such as date and time. This is where the demand arises for a
flexible solution to marshal Niagara’s rich object model to and from the JSON format.
You can extend the Toolkit or use APIs to access a station’s data. JSON can post data to APIs for data
transmission. For example, using the inbound components of the JSON Toolkit, external systems can send a
JSON-encoded message to a Niagara station to change a setpoint or acknowledge an alarm.
As the data manipulator, you set up data retrieval and use by creating links between JSON objects. Each
schema contains a single root object, which itself contains the JSON objects that establish the links.
10 December 9, 2019
Niagara JSON Toolkit Guide Chapter 1 Introduction
From the station to the destination, you link the output string, typically via MQTT, to a string-publish point,
which sends the payload to a topic in the broker that forwards (transports) it to the destination system. The
schema itself is transport agnostic. Linking produces the desired result.
For example, just as an external oBIX client can poll a station for data, the JSON output can be retrieved via
an HTTP GET request to a URL that exposes its contents as a web servlet. Using JSON, you can have the
same rich data that oBIX provides without the pre-defined oBIX format. Using JSON, you choose the format
of the data, which affords total flexibility.
MQTT brokers can link the output of a JSON schema to a cloud platform, such as Bluemix, Google Cloud,
and AWS.
December 9, 2019 11
Chapter 1 Introduction Niagara JSON Toolkit Guide
• Machine learning
• Analytics
• Data archival
Tr ans po r t pro to c o ls
The JSON Toolkit itself does not mandate the transport protocol used.
Potential transport protocols include:
• MQTT (by linking the JSON schema output to an Mqtt String Publish Point)
• HTTP(s)
• Box (bajaux widget)
• File
These options may be valid for both incoming and outgoing JSON payloads. When linking to a publish or
subscribe control point, you may need to use an Engine Cycle Message Queue component to ensure that
the schema outputs all messages to the linked transport.
Fea ture s um m ar y
The JSON Toolkit supports a significant list of features and options to aid the engineering effort.
• Customer-definable JSON payloads
• Payloads for types other than points, such as, tags and facets
• Payloads for histories, series transform, and alarms
• Data selection using either bindings (ords), bql, or the addition of markers that directly identify points
• Support for encoding alarm events via a JsonAlarmRecipient
• The ability to respond to incoming requests to change a setpoint or acknowledge an alarm by uuid
• On-demand payloads generated by a CoV
• Topic generation for a relative schema (for example, mqtt publish topic)
• Tuning policies to throttle output
• Program object-based overrides
• The ability for developers to extend the toolkit
Comparison to alternatives
JSON Toolkit alternatives include oBIX and bajaScript.
• oBIX (obix.org) provides a comprehensive connectivity option for Niagara, the JSON Toolkit differs by of-
fering a flexible, user-defined payload, and support for publish-on-change.
• bajaScript (bajascript.com) provides a means to access the Niagara component model with convenient
support for complex objects, subscription, action invocation, and querying.
• Project Haystack (project-haystack.org) offers both a common semantic model and a protocol to enable
the exchange of data. These tags can be included in the payloads generated by the JSON Toolkit.
In contrast to the above options, the JSON Toolkit does not dictate the protocol or layout used to exchange
data. This could be an advantage when dealing with charting libraries and cloud service providers who ex-
pect to send or receive data in a specific format.
12 December 9, 2019
Niagara JSON Toolkit Guide Chapter 1 Introduction
The Spy page for the service maintains a registry of the export markers contained within the station. This
registry might prove useful when debugging issues with relative schema used in conjunction with the export
marker paradigm. In the event that setpoint changes are received, the register aids in finding the marked
points.
December 9, 2019 13
Chapter 1 Introduction Niagara JSON Toolkit Guide
Configuring a setpoint handler for incoming JSON No The set operation only succeeds if the Run As User is a real
user who has operator write permission on the slot target.
Defining a schema for exporting JSON Yes When set, the data value of the exported slot defaults to an
empty string unless the Run As User is a real user who has
operator read permission on the slot.
N O T E : Run As User is important for security. This property may only be set by a super user.
Supervisor
The most convenient deployment of the jsonToolkit in cloud connectivity is to connect directly from the con-
troller schema to the remote transport. However, if a controller does not have remote connectivity, a Super-
visor is required. There are a few options to consider.
System database
Use the system database to index subordinate controllers and sys: ords for queries within a Supervisor
schema.
• Example schema query: station:|sys:|neql:n:point|bql:select name, out.value, out.
status
14 December 9, 2019
Niagara JSON Toolkit Guide Chapter 1 Introduction
December 9, 2019 15
Chapter 1 Introduction Niagara JSON Toolkit Guide
16 December 9, 2019
Chapter 2 Exporting with a JSON
schema
Topics covered in this chapter
♦ Config folder
♦ Tuning policy
♦ Overrides
♦ Debugging errors (Schema History Debug)
♦ JSON schema metrics
♦ Schema construction
♦ Queries
♦ Alarms
♦ Exporting schema output (JsonExporter)
♦ Exploring the examples
Adding a JsonSchema component to your station allows for the construction of a JSON payload to suit the
requirements of your particular application.
Overview
There are examples in the j s o n T o o l k i t palette, which may help with learning how to construct a schema. You
can simply drag the JsonExampleComponents and Schemas folders into a running station to work with
them.
You construct a schema by placing “entities” from the j s o n T o o l k i t palette below a JsonSchema in the
station and then use configuration properties and queries to get the output you want. Use the numbers in
the screen capture to learn about schema elements:
1. You can give each schema a unique name.
2. The Output property contains the resulting JSON payload (message or string).
December 9, 2019 17
Chapter 2 Exporting with a JSON schema Niagara JSON Toolkit Guide
N O T E : The format of this black box (with new lines and spacing) is purely for presentation. The actual
string output is minified and does not contain extra spaces.
3. The Enabled property turns the generation of output, execution of queries and subscription to bound
values on and off. The Config folder contains properties that configure general schema attributes.
4. The Queries folder can contain query entities to insert bql, historical or alarm database content into a
payload.
5. A { } root object or an [ ] array contains JSON entities that structure the Output message. Some
entities may be simple—for example braces { } represent a simple JSON object, while other entities
represent Niagara bql queries (refer to the Niagara Developer Guide) and, therefore, have the potential
to be more complex.
6. Actions build and manage schema contents. The Generate action builds and updates schema output.
For relative schemata, Generate evaluates the base query and publishes the results for each resolved
base item.
What can a Schema contain?
The schema supports a nested structure of child entities. These can be Objects, Arrays, or Properties of
various types. Niagara alarm, history or point data may populate these entities, which include:
All entities (minus Property) support nested child entities. This lets you build a schema using a tree structure
with entities found in the j s o n T o o l k i t palette.
Wha t st ru c tu re is al lo we d?
Every schema requires a root member that is allowed by the JSON standard: this means an object { } or an
array [ ].
The screen capture shows how Niagara represents a JSON root object in a standard P r o p e r t y S h e e t view.
Config folder
The JSON Toolkit provides several options to help you create consistent naming and formatting. The root
properties in each schema’s C o n f i g folder provide these consistency properties.
The schema C o n f i g folder is separate from the station C o n f i g folder and applies only to the parent schema.
18 December 9, 2019
Niagara JSON Toolkit Guide Chapter 2 Exporting with a JSON schema
Tuning policy
Most tuning policies properties are explained by the Niagara Drivers Guide.
Tuning properties provide rules for evaluating when JSON outputs data and for indicating an update strat-
egy. Configuring this policy can affect system performance. They are located in the Schema C o n f i g folder.
F ig ure 8 Tuning policy properties
Update Strategy determines when JSON string generation occurs: at change-of-value or on demand.
There is a built-in Min Write Time to ensure that hundreds of concurrent CoV changes over a short time do
not result in a deluge of JSON messages. For example, when set to five (5) seconds and a change-of-value
occurs within five seconds of the last change of value, schema generation defers for a full five seconds. How-
ever, if this amount of time exceeds the Max Write Time setting, the system forces schema generation. In
contrast, Max Write forces an update after the specified interval.
N O T E : A Force Generate Json action overrides all tuning policy settings.
Export markers applied to numeric points also have a CoV Tolerance property which can be used to throt-
tle output.
The Write On Start and Write On Enabled properties provide other ways to invoke schema generation,
for example, when the station starts.
Overrides
An O v e r r i d e s folder is a standard container under the JSON C o n f i g folder.
December 9, 2019 19
Chapter 2 Exporting with a JSON schema Niagara JSON Toolkit Guide
This folder adds a TypeOverride component to the schema, should it be necessary to override how the
schema converts specific datatypes to JSON. The override applies to anywhere the system encounters the
data type in the entire schema. Examples might be:
• replacing Facets with a locally-understood value, such as ‘degC’ to ‘Celsius’
• defining a different format for simple types, such as Color and RelTime
• managing expectations for +/- INF in a target platform
For further information, refer to the “Type Override Example” in the “Developer Guide” chapter of this
document.
Deb u gg i ng er ror s ( Sc he m a H i st or y D eb ug )
When output updates rapidly, such as when a link calls a generate JSON action in quick succession or a rela-
tive schema quickly changes output once per base item, it may be useful to view the most recent output his-
tory. This task describes how to view the output history.
P r e r e q u i s i t e s : You are viewing the Property Sheet for the schema.
Step 1 Do one of the following:
• Click the O u t p u t H i s t o r y button to the right of the Output property on the schema.
• Expand the schema’s C o n f i g → D e b u g folder, right-click the S c h e m a O u t p u t H i s t o r y D e b u g
slot, and click V i e w s → S p y R e m o t e .
The S c h e m a O u t p u t H i s t o r y D e b u g view opens.
The History Size allows you to store more but be careful not to fill memory with JSON strings.
Step 2 To configure the amount of debug data stored in the station, expand the S c h e m a O u t p u t H i s t o r y
D e b u g folder and configure the History Max Size property.
It is a good idea to reduce this value once you have finished debugging.
Fi gu re 9 Schema metrics
These help with sizing and provisioning capacity from a cloud platform by estimating the traffic a station is
likely to generate with a given JSON schema. They may also assist in identifying performance problems. De-
bugging can be assisted by using the reset action.
The metrics provide three categories of performance information: query performance, generate perform-
ance, and subscription performance.
20 December 9, 2019
Niagara JSON Toolkit Guide Chapter 2 Exporting with a JSON schema
Last Query Fail Reason Last Schema Generation Fail Reason Subscription Events Ignored
Resolve Errors
Schema construction
Setting up a schema involves binding station data to JSON entities.
Slot selection
When picking a bound object or array, you may choose which slots from the target to include in the resultant
JSON container. Currently the options are:
• All slots
• All visible slots (hidden slots excluded)
• Summary slots—only those with a summary flag
• Selected slots—manually-selected slots from a list
T a rg e t t y p e s
N O T E : When choosing the bind target for a binding you could select any type of slot, from devices to con-
trol points to out slots to simple values, there are no restrictions.
Bound arrays and objects output the value of each of the selected slots (refer to Slot Selection, page 21).
The default behaviour for each encountered slot type is as follows:
Selection Output
December 9, 2019 21
Chapter 2 Exporting with a JSON schema Niagara JSON Toolkit Guide
Selection Output
Double and float decimals A JSON number rounded to use the schema’s decimal places config
AbsTime A String representation of the date formatted as per the schema config
Control Point A JSON String, Numeric, Boolean, or Enum to represent the out slot’s value
Status Value A JSON String, Numeric, Boolean, or Enum to represent the value
Anything else The string representation of the value as returned from the framework. This is often the
type display name.
N O T E : Bound objects and arrays do not recurse. Only direct child slots are included. These behaviours make
a few assumptions about the most-expected case, for example, excluding the status string from certain
types. Program overrides may override all these behaviours.
Naming
For binding results you may choose what the key is in the key/value pair:
Selection Output
Target Path The absolute path of the target from the root of the component tree
T I P : You may use a Tag property with the name n:name to include point names. This property inserts a sin-
gle tag value from the bound component in the output. If the SearchParents property is true, the frame-
work searches up the hierarchy for the closest component with a matching tag id (if the tag not found on
binding target.
Entities
Entities are objects, arrays, properties and bound properties.
Objects
Objects are entities used to create containers in the JSON message and identify slots in a target ord.
• A JSON schema Object inserts into the schema an empty named container ({ }) for holding other sche-
ma entities.
• A BoundObject is a named JSON object whose child name and value pairs are the slots within a target
ord.
22 December 9, 2019
Niagara JSON Toolkit Guide Chapter 2 Exporting with a JSON schema
Arrays
Arrays contain a list of values. They do not include names.
• A JSON schema Array inserts into a schema an empty named container ([ ]) for the purpose of holding
other schema entities.
• A JSON schema BoundArray is a named JSON object that renders values as a list.
Fi xe d prope r ti e s
Fixed Properties are hard-coded name and value pairs, which you always want to appear as constants in
the JSON string. You can link to these if the value is expected to vary. The next generation event, triggered
by a CoV on a bound entity or by the invocation of the Generate action, includes the current value. A
change in the value of any fixed property does not trigger a CoV generation event in the same way that a
bound equivalent does.
• A FixedString property inserts a string value.
• A FixedNumeric property inserts a numeric value.
• A FixedBoolean property inserts a Boolean value.
Bou nd prope r t ie s
A bound property inserts the current value of the object specified in the binding.
December 9, 2019 23
Chapter 2 Exporting with a JSON schema Niagara JSON Toolkit Guide
Fi gu re 12 Bound properties
BoundProperties include:
• A BoundCSVProperty is a named JSON string that renders child slots as a string, comma–separated list
with no surrounding [ ] or { }.
• A Tag property is a list of name and value properties based upon selected tags found on a binding tar-
get. If the tag is not found on the binding target, andSearchParents is true, the framework searches
up the hierarchy for the closest component with a matching tag id.
• A TagList is a list of name and value property pairs that are based upon selected tags found on a bind-
ing target. A comma-separated list specified in the Tag Id List Filter property can limit the tags to
be included in the output. Example: n:name, n:type or * for all. If Include Namespace is true, the
tag dictionary prefix is added to the key (for example, the hs: is added to hvac to give: hs:hvac).
NOTE:
Facet and Tag properties are not bound like the other bindings, in that changes of value do not prompt
schema generation. The current value is retrieved from the station when the schema generates.
Fi gu re 13 Json Schema Tag List
• A Facet property inserts a single facet value from a bound component into the schema output, for ex-
ample, the units of the current point.
• A FacetList inserts a list of name and value facet properties based on a comma-separated list or * for
all. Add facet keys as follows: units, mix, max
• Message properties
24 December 9, 2019
Niagara JSON Toolkit Guide Chapter 2 Exporting with a JSON schema
• A Count property is a named numeric value, which increments by 1 on each schema generation. Could
be used for message IDs.
• A CurrentTime property inserts the current time as set up in the C o n f i g folder’s Time Format
property.
• UnixTime property inserts the current time in Unix time as seconds from January 1, 1970.
Step 3 To view the schema P r o p e r t y S h e e t , double-click the schema glyph in the Nav tree.
The P ro p e r t y S h e e t opens.
December 9, 2019 25
Chapter 2 Exporting with a JSON schema Niagara JSON Toolkit Guide
When you initially view the P r o p e r t y S h e e t for a new schema, the Output property is an empty
black box. JSON strings appear here when you generate output.
Step 4 In the P r o p e r t y S h e e t view, ensure that the Enabled property is set to true.
Setting Enabled to false prevents the generation of output, the execution of queries and the
subscription to bound values.
Step 5 To begin setting up the message, expand the O b j e c t s folder in the palette, drag an O b j e c t to the
P r o p e r t y S h e e t and name it, for example, root.
Braces { } represent this object in the Output. This single top-level object serves as the JSON pa-
rent container for other JSON objects that make up the message. Each JSON object requires a pair
of braces ({ }) and arrays require brackets ([ ]).
Step 6 Drag an object, array, or property from the palette to the P r o p e r t y S h e e t root container.
Some objects may be simple and other objects may yield the more complex results of Niagara bql
queries. The objects that you choose to add depend on your unique requirements.
• Empty braces { } icons represent a JSON object. A bound object is a named object whose child
name and value pairs are the slots within an ord target.
• Bracket [ ] icons represent an array, which is an empty named container of other schema enti-
ties. A bound array is a named object that renders values as a list.
• Other icons represent properties, which may be fixed or bound.
Step 7 To update the schema Output based on the current values retrieved from the station, click G e n e r-
a t e , or right-click the schema name and click A c t i o n s → G e n e r a t e J s o n .
This action causes a regular schema to re-evaluate any query and populate the Output box with
JSON.
Step 8 To set up some actual station data, drag in a BoundObject, name it appropriately, expand the
bound object and click the Select Source finder to the right of the Binding property.
This object requires a binding similar to the way components on Px pages require bindings to ac-
tual points in a station.
The C h o o s e c o m p o n e n t / s l o t f o r J S O N window opens.
Step 9 Navigate to and select the source component, click O K and then click S a v e .
26 December 9, 2019
Niagara JSON Toolkit Guide Chapter 2 Exporting with a JSON schema
When choosing the target for a binding, you can select any type of slot, from devices to control
points to out slots to simple values. There is no restriction. Due to subscription, saving the schema
also generates the JSON message (output).
If your logic contains one or more points whose values change periodically, the schema generates a
new JSON message every time a CoV occurs. If the schema is connected to MQTT, the schema can
send each new message to the web.
Step 10 To change the Json Name (a read-only property) to the name of the bound input slot on your W i r e
S h e e t , change Json Name Source property to Target Name, and, from the Slots To Include
property, choose Summary Slots.
To include specific slots, use the Slots to Include properties, identify and pick individual slots
for more fine-grained control.
You may link the output slot to an EngineCycleMessageQueue, if required, which buffers output sent to
the onward transport. These could be MQTT or HTTP depending on the onward linked point.
December 9, 2019 27
Chapter 2 Exporting with a JSON schema Niagara JSON Toolkit Guide
N O T E : A best practice is to limit the scope of the base query to a subset of points in the station and limit
the frequency of JSON message generation. Very frequent payload generation could degrade station
performance.
Export markers
Export markers on points and other entities set up efficient data retrieval.
28 December 9, 2019
Niagara JSON Toolkit Guide Chapter 2 Exporting with a JSON schema
Marked With Id If the source has an export marker present, with Id set
In the context of alarming, the filtering occurs normally on alarms passed from the alarm class as they are
generated.
The Send Since action queries the alarm database and passes existing records in to this filter (inclusive of
the supplied timestamp) so that they can be checked for a suitable export marker and then passed to the re-
ceiving schema as required to create a new record for each alarm. The timestamp, being in the past, should
help identify when this mode is active.
NOTE:
To prevent an accidental data deluge, Send Since does not function if the filter is in Pass All mode. A bql
query on the alarm database could be used if this is a requirement.
History export marker filter
This filter exports history data for points with an export marker.
The filter overlaps somewhat with the relative history query, which can select history for points using many
different selection criteria, or an appropriate base query may also be used to generate history for each ex-
port marked point. The HistoryExportMarkerFilter allows updating of the timestamp stored on each
export marker so that only recent history records are sent to the remote system (typically, records added
since the last export).
The schema nested under the filter determines the payload format. To complete the export, link the output
from that schema to a target transport point.
If one does not exist already, the HistoryExportMarkerFilter adds a new query to the Queries folder
of the schema. This query needs to be referenced by a BoundQueryResult.
In the event that an export-marked point has more than one history extension beneath it, the schema ex-
ports each extension in turn.
In most cases, it is likely the Current Export Id property needs to be linked into the schema output to
provide identifying information, or even the query used to select data may be included if the target system
could infer useful data from it.
N O T E : Because the export marker relies on being added to a local control point in the station, it is not pos-
sible to match histories imported over BACnet or NiagaraNetwork using this method. Use a relative schema
instead.
December 9, 2019 29
Chapter 2 Exporting with a JSON schema Niagara JSON Toolkit Guide
Use the Send Since Last Export action to send only unsent history data using the timestamp stored on
each export marker.
These are some important filter properties:
• History Export Filter is the schema that produces the output.
• Current Query identifies the query fed into the schema below. The first query in the Q u e r i e s folder is
linked on start, does not have to be the only query, and is output first by the schema.
• Columns sets up comma-separated values, for example, timestamp, value, status.
• Update Send Since Time determines if the schema updates most recent send time when the schema
generates data and enables sending only changed records on the next run. If true, every time the sche-
ma exports history it updates the timestamp stored on each export marker.
Queries
Queries search the station database for the data to include in a schema.
Query folder
The Q u e r i e s folder of a JSON schema stores queries whose results are available to be used in the schema.
This allows JSON content to be generated from the results of bql or neql queries. For example, to name just
a few, you can generate a report of overridden points, active alarms, or history logs for a given point.
Query Interval is an important property of the queries folder. It determines how often queries execute,
and, therefore, how up-to-date any data exported by the schema will be when an update strategy of CoV is
used.
N O T E : If multiple queries exist, the station runs each query in parallel each time the schema executes.
Queries do not execute each time a schema generates in change-of-value mode, otherwise a query could
run every time a point value changes, which could have a negative impact on the performance of the control
strategy running in a station. Instead, a BoundQueryResult caches the results and adds them to the
schema.
Schemata in on-demand mode and relative schemata do execute each query every time a schema
generates.
It is possible to manually invoke query execution using the Execute Queries action of the schema, which
could also be linked to some appropriate logic to trigger execution when needed.
IMP O RTAN T:
When executing queries against your station, bear in mind the potential performance implications of running
queries frequently. To reduce the scope of the query, focus the first part of the ord to the location where the
data are likely to be found, or by using the stop keyword to prevent depth recursion.
Query
You add queries below the Q u e r i e s folder found at the top level of the schema.
Fi gu re 16 Query properties
A query can be any valid transform, neql or bql statement which returns a BITable.
Here are some useful examples to include in a schema:
30 December 9, 2019
Niagara JSON Toolkit Guide Chapter 2 Exporting with a JSON schema
D a t a t o re t u r n Query
BoundQueryResult
Once you define a query, use the BoundQueryResult to determine where and how to insert the results into
the payload.
You can mix query results, such as bound properties or other query results with all other schema member
types in the same payload. For example, if required by the target platform, you could construct a floor sum-
mary with historical data and current alarms.
The JSON Toolkit provides various output formats as the following examples demonstrate, and a developer
can create new output formats.
The following examples use two columns for the sake of brevity. You may add more columns.
You can format the timestamp returned by a query using the format options in the schema’s C o n f i g folder.
REMEMBER:
Executing a bql query does not trigger subscription of the component in question. The values used are the
last values known to the station.
December 9, 2019 31
Chapter 2 Exporting with a JSON schema Niagara JSON Toolkit Guide
Example JSON
"2019-02-07 23:29:06.247+0000",
25
]
]
Row array
"data": [
[
"2019-02-07 23:27:42.116+0000",
45
], [
"2019-02-07 23:28:03.157+0000",
15
], [
"2019-02-07 23:28:24.197+0000",
85
], [
"2019-02-07 23:28:45.222+0000",
55
], [
"2019-02-07 23:29:06.247+0000",
25
]
]
Objects array
"data": [
{
"timestamp": "2019-02-07 23:27:42.116+0000",
"value": 45
},
{
"timestamp": "2019-02-07 23:28:03.157+0000",
"value": 15
},
{
"timestamp": "2019-02-07 23:28:24.197+0000",
"value": 85
},
{
"timestamp": "2019-02-07 23:28:45.222+0000",
"value": 55
},
{
"timestamp": "2019-02-07 23:29:06.247+0000",
"value": 25
}
]
32 December 9, 2019
Niagara JSON Toolkit Guide Chapter 2 Exporting with a JSON schema
Example JS ON
"value": 15
},
"2019-02-07 23:28:24.197+0000": {
"value": 85
},
"2019-02-07 23:28:45.222+0000": {
"value": 55
},
"2019-02-07 23:29:06.247+0000": {
"value": 25
}
]
Column array
"data": [
[
"2019-02-07 23:27:42.116+0000",
"2019-02-07 23:28:03.157+0000",
"2019-02-07 23:28:24.197+0000",
"2019-02-07 23:28:45.222+0000",
"2019-02-07 23:29:06.247+0000"
], [
45,
15,
85,
55,
25
]
]
Tuning
You may use the hidden query folder property queriesMaxExecutionTime to in-
crease the amount of time granted to complete all the queries during each cycle.
Failure to complete in this time causes schema generation to fail.
Setting up queries
In addition to the binding queries, which set up a single query bql, neql or ord, you can add additional
queries to a Queries folder. The schema turns the queries in this folder into a string.
Step 1 Create a regular schema.
December 9, 2019 33
Chapter 2 Exporting with a JSON schema Niagara JSON Toolkit Guide
The example above uses the points of a BACnet device. This JSON configuration includes the
Queries folder and the root object container for the schema.
a. Identifies the regular queries that define the source of the data for binding. In this example, the
query uses bql to identify the data.
b. Identifies a query that can become a JSON string. The query result injects the query referenced
from the Queries folder into the point in the schema output. You can nest these queries any-
where within your JSON message.
By default, each schema includes a Q u e r i e s folder, which comes with two properties: Query In-
terval (to configure how frequently to execute the query), and Last Query Completed
Timestamp.
Step 2 To configure the Query Interval, right-click the Q u e r i e s folder, click V i e w s → A X P r o p e r t y
S h e e t , configure the interval, and click S a v e .
Step 3 To add an ad hoc query to the schema, expand the Q u e r y node in the palette, drag a Q u e r y from
the palette to the Q u e r i e s folder in the schema, double-click the Q u e r y, enter the Query Ord, and
click S a v e .
For simplicity, the example Queries folder contains a single query. It could contain additional
queries.
A above identifies the ord for the single ad hoc query (BacnetQuery): station:|slot:/Driv-
ers/BacnetNetwork/MyName|bql:select name, proxyExt.objectId, out.value AS
‘v’, status from control:ControlPoint
This query searches a particular BACnet device for the name, object ID, current value and status of
all points under the device. The Last Result Size property indicates that the query finds two
points.
Step 4 To create a bound query result, expand the Q u e r y node in the palette and drag a B o u n d Q u e r y R e -
s u l t from the palette to the root object in the schema.
In the example, the bound query result (identified by the second box) references the query (Bac-
netQuery) and defines the Output Style to render the query in.
Step 5 To update the payload message, click the G e n e r a t e button.
The result of running the example query looks like this:
34 December 9, 2019
Niagara JSON Toolkit Guide Chapter 2 Exporting with a JSON schema
The first group of name and value pairs reports the result of the main binding query (under config). The data
block at the bottom shows the result of the ad hoc query in the Q u e r i e s folder. The data block displays as
an object array identified by the square brackets. The array contains one object per BACnet point, in this
case two objects, each inside a pair of braces.
This example could have used a relative schema. Which one to use depends on your requirements. Does
your API need all data in a single JSON message or does it require one message per point? This procedure
does not subscribe to the component model. It runs a bql query to populate the BITable and encodes that
data. The power of bql to select data feeds into the input to the schema the same as you could feed a series
transform into this schema, query the historical alarm data, or query history data.
This type of query configuration does not have to be done with device points. By “query” in this context, we
mean anything that returns a BITable so you could use a transform ord, bql on the history space or neql on
the component space. Any time you have something you can feed to the ReportService you can encode and
output it with a schema.
Alarms
The JsonAlarmRecipient exports alarms using the recipient’s schema.
AlarmRecipient
Linking the alarm topic of an alarm class into the route action of a JsonAlarmRecipient triggers the gen-
eration of a new payload each time the alarm class receives an alarm.
The JsonAlarmRecipient comes with a nested schema whose payload output depends on the alarms
passed through from the parent recipient.
December 9, 2019 35
Chapter 2 Exporting with a JSON schema Niagara JSON Toolkit Guide
Queries, bound objects and arrays, and/or properties can include present value data from the station in the
payload.
There are, however, some alarm-specific data types you can include, notably the properties from a Niagara
Alarm Record: BAlarmRecord
By including the unique identifier in an outgoing message, an inbound payload can acknowledge alarms.
A l a r m R e c o rd P ro p e r t y
Only the JsonAlarmRecipient’s schema supports these alarm-related properties. Adding each of these
to the schema allows inclusion of the selected alarm property in the output.
For example, the sourceState, uuid, alarmClass etc. As with other schema properties the name is deter-
mined by renaming the property, for example AlarmRecordProperty becomes timestamp.
BF orma t Prope rt y
This property defines the alarm data to be extracted from the Niagara alarm database. For example, if an en-
gineer used the Metadata property of an AlarmExt to record the location of a point in the building, this
could be fetched using alarmData.location to include in the payload.
E x p o r t i n g a l a r m re c o rd s t o t h e J s o n A l a r m R e c i p i e n t
This component comes with a nested schema whose payload output depends on the alarms passed through
from the parent recipient.
You may include queries, bound objects or arrays, and properties to return a station’s present value data in
the payload. You may also include some specific alarm data types, notably the properties from the alarm re-
cord: BAlarmRecord.
Step 1 Drag the JsonAlarmRecipient to the W i re S h e e t .
Step 2 Connect the Alarm Class to the Route Alarm action on the recipient.
Linking the alarm class to the route action of a JsonAlarmRecipient component triggers the
generation of a new JSON payload each time the recipient receives an alarm from the alarm class.
Step 3 Add an AlarmRecordProperty component to the schema and select one or more properties.
36 December 9, 2019
Niagara JSON Toolkit Guide Chapter 2 Exporting with a JSON schema
Each property you add to the schema can include selected alarm data in the output, such as the
sourceState, uuid, alarmClass etc. As with other JSON schema properties, you can rename the
property; for example “AlarmRecordProperty” can be renamed to “current value”, as shown
above.
Step 4 To filter out unwanted alarms before sending data to the alarm recipient, add the AlarmExport-
MarkerFilter to the W i r e S h e e t and connect it as shown below.
Normal filtering occurs on alarms passed from the alarm class to the recipient as the station gener-
ates the alarms.
December 9, 2019 37
Chapter 2 Exporting with a JSON schema Niagara JSON Toolkit Guide
The Send Since action queries the alarm database and passes existing records to the filter (includ-
ing the supplied timestamp). The system checks the records for a suitable ExportMarker, and
passes them to the receiving JsonSchema to create a new record for each alarm. Since the time-
stamp is in the past, the filter should be able to identify when its mode was active.
NOTE:
To prevent accidental data deluge, Send Since does not function if the filter’s Mode is set to Pass
All. You could use a bql query on the alarm database if this is a requirement.
Step 1 To export current JSON data, either click the E x p o r t button ( ) or click F i l e → E x p o r t
The E x p o r t window opens.
38 December 9, 2019
Niagara JSON Toolkit Guide Chapter 2 Exporting with a JSON schema
The examples folders must be at the root of the station Config component for them to work
correctly.
The screen capture shows the example folders in the station C o n f i g folder.
December 9, 2019 39
Chapter 2 Exporting with a JSON schema Niagara JSON Toolkit Guide
There is a basic example with bindings. Another that runs a query. There is a relative schema. Along
the bottom are examples of how to apply a schema to a practical job. For example, there is are for-
mats for communicating with an IBM cloud and the Sparkplug standard.
Connecting a device
This procedure uses an example to demonstrate how to connect a device. The example sets up a relative
schema to look for all folders in the station that have a particular tag, such as “lights,” “sensor,” etc.
Step 1 Set up writable points in the station folders and connect them to the source points.
Step 2 Drag a relative schema to a logic W i re S h e e t .
Step 3 Set up a base query to locate the point values.
40 December 9, 2019
Niagara JSON Toolkit Guide Chapter 2 Exporting with a JSON schema
For example: slot:/Hue|neql:n:light (assuming a “light” tag has been applied to the point’s
parent folder).
Step 4 Specify the binding.
For example: slot:
The Output property displays the JSON message payload.
Step 5 Drag an E n g i n e C y c l e M e s s a g e Q u e u e to the W i r e S h e e t .
Step 6 On the W i r e S h e e t , link the R e l a t i v e J s o n S c h e m a ’s Current Base And Output to the Enqueue slot
of the queue.
Step 7 Post the output from the program to HTTP.
This shows all the square brackets of several arrays with values. The JSON that generates this payload
queries the history for a particular ramp in a station:
history:/json/Ramp|bql:select top 5 value
This is another (different) example of a schema and the JSON message that creates its chart.
Gold (orange) identifies the basic Query ord. The block identified by the green box and arrow (data) is the
bound query result. The resulting graph looks like the following on a web page.
December 9, 2019 41
Chapter 2 Exporting with a JSON schema Niagara JSON Toolkit Guide
42 December 9, 2019
Chapter 3 Importing JSON
Topics covered in this chapter
♦ Routing complete incoming messages
♦ Routing part of a message
♦ About the Json Path selector
♦ Handlers and alarm acknowledgments
♦ Setpoint handler and writing to points
♦ Export setpoint handler and export registration
Data coming into a station can be used to modify a setpoint or execute some other action. A handler
processes imported JSON.
December 9, 2019 43
Chapter 3 Importing JSON Niagara JSON Toolkit Guide
Step 5 Give the slot a name, use the transient and read-only flags to avoid onward handlers running again
at station start and click O K .
The new slot is added.
For example, if Key = messageType, the JSON routes this message to a string slot with a name
“alarmAck” and then on to connected handlers, as shown above.
{
"messageType": "alarmAck",
"user": "AJones",
"alarmId": [ "5cf9c8b2-1542-42ba-a1fd-5f753c777bc0" ]
}
44 December 9, 2019
Niagara JSON Toolkit Guide Chapter 3 Importing JSON
N O T E : Enabling Learn Mode adds a dynamic slot on input. This procedure documents how to add
the slot manually.
Step 3 Manually add a baja:double slot by opening the A X S l o t S h e e t view, or by simply right-clicking the
sheet and clicking A d d S l o t .
An A d d S l o t window opens for either method, as shown below.
A d di n g a s l o t f ro m t h e S l o t S h e e t Vi e w Adding a s lot using the Action menu
Step 4 To add the slot to the JsonDemuxRouter component, give the slot a name (“hue” for this exam-
ple), choose Type: baja:Double and click O K .
The new slot is added.
Step 5 In the W i r e S h e e t view, connect the schema output to the JsonDemuxRouter component’s Route
slot.
The following image shows a W i re S h e e t view of components routing part of an incoming message
to the slot for onward processing. The slot that you add must match the key name, to select that
key, and should be either Boolean, Numeric or String to match the JSON value.
December 9, 2019 45
Chapter 3 Importing JSON Niagara JSON Toolkit Guide
Once the JsonDemuxRouter component has a slot of type baja:Double named "hue", it passes
the hue to expose the value “43211” for use in the station.
N O T E : To extract nested JSON objects, add a string with an appropriate name, for example, a demuxed
string named ‘data’ could contain this entire nested object:
{
"type" : "line",
"data" :
{
"labels" : ["Sunday", "Monday"],
"values" : [ 1, 2 ]
}
}
In this example a single numeric value was selected. However it is possible to select a complete subset of
the incoming JSON, for example: $.data would select the entire data object into the out slot, or $.data.
values would select the entire JSON “values” array. Any expression containing a search with $..labels,
for example, will return search results enclosed within an outer array.
Much more explanation of this powerful tool can be found at the following websites:
• https://goessner.net/articles/JsonPath/
• http://jsonpath.com/
• https://www.baeldung.com/guide-to-jayway-jsonpath
46 December 9, 2019
Niagara JSON Toolkit Guide Chapter 3 Importing JSON
The following task shows how to use a JsonPath component for data selection.
Step 1 Open the j s o n T o o l k i t palette, expand I n b o u n d → S e l e c t o r s and drag a JsonPath selector to a
Wire Sheet and then open the selector’s property sheet view.
Step 2 Configure the path property using the syntax $.data.values.[0], as shown below, and save
your changes.
December 9, 2019 47
Chapter 3 Importing JSON Niagara JSON Toolkit Guide
The user value stored on the alarm record identifies which user acknowledged the alarm in the remote appli-
cation. If the user key is omitted the component still tries to acknowledge the alarms using the fallback name
“AlarmUuidAckUser”.
N O T E : The Json Schema Service runAsUser is a prerequisite for this handler to work. The specified user
must have admin write permissions for the alarm class of the records being acknowledged.
Two alarm handler properties configure this task:
• AckSource is a string appended to every AlarmRecord acknowledged. Its purpose is to allow auditing in
future and is stored as AckSource in the alarm data.
• AckResult is a topic that reports the results of the alarm acknowledgment. Its purpose is to log or post
process activity. Here is an example of the output it reports:
"Ack-ed alarm " + record
"Already ack-ed in alarmDb " + record
"Could not create BUuid from " + uuid
48 December 9, 2019
Niagara JSON Toolkit Guide Chapter 3 Importing JSON
This allows the cloud (or other external system) to assign it’s own identifier or primary key to export-marked
points in the Niagara station, which can be used to locate them in future or include them in exports to the
cloud system.
The messages should be in this format:
{
"messageType" : "registerId"
"niagaraId" : "h:a032",
"platformId" : "mooseForce123"
}
or
{
"messageType" : "deregisterId"
"platformId" : "mooseForce123",
}
N O T E : This class does not use the messageType, which would be used simply to route it to this handler and
so can be changed as needed.
Example
This W i r e S h e e t and JSON loosely demonstrate some of the routers and selectors based upon a fictional
point search JSON message.
F ig ure 21 Json Export Registration Handler example Wire Sheet and JSON
December 9, 2019 49
Chapter 3 Importing JSON Niagara JSON Toolkit Guide
50 December 9, 2019
Chapter 4 Developer guide
Topics covered in this chapter
♦ JSON schema types
♦ Relative topic builder
♦ Type Override example
♦ Inline JSON Writer
♦ Custom query style
♦ Builder class / API
♦ Useful methods
♦ How schema generation works
♦ Working with Apache Velocity
♦ Subscription examples with bajascript
♦ Inbound components
Developers can use JSON to create complex queries and apps. They can extend the Toolkit by creating their
own query styles.
Three interfaces represent the three structural types of a JSON payload: property (key and value pair), ob-
ject and array. All schema members have a name defined by getJsonName().
December 9, 2019 51
Chapter 4 Developer guide Niagara JSON Toolkit Guide
All schema members inherit the default processChildJsonMembers() behaviour, which allows us to recur-
sively call process() on each member down through the nested schema structure.
All schema member types extend BJsonSchemaMember and most implement one of the three interface
types. The base class lets us define the parent-child legal checks. This restricts nested types to just other
BJsonSchemaMembers. This is where the JSON passes global schema events, for example, unsubscribe.
Different types of JSON schema members may be nested under a schema. These are logically grouped by
common behaviour.
When developing against the toolkit, most of these classes are open to extension.
Example 1
Consider a requirement for a new key and value pair to represent a device’s startup time as a string value.
You might simply extend the BJsonSchemaProperty<T> as type <String> using your own date format or
type <BAbsTime> allowing the schema to render the date automatically using the schema date config. Now,
you just need to implement getJsonValue() to return the appropriate value.
@NiagaraType
public class BDeviceTimeProperty extends BJsonSchemaProperty<BAbsTime>
{
/*+ ------------ BEGIN BAJA AUTO GENERATED CODE ------------ +*/
.....
/*+ ------------ END BAJA AUTO GENERATED CODE -------------- +*/
@Override
public BAbsTime getJsonValue()
{
return (BAbsTime) ..... // this will use the schemas date format config
}
}
Example 2
This requirement is for an object that contains a key and value pair for each slot on the target component,
but only those with a user defined 1 flag. You might extend BJsonSchemaBoundObject, hide the
52 December 9, 2019
Niagara JSON Toolkit Guide Chapter 4 Developer guide
slotsToInclude slot, and override the method getPropertiesToIncludeInJson() to only return properties with
the user defined flag.
@NiagaraType
@NiagaraProperty(name = "slotsToInclude", type = "jsonToolkit:SlotSelectionType",
defaultValue = "BSlotSelectionType.allVisibleSlots",flags = Flags.HIDDEN,
override = true) public class BUserDefinedFlags extends BJsonSchemaBoundObject
{
/*+ ------------ BEGIN BAJA AUTO GENERATED CODE ------------ +*/
.....
/*+ ------------ END BAJA AUTO GENERATED CODE -------------- +*/
@Override
public List <String>getPropertiesToIncludeInJson(BComplex resolvedTarget)
{
if (resolvedTarget == null)
{
return Collections.emptyList(); // or try to resolve it!
}
return Arrays.stream(resolvedTarget.getPropertiesArray())
.filter(prop -> (resolvedTarget.getFlags(prop) & Flags.USER_DEFINED_1) != 0)
.map(prop -> prop.getName())
.collect(Collectors.toList());
}
}
Ty p e O v e r r i d e e x a m p l e
At the core of the JSON Toolkit is a method that maps baja object types to JSON. This determines, for ex-
ample, how any encountered BControlPoint, Facets, BAbsTime etc. should be encoded in the output.
The payload includes many variations for the supported Niagara types. Our approach to accommodating
this is to allow a developer or power user the ability to override specific types as they are converted to
JSON.
For a small JsonSchema, the example in the jsonToolkit palette demonstrates how to use a program ob-
ject[^1] to replace units:
/**
* Allows Json types to to be overridden when placed under JsonSchema/config/overrides/
*/
public BValue onOverride(final BValue input)
{
if (input instanceof BUnit)
{
December 9, 2019 53
Chapter 4 Developer guide Niagara JSON Toolkit Guide
Developers could also override the doOverride(BValue value) method in their own BTypeOverride
variant.
I n l i n e J S O N Wr i t e r
This writer allows the schema to defer control to a developer’s own code in the tree of schema members.
This means that you can add any form of dynamic content into the schema output.
To add custom dynamic content, use a program object as per the example in the P r o g r a m s folder of the
jsonToolkit palette. Or you can extend BAbstractInlineJsonWriter. As code contained in a module
is easier to maintain, extending the abstract class would be preferred where the program object may be
widely distributed.
This palette example implements a method: public BValue onOverride(final BInlineJsonWriter
input), which you can customize to meet your project’s needs. The InlineJsonWriter has two impor-
tant methods:
• JSONWriter jsonWriter = in.getJsonWriter();
• BComplex base = in.getCurrentBase();
Demonstrated below:
/**
* The override method allows control of the writer and current base to be passed
* to the code below * allowing JSON to be dynamically constructed within a schema.
*
* @param BInlineJsonWriter wraps two things:
* JSONWriter jsonWriter = in.getJsonWriter();
54 December 9, 2019
Niagara JSON Toolkit Guide Chapter 4 Developer guide
jsonWriter.key("highLimit")
jsonWriter.value("1024")
return null
}
import java.util.concurrent.atomic.AtomicInteger
import javax.baja.nre.annotations.AgentOn
import javax.baja.nre.annotations.NiagaraType
import javax.baja.sys.BString
import javax.baja.sys.Sys
import javax.baja.sys.Type
import com.tridiumx.jsonToolkit.outbound.schema.query.style.BQueryResultWriter
import com.tridium.json.JSONWriter
/**
* An example custom query result writer.
*
* @author Nick Dodd
*/
@NiagaraType(agent = @AgentOn(types = "jsonToolkit:JsonSchemaBoundQueryResult"))
public class BCowSayJson extends BQueryResultWriter
{
/*+ ------------ BEGIN BAJA AUTO GENERATED CODE ------------ +*/
/*@ $com.tridiumx.jsonToolkit.outbound.schema.query.style.BObjectsArray(4046064316)1.0$ @*/
/* Generated Thu Dec 13 11:24:58 GMT 2018 by Slot-o-Matic (c) Tridium, Inc. 2012 */
////////////////////////////////////////////////////////////////
December 9, 2019 55
Chapter 4 Developer guide Niagara JSON Toolkit Guide
// Type
////////////////////////////////////////////////////////////////
@Override
public Type getType() { return TYPE }
public static final Type TYPE = Sys.loadType(BCowSayJson.class)
@Override
public BString previewText()
{
return BString.make("A demonstration result writer")
}
@Override
public void appendJson(JSONWriter jsonWriter, QueryResultHolder result)
{
jsonWriter.object()
try
{
jsonWriter.key("mooo01").value("____________________________")
headerCsv(jsonWriter, result)
dataCsv(jsonWriter, result)
jsonWriter.key("mooo02").value("----------------------------")
jsonWriter.key("mooo03").value(" \\ ^__^ ")
jsonWriter.key("mooo04").value(" \\ (oo)\\_______ ")
jsonWriter.key("mooo05").value(" (__)\\ )\\/\\")
jsonWriter.key("mooo06").value(" ||----w | ")
jsonWriter.key("mooo07").value(" || || ")
}
finally
{
jsonWriter.endObject()
}
}
result.getResultList().forEach( map - {
jsonWriter.key("data" + rowCount.incrementAndGet())
jsonWriter.array()
try
{
map.values()
.forEach(value - jsonWriter.value(toJsonType(value, getSchema().getConfig())))
}
finally
{
jsonWriter.endArray()
56 December 9, 2019
Niagara JSON Toolkit Guide Chapter 4 Developer guide
}
})
Useful methods
These are some methods you might regularly use to create custom content.
December 9, 2019 57
Chapter 4 Developer guide Niagara JSON Toolkit Guide
Override to return a different key. This skips the schema’s config BIJsonSchemaMember getJsonName()
settings for name case/space handling.
Override to append customized content to the current JSON BIJsonSchemaMember process(JSONWriter json, boo-
stream via json.key() and json.value(), etc. The Boolean lean jsonKeysValid)
parameter indicates if the syntax of the keys are currently valid
(for example, not inside an array)
Override to react to events, such as the base item changing or BJsonSchemaMember onSchemaEvent(BSchemaE-
subscription disabled. vent event)
Write a JSON key with the schema’s current case and space-han- JsonSchemaNameUtil writeKey(BIJsonSchemaMem-
dling rules applied. ber member, JSONWriter json-
Writer, String name)
Convert any Java value to a native JSON type (String or Number JsonSchemaUtil toJsonType(Object value,
or Boolean) with some default handling of some baja types, and BJsonSchemaConfigFolder
filter out sensitive types. config)
Convert core Java type values (Numerics or Strings or Booleans) JsonSchemaUtil toBValue(Object value)
to BValue equivalents. If the parameter is already a BValue, this
method returns a copy.
Get a live resolved reference to the ord bindings target. BJsonSchemaBoundMember getOrdTarget() / getTarget()
Override the schema’s default behaviour for handling a subscrip- BJsonSchemaBoundMember handleSubscriptionEvent(Sub-
tion event from a binding target. Depending on the content, the scription subscription, BCom-
schema’s default behaviour is to unsubscribe, ignore or request ponentEvent event)
schema generation.
Override to return a different set of slot values for the resolved BJsonSchemaBoundSlotsCon- getPropertiesToIncludeInJson
target. tainer (target)
Extract values from incoming JSON payloads using various JsonKeyExtractUtil lookup*()
methods.
Implement to handle an incoming JSON payload or throw a Rou- BJsonInbound routeValue(BString message,
tingFailedException if unable to process the message. Context cx)
Override to locate a control point by another means than the BJsonSetPointHandler lookupTarget(BString msg,
handle ord, for example by slot path or name. String id)
58 December 9, 2019
Niagara JSON Toolkit Guide Chapter 4 Developer guide
doGenerateJson() doForceGenerateJson()
Min write
time no generateOutputJson()
exceeded?
yes
yes
Set up
security context
yes
no no yes
process(JSONWriter) JSON/PermissionException
recurse
recurse
December 9, 2019 59
Chapter 4 Developer guide Niagara JSON Toolkit Guide
Binding ords resolve against the current base item of the schema. Unless you are using a relative JSON sche-
ma, this is the station that uses the current result of the base query. Currently, base queries resolve against
the station.
Fi gu re 26 Regular JSON schema with absolute ord bindings that resolve against the station
Relative JSON schema with relative ord bindings resolve against the current base item. This process repeats
until there are no more base items, and results in several output strings.
Fi gu re 27 Relative JSON schema with relative ord bindings that resolve against the current base item
60 December 9, 2019
Niagara JSON Toolkit Guide Chapter 4 Developer guide
This could allow access to an external application consuming data from Niagara.
Wo r k i n g w i t h A p a c h e Ve l o c i t y
Apache Velocity is a Java-based template language anyone can use to reference objects defined in Java
code. You can use it to expose the output of a JSON schema via the Jetty Web Server in Niagara 4. This tool
may be beneficial for applications that expect to consume data provided by the Niagara station, for exam-
ple, a visualization or machine-learning library.
P re re q u i s i t e s :
Given JSON’s origin as a data exchange format for the web, many libraries expect to receive input in this for-
mat. The Google Chart library is such an example. The following example is from the Google Chart project
web site. Notice that the var data is populated with JSON data. Replacing hard-coded data with the out-
put from a suitably-configured JSON schema in your station draws a chart from the Niagara station data.
<html>
<head>
<script type="text/javascript" src="https://www.gstatic.com/charts/loader.js"></script>
<script type="text/javascript">
google.charts.load('current', {'packages':['corechart']});
google.charts.setOnLoadCallback(drawChart);
function drawChart() {
var data = google.visualization.arrayToDataTable([
['Year', 'Sales', 'Expenses'],
['2004', 1000, 400],
['2005', 1170, 460],
['2006', 660, 1120],
['2007', 1030, 540]
]);
var options = {
title: 'Company Performance',
curveType: 'function',
legend: { position: 'bottom' }
};
chart.draw(data, options);
}
</script>
</head>
<body>
<div id="curve_chart" style="width: 900px height: 500px"></div>
</body>
</html>
Step 1 Create a new file chart.vm and paste into it the code example of a sample chart from the json-
consuming-charting library of your choice.
Step 2 Replace the JSON data with a velocity variable, for example, $schema.output,
var data = google.visualization.arrayToDataTable([
$schema.output
])
Step 3 After saving the file, open the axvelocity palette and add a VelocityServlet named “chart”
to your station.
December 9, 2019 61
Chapter 4 Developer guide Niagara JSON Toolkit Guide
Step 4 Add a VelocityDocument below the servlet and change the Template File property to point to
the chart.vm file you created earlier.
Step 5 Add a new ContextOrdElement named Schema to the VelocityContext of your
VelocityDocument.
Step 6 Update the Schema Ord element to point to a suitable jsonSchema added to your station.
This schema could output live station data or the result of a query or transform. Both would be suit-
able for charting libraries, although it may be necessary to modify the time and date format form
the schema default settings or to reduce the presented interval of data by using a SeriesTrans-
form Rollup function.
So, what did we achieve? The template HTML file has a variable, which when accessed via the station’s veloc-
ity servlet will be replaced with the output from our schema.
If you add a WebBrowser from the workbench palette to a Px Page and set the ord property to http:\
\127.0.0.1\velocity\chart, you should see a chart when you view the page in a web browser. If not,
use the developer tools to view the source code and ensure that the output of your schema is replacing the
$schema.output variable.
<script src="https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.7.3/Chart.min.js">
</script>
<!-- note the special syntax for downloading JS file from the 'bajascript' folder
you add in your station -->
<script type='text/javascript' src='/ord/file:%5Ebajascript/basic.js%7Cview:web
:FileDownloadView'></script>
</head>
<body>
</body>
</html>
Example basic.js file to fetch chart data
The data array in the payload below uses bound properties. A single-column query would allow historical da-
ta to be used instead from a bql query on the history database.
62 December 9, 2019
Niagara JSON Toolkit Guide Chapter 4 Developer guide
// Graphs
var ctx = $('#myChart')
December 9, 2019 63
Chapter 4 Developer guide Niagara JSON Toolkit Guide
"borderColor": "#ff0033",
"borderWidth": 3,
"lineTension": 0
}
]
},
"options": {
"scales": {
"yAxes": [
{
"ticks": {
"beginAtZero": false
}
}
]
},
"legend": {
"display": false
},
"title": {
"display": true,
"text": "Philips Hue Light Demo"
},
"tooltips": {
"intersect": true,
"mode": "index"
},
"hover": {
"intersect": true,
"mode": "nearest"
}
}
}
Inbound components
Inbound components route JSON messages to control points and devices.
To create a new inbound type, you extend one of the three main types: BJsonRouter, BJsonSelector or
BJsonHandler and implement routeValue(BString message, Context cx) throws RoutingFaile-
dException. You can create a new RoutingFailedException at any stage to report an error and update
the lastResult slot.
When extending any of the BJsonInbound types, you may specify which property triggers an automatic re-
routing of the last input with Property[] getRerunTriggers(). The helper interface JsonKeyExtrac-
tUtil contains several methods for extracting values from a JSON payload.
64 December 9, 2019
Chapter 5 Components
Topics covered in this chapter
♦ JsonSchema (Json Schema)
♦ Config (Json Schema Config Folder)
♦ Debug (Json Schema Debug Folder)
♦ Queries (Json Schema Query Folder)
♦ RelativeJsonSchema (Relative Json Schema)
♦ JsonSchemaService (Json Schema Service)
♦ Object (Json Schema Object)
♦ BoundObject (Json Schema Bound Object)
♦ Array (Json Schema Array)
♦ BoundArray (Json Schema Bound Array
♦ FixedString (Json Schema String Property)
♦ FixedNumeric (Json Schema Numeric Property)
♦ FixedBoolean (Json Schema Boolean Property)
♦ Count (Json Schema Count Property)
♦ CurrentTime (Json Schema Current Time Property)
♦ UnixTime (Json Schema Unix Time Property)
♦ BoundProperty (Json Schema Bound Property)
♦ BoundCSVProperty (Json Schema Bound Csv Property)
♦ Facet (Json Schema Facet Property)
♦ FacetList (Json Schema Facet List)
♦ Tag (Json Schema Tag Property)
♦ TagList (Json Schema Tag List)
♦ Query (Json Schema Query)
♦ RelativeHistoryQuery (Relative History Query)
♦ BoundQueryResult (Json Schema Bound Query Result)
♦ JsonAlarmRecipient (Json Alarm Recipient)
♦ AlarmRecordProperty (Json Schema Alarm Record Property)
♦ BFormatProperty (B Format String)
♦ ExportMarker (Json Export Marker)
♦ AlarmExportMarkerFilter (Alarm Export Marker Filter)
♦ HistoryExportMarkerFilter (History Export Marker Filter)
♦ JsonExportSetpointHandler (Json Export Setpoint Handler)
♦ JsonExportRegistrationHandler (Json Export Registration Handler)
♦ JsonExportDeregistrationHandler (Json Export Deregistration Handler)
♦ JsonMessageRouter (Json Message Router)
♦ JsonDemuxRouter (Json Dmux Router)
♦ JsonPath (Json Path)
♦ JsonAtArrayIndex (Json At Array Index)
♦ JsonContainsKey (Json Contains Key)
♦ JsonIndexOf (Json Index Of Key Selector)
♦ JsonSum (Json Sum Selector)
♦ JsonLength (Json Length Selector)
♦ JsonFindAll (Json Find All Selector)
♦ AlarmUuidAckHandler (Alarm Uuid Ack Handler)
♦ SetPointHandler (Json Set Point Handler)
♦ EngineCycleMessageQueue (Engine Cycle Message Queue)
♦ EngineCycleMessageAndBaseQueue (Engine Cycle Pair Queue)
♦ InlineJsonWriter (Inline Json Writer)
♦ TypeOverride (Type Override)
♦ relativeTopicBuilder (Program)
Components include services, folders and other model building blocks associated with a module. You may
drag them to a P r o p e r t y or W i r e S h e e t from a palette.
December 9, 2019 65
Chapter 5 Components Niagara JSON Toolkit Guide
Descriptions included in the following topics appear as context-sensitive help topics when accessed by:
• Right-clicking on the object and selecting V i e w s → G u i d e H e l p
• Clicking H e l p → G u i d e O n T a r g e t
Fi gu re 28 JsonSchema properties
You add a schema to a station by dragging a JsonSchema from the palette to the C o n f i g folder in the Nav
tree. From there, to access schema properties, expand the C o n f i g folder and double-click the schema.
ro o t
This container holds JSON entities: objects, arrays, properties and bound properties.
66 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
• G e n e r a t e J s o n requests a rebuild and update of schema output. For relative schemas, this evaluates the
Base Query and publishes results.
• F o rc e G e n e r a t e J s o n forces the generate action regardless of the current tuning settings.
• C l e a r C a c h e discards the last known values of bindings and cached query results.
• C l e a r O u t p u t sets the schema output to an empty string.
• E x e c u t e Q u e r i e s forces an immediate execution of all the schemas queries. You can link this action to
some appropriate logic to trigger execution when needed.
• U n r e g i s t e r (relative schema only) unsubscribes the registration from any base items that the relative
schema monitors for updates.
• U n s u b s c r i b e A l l removes cloud registration from all export-marked entities in the station.
In addition to the standard properties (Enabled, Status, and Fault Cause), these properties are unique to the
jsonSchema. The child containers are documented in following topics.
December 9, 2019 67
Chapter 5 Components Niagara JSON Toolkit Guide
68 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
December 9, 2019 69
Chapter 5 Components Niagara JSON Toolkit Guide
Config, Debug, read-only folder Reports JSON statistics related to three aspects of activity:
Metrics queries, data generation, and data subscription.
(JsonSchema)
70 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
De bu g rep ort
F ig ure 36 Debug report
To access this view, click the O u t p u t H i s t o r y button or right-click the Schema Output History Debug slot
and click V i e w s → S p y R e m o t e or S p y L o c a l .
Column Description
No. Identifies the row. You configure the number of allowed rows by setting the History Max Size value on
the Debug P r o p e r t y S h e e t .
Base Item Identifies the slot from which the system generated the JSON.
December 9, 2019 71
Chapter 5 Components Niagara JSON Toolkit Guide
Last Query Fail Reason Last Schema Generation Fail Reason Subscription Events Ignored
Resolve Errors
72 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
Most metrics are self-explanatory. Execution millis report the number of milliseconds spent performing a
query. Cache hits indicate the number of schema string generations that found a cached value for a binding.
Cache misses indicate the number of schema string generations that found no cached value for a binding.
Last Query Com- read-only (defaults Reports the time the last query completed.
pleted Timestamp to null)
Queries, queries- time Increases the amount of time granted to complete all queries
MaxExecutionTime on each cycle. Failure to complete within this time causes the
(hidden property schema generation to fail.
on the Queries
folder)
December 9, 2019 73
Chapter 5 Components Niagara JSON Toolkit Guide
Fi gu re 40 RelativeHistoryQuery properties
You add a RelativeHistoryQuery under the Queries folder in the R e l a t i v e J s o n S c h e m a . You access these
properties by double-clicking the R e l a t i v e J s o n S c h e m a node in the Nav tree and expanding the Queries
folder.
Property Value Description
Last Result Size read-only (defaults Reports the size of the query result the last time the frame-
to 0) work executed the search.
Query Pattern bql Prepends to a bql query so query data can be included in the
payload for a given set of points or devices.
For example: %baseHistoryOrd%?period=today|bql:
select timestamp, value
Example
Here is an example of how to use the Query Pattern property to pre-pend the current base item to a bql
query. This example includes query data in the payload for a given set of points or devices:
%baseHistoryOrd%?period=today|bql:select timestamp, value
You may use this with a base query to return a HistoryConfig or a HistoryExt (or the parent of these slots):
station:|slot:/JsonExampleComponents|bql:select * from history:HistoryConfig
C A U T I O N : When creating queries, bear in mind the potential performance implications of running queries
frequently. To reduce the scope of the query, focus the first part of the ord on the location where the data
are likely to be found, or use the stop keyword to prevent depth recursion.
74 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
To add this component, expand the Q u e r y folder in the palette and drag a BoundQueryResult to the root
JSON schema O b j e c t of a relative JSON schema.
Output Style drop-down list Defines the output style to render the query in.
The Base Query component is located in the palette as part of any of the relative schema components (for
example, BasicRelativeSchema, RelativeHistorySchema, and others).
In addition to the standard properties (Status and Fault Cause), these properties support the Base Query.
Publish Interval hours, minutes, Specifies the amount of time between query executions. It
seconds triggers a complete publish output (of every returned compo-
nent) at the interval selected.
Last Publish Count read-only Indicates the number of times the query executed.
Last Publish Time read-only Indicates the last time the query executed.
December 9, 2019 75
Chapter 5 Components Niagara JSON Toolkit Guide
Fi gu re 42 RelativeJsonSchema properties
You add a relative schema to a station by dragging a RelativeJsonSchema from the palette to the C o n f i g
folder in the Nav tree. From there, to access schema properties, expand the C o n f i g folder and double-click
the schema.
In addition to the standard properties (Enabled, Status, and Fault Cause), these properties are unique to
JSON.
Property Value Description
Last Updated read-only (defaults Reports when the relative schema was updated last.
to null)
Config folder Contains properties for configuring the relative schema.
76 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
Global Cov Slot Additional Provides some station global filtering by identifying which
Filter properties slots should be ignored when subscribed to bound values. The
default list of slots includes a good example of why this func-
tion is necessary in that changes to a component’s wsAnnota-
tion property (which details the position and size of a
component glyph on the W i r e S h e e t ), should generally be ex-
cluded from the changes of value reported to any upstream
consumer of data.
December 9, 2019 77
Chapter 5 Components Niagara JSON Toolkit Guide
78 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
To add the root object to a schema, expand the O b j e c t s folder in the palette and drag an O b j e c t to the
J s o n S c h e m a folder. To add another object to the schema, drag an O b j e c t from the palette to the root O b -
j e c t container under the schema.
An object is a container. It has no properties of its own or additional containers. Inside this container, the
JSON objects and properties model the structure of the JSON message underneath the schema object. If
you nest items in this container within each other in a tree structure, they will appear nested in the JSON
string.
To add a bound object to a schema, expand the O b j e c t s folder in the palette and drag a BoundObject to
the schema folder, then double-click the bound object.
Json Name Source drop-down list Selects a name for the source object based on how it is defined
elsewhere. Options are:
Display Name is an explicitly-assigned name for the object.
Target Name
December 9, 2019 79
Chapter 5 Components Niagara JSON Toolkit Guide
To add a bound array to a schema object, expand the A r r a y s folder in the palette and drag a BoundArray
to the root object folder in a schema, then double-click the BoundArray component.
80 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
To add this property to a schema object or array, expand the P ro p e r t i e s folder in the palette and drag a
FixedString to the root object { } or to an array [ ] under the root object, then double-click the Fixed-
String component.
Fixed properties, such as names, appear as constants.
You can link in to these if you expect a name to vary. JSON includes the current value during the next gener-
ation event triggered by a CoV on a bound entity or by the invocation of the Generate action. Changing
the value of a fixed property does not trigger a CoV generation event the same way that a bound equivalent
does.
To add this property to a schema object or array, expand the P ro p e r t i e s folder in the palette and drag a
FixedNumeric to the root object { } or to an array [ ] under the root object. To configure its property,
double-click it.
You can link in to this value if you expect it to vary. The next generation event includes the current value trig-
gered by CoV on a bound entity or by the invocation of the Generate action. A change in the value of any
fixed property does not trigger a CoV generation event in the way that a bound equivalent would.
December 9, 2019 81
Chapter 5 Components Niagara JSON Toolkit Guide
To add this property to a schema object or array, expand the P r o p e r t i e s folder in the palette and drag a
FixedBoolean to the root object { } or to an array [ ] under the root object, then double-click the Fixed-
Boolean component.
You can link in to this value if you expect it to vary. The next generation event includes the current value trig-
gered by CoV on a bound entity or by the invocation of the Generate action. A change in the value of any
fixed property does not trigger a CoV generation event in the way that a bound equivalent would.
Fi gu re 52 Count property
To add this property to a schema object or array, expand the P r o p e r t i e s folder in the palette and drag a
Count to the root object { } or to an array [ ] under the root object, then double-click the Count
component.
This property is a number that defaults to zero (0).
To return this value to zero, right-click the Count property and click A c t i o n s → R e s e t .
To add this property to a schema object or array, expand the P r o p e r t i e s folder in the palette and drag a
CurrentTime to the root object { } or to an array [ ] under the root object, then double-click the Cur-
rentTime component.
The format for the current time is: year-month-day hour:minute:second
82 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
To add this property to a schema object or array, expand the P ro p e r t i e s folder in the palette and drag a
UnixTime to the root object { } or to an array [ ] under the root object, then double-click the UnixTime
component.
To add a bound property to a schema object or array, expand the B o u n d P ro p e r t i e s folder in the palette
and drag a BoundProperty to the root object { } or to an array [ ] under the root object, then double-click
the BoundProperty component.
December 9, 2019 83
Chapter 5 Components Niagara JSON Toolkit Guide
Json Name Source drop-down list Selects a name for the source object based on how it is defined
elsewhere. Options are:
Display Name is an explicitly-assigned name for the object.
Target Name
Target Display Name
Target Parent Name
Target Path displays the ord for the object rather than a
name.
To add this bound property to a schema object or array, expand the B o u n d P ro p e r t i e s folder in the palette
and drag a BoundCSVProperty to the root object { } or to an array [ ] under the root object, then double-
click the BoundCSVProperty component.
84 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
Json Name Source drop-down list Selects a name for the source object based on how it is defined
elsewhere. Options are:
Display Name is an explicitly-assigned name for the object.
Target Name
Target Display Name
Target Parent Name
Target Path displays the ord for the object rather than a
name.
Slots To Include dop-down lists Identifies which slots from the target to include in the resultant
JSON. Options are:
All Slots reports data from all slots in the target object.
All Visible Slots excludes hidden slots.
All Summary Slots includes only those with the summary
flag set.
Selected Slots manually selects slots from a list.
F ac et ( J s on Sc he m a F ac e t Prop er ty )
This bound property defines a single facet value from a bound component to insert in the schema output,
for example the units of the current point.
To add this bound property to a schema object or array, expand the B o u n d P r o p e r t i e s folder in the palette
and drag a Facet to the root object { } or to an array [ ] under the root object, then double-click the
bound component.
December 9, 2019 85
Chapter 5 Components Niagara JSON Toolkit Guide
To add this bound property to a schema object or array, expand the B o u n d P ro p e r t i e s folder in the palette
and drag a FacetList to an object { } or to an array [ ] under the root object, then double-click the bound
component.
To add this bound property to a schema object or array, expand the B o u n d P ro p e r t i e s folder in the palette
and drag a Tag to an object { } or to an array [ ] under the root object, then double-click the bound
component.
86 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
To add this bound property to a schema object or array, expand the B o u n d P r o p e r t i e s folder in the palette
and drag a TagList to the root object { } or to an array [ ] under the root object, then double-click the
bound component.
Tag Id List Filter text Identifies a comma-separated list to limit the tags to be in-
cluded in the output. For example, n:name,n:type or * for all.
If Include Name Space is set to true, the schema adds the
tag dictionary prefix to the key (for example, hs:hvac).
Include Name true (default) or Configures the search to include the tag dictionary prefix in
Space false the key.
December 9, 2019 87
Chapter 5 Components Niagara JSON Toolkit Guide
You add a RelativeHistoryQuery under the Queries folder in the R e l a t i v e J s o n S c h e m a . You access these
properties by double-clicking the R e l a t i v e J s o n S c h e m a node in the Nav tree and expanding the Queries
folder.
Property Value Description
Last Result Size read-only (defaults Reports the size of the query result the last time the frame-
to 0) work executed the search.
Query Pattern bql Prepends to a bql query so query data can be included in the
payload for a given set of points or devices.
For example: %baseHistoryOrd%?period=today|bql:
select timestamp, value
Example
Here is an example of how to use the Query Pattern property to pre-pend the current base item to a bql
query. This example includes query data in the payload for a given set of points or devices:
%baseHistoryOrd%?period=today|bql:select timestamp, value
You may use this with a base query to return a HistoryConfig or a HistoryExt (or the parent of these slots):
station:|slot:/JsonExampleComponents|bql:select * from history:HistoryConfig
C A U T I O N : When creating queries, bear in mind the potential performance implications of running queries
frequently. To reduce the scope of the query, focus the first part of the ord on the location where the data
are likely to be found, or use the stop keyword to prevent depth recursion.
Fi gu re 63 BoundQueryResult properties
88 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
To add this component, expand the Q u e r y folder in the palette and drag a BoundQueryResult to the root
JSON schema O b j e c t of a relative JSON schema.
Output Style drop-down list Defines the output style to render the query in.
To use this component, expand the A l a r m node in the palette and drag a JsonAlarmRecipient to the C o n -
f i g → S e r v i c e s → A l a r m S e r v i c e folder in the Nav tree.
In addition to the standard properties (Days of the Week, Transitions, Publish Point and Enabled), these
properties configure this alarm recipient.
Time Range, End hour of the day When configured with Start Time, filters alarm records by
Time (JsonAlarm- defining when to stop returning records.
Recipient)
A l a r m R e c o rd P ro p e r t y ( J s o n S c h e m a A l a r m R e c o rd P ro p e r t y )
These properties are only supported on the JsonAlarmRecipients Schema.
F ig ure 65 Alarm record property
December 9, 2019 89
Chapter 5 Components Niagara JSON Toolkit Guide
To use this property, expand the A l a r m node in the palette and drag a AlarmRecordProperty to a sche-
ma’s object folder.
Each of these added to the schema includes the selected Alarm Property in the output. For example the
sourceState, uuid, alarmClass etc. As with other schema Properties, the name is determined by renaming
the property, for example AlarmRecordProperty -> timestamp.
BF or m atP rop er ty ( B F or m at St ri ng )
This property defines alarm data to extract from the Niagara alarm database. For example, if an engineer
uses the Metadata property of an AlarmExt to record the location of a point in a building, alarmData.lo-
cation could fetch this information and include it in the payload.
Fi gu re 66 BFormatProperty
To use this property, expand the A l a r m node in the palette and drag a BFormatProperty to a schema’s ob-
ject folder.
90 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
The toolkit provides three ways to select control point data for export:
• Add an absolute ord binding to a JSON schema.
• Use bql or neql to identify control points to a relative JSON schema.
• Add a JsonExportMarker to a component.
Marking a component offers several benefits beyond just marking points to include in a RelativeJson-
Schema. For example, markers support the export of alarm and history data for specific points. Markers can
store a unique identifier supplied by a third party platform. This can be used to differentiate between regis-
teredpoints with an ID and unregistered points without an ID. For example, with markers JSON can send dif-
ferent payloads prior to registration including more detailed information (units, min/max, descriptive tags)
than should be sent upon every change of value.
When applied to a numeric point, a JSON export marker introduces a CovTolerance property to reduce
unwanted updates from the station if a value changes only slightly. You may also use the export marker with
incoming JSON payloads.
F ig ure 67 ExportMarker properties
To use this marker, expand the E x p o r t M a r k e r node in the palette and drag an E x p o r t M a r k e r to a point in
the station.
Property Value Description
Id (ExportMarker) Provides an id from the cloud platform. The expectation is that
this value will be unique, at least within each station as it may
be used by the cloud platform as a primary key.
Platform Writable true or false Used with the setpoint/override feature to prevent writes from
(ExportMarker) (default) the upstream platform.
filterEnabled true (default) or Turns the filter on and off. When disabled, the schema ignores
false CovTolerance.
CovTolerance number to two Sets up an amount that defines a range of values within which
decimal places a given value may vary without requiring the station to update
the value. This eliminates the overhead required to update
when a value changes only slightly.
lastPublishedValue read-only Reports the most recent value that was exported.
Examples
Example JSON
Base query station:|slot:/|bql:select * from jsonToolkit:
JsonExportMarker
BoundProperty binding ord slot:.. (References the parent of the JsonMarker Base)
December 9, 2019 91
Chapter 5 Components Niagara JSON Toolkit Guide
To use this filter, expand the E x p o r t M a r k e r node in the palette and drag an AlarmExportMarkerFilter
to a point in the station.
Action
Action Description
Send Since Queries the alarm database and passes existing records in to this filter (inclusive of the supplied time-
stamp) so that the framework can check them for a suitable export marker and then pass them on to the
receiving JSON schema as required to create a new record for each alarm. The timestamp, being in the
past, should help identify when this mode is active.
92 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
To use this filter, expand the ExportMarker node in the palette and drag an HistoryExportMarkerFil-
ter to a location in the station.
You access these properties by double-clicking the HistoryExportMarkerFilter node in the Nav tree.
There is some overlap with the RelativeHistoryComponent, which can select point histories using many
different criteria, and an appropriate BaseQuery may also be used to generate history for each export
marked point. The HistoryExportMarkerFilter updates the timestamp stored on each ExportMarker,
so that the schema sends only recent history records to the remote system (typically records added since
the last export).
The History Export Filter container is a JsonSchema nested under the filter. It determines the payload for-
mat, and the output from that schema to link to a target transport point to complete the export.
If a point with an ExportMarker has more than one history extension, the schema exports each in turn.
N O T E : Since the ExportMarker relies on being added to a local control point in the station, it is not possi-
ble to match histories imported over BACnet or the NiagaraNetwork using this filter. Instead, use a
RelativeJsonSchema.
In addition to the standard properties (Enabled, Status, and Fault Cause), the history export filter provides
these properties.
December 9, 2019 93
Chapter 5 Components Niagara JSON Toolkit Guide
Actions
Send Since Last Export uses the timestamp stored in each ExportMarker to send only history records
that have not yet been sent.
Fi gu re 70 JsonExportSetpointHandler properties
To add this handler to a station, expand the E x p o r t M a r k e r folder in the palette and drag this component to
the router folder in the Nav tree.
In addition to the standard property (Enabled), these properties support the JsonExportSetpointHandler.
Id Key text Defines which top-level key in the JSON payload represents
the point Id.
Value Key text (defaults to Defines which top-level key in the JSON payload represents
value) the value to set.
Slot Name Key text (defaults to Defines the optional top-level key in the JSON payload that
slotName) represents the slot name to write to.
Default Write Slot text Defines the slot to write to by default if the payload does not
(JsonExportSet- specify the slot.
pointHandler,
SetPointHandler)
94 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
To add this handler to a station, expand the E x p o r t M a r k e r folder in the palette and drag this component to
the router folder in the Nav tree.
In addition to the standard property (Enabled), these properties support the
JsonExportRegistrationHandler.
Local Key text (defaults to Identifies the name of the JSON property that denotes the
niagaraId) point’s identifier in the Niagara station.
Syntax
The messages should be in this format:
{
"messageType" : "registerId"
"niagaraId" : "h:a032",
"platformId" : "mooseForce123"
}
or
{
"messageType" : "deregisterId"
"platformId" : "mooseForce123",
}
December 9, 2019 95
Chapter 5 Components Niagara JSON Toolkit Guide
N O T E : This class does not use the messageType, which would be used simply to route it to this handler
and so can be changed as needed.
Example
This W i r e S h e e t and JSON loosely demonstrate some of the routers and selectors based upon a fictional
point search JSON message.
Fi gu re 72 Json Export Registration Handler example Wire Sheet and JSON
To add this handler to a station, expand the E x p o r t M a r k e r folder in the palette and drag this component to
the router folder in the Nav tree.
In addition to the standard property (Enabled), these properties support the
JsonExportDeregistrationHandler.
96 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
Syntax
The messages should be in this format:
{
"messageType" : "registerId"
"niagaraId" : "h:a032",
"platformId" : "mooseForce123"
}
or
{
"messageType" : "deregisterId"
"platformId" : "mooseForce123",
}
N O T E : This class does not use the messageType, which would be used simply to route it to this handler
and so can be changed as needed.
Example
This Wire Sheet and JSON loosely demonstrate some of the routers and selectors based upon a fictional
point search JSON message.
December 9, 2019 97
Chapter 5 Components Niagara JSON Toolkit Guide
Fi gu re 75 JsonMessageRouter properties
You add this router to a station by expanding the I n b o u n d → R o u t e r s in the palette and dragging this com-
ponent to the C o n f i g folder in the Nav tree.
In addition to the standard property (Enabled), these properties support the JsonMessageRouter.
98 December 9, 2019
Niagara JSON Toolkit Guide Chapter 5 Components
Learn Mode true or false true configures the JSON to add a dynamic slot on input for
(default) any newly-found message key.
Key text (defaults to Defines which part of the incoming message to switch on)
messageType)
Resend With Blank true or false Turns on and off the resending of a message if a duplicate or
(default) matching message is received.
true causes the router to send an empty string to the target
slot, then resend the output.
Without injecting an empty message, the link does not propa-
gate the change, which could be an issue if the handler needed
other values in place to respond to this message.
false does not send the empty string to the target slot, which
does not resend the output.
You add this router to a station by expanding the I n b o u n d → R o u t e r s in the palette and dragging this com-
ponent to the C o n f i g folder in the Nav tree.
In addition to the standard property (Enabled), these properties support the JsonDemuxRouter.
December 9, 2019 99
Chapter 5 Components Niagara JSON Toolkit Guide
Learn Mode true or false true configures the JSON to add a dynamic slot on input for
(default) any newly-found message key.
Default Missing true (default) or Can set a dynamic slot’s value to its default if the value is
false missing.
true sets the value to its default if the inbound JSON mes-
sage did not include a value for the slot.
false ignores setting the default value.
You add this selector to a station by expanding I n b o u n d → S e l e c t o r s in the palette and dragging the Json-
Path to a JSON message router node in the Nav tree.
In addition to the standard properties (Enabled and Status), these properties support the JsonPath.
You add this selector to a station by expanding I n b o u n d → S e l e c t o r s in the palette and dragging the JsonA-
tArrayIndex to a message router in the Nav tree.
In addition to the standard property (Enabled), these properties support the JsonAtArrayIndex.
You add this selector to a station by expanding I n b o u n d → S e l e c t o r s in the palette and dragging the Json-
ContainsKey to a message router in the Nav tree.
In addition to the standard property (Enabled), these properties support the JsonContainsKey.
You add this selector to a station by expanding I n b o u n d → S e l e c t o r s in the palette and dragging the Jso-
nIndexOf to a message router in the Nav tree.
In addition to the standard property (Enabled), these properties support the JsonIndexOf.
You add this selector to a station by expanding I n b o u n d → S e l e c t o r s in the palette and dragging the Json-
Sum to a message router in the Nav tree.
In addition to the standard property (Enabled), these properties support the JsonSum.
You add this selector to a station by expanding I n b o u n d → S e l e c t o r s in the palette and dragging the Json-
Length to a message router in the Nav tree.
In addition to the standard property (Enabled), these properties support the JsonLength.
You add this selector to a station by expanding I n b o u n d → S e l e c t o r s in the palette and dragging the Json-
FindAll to a message router in the Nav tree.
In addition to the standard property (Enabled), these properties support the JsonFindAll.
Message handlers are components designed to perform a specific task with the data routed and selected via
the other inbound components.
F ig ure 84 AlarmUuidAckHandler properties
You add this handler to a station by expanding the I n b o u n d → H a n d l e r s folder in the palette and dragging
this component to a message router in the Nav tree.
In addition to the standard property (Enabled), these properties support the AlarmUuidAckHandler:
Ack Source text Configures a string to append to every alarm record acknowl-
edgement. This string can provide additional information for
future auditing. The alarm data record stores it as AckSource.
Ack Result Reports the results of the alarm acknowledgment for logging
or post processing activity.
Example
The expected format for this component is:
{
"user": "Maya",
"alarms": [ "5cf9c8b2-1542-42ba-a1fd-5f753c777bc0" ]
}
This array allows the system to acknowledge multiple alarms at once.
The alarm record stores the user value, which identifies the user who acknowledged the alarm in the remote
application. If the user key is omitted, the component still tries to acknowledge the alarms using the fallback
name: AlarmUuidAckUser.
NOTE:
The J s o n S c h e m a S e r v i c e ’s Run As User property is a prerequisite for this handler to work. The specified
user must have admin write permissions for the alarm class of the records being acknowledged.
Fi gu re 85 SetpointHandler properties
You add this handler to a station by expanding the I n b o u n d → H a n d l e r s folder in the palette and dragging
this component to a message router in the Nav tree.
N O T E : The Run As User property in the J s o n S c h e m a S e r v i c e is required to use the SetPointHandler.
In addition to the standard property (Enabled), these properties support the SetPointHandler:
Id Key text Defines which top-level key in the JSON payload represents
the point Id.
Value Key text (defaults to Defines which top-level key in the JSON payload represents
value) the value to set.
Slot Name Key text (defaults to Defines the optional top-level key in the JSON payload that
slotName) represents the slot name to write to.
Default Write Slot text Defines the slot to write to by default if the payload does not
specify the slot.
You add this queue to a station by expanding the I n b o u n d → H a n d l e r s folder in the palette and dragging this
component to a message router in the Nav tree.
For example, you can link a string output slot to onward points or, where necessary, to an
EngineCycleMessageQueue.
To buffer incoming messages when using this component, it is advisable to link from the readValue on a
proxyExt rather than from the out slot of its parent point.
You add this queue to a station by expanding the I n b o u n d → H a n d l e r s folder in the palette and dragging this
component to a message router in the Nav tree.
To use this program object, drag it from the P ro g r a m s folder in the jsonToolkit palette to the C o n f i g
folder in the station. To open this A X P r o p e r t y S h e e t , double-click the InlineJsonWriter component in
the station.
To view the example code, right–click the P ro g r a m node, clickV
Vi e w s → P r o g r a m E d i t o r and click the E d i t
tab.
In addition to the standard properties (Status and Fault Cause), these properties support the
InlineJsonWriter.
Signature read-only Identifies the mathematical scheme used to verify the authen-
ticity of the program.
Ty p e O v e r r i d e ( Ty p e O v e r r i d e )
This component is an example of a program to override a data type.
Fi gu re 89 TypeOverride properties
To use this program object, drag it from the P r o g r a m s folder in the jsonToolkit palette to the C o n f i g
folder in the station. To open its A X P ro p e r t y S h e e t , double-click the TypeOverride object in the station.
To view the example code, right–click the E x a m p l e O v e r r i d e node, clickV
Vi e w s → P r o g r a m E d i t o r and, if
needed, click the E d i t tab.
In addition to the standard properties (Status and Fault Cause), these properties support the TypeOverr-
ide example. These properties are part of the Program component from the Program module and are not
specific to the JSON Toolkit.
Signature read-only Identifies the mathematical scheme used to verify the authen-
ticity of the program.
User Defined read-only Displays user-defined custom imports of the types used in the
Imports source code.
To use this object, drag it from the P r o g r a m s folder in the jsonToolkit palette to the C o n f i g folder in the
station. To open this A X P ro p e r t y S h e e t , double-click the relativeTopicBuilder component in the
station.
In addition to the standard properties (Status and Fault Cause), these properties support the
relativeTopicBuilder.
Signature read-only Identifies the mathematical scheme used to verify the authen-
ticity of the program.
User Defined read-only Displays user-defined custom imports of the types used in the
Imports source code.
Actions
Execute takes some input, does whatever the action should (including
G
B global cov slot filter ............................................78
base query .........................................................75
BFormat property...............................................90
binding ..............................................................21 H
bound array........................................................80
bound CSV property ...........................................84 handlers .............................................................47
bound object......................................................79 history export marker filter .................................92
bound property ..................................................83
bound query result ....................................... 74, 88
builder class .......................................................57 I
inbound components ..........................................43
index of............................................................ 102
C inline json writer ......................................... 54, 107
client-side target media ......................................12 introduction .........................................................9
components .......................................................65
contains key ..................................................... 101
count .................................................................82 J
current time .......................................................82 JSON alarm recipient..........................................36
custom content ..................................................57 Json Schema ......................................................66
json schema config folder ...................................67
Json schema service ...........................................13
D Json Schema Service...........................................14
debug .......................................................... 70–71 jsonPath
demux router .....................................................99 applying selection criteria ................................46
developer guide .................................................51 JsonSchemaService ............................................76
device
connecting ......................................................40
document change log ...........................................7 K
key/value pair.....................................................11
E
engine cycle message queue ......................106–107 L
examples...................................................... 11, 38 length .............................................................. 103
export................................................................38 license
export deregistration handler .............................96 expiration .......................................................77
requirements ..................................................13 S
S M A expiration .................................................77
schema...............................................................25
M construction....................................................21
message generation ......................................................58
router .............................................................98 properties .......................................................66
routing a a subset of a payload ........................44 types ..............................................................51
routing the whole payload ...............................43 selectors ............................................................46
methods.............................................................57 setpoint handler ......................................... 48, 105
metrics...............................................................71 SMA expiration monitor ......................................13
structure ............................................................20
subscription .......................................................62
N subscription slot blacklist ....................................78
sum.................................................................. 102
naming conventions............................................18
O T
tag .....................................................................86
object ..................................................... 11, 22, 78
tag list................................................................87
output................................................................38
transport protocols.............................................12
control ............................................................19
troubleshooting..................................................71
history, viewing ...............................................20
tuning policy................................................. 19, 69
overrides............................................................19
type override.............................................. 53, 108
folder..............................................................70
P U
path ................................................................. 100 Unix time............................................................83
property....................................................... 81–82 use cases, transport protocols, JSON Toolkit
Px target media features ............................................................11
about types of.................................................12
Hx PxMedia ....................................................12
Mobile PxMedia ..............................................12 V
Report PxMedia ..............................................12 visualization.................................................. 33, 41
WbPxMedia ....................................................12
Q
queries......................................................... 30, 73
bound query result ..........................................31
folder..............................................................30
query ..............................................................30
relative history query .......................................31
query ...................................................... 55, 73, 87
ad hoc.............................................................33
base query ......................................................33
R
related documentation .........................................7
relative history query .................................... 74, 88
relative Json schema..................................... 27, 75
relative topic builder................................... 53, 109
Run As User........................................................14
payload The objects, arrays and key/value pairs contained between open and close
curly brackets that conform to JSON syntax.
subscription A method for updating a station with the current value of a remote
component. When a remote component’s value changes, subscription
synchronizes the related proxy point’s value in the station with the current
value of the remote component. Subscription occurs in real time.