# HG changeset patch # User Edouard Tisserant # Date 1571152488 -7200 # Node ID ddb2c4668a6b2fd6902a86ffd6eeafff7859f238 # Parent c5ba1e77f05499b354c005de95c047686fe27d65 SVGHMI : many details about communication implemented in JS, with side effects. diff -r c5ba1e77f054 -r ddb2c4668a6b svghmi/Makefile --- a/svghmi/Makefile Fri Oct 11 12:03:14 2019 +0200 +++ b/svghmi/Makefile Tue Oct 15 17:14:48 2019 +0200 @@ -16,7 +16,7 @@ all:$(xsltfiles) -%.xslt: %.ysl2 ../yslt_noindent.yml2 +%.xslt: %.ysl2 svghmi.js ../yslt_noindent.yml2 $(yml2path)/yml2c -I $(yml2path):../ $< -o $@.tmp xmlstarlet fo $@.tmp > $@ rm $@.tmp diff -r c5ba1e77f054 -r ddb2c4668a6b svghmi/gen_index_xhtml.xslt --- a/svghmi/gen_index_xhtml.xslt Fri Oct 11 12:03:14 2019 +0200 +++ b/svghmi/gen_index_xhtml.xslt Tue Oct 15 17:14:48 2019 +0200 @@ -1,6 +1,6 @@ <?xml version="1.0"?> -<xsl:stylesheet xmlns:func="http://exslt.org/functions" xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" xmlns:svg="http://www.w3.org/2000/svg" xmlns:str="http://exslt.org/strings" xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns:exsl="http://exslt.org/common" xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" xmlns:ns="beremiz" xmlns:cc="http://creativecommons.org/ns#" xmlns:regexp="http://exslt.org/regular-expressions" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:dc="http://purl.org/dc/elements/1.1/" extension-element-prefixes="ns func" version="1.0" exclude-result-prefixes="ns str regexp exsl func"> - <xsl:output method="xml" cdata-section-elements="script"/> +<xsl:stylesheet xmlns:func="http://exslt.org/functions" xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" xmlns:svg="http://www.w3.org/2000/svg" xmlns:str="http://exslt.org/strings" xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns:exsl="http://exslt.org/common" xmlns:xhtml="http://www.w3.org/1999/xhtml" xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" xmlns:ns="beremiz" xmlns:cc="http://creativecommons.org/ns#" xmlns:regexp="http://exslt.org/regular-expressions" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:dc="http://purl.org/dc/elements/1.1/" extension-element-prefixes="ns func" version="1.0" exclude-result-prefixes="ns str regexp exsl func"> + <xsl:output method="xml" cdata-section-elements="xhtml:script"/> <xsl:variable name="geometry" select="ns:GetSVGGeometry()"/> <xsl:variable name="hmitree" select="ns:GetHMITree()"/> <xsl:variable name="hmi_elements" select="//svg:*[starts-with(@inkscape:label, 'HMI:')]"/> @@ -179,6 +179,10 @@ <func:result select="exsl:node-set($ast)"/> </func:function> <xsl:template name="scripts"> + <xsl:text>(function(){ +</xsl:text> + <xsl:text> +</xsl:text> <xsl:text>var hmi_hash = [</xsl:text> <xsl:value-of select="$hmitree/@hash"/> <xsl:text>]; @@ -212,7 +216,7 @@ </xsl:for-each> <xsl:text> ], </xsl:text> - <xsl:text> paths: [ + <xsl:text> indexes: [ </xsl:text> <xsl:for-each select="$widget/path"> <xsl:variable name="hmipath" select="@value"/> @@ -245,36 +249,16 @@ </xsl:text> <xsl:text> </xsl:text> - <xsl:text>var hmi_index = [ -</xsl:text> - <xsl:variable name="svg" select="/"/> + <xsl:text>var hmitree_types = [ +</xsl:text> <xsl:for-each select="$indexed_hmitree/*"> - <xsl:text>{ /* </xsl:text> + <xsl:text>/* </xsl:text> <xsl:value-of select="@index"/> <xsl:text> </xsl:text> <xsl:value-of select="@hmipath"/> - <xsl:text> */ -</xsl:text> - <xsl:text> type: "</xsl:text> - <xsl:value-of select="local-name()"/> - <xsl:text>", -</xsl:text> - <xsl:text> ids: [ -</xsl:text> - <xsl:variable name="hmipath" select="@hmipath"/> - <xsl:for-each select="$svg//*[substring-after(@inkscape:label,'@') = $hmipath]"> - <xsl:text> hmi_widgets["</xsl:text> - <xsl:value-of select="@id"/> - <xsl:text>"]</xsl:text> - <xsl:if test="position()!=last()"> - <xsl:text>,</xsl:text> - </xsl:if> - <xsl:text> -</xsl:text> - </xsl:for-each> - <xsl:text> ] -</xsl:text> - <xsl:text>}</xsl:text> + <xsl:text> */ "</xsl:text> + <xsl:value-of select="substring(local-name(), 5)"/> + <xsl:text>"</xsl:text> <xsl:if test="position()!=last()"> <xsl:text>,</xsl:text> </xsl:if> @@ -304,9 +288,8 @@ <xsl:text> widgets: [ </xsl:text> <xsl:for-each select="$page_ids"> - <xsl:text> "</xsl:text> + <xsl:text> hmi_widgets.</xsl:text> <xsl:value-of select="."/> - <xsl:text>"</xsl:text> <xsl:if test="position()!=last()"> <xsl:text>,</xsl:text> </xsl:if> @@ -315,24 +298,6 @@ </xsl:for-each> <xsl:text> ] </xsl:text> - <xsl:text> subscriptions: [ -</xsl:text> - <xsl:for-each select="$page_elements"> - <xsl:variable name="hmipaths" select="func:parselabel(@inkscape:label)/widget/path/@value"/> - <xsl:variable name="notlast" select="position()!=last()"/> - <xsl:for-each select="$hmipaths"> - <xsl:variable name="hmipath" select="."/> - <xsl:text> </xsl:text> - <xsl:value-of select="$indexed_hmitree/*[@hmipath = $hmipath]/@index"/> - <xsl:if test="$notlast or position()!=last()"> - <xsl:text>,</xsl:text> - </xsl:if> - <xsl:text> -</xsl:text> - </xsl:for-each> - </xsl:for-each> - <xsl:text> ] -</xsl:text> <xsl:text> }</xsl:text> <xsl:if test="position()!=last()"> <xsl:text>,</xsl:text> @@ -352,147 +317,305 @@ </xsl:text> <xsl:text> </xsl:text> - <xsl:text>(function(){ -</xsl:text> - <xsl:text> // Open WebSocket to relative "/ws" address -</xsl:text> - <xsl:text> var ws = new WebSocket(window.location.href.replace(/^http(s?:\/\/[^\/]*)\/.*$/, 'ws$1/ws')); -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> // Register message reception handler -</xsl:text> - <xsl:text> ws.onmessage = function (evt) { -</xsl:text> - <xsl:text> // TODO : dispatch and cache hmi tree updates -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> var received_msg = evt.data; -</xsl:text> - <xsl:text> // TODO : check for hmitree hash header -</xsl:text> - <xsl:text> // if not matching, reload page -</xsl:text> - <xsl:text> alert("Message is received..."+received_msg); + <xsl:text>function dispatch_value(index, value) { +</xsl:text> + <xsl:text> console.log("dispatch_value("+index+value+")"); +</xsl:text> + <xsl:text>}; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>// Open WebSocket to relative "/ws" address +</xsl:text> + <xsl:text>var ws = new WebSocket(window.location.href.replace(/^http(s?:\/\/[^\/]*)\/.*$/, 'ws$1/ws')); +</xsl:text> + <xsl:text>ws.binaryType = 'arraybuffer'; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>const dvgetters = { +</xsl:text> + <xsl:text> INT: [DataView.prototype.getInt16, 2], +</xsl:text> + <xsl:text> BOOL: [DataView.prototype.getInt8, 1] +</xsl:text> + <xsl:text> /* TODO */ +</xsl:text> + <xsl:text>}; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>// Register message reception handler +</xsl:text> + <xsl:text>ws.onmessage = function (evt) { +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text> let data = evt.data; +</xsl:text> + <xsl:text> let dv = new DataView(data); +</xsl:text> + <xsl:text> let i = 0; +</xsl:text> + <xsl:text> for(let hash_int of hmi_hash) { +</xsl:text> + <xsl:text> if(hash_int != dv.getUint8(i)){ +</xsl:text> + <xsl:text> console.log("Recv non maching hash. Reload."); +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text> // 1003 is for "Unsupported Data" +</xsl:text> + <xsl:text> ws.close(1003,"Hash doesn't match"); +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text> // TODO : remove debug alert ? +</xsl:text> + <xsl:text> alert("HMI will be reloaded."); +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text> // force reload ignoring cache +</xsl:text> + <xsl:text> location.reload(true); +</xsl:text> + <xsl:text> }; +</xsl:text> + <xsl:text> i++; </xsl:text> <xsl:text> }; </xsl:text> <xsl:text> </xsl:text> - <xsl:text> // Once connection established -</xsl:text> - <xsl:text> ws.onopen = function (evt) { -</xsl:text> - <xsl:text> // TODO : enable the HMI (was previously offline, or just starts) -</xsl:text> - <xsl:text> // show main page -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> // TODO : prefix with hmitree hash header -</xsl:text> - <xsl:text> ws.send("test"); + <xsl:text> while(i < data.length){ +</xsl:text> + <xsl:text> let index = dv.getUint32(i); +</xsl:text> + <xsl:text> i += 4; +</xsl:text> + <xsl:text> let iectype = hmitree_types[index]; +</xsl:text> + <xsl:text> let [dvgetter, bytesize] = dvgetters[iectypes]; +</xsl:text> + <xsl:text> value = dvgetter.call(dv,i); +</xsl:text> + <xsl:text> dispatch_value(index, value); +</xsl:text> + <xsl:text> i += bytesize; </xsl:text> <xsl:text> }; </xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> var pending_updates = {}; -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> // subscription state, as it should be in hmi server -</xsl:text> - <xsl:text> // expected {index:period} -</xsl:text> - <xsl:text> const subscriptions = new Map(); -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> // subscription state as needed by widget now -</xsl:text> - <xsl:text> // expected {index:[widgets]}; -</xsl:text> - <xsl:text> var subscribers = new Map(); -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> // return the diff in between curently subscribed and subscription -</xsl:text> - <xsl:text> function update_subscriptions() { -</xsl:text> - <xsl:text> let delta = []; -</xsl:text> - <xsl:text> for(let [index, widgets] in subscribers.entries()){ -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> // periods are in ms -</xsl:text> - <xsl:text> let previous_period = subscriptions.get(index); -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> let new_period = Math.min(...widgets.map(widget => 1000/widget.frequency)); -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> if(previous_period != new_period) { -</xsl:text> - <xsl:text> subscriptions.set(index, new_period); -</xsl:text> - <xsl:text> delta.push({index: index, period: new_period}); -</xsl:text> - <xsl:text> } -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> }) -</xsl:text> - <xsl:text> return result; + <xsl:text>}; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>function send_blob(data) { +</xsl:text> + <xsl:text> if(data.length > 0) { +</xsl:text> + <xsl:text> ws.send(new Blob([ +</xsl:text> + <xsl:text> new Uint8Array(hmi_hash), +</xsl:text> + <xsl:text> data])); +</xsl:text> + <xsl:text> }; +</xsl:text> + <xsl:text>}; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>const typedarray_types = { +</xsl:text> + <xsl:text> INT: Int16Array, +</xsl:text> + <xsl:text> BOOL: Uint8Array +</xsl:text> + <xsl:text> /* TODO */ +</xsl:text> + <xsl:text>}; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>function send_reset() { +</xsl:text> + <xsl:text> send_blob(new Uint8Array([1])); /* reset = 1 */ +</xsl:text> + <xsl:text>}; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>// subscription state, as it should be in hmi server +</xsl:text> + <xsl:text>// hmitree indexed array of integers +</xsl:text> + <xsl:text>var subscriptions = hmitree_types.map(_ignored => 0); +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>// subscription state as needed by widget now +</xsl:text> + <xsl:text>// hmitree indexed array of Sets of widgets objects +</xsl:text> + <xsl:text>var subscribers = hmitree_types.map(_ignored => new Set()); +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>function update_subscriptions() { +</xsl:text> + <xsl:text> let delta = []; +</xsl:text> + <xsl:text> for(let index = 0; index < subscribers.length; index++){ +</xsl:text> + <xsl:text> let widgets = subscribers[index]; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text> // periods are in ms +</xsl:text> + <xsl:text> let previous_period = subscriptions[index]; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text> let new_period; +</xsl:text> + <xsl:text> if(widgets.size > 0) { +</xsl:text> + <xsl:text> let maxfreq = 0; +</xsl:text> + <xsl:text> for(let widget of widgets) +</xsl:text> + <xsl:text> if(maxfreq < widgets.frequency) +</xsl:text> + <xsl:text> maxfreq = widgets.frequency; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text> new_period = 1000/maxfreq; +</xsl:text> + <xsl:text> } else { +</xsl:text> + <xsl:text> new_period = 0; +</xsl:text> + <xsl:text> } +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text> if(previous_period != new_period) { +</xsl:text> + <xsl:text> subscriptions[index] = new_period; +</xsl:text> + <xsl:text> delta.push([ +</xsl:text> + <xsl:text> new Uint8Array([2]), /* subscribe = 2 */ +</xsl:text> + <xsl:text> new Uint32Array([index]), +</xsl:text> + <xsl:text> new Uint16Array([new_period])]); +</xsl:text> + <xsl:text> } +</xsl:text> + <xsl:text> </xsl:text> <xsl:text> } </xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> function update_value(index, value) { -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> }; -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> var current_page = default_page; -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> function switch_page(page_name) { -</xsl:text> - <xsl:text> let new_desc = page_desc[page_name]; -</xsl:text> - <xsl:text> let old_desc = page_desc[page_name]; -</xsl:text> - <xsl:text> /* TODO hide / show widgets */ -</xsl:text> - <xsl:text> /* TODO move viewport */ -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> /* Update subscribers */ -</xsl:text> - <xsl:text> /* Update subscriptions */ -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> -</xsl:text> - <xsl:text> }; + <xsl:text> send_blob(delta); +</xsl:text> + <xsl:text>}; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>function update_value(index, value) { +</xsl:text> + <xsl:text> iectype = hmitree_types[index]; +</xsl:text> + <xsl:text> jstype = typedarray_types[iectypes]; +</xsl:text> + <xsl:text> send_blob([ +</xsl:text> + <xsl:text> new Uint8Array([0]), /* setval = 0 */ +</xsl:text> + <xsl:text> new jstype([value]) +</xsl:text> + <xsl:text> ]); +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>}; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>var current_page; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>function switch_page(page_name) { +</xsl:text> + <xsl:text> let old_desc = page_desc[current_page]; +</xsl:text> + <xsl:text> let new_desc = page_desc[page_name]; +</xsl:text> + <xsl:text> /* TODO hide / show widgets */ +</xsl:text> + <xsl:text> /* TODO move viewport */ +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text> /* remove subsribers of previous page if any */ +</xsl:text> + <xsl:text> if(old_desc) for(let widget of old_desc.widgets){ +</xsl:text> + <xsl:text> for(let index of widget.indexes){ +</xsl:text> + <xsl:text> subscribers[index].delete(widget); +</xsl:text> + <xsl:text> } +</xsl:text> + <xsl:text> } +</xsl:text> + <xsl:text> /* add new subsribers if any */ +</xsl:text> + <xsl:text> if(new_desc) for(let widget of new_desc.widgets){ +</xsl:text> + <xsl:text> for(let index of widget.indexes){ +</xsl:text> + <xsl:text> subscribers[index].add(widget); +</xsl:text> + <xsl:text> } +</xsl:text> + <xsl:text> } +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text> current_page = page_name; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text> update_subscriptions(); +</xsl:text> + <xsl:text>}; +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>// Once connection established +</xsl:text> + <xsl:text>ws.onopen = function (evt) { +</xsl:text> + <xsl:text> send_reset(); +</xsl:text> + <xsl:text> // show main page +</xsl:text> + <xsl:text> switch_page(default_page); +</xsl:text> + <xsl:text> +</xsl:text> + <xsl:text>}; </xsl:text> <xsl:text> </xsl:text> diff -r c5ba1e77f054 -r ddb2c4668a6b svghmi/gen_index_xhtml.ysl2 --- a/svghmi/gen_index_xhtml.ysl2 Fri Oct 11 12:03:14 2019 +0200 +++ b/svghmi/gen_index_xhtml.ysl2 Tue Oct 15 17:14:48 2019 +0200 @@ -1,7 +1,7 @@ include yslt_noindent.yml2 // overrides yslt's output function to set CDATA -decl output(method, cdata-section-elements="script"); +decl output(method, cdata-section-elements="xhtml:script"); istylesheet /* From Inkscape */ @@ -11,6 +11,7 @@ xmlns:svg="http://www.w3.org/2000/svg" xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" + xmlns:xhtml="http://www.w3.org/1999/xhtml" /* Our namespace to invoke python code */ xmlns:ns="beremiz" @@ -163,7 +164,8 @@ function "scripts" { - /* paste hmitree hash stored in hmi tree root node */ + | (function(){ + | | var hmi_hash = [«$hmitree/@hash»]; /* TODO re-enable @@ -194,7 +196,7 @@ foreach "$widget/arg" | "«@value»"`if "position()!=last()" > ,` | ], - | paths: [ + | indexes: [ foreach "$widget/path" { const "hmipath","@value"; const "hmitree_match","$indexed_hmitree/*[@hmipath = $hmipath]"; @@ -207,19 +209,10 @@ } | } | - | var hmi_index = [ - - const "svg","/"; /* foreach loses document root */ + | var hmitree_types = [ + foreach "$indexed_hmitree/*" { - | { /* «@index» «@hmipath» */ - | type: "«local-name()»", - | ids: [ - const "hmipath","@hmipath"; - foreach "$svg//*[substring-after(@inkscape:label,'@') = $hmipath]" { - | hmi_widgets["«@id»"]`if "position()!=last()" > ,` - } - | ] - | }`if "position()!=last()" > ,` + | /* «@index» «@hmipath» */ "«substring(local-name(), 5)»"`if "position()!=last()" > ,` } | ] @@ -238,17 +231,7 @@ | id: "«@id»", | widgets: [ foreach "$page_ids" { - | "«.»"`if "position()!=last()" > ,` - } - | ] - | subscriptions: [ - foreach "$page_elements" { - const "hmipaths", "func:parselabel(@inkscape:label)/widget/path/@value"; - const "notlast", "position()!=last()"; - foreach "$hmipaths" { - const "hmipath","."; - | «$indexed_hmitree/*[@hmipath = $hmipath]/@index»`if "$notlast or position()!=last()" > ,` - } + | hmi_widgets.«.»`if "position()!=last()" > ,` } | ] | }`if "position()!=last()" > ,` @@ -259,6 +242,7 @@ | var default_page = "«$default_page»"; include text svghmi.js + | })(); } /* diff -r c5ba1e77f054 -r ddb2c4668a6b svghmi/svghmi.c --- a/svghmi/svghmi.c Fri Oct 11 12:03:14 2019 +0200 +++ b/svghmi/svghmi.c Tue Oct 15 17:14:48 2019 +0200 @@ -242,8 +242,7 @@ typedef enum { setval = 0, reset = 1, - subscribe = 2, - unsubscribe = 3 + subscribe = 2 } cmd_from_JS; int svghmi_recv_dispatch(uint32_t size, const uint8_t *ptr){ @@ -320,20 +319,6 @@ } break; - case unsubscribe: - { - uint32_t index = *(uint32_t*)(cursor); - - if(index < HMI_ITEM_COUNT) - { - hmi_tree_item_t *dsc = &hmi_tree_item[index]; - reset_iterator(index, dsc); - } - else return -EINVAL; - - progress = sizeof(uint32_t) /* index */; - } - break; } cursor += progress; } diff -r c5ba1e77f054 -r ddb2c4668a6b svghmi/svghmi.js --- a/svghmi/svghmi.js Fri Oct 11 12:03:14 2019 +0200 +++ b/svghmi/svghmi.js Tue Oct 15 17:14:48 2019 +0200 @@ -1,26 +1,153 @@ // svghmi.js -(function(){ - // Open WebSocket to relative "/ws" address - var ws = new WebSocket(window.location.href.replace(/^http(s?:\/\/[^\/]*)\/.*$/, 'ws$1/ws')); +function dispatch_value(index, value) { + console.log("dispatch_value("+index+value+")"); +}; - // Register message reception handler - ws.onmessage = function (evt) { - // TODO : dispatch and cache hmi tree updates +// Open WebSocket to relative "/ws" address +var ws = new WebSocket(window.location.href.replace(/^http(s?:\/\/[^\/]*)\/.*$/, 'ws$1/ws')); +ws.binaryType = 'arraybuffer'; - var received_msg = evt.data; - // TODO : check for hmitree hash header - // if not matching, reload page - alert("Message is received..."+received_msg); +const dvgetters = { + INT: [DataView.prototype.getInt16, 2], + BOOL: [DataView.prototype.getInt8, 1] + /* TODO */ +}; + +// Register message reception handler +ws.onmessage = function (evt) { + + let data = evt.data; + let dv = new DataView(data); + let i = 0; + for(let hash_int of hmi_hash) { + if(hash_int != dv.getUint8(i)){ + console.log("Recv non maching hash. Reload."); + + // 1003 is for "Unsupported Data" + ws.close(1003,"Hash doesn't match"); + + // TODO : remove debug alert ? + alert("HMI will be reloaded."); + + // force reload ignoring cache + location.reload(true); + }; + i++; }; - // Once connection established - ws.onopen = function (evt) { - // TODO : enable the HMI (was previously offline, or just starts) - // show main page + while(i < data.length){ + let index = dv.getUint32(i); + i += 4; + let iectype = hmitree_types[index]; + let [dvgetter, bytesize] = dvgetters[iectypes]; + value = dvgetter.call(dv,i); + dispatch_value(index, value); + i += bytesize; + }; +}; - // TODO : prefix with hmitree hash header - ws.send("test"); +function send_blob(data) { + if(data.length > 0) { + ws.send(new Blob([ + new Uint8Array(hmi_hash), + data])); }; -})(); +}; + +const typedarray_types = { + INT: Int16Array, + BOOL: Uint8Array + /* TODO */ +}; + +function send_reset() { + send_blob(new Uint8Array([1])); /* reset = 1 */ +}; + +// subscription state, as it should be in hmi server +// hmitree indexed array of integers +var subscriptions = hmitree_types.map(_ignored => 0); + +// subscription state as needed by widget now +// hmitree indexed array of Sets of widgets objects +var subscribers = hmitree_types.map(_ignored => new Set()); + +function update_subscriptions() { + let delta = []; + for(let index = 0; index < subscribers.length; index++){ + let widgets = subscribers[index]; + + // periods are in ms + let previous_period = subscriptions[index]; + + let new_period; + if(widgets.size > 0) { + let maxfreq = 0; + for(let widget of widgets) + if(maxfreq < widgets.frequency) + maxfreq = widgets.frequency; + + new_period = 1000/maxfreq; + } else { + new_period = 0; + } + + if(previous_period != new_period) { + subscriptions[index] = new_period; + delta.push([ + new Uint8Array([2]), /* subscribe = 2 */ + new Uint32Array([index]), + new Uint16Array([new_period])]); + } + + } + send_blob(delta); +}; + +function update_value(index, value) { + iectype = hmitree_types[index]; + jstype = typedarray_types[iectypes]; + send_blob([ + new Uint8Array([0]), /* setval = 0 */ + new jstype([value]) + ]); + +}; + +var current_page; + +function switch_page(page_name) { + let old_desc = page_desc[current_page]; + let new_desc = page_desc[page_name]; + /* TODO hide / show widgets */ + /* TODO move viewport */ + + /* remove subsribers of previous page if any */ + if(old_desc) for(let widget of old_desc.widgets){ + for(let index of widget.indexes){ + subscribers[index].delete(widget); + } + } + /* add new subsribers if any */ + if(new_desc) for(let widget of new_desc.widgets){ + for(let index of widget.indexes){ + subscribers[index].add(widget); + } + } + + current_page = page_name; + + update_subscriptions(); +}; + + +// Once connection established +ws.onopen = function (evt) { + send_reset(); + // show main page + switch_page(default_page); + +}; + diff -r c5ba1e77f054 -r ddb2c4668a6b svghmi/svghmi_server.py --- a/svghmi/svghmi_server.py Fri Oct 11 12:03:14 2019 +0200 +++ b/svghmi/svghmi_server.py Tue Oct 15 17:14:48 2019 +0200 @@ -85,7 +85,7 @@ def onMessage(self, msg, isBinary): self._hmi_session.onMessage(msg) - print msg + # print msg #self.sendMessage(msg, binary) class HMIWebSocketServerFactory(WebSocketServerFactory):