/*
Project Name: SPIKE Prime Web Interface
File name: ServiceDock_SystemLink.js
Author: Jeremy Jung
Last update: 7/19/20
Description: HTML Element definition for <service-systemlink> to be used in ServiceDocks
Credits/inspirations:
History:
Created by Jeremy on 7/16/20
LICENSE: MIT
(C) Tufts Center for Engineering Education and Outreach (CEEO)
*/
// import { Service_SystemLink } from "./Service_SystemLink.js";
class servicesystemlink extends HTMLElement {
constructor () {
super();
this.active = false; // whether the service was activated
this.service = new Service_SystemLink(); // instantiate a service object ( one object per button )
this.proceed = false; // if there are credentials input
// Create a shadow root
var shadow = this.attachShadow({ mode: 'open' });
/* wrapper definition and CSS */
var wrapper = document.createElement('div');
wrapper.setAttribute('class', 'wrapper');
wrapper.setAttribute("style", "width: 50px; height: 50px; position: relative; margin-top: 10px;")
/* ServiceDock button definition and CSS */
var button = document.createElement("button");
button.setAttribute("id", "sl_button");
button.setAttribute("class", "SD_button");
/* CSS */
//var imageRelPath = "./modules/views/systemlinkIcon.png" // relative to the document in which a servicesystemlink is created ( NOT this file )
var length = 50; // for width and height of button
var backgroundColor = "#A2E1EF" // background color of the button
var buttonStyle = "width:" + length + "px; height:" + length + "px; background:" + "url('')" + "no-repeat; background-size: 50px 50px; background-color:" + backgroundColor
+ "; border: none; background-position: center; cursor: pointer; border-radius: 10px; position: relative; margin: 4px 0px; "
button.setAttribute("style", buttonStyle);
/* status circle definition and CSS */
this.status = document.createElement("div");
this.status.setAttribute("class", "status");
/* CSS */
var length = 20; // for width and height of circle
var statusBackgroundColor = "red" // default background color of service (inactive color)
var posLeft = 30;
var posTop = 20;
var statusStyle = "border-radius: 50%; height:" + length + "px; width:" + length + "px; background-color:" + statusBackgroundColor +
"; position: relative; left:" + posLeft + "px; top:" + posTop + "px;";
this.status.setAttribute("style", statusStyle);
/* event listeners */
button.addEventListener("mouseleave", function (event) {
button.style.backgroundColor = "#A2E1EF";
button.style.color = "#000000";
});
button.addEventListener("mouseenter", function(event){
button.style.backgroundColor = "#FFFFFF";
button.style.color = "#000000";
})
this.addEventListener("click", async function() {
if ( !this.active ) {
this.popUpBox();
}
// check active flag so once activated, the service doesnt reinit
if ( !this.active && this.proceed) {
console.log("%cTuftsCEEO ", "color: #3ba336;", "Activating SystemLink Service");
var initSuccessful = await this.service.init(this.APIKey);
if (initSuccessful) {
this.active = true;
this.status.style.backgroundColor = "green";
}
}
});
shadow.appendChild(wrapper);
button.appendChild(this.status);
wrapper.appendChild(button);
}
/* Ask user for API credentials */
popUpBox() {
var APIKeyExists = true;
// if apikey was not given in attributes
if (this.getAttribute("apikey") == undefined || this.getAttribute("apikey") == "") {
var APIKeyResult = prompt("Please enter your System Link Cloud API Key:");
// APIkey
if (APIKeyResult == null || APIKeyResult == "") {
console.log("%cTuftsCEEO ", "color: #3ba336;", "You inserted no API key");
APIKeyExists = false;
}
else {
this.APIKey = APIKeyResult;
}
}
else {
var APIKeyResult = this.getAttribute("apikey");
this.APIKey = APIKeyResult;
}
if ( APIKeyExists ) {
this.proceed = true;
}
}
/* for Service's API credentials */
static get observedAttributes() {
return ["apikey"];
}
get apikey() {
return this.getAttribute("apikey");
}
set apikey(val) {
// console.log("%cTuftsCEEO ", "color: #3ba336;", val);
if ( val ) {
this.setAttribute("apikey", val);
}
else {
this.removeAttribute("apikey");
}
}
attributeChangedCallback (name, oldValue, newValue) {
console.log("%cTuftsCEEO ", "color: #3ba336;", "new value of apikey: ", newValue);
this.APIKey = newValue;
}
/* get the Service_SystemLink object */
getService() {
return this.service;
}
/* get whether the ServiceDock button was clicked */
getClicked() {
return this.active;
}
// initialize the service (is not used in this class but available for use publicly)
async init() {
var initSuccess = await this.service.init(this.APIKey);
if (initSuccess) {
this.status.style.backgroundColor = "green";
this.active = true;
return true;
}
else {
return false;
}
}
}
// when defining custom element, the name must have at least one - dash
window.customElements.define('service-systemlink', servicesystemlink);
/*
Project Name: SPIKE Prime Web Interface
File name: Service_SystemLink.js
Author: Jeremy Jung
Last update: 8/04/20
Description: SystemLink Service Library (OOP)
History:
Created by Jeremy on 7/15/20
LICENSE: MIT
(C) Tufts Center for Engineering Education and Outreach (CEEO)
*/
/**
*
* @class Service_SystemLink
* @example
* // assuming you declared <service-systemlink> with the id, "service_systemlink"
* var mySL = document.getElemenyById("service_systemlink").getService();
* mySL.setAttribute("apikey", "YOUR API KEY");
* mySL.init();
*/
function Service_SystemLink() {
//////////////////////////////////////////
// //
// Global Variables //
// //
//////////////////////////////////////////
/* private members */
let tagsInfo = {}; // contains real-time information of the tags in the cloud
let APIKey = "API KEY";
let serviceActive = false; // set to true when service goes through init
let pollInterval = 1000;
var funcAtInit = undefined; // function to call after init
//////////////////////////////////////////
// //
// Public Functions //
// //
//////////////////////////////////////////
/** initialize SystemLink_Service
* <p> Starts polling the System Link cloud </p>
* <p> <em> this function needs to be executed after executeAfterInit but before all other public functions </em> </p>
*
* @public
* @param {string} APIKeyInput SYstemlink APIkey
* @param {integer} pollIntervalInput interval at which to get tags from the cloud in MILISECONDS. Default value is 1000 ms.
* @returns {boolean} True if service was successsfully initialized, false otherwise
* @example
* var SystemLinkElement = document.getElemenyById("service_systemlink");
* var mySL = SystemLinkElement.getService();
* mySL.init("APIKEY", 1000); // initialize SystemLink Service with a poll interval of 10 ms
*
*/
async function init(APIKeyInput, pollIntervalInput) {
// if an APIKey was specified
if (APIKeyInput !== undefined) {
APIKey = APIKeyInput;
}
var response = await checkAPIKey(APIKey);
// if response from checkAPIKey is valid
if (response) {
if (pollIntervalInput !== undefined) {
pollInterval = await pollIntervalInput;
}
// initialize the tagsInfo global variable
updateTagsInfo(function () {
serviceActive = true;
// call funcAtInit if defined
if (funcAtInit !== undefined) {
funcAtInit();
}
});
return true;
}
else {
return false;
}
}
/** Get the callback function to execute after service is initialized
* <p> <em> This function needs to be executed before calling init() </em> </p>
*
* @public
* @param {function} callback function to execute after initialization
* @example
* mySL.executeAfterInit( function () {
* var tagsInfo = mySL.getTagsInfo();
* })
*/
function executeAfterInit(callback) {
// Assigns global variable funcAtInit a pointer to callback function
funcAtInit = callback;
}
/** Return the tagsInfo global variable
*
* @public
* @returns basic information about currently existing tags in the cloud
* @example
* var tagsInfo = mySL.getTagsInfo();
* var astringValue = tagsInfo["astring"]["value"];
* var astringType = tagsInfo["astring"]["type"];
*/
function getTagsInfo() {
return tagsInfo;
}
/** Change the current value of a tag on SystemLink cloud.
*
* @private
* @param {string} name name of tag to update
* @param {any} value new value's data type must match the Tag's data type.
* @param {function} callback function to execute after tag is updated
* @example
* // set a string type Value of a Tag and display
* mySL.setTagValue("message", "hello there", function () {
* let messageValue = mySL.getTagValue("message");
* console.log("message: ", messageValue); // display the updated value
* })
* // set value of a boolean Tag
* mySL.setTagValue("aBoolean", true);
*
* // set value of an integer Tag
* mySL.setTagValue("anInteger", 10);
*
* // set value of a double Tag
* mySL.setTagValue("aDouble", 5.2);
*/
function setTagValue(tagName, newValue, callback) {
// changes the value of a tag on the cloud
setTagValueStrict(tagName, newValue, callback);
}
/** Change the current value of a tag on SystemLink cloud with strict data types. Values will be implicitly converted
* <br>
* NotStrict property indicates that the data type of the Value supplied will be implicitly converted. For example, allowing for setting an INT tag's value with a string, "123" or a STRING tag's value with
* a number. This method exists for convenience but please avoid using it extensively as it can lead to unpredictable outcomes.
* @public
* @param {any} tagName
* @param {any} newValue
* @param {any} callback
* @example
* // set a string type Value of a Tag and display
* mySL.setTagValueNotStrict("message", 123, function () {
* let messageValue = mySL.getTagValue("message");
* console.log("message: ", messageValue); // display the updated value, which will be 123.
* })
* // set value of a boolean Tag
* mySL.setTagValueNotStrict("aBoolean", true);
*
* // set value of an integer Tag
* mySL.setTagValueNotStrict("anInteger", 10);
* mySL.setTagValueNotStrict("anInteger", "10");
*
* // set value of a double Tag
* mySL.setTagValueNotStrict("aDouble", 5.2);
* mySL.setTagValueNotStrict("aDouble", "5.2");
*/
function setTagValueNotStrict(tagName, newValue, callback) {
// changes the value of a tag on the cloud
changeValue(tagName, newValue, false, function (valueChanged) {
if (valueChanged) {
// wait for changed value to be retrieved
setTimeout(function () {
if (typeof callback === 'function') {
callback();
}
}, 1000)
}
});
}
/** Change the current value of a tag on SystemLink cloud with strict data types. There will be no implicit data type conversions. E.g. Updating tags of INT type will only work with javascript number.
*
* @public
* @param {any} name name of tag to update
* @param {any} value value to update tag to
* @param {any} callback function to execute after tag is updated
* @example
* // set a string type Value of a Tag and display
* mySL.setTagValueStrict("message", "hello there", function () {
* let messageValue = mySL.getTagValue("message");
* console.log("message: ", messageValue); // display the updated value
* })
* // set value of a boolean Tag
* mySL.setTagValueStrict("aBoolean", true);
*
* // set value of an integer Tag
* mySL.setTagValueStrict("anInteger", 10);
*
* // set value of a double Tag
* mySL.setTagValueStrict("aDouble", 5.2);
*/
function setTagValueStrict(tagName, newValue, callback) {
// changes the value of a tag on the cloud
changeValue(tagName, newValue, true, function (valueChanged) {
if (valueChanged) {
// wait for changed value to be retrieved
setTimeout(function () {
if (typeof callback === 'function') {
callback();
}
}, 1000)
}
});
}
/** Get the current value of a tag on SystemLink cloud
*
* @public
* @param {string} tagName
* @returns {any} current value of tag
* @example
* messageValue = mySL.getTagValue("message");
* console.log("message: ", messageValue);
*/
function getTagValue(tagName) {
var currentValue = tagsInfo[tagName].value;
return currentValue;
}
/** Get whether the Service was initialized or not
*
* @public
* @returns {boolean} whether Service was initialized or not
* @example
* if (mySL.isActive() === true)
* // do something if SystemLink Service is active
*/
function isActive() {
return serviceActive;
}
/** Change the APIKey
* @ignore
* @param {string} APIKeyInput
*/
function setAPIKey(APIKeyInput) {
// changes the global variable APIKey
APIKey = APIKeyInput;
}
/** Create a new tag. The type of new tag is determined by the javascript data type of tagValue.
* @public
* @param {string} tagName name of tag to create
* @param {any} tagValue value to assign the tag after creation
* @param {function} callback optional callback
* @example
* mySL.createTag("message", "hi", function () {
* mySL.setTagValueStrict("message", "bye"); // change the value of 'message' from "hi" to "bye"
* })
*/
function createTag(tagName, tagValue, callback) {
// get the SystemLink formatted data type of tag
var valueType = getValueType(tagValue);
// create a tag with the name and data type. If tag exists, it still returns successful response
createNewTagHelper(tagName, valueType, function (newTagCreated) {
// after tag is created, assign a value to it
changeValue(tagName, tagValue, false, function (newTagValueAssigned) {
// execute callback if successful
if (newTagCreated) {
if (newTagValueAssigned) {
// wait for changed value to be retrieved
setTimeout( function() {
if (typeof callback == 'function') {
callback();
}
}, 1000)
}
}
})
})
}
/** Delete tag
*
* @public
* @param {string} tagName name of tag to delete
* @param {function} callback optional callback
* @example
* mySL.deleteTag("message", function () {
* let tagsInfo = mySL.getTagsInfo();
* console.log("tagsInfo: ", tagsInfo); // tags information will now not contain the 'message' tag
* })
*/
function deleteTag(tagName, callback) {
// delete the tag on System Link cloud
deleteTagHelper(tagName, function (tagDeleted) {
if ( tagDeleted ) {
typeof callback === 'function' && callback();
}
});
}
//////////////////////////////////////////
// //
// Private Functions //
// //
//////////////////////////////////////////
/** sleep function
*
* @private
* @param {integer} ms
* @returns {Promise}
*/
function sleep(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
/** Check if Systemlink API key is valid for use
*
* @private
* @param {string} APIKeyInput
* @returns {Promise} resolve(true) or reject(error)
*/
async function checkAPIKey(APIKeyInput) {
return new Promise(async function (resolve, reject) {
var apiKeyAuthURL = "https://api.systemlinkcloud.com/niauth/v1/auth";
var request = await sendXMLHTTPRequest("GET", apiKeyAuthURL, APIKeyInput)
request.onload = function () {
var response = JSON.parse(request.response);
if (response.error) {
reject(new Error("Error at apikey auth:", response));
}
else {
console.log("%cTuftsCEEO ", "color: #3ba336;", "APIkey is valid")
resolve(true)
}
}
request.onerror = function () {
var response = JSON.parse(request.response);
// console.log("Error at apikey auth:", request.response);
reject(new Error("Error at apikey auth:", response));
}
})
}
/** Assign list of tags existing in the cloud to {tagPaths} global variable
*
* @private
* @param {function} callback
*/
async function updateTagsInfo(callback) {
// get the tags the first time before running callback
getTagsInfoFromCloud(function (collectedTagsInfo) {
// if the collectedTagsInfo is defined and not boolean false
if (collectedTagsInfo) {
tagsInfo = collectedTagsInfo;
}
// after tagsInfo is initialized, begin the interval to update it
setInterval(async function () {
getTagsInfoFromCloud(function (collectedTagsInfo) {
// if the object is defined and not boolean false
if (collectedTagsInfo) {
tagsInfo = collectedTagsInfo;
}
});
}, pollInterval)
// run the callback of updateTagsInfo inside init()
callback();
});
}
/** Get the info of a tag in the cloud
*
* @private
* @param {function} callback
*/
async function getTagsInfoFromCloud(callback) {
// make a new promise
new Promise(async function (resolve, reject) {
var collectedTagsInfo = {}; // to return
var getMultipleTagsURL = "https://api.systemlinkcloud.com/nitag/v2/tags-with-values";
// send request to SystemLink API
var request = await sendXMLHTTPRequest("GET", getMultipleTagsURL, APIKey);
// when transaction is complete, parse response and update return value (collectedTagsInfo)
request.onload = async function () {
// parse response (string) into JSON object
var responseJSON = JSON.parse(this.response)
var tagsInfoArray = responseJSON.tagsWithValues;
// get total number of tags
var tagsAmount = responseJSON.totalCount;
for (var i = 0; i < tagsAmount; i++) {
// parse information of the tags
try {
var value = tagsInfoArray[i].current.value.value;
var valueType = tagsInfoArray[i].current.value.type;
var tagName = tagsInfoArray[i].tag.path;
var valueToAdd = await getValueFromType(valueType, value);
// store tag information
var pathInfo = {};
pathInfo["value"] = valueToAdd;
pathInfo["type"] = valueType;
// add a tag info to the return object
collectedTagsInfo[tagName] = pathInfo;
}
// when value is not yet assigned to tag
catch (e) {
var value = null
var valueType = tagsInfoArray[i].tag.type;
var tagName = tagsInfoArray[i].tag.path;
// store tag information
var pathInfo = {};
pathInfo["value"] = value;
pathInfo["type"] = valueType;
// add a tag info to the return object
collectedTagsInfo[tagName] = pathInfo;
}
}
resolve(collectedTagsInfo)
}
request.onerror = function () {
console.log("%cTuftsCEEO ", "color: #3ba336;", this.response);
reject(false);
}
}).then(
// success handler
function (resolve) {
//run callback with resolve object
callback(resolve);
},
// failure handler
function (reject) {
// run calllback with reject object
callback(reject);
}
)
}
/** Send PUT request to SL cloud API and change the value of a tag
* This function will receive a newValue of any kind of type. Before the POST request is sent,
* the SL data type of the tag to convert must be found, and newValue must be in string format
* @private
* @param {string} tagPath string of the name of the tag
* @param {any} newValue value to assign tag
* @param {function} callback
*/
async function changeValue(tagPath, newValue, strict, callback) {
new Promise(async function (resolve, reject) {
var URL = "https://api.systemlinkcloud.com/nitag/v2/tags/" + tagPath + "/values/current";
// assume newValue is already in correct datatype and just give the data type in SystemLink format
//var valueType = getValueType(newValue);
var valueType;
var newValueStringified;
// if Tag to change does not yet exist (possibly due to it being created very recently)
if (tagsInfo[tagPath] === undefined) {
// refer to newValue's JS type to deduce Tag's data type
valueType = getValueTypeStrict(newValue);
}
// Tag to change exists; find the SL data type of tag from locally stored tagsInfo
else {
if (strict === true) {
/* strict changeValue. So no implicit data type conversions. All newValue's types need to match the tag's type */
expectedValueType = tagsInfo[tagPath].type;
inputValueType = getValueTypeStrict(newValue);
// console.log("%cTuftsCEEO ", "color: #3ba336;", expectedValueType, " vs ", inputValueType);
if (inputValueType !== expectedValueType) {
console.error("%cTuftsCEEO ", "color: #3ba336;", "Could not update value of tag on SystemLink Cloud. The given value is not of the data type defined for the tag in the database");
throw new Error("Could not update value of tag on SystemLink Cloud.The given value is not of the data type defined for the tag in the database");
}
else {
valueType = tagsInfo[tagPath].type;
}
}
else {
valueType = tagsInfo[tagPath].type;
}
}
newValueStringified = changeToString(newValue);
var data = { "value": { "type": valueType, "value": newValueStringified } };
var requestBody = data;
var request = await sendXMLHTTPRequest("PUT", URL, APIKey, requestBody);
request.onload = function () {
resolve(true);
}
request.onerror = function () {
reject(false);
}
// catch error
request.onreadystatechange = function () {
if (this.readyState === XMLHttpRequest.DONE && (this.status != 200) ) {
console.log("%cTuftsCEEO ", "color: #3ba336;", this.status + " Error at changeValue: ", this.response)
}
}
}).then(
// success handler
function (resolve) {
callback(resolve);
},
function (reject) {
callback(reject);
}
)
}
/** Send PUT request to SL cloud API and change the value of a tag
*
* @private
* @param {string} tagPath name of the tag
* @param {string} tagType SystemLink format data type of tag
* @param {function} callback
*/
async function createNewTagHelper(tagPath, tagType, callback) {
new Promise(async function (resolve, reject) {
var URL = "https://api.systemlinkcloud.com/nitag/v2/tags/";
var data = { "type": tagType, "properties": {}, "path": tagPath, "keywords": [], "collectAggregates": false };
var requestBody = data;
var request = await sendXMLHTTPRequest("POST", URL, APIKey, requestBody);
request.onload = function () {
resolve(true);
}
request.onerror = function () {
console.log("%cTuftsCEEO ", "color: #3ba336;", "Error at createNewTagHelper", request.response);
reject(false);
}
// catch error
request.onreadystatechange = function () {
if (this.readyState === XMLHttpRequest.DONE && (this.status != 200 && this.status != 201)) {
console.log("%cTuftsCEEO ", "color: #3ba336;", this.status + " Error at createNewTagHelper: ", this.response)
}
}
}).then(
// success handler
function (resolve) {
callback(resolve)
},
// error handler
function (reject) {
callback(reject)
}
)
}
/** Delete the tag on the System Link cloud
*
* @private
* @param {string} tagName
* @param {function} callback
*/
async function deleteTagHelper ( tagName, callback ) {
new Promise(async function (resolve, reject) {
var URL = "https://api.systemlinkcloud.com/nitag/v2/tags/" + tagName;
var request = await sendXMLHTTPRequest("DELETE", URL, APIKey);
request.onload = function () {
resolve(true);
}
request.onerror = function () {
console.log("%cTuftsCEEO ", "color: #3ba336;", "Error at deleteTagHelper", request.response);
reject(false);
}
// catch error
request.onreadystatechange = function () {
if (this.readyState === XMLHttpRequest.DONE && this.status != 200) {
console.log("%cTuftsCEEO ", "color: #3ba336;", this.status + " Error at deleteTagHelper: ", this.response)
}
}
}).then(
// success handler
function (resolve) {
callback(resolve)
},
// error handler
function (reject) {
callback(reject)
}
)
}
/** Helper function for sending XMLHTTPRequests
*
* @private
* @param {string} method
* @param {string} URL
* @param {string} APIKeyInput
* @param {object} body
* @returns {object} XMLHttpRequest
*/
async function sendXMLHTTPRequest(method, URL, APIKeyInput, body) {
var request = new XMLHttpRequest();
request.open(method, URL, true);
//Send the proper header information along with the request
request.setRequestHeader("x-ni-api-key", APIKeyInput);
if (body === undefined) {
request.setRequestHeader("Accept", "application/json");
request.send();
}
else {
request.setRequestHeader("Content-type", "application/json");
var requestBody = JSON.stringify(body);
try {
request.send(requestBody);
} catch (e) {
console.log("%cTuftsCEEO ", "color: #3ba336;", "error sending request:", request.response);
}
}
return request;
}
/** Helper function for getting data types in systemlink format
*
* @private
* @param {any} new_value the variable containing the new value of a tag
* @returns {any} data type of tag
*/
function getValueType(new_value) {
//if the value is not a number
if (isNaN(new_value)) {
//if the value is a boolean
if (new_value === "true" || new_value === "false") {
return "BOOLEAN";
}
//if the value is a string
return "STRING";
}
//value is a number
else {
//if value is an integer
if (Number.isInteger(parseFloat(new_value))) {
return "INT"
}
//if value is a double
else {
return "DOUBLE"
}
}
}
/**
* @private
* @param {any} new_value
* @returns {string} data type of tag
*/
function getValueTypeStrict(new_value) {
//if the value is a boolean
if (typeof new_value === "boolean") {
return "BOOLEAN";
}
else if (typeof new_value === "string") {
return "STRING";
}
else if (typeof new_value === "number") {
if (Number.isInteger(parseFloat(new_value))) {
return "INT"
}
//if value is a double
else {
return "DOUBLE"
}
}
}
/** stringify newValue
* Note: for POST request
* @private
* @param {any} newValue
* @returns {string} newValue stringified
*/
function changeToString(newValue) {
var newValueConverted;
// already a string
if (typeof newValue == "string") {
newValueConverted = newValue;
}
else {
newValueConverted = JSON.stringify(newValue);
}
return newValueConverted;
}
/** Helper function for converting values to correct type based on data type
*
* @private
* @param {string} valueType data type of value in systemlink format
* @param {string} value value to convert
* @returns {any} converted value
*/
function getValueFromType(valueType, value) {
if (valueType == "BOOLEAN") {
if (value == "true") {
return true;
}
else {
return false;
}
}
else if (valueType == "STRING") {
return value;
}
else if (valueType == "INT" || valueType == "DOUBLE") {
return parseFloat(value);
}
return value;
}
/* public members */
return {
init: init,
getTagsInfo: getTagsInfo,
setTagValueNotStrict: setTagValueNotStrict,
setTagValueStrict: setTagValueStrict,
getTagValue: getTagValue,
executeAfterInit: executeAfterInit,
setAPIKey: setAPIKey,
isActive: isActive,
createTag: createTag,
deleteTag: deleteTag
}
}