/*
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
    }
}