From ea671bf39569e2efebbedcf1cd462ed565936083 Mon Sep 17 00:00:00 2001 From: Steve-Mcl Date: Fri, 4 Mar 2022 08:44:21 +0000 Subject: [PATCH 1/7] Use v5 properties to aid auto parsing payload - closes #3421 - fixes bug in `function setBoolProp()` --- .../@node-red/nodes/core/network/10-mqtt.js | 98 +++++++++++++++---- 1 file changed, 78 insertions(+), 20 deletions(-) diff --git a/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js b/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js index bdf2b0e5b..fa4472e54 100644 --- a/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js +++ b/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js @@ -20,7 +20,30 @@ module.exports = function(RED) { var isUtf8 = require('is-utf8'); var HttpsProxyAgent = require('https-proxy-agent'); var url = require('url'); - + const knownMediaTypes = { + "text/css":"string", + "text/html":"string", + "text/plain":"string", + "text/html":"string", + "application/json":"object", + "application/octet-stream":"buffer", + "application/pdf":"buffer", + "application/x-gtar":"buffer", + "application/x-gzip":"buffer", + "application/x-tar":"buffer", + "application/xml":"string", + "application/zip":"buffer", + "audio/aac":"buffer", + "audio/ac3":"buffer", + "audio/basic":"buffer", + "audio/mp4":"buffer", + "audio/ogg":"buffer", + "image/bmp":"buffer", + "image/gif":"buffer", + "image/jpeg":"buffer", + "image/tiff":"buffer", + "image/png":"buffer", + } //#region "Supporting functions" function matchTopic(ts,t) { if (ts == "#") { @@ -103,7 +126,7 @@ module.exports = function(RED) { if(src[propName] === "true" || src[propName] === true) { dst[propName] = true; } else if(src[propName] === "false" || src[propName] === false) { - dst[propName] = true; + dst[propName] = false; } } else { if(def != undefined) dst[propName] = def; @@ -188,24 +211,7 @@ module.exports = function(RED) { */ function subscriptionHandler(node, datatype ,topic, payload, packet) { const v5 = node.brokerConn.options && node.brokerConn.options.protocolVersion == 5; - - if (datatype === "buffer") { - // payload = payload; - } else if (datatype === "base64") { - payload = payload.toString('base64'); - } else if (datatype === "utf8") { - payload = payload.toString('utf8'); - } else if (datatype === "json") { - if (isUtf8(payload)) { - payload = payload.toString(); - try { payload = JSON.parse(payload); } - catch(e) { node.error(RED._("mqtt.errors.invalid-json-parse"),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; } - } - else { node.error((RED._("mqtt.errors.invalid-json-string")),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; } - } else { - if (isUtf8(payload)) { payload = payload.toString(); } - } - var msg = {topic:topic, payload:payload, qos:packet.qos, retain:packet.retain}; + var msg = {topic:topic, payload:null, qos:packet.qos, retain:packet.retain}; if(v5 && packet.properties) { setStrProp(packet.properties, msg, "responseTopic"); setBufferProp(packet.properties, msg, "correlationData"); @@ -215,6 +221,58 @@ module.exports = function(RED) { setStrProp(packet.properties, msg, "reasonString"); setUserProperties(packet.properties.userProperties, msg); } + const v5isUtf8 = v5 ? msg.payloadFormatIndicator === true : null; + const v5HasMediaType = v5 ? !!msg.contentType : null; + const v5MediaTypeLC = v5 ? (msg.contentType + "").toLowerCase() : null; + + if (datatype === "buffer") { + // payload = payload; + } else if (datatype === "base64") { + payload = payload.toString('base64'); + } else if (datatype === "utf8") { + payload = payload.toString('utf8'); + } else if (datatype === "json") { + if (v5isUtf8 || isUtf8(payload)) { + try { + payload = JSON.parse(payload.toString()); + } + catch(e) { + node.error(RED._("mqtt.errors.invalid-json-parse"),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; + } + } + else { node.error((RED._("mqtt.errors.invalid-json-string")),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; } + } else { + //auto + if(v5isUtf8 || v5HasMediaType) { + const outputType = knownMediaTypes[v5MediaTypeLC] + switch (outputType) { + case "string": + payload = payload.toString(); + break; + case "buffer": + //no change + break; + case "object": + try { + payload = JSON.parse(payload.toString()); + } + catch(e) { + node.error(RED._("mqtt.errors.invalid-json-parse"),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; + } + break; + default: + if (v5isUtf8 || isUtf8(payload)) { + payload = payload.toString(); //auto String + } + break; + } + } else if (isUtf8(payload)) { + payload = payload.toString(); //auto String + } //else { + //leave as buffer + //} + } + msg.payload = payload; if ((node.brokerConn.broker === "localhost")||(node.brokerConn.broker === "127.0.0.1")) { msg._topic = topic; } From ce67737cc95439c9de9ce4114eac4db0c47252ec Mon Sep 17 00:00:00 2001 From: Stephen McLaughlin <44235289+Steve-Mcl@users.noreply.github.com> Date: Thu, 14 Apr 2022 16:39:35 +0100 Subject: [PATCH 2/7] auto mode to auto parse string to JS Object --- .../@node-red/nodes/core/network/10-mqtt.js | 13 ++++++++----- .../@node-red/nodes/locales/de/messages.json | 2 +- .../@node-red/nodes/locales/en-US/messages.json | 2 +- .../@node-red/nodes/locales/ja/messages.json | 2 +- .../@node-red/nodes/locales/ko/messages.json | 2 +- .../@node-red/nodes/locales/ru/messages.json | 2 +- .../@node-red/nodes/locales/zh-CN/messages.json | 2 +- .../@node-red/nodes/locales/zh-TW/messages.json | 2 +- 8 files changed, 15 insertions(+), 12 deletions(-) diff --git a/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js b/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js index fa4472e54..29870d63c 100644 --- a/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js +++ b/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js @@ -235,8 +235,7 @@ module.exports = function(RED) { if (v5isUtf8 || isUtf8(payload)) { try { payload = JSON.parse(payload.toString()); - } - catch(e) { + } catch(e) { node.error(RED._("mqtt.errors.invalid-json-parse"),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; } } @@ -255,19 +254,23 @@ module.exports = function(RED) { case "object": try { payload = JSON.parse(payload.toString()); - } - catch(e) { + } catch(e) { node.error(RED._("mqtt.errors.invalid-json-parse"),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; } break; default: - if (v5isUtf8 || isUtf8(payload)) { + if (v5isUtf8 || isUtf8(payload)) { payload = payload.toString(); //auto String } break; } } else if (isUtf8(payload)) { payload = payload.toString(); //auto String + try { + payload = JSON.parse(payload); + } catch(e) { + /* mute error - it simply isnt JSON, just leave payload as a string */ + } } //else { //leave as buffer //} diff --git a/packages/node_modules/@node-red/nodes/locales/de/messages.json b/packages/node_modules/@node-red/nodes/locales/de/messages.json index aa3f1fa49..35ce50f4d 100755 --- a/packages/node_modules/@node-red/nodes/locales/de/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/de/messages.json @@ -422,7 +422,7 @@ "buffer": "Einen binären Buffer", "string": "Ein String", "base64": "Ein Base64-kodierter String", - "auto": "Auto-Erkennung (string oder buffer)", + "auto": "Auto-Erkennung (parsed JSON-Objekt, string oder buffer)", "json": "Ein analysiertes (parsed) JSON-Objekt" }, "true": "wahr", diff --git a/packages/node_modules/@node-red/nodes/locales/en-US/messages.json b/packages/node_modules/@node-red/nodes/locales/en-US/messages.json index ecc5d905c..f6807562f 100755 --- a/packages/node_modules/@node-red/nodes/locales/en-US/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/en-US/messages.json @@ -450,7 +450,7 @@ "buffer": "a Buffer", "string": "a String", "base64": "a Base64 encoded string", - "auto": "auto-detect (string or buffer)", + "auto": "auto-detect (parsed JSON object, string or buffer)", "json": "a parsed JSON object" }, "true": "true", diff --git a/packages/node_modules/@node-red/nodes/locales/ja/messages.json b/packages/node_modules/@node-red/nodes/locales/ja/messages.json index 7784dbe4b..059522f46 100644 --- a/packages/node_modules/@node-red/nodes/locales/ja/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/ja/messages.json @@ -450,7 +450,7 @@ "buffer": "バイナリバッファ", "string": "文字列", "base64": "Base64文字列", - "auto": "自動判定(文字列もしくはバイナリバッファ)", + "auto": "自動判定(JSONオブジェクト、文字列もしくはバイナリバッファ)", "json": "JSONオブジェクト" }, "true": "する", diff --git a/packages/node_modules/@node-red/nodes/locales/ko/messages.json b/packages/node_modules/@node-red/nodes/locales/ko/messages.json index 0566f64ec..b543bba80 100755 --- a/packages/node_modules/@node-red/nodes/locales/ko/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/ko/messages.json @@ -361,7 +361,7 @@ "buffer": "바이너리 버퍼", "string": "문자열", "base64": "Base64문자열", - "auto": "자동판정(문자열혹은 바이너리버퍼)", + "auto": "자동판정(JSON오브젝트, 문자열혹은 바이너리버퍼)", "json": "JSON오브젝트" }, "true": "한다", diff --git a/packages/node_modules/@node-red/nodes/locales/ru/messages.json b/packages/node_modules/@node-red/nodes/locales/ru/messages.json index 72bda67c7..33d750950 100755 --- a/packages/node_modules/@node-red/nodes/locales/ru/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/ru/messages.json @@ -384,7 +384,7 @@ "buffer": "буфер", "string": "строка", "base64": "строка в кодировке Base64", - "auto": "автоопределение (строка или буфер)", + "auto": "автоопределение (разобрать объект JSON, строка или буфер)", "json": "объект JSON" }, "true": "да", diff --git a/packages/node_modules/@node-red/nodes/locales/zh-CN/messages.json b/packages/node_modules/@node-red/nodes/locales/zh-CN/messages.json index ca4de2963..9b972a3bb 100644 --- a/packages/node_modules/@node-red/nodes/locales/zh-CN/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/zh-CN/messages.json @@ -381,7 +381,7 @@ "buffer": "Buffer", "string": "字符串", "base64": "Base64编码字符串", - "auto": "自动检测 (字符串或buffer)", + "auto": "自动检测 (已解析的JSON对象、字符串或buffer)", "json": "解析的JSON对象" }, "true": "是", diff --git a/packages/node_modules/@node-red/nodes/locales/zh-TW/messages.json b/packages/node_modules/@node-red/nodes/locales/zh-TW/messages.json index eef6322a3..03db9d0c4 100644 --- a/packages/node_modules/@node-red/nodes/locales/zh-TW/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/zh-TW/messages.json @@ -385,7 +385,7 @@ "buffer": "Buffer", "string": "字串", "base64": "Base64編碼字串", - "auto": "自動檢測 (字符串或buffer)", + "auto": "自动检测 (已解析的JSON对象、字符串或buffer)", "json": "解析的JSON對象" }, "true": "是", From 8f5d3dc49c8e5d2f06d785c252be26d291c84dcf Mon Sep 17 00:00:00 2001 From: Steve-Mcl Date: Fri, 4 Mar 2022 08:44:21 +0000 Subject: [PATCH 3/7] Use v5 properties to aid auto parsing payload - closes #3421 - fixes bug in `function setBoolProp()` --- .../@node-red/nodes/core/network/10-mqtt.js | 96 +++++++++++++++---- 1 file changed, 77 insertions(+), 19 deletions(-) diff --git a/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js b/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js index 1ea5445ce..3eb9d0516 100644 --- a/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js +++ b/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js @@ -20,7 +20,30 @@ module.exports = function(RED) { var isUtf8 = require('is-utf8'); var HttpsProxyAgent = require('https-proxy-agent'); var url = require('url'); - + const knownMediaTypes = { + "text/css":"string", + "text/html":"string", + "text/plain":"string", + "text/html":"string", + "application/json":"object", + "application/octet-stream":"buffer", + "application/pdf":"buffer", + "application/x-gtar":"buffer", + "application/x-gzip":"buffer", + "application/x-tar":"buffer", + "application/xml":"string", + "application/zip":"buffer", + "audio/aac":"buffer", + "audio/ac3":"buffer", + "audio/basic":"buffer", + "audio/mp4":"buffer", + "audio/ogg":"buffer", + "image/bmp":"buffer", + "image/gif":"buffer", + "image/jpeg":"buffer", + "image/tiff":"buffer", + "image/png":"buffer", + } //#region "Supporting functions" function matchTopic(ts,t) { if (ts == "#") { @@ -188,24 +211,7 @@ module.exports = function(RED) { */ function subscriptionHandler(node, datatype ,topic, payload, packet) { const v5 = node.brokerConn.options && node.brokerConn.options.protocolVersion == 5; - - if (datatype === "buffer") { - // payload = payload; - } else if (datatype === "base64") { - payload = payload.toString('base64'); - } else if (datatype === "utf8") { - payload = payload.toString('utf8'); - } else if (datatype === "json") { - if (isUtf8(payload)) { - payload = payload.toString(); - try { payload = JSON.parse(payload); } - catch(e) { node.error(RED._("mqtt.errors.invalid-json-parse"),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; } - } - else { node.error((RED._("mqtt.errors.invalid-json-string")),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; } - } else { - if (isUtf8(payload)) { payload = payload.toString(); } - } - var msg = {topic:topic, payload:payload, qos:packet.qos, retain:packet.retain}; + var msg = {topic:topic, payload:null, qos:packet.qos, retain:packet.retain}; if(v5 && packet.properties) { setStrProp(packet.properties, msg, "responseTopic"); setBufferProp(packet.properties, msg, "correlationData"); @@ -215,6 +221,58 @@ module.exports = function(RED) { setStrProp(packet.properties, msg, "reasonString"); setUserProperties(packet.properties.userProperties, msg); } + const v5isUtf8 = v5 ? msg.payloadFormatIndicator === true : null; + const v5HasMediaType = v5 ? !!msg.contentType : null; + const v5MediaTypeLC = v5 ? (msg.contentType + "").toLowerCase() : null; + + if (datatype === "buffer") { + // payload = payload; + } else if (datatype === "base64") { + payload = payload.toString('base64'); + } else if (datatype === "utf8") { + payload = payload.toString('utf8'); + } else if (datatype === "json") { + if (v5isUtf8 || isUtf8(payload)) { + try { + payload = JSON.parse(payload.toString()); + } + catch(e) { + node.error(RED._("mqtt.errors.invalid-json-parse"),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; + } + } + else { node.error((RED._("mqtt.errors.invalid-json-string")),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; } + } else { + //auto + if(v5isUtf8 || v5HasMediaType) { + const outputType = knownMediaTypes[v5MediaTypeLC] + switch (outputType) { + case "string": + payload = payload.toString(); + break; + case "buffer": + //no change + break; + case "object": + try { + payload = JSON.parse(payload.toString()); + } + catch(e) { + node.error(RED._("mqtt.errors.invalid-json-parse"),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; + } + break; + default: + if (v5isUtf8 || isUtf8(payload)) { + payload = payload.toString(); //auto String + } + break; + } + } else if (isUtf8(payload)) { + payload = payload.toString(); //auto String + } //else { + //leave as buffer + //} + } + msg.payload = payload; if ((node.brokerConn.broker === "localhost")||(node.brokerConn.broker === "127.0.0.1")) { msg._topic = topic; } From 78f456911afb6e6b3e591949769aabec03ea6b19 Mon Sep 17 00:00:00 2001 From: Stephen McLaughlin <44235289+Steve-Mcl@users.noreply.github.com> Date: Thu, 14 Apr 2022 16:39:35 +0100 Subject: [PATCH 4/7] auto mode to auto parse string to JS Object --- .../@node-red/nodes/core/network/10-mqtt.js | 13 ++++++++----- .../@node-red/nodes/locales/de/messages.json | 2 +- .../@node-red/nodes/locales/en-US/messages.json | 2 +- .../@node-red/nodes/locales/ja/messages.json | 2 +- .../@node-red/nodes/locales/ko/messages.json | 2 +- .../@node-red/nodes/locales/ru/messages.json | 2 +- .../@node-red/nodes/locales/zh-CN/messages.json | 2 +- .../@node-red/nodes/locales/zh-TW/messages.json | 2 +- 8 files changed, 15 insertions(+), 12 deletions(-) diff --git a/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js b/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js index 3eb9d0516..d3016e5c6 100644 --- a/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js +++ b/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js @@ -235,8 +235,7 @@ module.exports = function(RED) { if (v5isUtf8 || isUtf8(payload)) { try { payload = JSON.parse(payload.toString()); - } - catch(e) { + } catch(e) { node.error(RED._("mqtt.errors.invalid-json-parse"),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; } } @@ -255,19 +254,23 @@ module.exports = function(RED) { case "object": try { payload = JSON.parse(payload.toString()); - } - catch(e) { + } catch(e) { node.error(RED._("mqtt.errors.invalid-json-parse"),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; } break; default: - if (v5isUtf8 || isUtf8(payload)) { + if (v5isUtf8 || isUtf8(payload)) { payload = payload.toString(); //auto String } break; } } else if (isUtf8(payload)) { payload = payload.toString(); //auto String + try { + payload = JSON.parse(payload); + } catch(e) { + /* mute error - it simply isnt JSON, just leave payload as a string */ + } } //else { //leave as buffer //} diff --git a/packages/node_modules/@node-red/nodes/locales/de/messages.json b/packages/node_modules/@node-red/nodes/locales/de/messages.json index aa3f1fa49..35ce50f4d 100755 --- a/packages/node_modules/@node-red/nodes/locales/de/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/de/messages.json @@ -422,7 +422,7 @@ "buffer": "Einen binären Buffer", "string": "Ein String", "base64": "Ein Base64-kodierter String", - "auto": "Auto-Erkennung (string oder buffer)", + "auto": "Auto-Erkennung (parsed JSON-Objekt, string oder buffer)", "json": "Ein analysiertes (parsed) JSON-Objekt" }, "true": "wahr", diff --git a/packages/node_modules/@node-red/nodes/locales/en-US/messages.json b/packages/node_modules/@node-red/nodes/locales/en-US/messages.json index ec09e01b1..486764d18 100755 --- a/packages/node_modules/@node-red/nodes/locales/en-US/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/en-US/messages.json @@ -454,7 +454,7 @@ "buffer": "a Buffer", "string": "a String", "base64": "a Base64 encoded string", - "auto": "auto-detect (string or buffer)", + "auto": "auto-detect (parsed JSON object, string or buffer)", "json": "a parsed JSON object" }, "true": "true", diff --git a/packages/node_modules/@node-red/nodes/locales/ja/messages.json b/packages/node_modules/@node-red/nodes/locales/ja/messages.json index 2fdf78f8f..613871771 100644 --- a/packages/node_modules/@node-red/nodes/locales/ja/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/ja/messages.json @@ -454,7 +454,7 @@ "buffer": "バイナリバッファ", "string": "文字列", "base64": "Base64文字列", - "auto": "自動判定(文字列もしくはバイナリバッファ)", + "auto": "自動判定(JSONオブジェクト、文字列もしくはバイナリバッファ)", "json": "JSONオブジェクト" }, "true": "する", diff --git a/packages/node_modules/@node-red/nodes/locales/ko/messages.json b/packages/node_modules/@node-red/nodes/locales/ko/messages.json index 0566f64ec..b543bba80 100755 --- a/packages/node_modules/@node-red/nodes/locales/ko/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/ko/messages.json @@ -361,7 +361,7 @@ "buffer": "바이너리 버퍼", "string": "문자열", "base64": "Base64문자열", - "auto": "자동판정(문자열혹은 바이너리버퍼)", + "auto": "자동판정(JSON오브젝트, 문자열혹은 바이너리버퍼)", "json": "JSON오브젝트" }, "true": "한다", diff --git a/packages/node_modules/@node-red/nodes/locales/ru/messages.json b/packages/node_modules/@node-red/nodes/locales/ru/messages.json index 72bda67c7..33d750950 100755 --- a/packages/node_modules/@node-red/nodes/locales/ru/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/ru/messages.json @@ -384,7 +384,7 @@ "buffer": "буфер", "string": "строка", "base64": "строка в кодировке Base64", - "auto": "автоопределение (строка или буфер)", + "auto": "автоопределение (разобрать объект JSON, строка или буфер)", "json": "объект JSON" }, "true": "да", diff --git a/packages/node_modules/@node-red/nodes/locales/zh-CN/messages.json b/packages/node_modules/@node-red/nodes/locales/zh-CN/messages.json index ca4de2963..9b972a3bb 100644 --- a/packages/node_modules/@node-red/nodes/locales/zh-CN/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/zh-CN/messages.json @@ -381,7 +381,7 @@ "buffer": "Buffer", "string": "字符串", "base64": "Base64编码字符串", - "auto": "自动检测 (字符串或buffer)", + "auto": "自动检测 (已解析的JSON对象、字符串或buffer)", "json": "解析的JSON对象" }, "true": "是", diff --git a/packages/node_modules/@node-red/nodes/locales/zh-TW/messages.json b/packages/node_modules/@node-red/nodes/locales/zh-TW/messages.json index eef6322a3..03db9d0c4 100644 --- a/packages/node_modules/@node-red/nodes/locales/zh-TW/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/zh-TW/messages.json @@ -385,7 +385,7 @@ "buffer": "Buffer", "string": "字串", "base64": "Base64編碼字串", - "auto": "自動檢測 (字符串或buffer)", + "auto": "自动检测 (已解析的JSON对象、字符串或buffer)", "json": "解析的JSON對象" }, "true": "是", From 8ba6a7436e7e403fc51a6246588ef001363fd727 Mon Sep 17 00:00:00 2001 From: Steve-Mcl Date: Fri, 15 Apr 2022 18:21:36 +0100 Subject: [PATCH 5/7] Add tests for MQTT v5 auto parsing hints --- test/nodes/core/network/21-mqtt_spec.js | 168 ++++++++++++++++++++++-- 1 file changed, 157 insertions(+), 11 deletions(-) diff --git a/test/nodes/core/network/21-mqtt_spec.js b/test/nodes/core/network/21-mqtt_spec.js index 56fa1abb3..28321d115 100644 --- a/test/nodes/core/network/21-mqtt_spec.js +++ b/test/nodes/core/network/21-mqtt_spec.js @@ -4,6 +4,7 @@ "use strict"; const should = require("should"); const helper = require("node-red-node-test-helper"); +const { doesNotThrow } = require("should"); const mqttNodes = require("nr-test-utils").require("@node-red/nodes/core/network/10-mqtt.js"); const BROKER_HOST = process.env.MQTT_BROKER_SERVER || "localhost"; const BROKER_PORT = process.env.MQTT_BROKER_PORT || 1883; @@ -92,7 +93,21 @@ describe('MQTT Nodes', function () { options.expectMsg = Object.assign({}, options.sendMsg); testSendRecv({}, { datatype: "auto", topicType: "static" }, {}, options, { done: done }); }); - itConditional('should send JSON and receive string (auto)', function (done) { + //Prior to V3, "auto" mode would only parse to string or buffer. + // itConditional('should send JSON and receive string (auto mode)', function (done) { + // if (skipTests) { return this.skip() } + // this.timeout = 2000; + // const options = {} + // options.sendMsg = { + // topic: nextTopic(), + // payload: '{"prop":"value1", "num":1}', + // qos: 1 + // } + // options.expectMsg = Object.assign({}, options.sendMsg); + // testSendRecv({}, { datatype: "auto", topicType: "static" }, {}, options, { done: done }); + // }) + //In V3, "auto" mode should try to parse JSON, then string and fall back to buffer + itConditional('should send JSON and receive object (auto mode)', function (done) { if (skipTests) { return this.skip() } this.timeout = 2000; const options = {} @@ -102,9 +117,22 @@ describe('MQTT Nodes', function () { qos: 1 } options.expectMsg = Object.assign({}, options.sendMsg); + options.expectMsg.payload = JSON.parse(options.sendMsg.payload); testSendRecv({}, { datatype: "auto", topicType: "static" }, {}, options, { done: done }); }) - itConditional('should send JSON and receive string (utf8)', function (done) { + itConditional('should send invalid JSON and receive string (auto mode)', function (done) { + if (skipTests) { return this.skip() } + this.timeout = 2000; + const options = {} + options.sendMsg = { + topic: nextTopic(), + payload: '{prop:"value3", "num":3}'// send invalid JSON ... + } + options.expectMsg = Object.assign({}, options.sendMsg);//expect same payload + testSendRecv({}, { datatype: "auto", topicType: "static" }, {}, options, { done: done }); + }); + + itConditional('should send JSON and receive string (utf8 mode)', function (done) { if (skipTests) { return this.skip() } this.timeout = 2000; const options = {} @@ -116,7 +144,7 @@ describe('MQTT Nodes', function () { options.expectMsg = Object.assign({}, options.sendMsg); testSendRecv({}, { datatype: "utf8", topicType: "static" }, {}, options, { done: done }); }); - itConditional('should send JSON and receive Object (json)', function (done) { + itConditional('should send JSON and receive Object (json mode)', function (done) { if (skipTests) { return this.skip() } this.timeout = 2000; const options = {} @@ -127,7 +155,31 @@ describe('MQTT Nodes', function () { options.expectMsg = Object.assign({}, options.sendMsg, { payload: { "prop": "value3", "num": 3 } });//expect an object testSendRecv({}, { datatype: "json", topicType: "static" }, {}, options, { done: done }); }); - itConditional('should send String and receive Buffer (buffer)', function (done) { + itConditional('should send invalid JSON and raise error (json mode)', function (done) { + if (skipTests) { return this.skip() } + this.timeout = 2000; + const options = {} + options.sendMsg = { + topic: nextTopic(), + payload: '{prop:"value3", "num":3}', // send invalid JSON ... + } + const hooks = { done: done, beforeLoad: null, afterLoad: null, afterConnect: null } + hooks.afterLoad = (helperNode, mqttBroker, mqttIn, mqttOut) => { + helperNode.on("input", function (msg) { + try { + msg.should.have.a.property("error").type("object"); + msg.error.should.have.a.property("source").type("object"); + msg.error.source.should.have.a.property("id", mqttIn.id); + done(); + } catch (err) { + done(err) + } + }); + return true; //handled + } + testSendRecv({}, { datatype: "json", topicType: "static" }, {}, options, hooks); + }); + itConditional('should send String and receive Buffer (buffer mode)', function (done) { if (skipTests) { return this.skip() } this.timeout = 2000; const options = {} @@ -138,7 +190,7 @@ describe('MQTT Nodes', function () { options.expectMsg = Object.assign({}, options.sendMsg, { payload: Buffer.from(options.sendMsg.payload) });//expect Buffer.from(msg.payload) testSendRecv({}, { datatype: "buffer", topicType: "static" }, {}, options, { done: done }); }); - itConditional('should send utf8 Buffer and receive String (auto)', function (done) { + itConditional('should send utf8 Buffer and receive String (auto mode)', function (done) { if (skipTests) { return this.skip() } this.timeout = 2000; const options = {} @@ -149,7 +201,7 @@ describe('MQTT Nodes', function () { options.expectMsg = Object.assign({}, options.sendMsg, { payload: "x y z" });//set expected payload to "x y z" testSendRecv({}, { datatype: "auto", topicType: "static" }, {}, options, { done: done }); }); - itConditional('should send non utf8 Buffer and receive Buffer (auto)', function (done) { + itConditional('should send non utf8 Buffer and receive Buffer (auto mode)', function (done) { if (skipTests) { return this.skip() } this.timeout = 2000; const options = {} @@ -158,7 +210,7 @@ describe('MQTT Nodes', function () { topic: nextTopic(), payload: Buffer.from([0xC0, 0xC1, 0xF5, 0xF6, 0xF7, 0xF8, 0xF9, 0xFA, 0xFB, 0xFC, 0xFD, 0xFE, 0xFF]) //non valid UTF8 } - options.expectMsg = Object.assign({}, options.sendMsg); + options.expectMsg = Object.assign({}, options.sendMsg, {payload: Buffer.from([0xC0, 0xC1, 0xF5, 0xF6, 0xF7, 0xF8, 0xF9, 0xFA, 0xFB, 0xFC, 0xFD, 0xFE, 0xFF])}); testSendRecv({}, { datatype: "auto", topicType: "static" }, {}, options, hooks); }); itConditional('should send/receive all v5 flags and settings', function (done) { @@ -168,16 +220,16 @@ describe('MQTT Nodes', function () { const options = {} const hooks = { done: done, beforeLoad: null, afterLoad: null, afterConnect: null } options.sendMsg = { - topic: t + "/command", payload: Buffer.from("v5"), qos: 1, retain: true, + topic: t + "/command", payload: Buffer.from('{"version":"v5"}'), qos: 1, retain: true, responseTopic: t + "/response", userProperties: { prop1: "val1" }, - contentType: "application/json", + contentType: "text/plain", correlationData: Buffer.from([1, 2, 3]), payloadFormatIndicator: true, messageExpiryInterval: 2000, } options.expectMsg = Object.assign({}, options.sendMsg); - options.expectMsg.payload = options.expectMsg.payload.toString(); //auto mode + payloadFormatIndicator should make a string + options.expectMsg.payload = options.expectMsg.payload.toString(); //auto mode + payloadFormatIndicator + contentType: "text/plain" should make a string delete options.expectMsg.payloadFormatIndicator; //Seems mqtt.js only publishes payloadFormatIndicator the will msg const inOptions = { datatype: "auto", topicType: "static", @@ -185,6 +237,98 @@ describe('MQTT Nodes', function () { } testSendRecv({ protocolVersion: 5 }, inOptions, {}, options, hooks); }); + itConditional('should send regular string with v5 media type "text/plain" and receive a string (auto mode)', function (done) { + if (skipTests) { return this.skip() } + this.timeout = 2000; + const options = {} + const hooks = { done: done, beforeLoad: null, afterLoad: null, afterConnect: null } + options.sendMsg = { + topic: nextTopic(), payload: "abc", contentType: "text/plain" + } + options.expectMsg = Object.assign({}, options.sendMsg); + testSendRecv({ protocolVersion: 5 }, { datatype: "auto", topicType: "static" }, {}, options, hooks); + }); + itConditional('should send JSON with v5 media type "text/plain" and receive a string (auto mode)', function (done) { + if (skipTests) { return this.skip() } + this.timeout = 2000; + const options = {} + const hooks = { done: done, beforeLoad: null, afterLoad: null, afterConnect: null } + options.sendMsg = { + topic: nextTopic(), payload: '{"prop":"val"}', contentType: "text/plain" + } + options.expectMsg = Object.assign({}, options.sendMsg); + testSendRecv({ protocolVersion: 5 }, { datatype: "auto", topicType: "static" }, {}, options, hooks); + }); + itConditional('should send JSON with v5 media type "application/json" and receive an object (auto mode)', function (done) { + if (skipTests) { return this.skip() } + this.timeout = 2000; + const options = {} + const hooks = { done: done, beforeLoad: null, afterLoad: null, afterConnect: null } + options.sendMsg = { + topic: nextTopic(), payload: '{"prop":"val"}', contentType: "application/json", + } + options.expectMsg = Object.assign({}, options.sendMsg, { payload: JSON.parse(options.sendMsg.payload)}); + testSendRecv({ protocolVersion: 5 }, { datatype: "auto", topicType: "static" }, {}, options, hooks); + }); + itConditional('should send invalid JSON with v5 media type "application/json" and raise an error (auto mode)', function (done) { + if (skipTests) { return this.skip() } + this.timeout = 2000; + const options = {} + options.sendMsg = { + topic: nextTopic(), + payload: '{prop:"value3", "num":3}', contentType: "application/json", // send invalid JSON ... + } + const hooks = { done: done, beforeLoad: null, afterLoad: null, afterConnect: null } + hooks.afterLoad = (helperNode, mqttBroker, mqttIn, mqttOut) => { + helperNode.on("input", function (msg) { + try { + msg.should.have.a.property("error").type("object"); + msg.error.should.have.a.property("source").type("object"); + msg.error.source.should.have.a.property("id", mqttIn.id); + done(); + } catch (err) { + done(err) + } + }); + return true; //handled + } + testSendRecv({ protocolVersion: 5 }, { datatype: "auto", topicType: "static" }, {}, options, hooks); + }); + + itConditional('should send buffer with v5 media type "application/json" and receive an object (auto mode)', function (done) { + if (skipTests) { return this.skip() } + this.timeout = 2000; + const options = {} + const hooks = { done: done, beforeLoad: null, afterLoad: null, afterConnect: null } + options.sendMsg = { + topic: nextTopic(), payload: Buffer.from([0x7b,0x22,0x70,0x72,0x6f,0x70,0x22,0x3a,0x22,0x76,0x61,0x6c,0x22,0x7d]), contentType: "application/json", + } + options.expectMsg = Object.assign({}, options.sendMsg, { payload: {"prop":"val"}}); + testSendRecv({ protocolVersion: 5 }, { datatype: "auto", topicType: "static" }, {}, options, hooks); + }); + itConditional('should send buffer with v5 media type "text/plain" and receive a string (auto mode)', function (done) { + if (skipTests) { return this.skip() } + this.timeout = 2000; + const options = {} + const hooks = { done: done, beforeLoad: null, afterLoad: null, afterConnect: null } + options.sendMsg = { + topic: nextTopic(), payload: Buffer.from([0x7b,0x22,0x70,0x72,0x6f,0x70,0x22,0x3a,0x22,0x76,0x61,0x6c,0x22,0x7d]), contentType: "text/plain", + } + options.expectMsg = Object.assign({}, options.sendMsg, { payload: '{"prop":"val"}'}); + testSendRecv({ protocolVersion: 5 }, { datatype: "auto", topicType: "static" }, {}, options, hooks); + }); + itConditional('should send buffer with v5 media type "application/zip" and receive a buffer (auto mode)', function (done) { + if (skipTests) { return this.skip() } + this.timeout = 2000; + const options = {} + const hooks = { done: done, beforeLoad: null, afterLoad: null, afterConnect: null } + options.sendMsg = { + topic: nextTopic(), payload: Buffer.from([0x7b,0x22,0x70,0x72,0x6f,0x70,0x22,0x3a,0x22,0x76,0x61,0x6c,0x22,0x7d]), contentType: "application/zip", + } + options.expectMsg = Object.assign({}, options.sendMsg, { payload: Buffer.from([0x7b,0x22,0x70,0x72,0x6f,0x70,0x22,0x3a,0x22,0x76,0x61,0x6c,0x22,0x7d])}); + testSendRecv({ protocolVersion: 5 }, { datatype: "auto", topicType: "static" }, {}, options, hooks); + }); + itConditional('should subscribe dynamically via action', function (done) { if (skipTests) { return this.skip() } this.timeout = 2000; @@ -463,14 +607,16 @@ function buildBasicMQTTSendRecvFlow(brokerOptions, inOptions, outOptions) { const inNode = buildMQTTInNode(inOptions.id, inOptions.name, inOptions.broker || broker.id, inOptions.topic, inOptions, ["helper.node"]); const outNode = buildMQTTOutNode(outOptions.id, outOptions.name, outOptions.broker || broker.id, outOptions.topic, outOptions); const helper = buildNode("helper", "helper.node", "helper_node", {}); + const catchNode = buildNode("catch", "catch.node", "catch_node", {"scope": ["mqtt.in"]}, ["helper.node"]); return { nodes: { [broker.name]: broker, [inNode.name]: inNode, [outNode.name]: outNode, [helper.name]: helper, + [catchNode.name]: catchNode, }, - flow: [broker, inNode, outNode, helper] + flow: [broker, inNode, outNode, helper, catchNode] } } From 552408f488fc9825873d8202d6aec1aec8308b8b Mon Sep 17 00:00:00 2001 From: Steve-Mcl Date: Thu, 21 Apr 2022 21:20:41 +0100 Subject: [PATCH 6/7] Make new Auto Output mode optional - adds entry to output data type - adds depreciation tip --- .../@node-red/nodes/core/network/10-mqtt.html | 24 +++++++++- .../@node-red/nodes/core/network/10-mqtt.js | 47 ++++++++++++------- .../@node-red/nodes/locales/de/messages.json | 3 +- .../nodes/locales/en-US/messages.json | 6 ++- .../@node-red/nodes/locales/ja/messages.json | 3 +- .../@node-red/nodes/locales/ko/messages.json | 3 +- .../@node-red/nodes/locales/ru/messages.json | 3 +- .../nodes/locales/zh-CN/messages.json | 3 +- .../nodes/locales/zh-TW/messages.json | 3 +- 9 files changed, 69 insertions(+), 26 deletions(-) diff --git a/packages/node_modules/@node-red/nodes/core/network/10-mqtt.html b/packages/node_modules/@node-red/nodes/core/network/10-mqtt.html index 747281ad0..fd39c9a9c 100644 --- a/packages/node_modules/@node-red/nodes/core/network/10-mqtt.html +++ b/packages/node_modules/@node-red/nodes/core/network/10-mqtt.html @@ -65,6 +65,11 @@ /* opacity: 0.3; pointer-events: none; */ } + .form-row.form-row-mqtt-datatype-tip > .form-tips { + width: calc(70% - 18px); + display: inline-block; + margin-top: -8px; + } @@ -121,6 +126,7 @@
+
+ +
+
@@ -750,7 +760,7 @@ } }, qos: {value: "2"}, - datatype: {value:"auto",required:true}, + datatype: {value:"auto-detect",required:true}, broker: {type:"mqtt-broker", required:true}, // subscriptionIdentifier: {value:0}, nl: {value:false}, @@ -787,6 +797,16 @@ $("div.form-row-mqtt5").toggleClass("form-row-mqtt5-active",!!v5); $("div.form-row.form-row-mqtt-static").toggleClass("form-row-mqtt-static-disabled", !!dynamic) } + + $("#node-input-datatype").on("change", function() { + if($(this).val() === "auto") { + $(".form-row.form-row-mqtt-datatype-tip").show(); + } else { + $(".form-row.form-row-mqtt-datatype-tip").hide(); + } + }) + $("#node-input-datatype").trigger("change"); + $("#node-input-broker").on("change",function(d){ updateVisibility(); }); @@ -807,7 +827,7 @@ $("#node-input-qos").val("2"); } if (this.datatype === undefined) { - $("#node-input-datatype").val("auto"); + $("#node-input-datatype").val("auto-detect"); } }, oneditsave: function() { diff --git a/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js b/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js index d3016e5c6..9b365585b 100644 --- a/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js +++ b/packages/node_modules/@node-red/nodes/core/network/10-mqtt.js @@ -25,7 +25,7 @@ module.exports = function(RED) { "text/html":"string", "text/plain":"string", "text/html":"string", - "application/json":"object", + "application/json":"json", "application/octet-stream":"buffer", "application/pdf":"buffer", "application/x-gtar":"buffer", @@ -234,15 +234,16 @@ module.exports = function(RED) { } else if (datatype === "json") { if (v5isUtf8 || isUtf8(payload)) { try { - payload = JSON.parse(payload.toString()); - } catch(e) { - node.error(RED._("mqtt.errors.invalid-json-parse"),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; + payload = JSON.parse(payload.toString()); + } catch (e) { + node.error(RED._("mqtt.errors.invalid-json-parse"), { payload: payload, topic: topic, qos: packet.qos, retain: packet.retain }); return; } + } else { + node.error((RED._("mqtt.errors.invalid-json-string")), { payload: payload, topic: topic, qos: packet.qos, retain: packet.retain }); return; } - else { node.error((RED._("mqtt.errors.invalid-json-string")),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; } } else { - //auto - if(v5isUtf8 || v5HasMediaType) { + //"auto" (legacy) or "auto-detect" (new default) + if (v5isUtf8 || v5HasMediaType) { const outputType = knownMediaTypes[v5MediaTypeLC] switch (outputType) { case "string": @@ -251,25 +252,39 @@ module.exports = function(RED) { case "buffer": //no change break; - case "object": + case "json": try { - payload = JSON.parse(payload.toString()); - } catch(e) { - node.error(RED._("mqtt.errors.invalid-json-parse"),{payload:payload, topic:topic, qos:packet.qos, retain:packet.retain}); return; + //since v5 type states this should be JSON, parse it & error out if NOT JSON + payload = payload.toString() + const obj = JSON.parse(payload); + if (datatype === "auto-detect") { + payload = obj; //as mode is "auto-detect", return the parsed JSON + } + } catch (e) { + node.error(RED._("mqtt.errors.invalid-json-parse"), { payload: payload, topic: topic, qos: packet.qos, retain: packet.retain }); return; } break; default: if (v5isUtf8 || isUtf8(payload)) { payload = payload.toString(); //auto String + if (datatype === "auto-detect") { + try { + payload = JSON.parse(payload); //auto to parsed object (attempt) + } catch (e) { + /* mute error - it simply isnt JSON, just leave payload as a string */ + } + } } break; } - } else if (isUtf8(payload)) { + } else if (isUtf8(payload)) { payload = payload.toString(); //auto String - try { - payload = JSON.parse(payload); - } catch(e) { - /* mute error - it simply isnt JSON, just leave payload as a string */ + if (datatype === "auto-detect") { + try { + payload = JSON.parse(payload); + } catch (e) { + /* mute error - it simply isnt JSON, just leave payload as a string */ + } } } //else { //leave as buffer diff --git a/packages/node_modules/@node-red/nodes/locales/de/messages.json b/packages/node_modules/@node-red/nodes/locales/de/messages.json index 35ce50f4d..65f251e98 100755 --- a/packages/node_modules/@node-red/nodes/locales/de/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/de/messages.json @@ -422,7 +422,8 @@ "buffer": "Einen binären Buffer", "string": "Ein String", "base64": "Ein Base64-kodierter String", - "auto": "Auto-Erkennung (parsed JSON-Objekt, string oder buffer)", + "auto": "Auto-Erkennung (string oder buffer)", + "auto-detect": "Auto-Erkennung (parsed JSON-Objekt, string oder buffer)", "json": "Ein analysiertes (parsed) JSON-Objekt" }, "true": "wahr", diff --git a/packages/node_modules/@node-red/nodes/locales/en-US/messages.json b/packages/node_modules/@node-red/nodes/locales/en-US/messages.json index 486764d18..d68fff71b 100755 --- a/packages/node_modules/@node-red/nodes/locales/en-US/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/en-US/messages.json @@ -424,7 +424,8 @@ "action": "Action", "staticTopic": "Subscribe to single topic", "dynamicTopic": "Dynamic subscription", - "auto-connect": "Connect automatically" + "auto-connect": "Connect automatically", + "auto-mode-depreciated": "This option is depreciated. Please use the new auto-detect mode." }, "sections-label":{ "birth-message": "Message sent on connection (birth message)", @@ -454,7 +455,8 @@ "buffer": "a Buffer", "string": "a String", "base64": "a Base64 encoded string", - "auto": "auto-detect (parsed JSON object, string or buffer)", + "auto": "auto-detect (string or buffer)", + "auto-detect": "auto-detect (parsed JSON object, string or buffer)", "json": "a parsed JSON object" }, "true": "true", diff --git a/packages/node_modules/@node-red/nodes/locales/ja/messages.json b/packages/node_modules/@node-red/nodes/locales/ja/messages.json index 613871771..f49a973e5 100644 --- a/packages/node_modules/@node-red/nodes/locales/ja/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/ja/messages.json @@ -454,7 +454,8 @@ "buffer": "バイナリバッファ", "string": "文字列", "base64": "Base64文字列", - "auto": "自動判定(JSONオブジェクト、文字列もしくはバイナリバッファ)", + "auto": "自動判定(文字列もしくはバイナリバッファ)", + "auto-detect": "自動判定(JSONオブジェクト、文字列もしくはバイナリバッファ)", "json": "JSONオブジェクト" }, "true": "する", diff --git a/packages/node_modules/@node-red/nodes/locales/ko/messages.json b/packages/node_modules/@node-red/nodes/locales/ko/messages.json index b543bba80..15b7bee8e 100755 --- a/packages/node_modules/@node-red/nodes/locales/ko/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/ko/messages.json @@ -361,7 +361,8 @@ "buffer": "바이너리 버퍼", "string": "문자열", "base64": "Base64문자열", - "auto": "자동판정(JSON오브젝트, 문자열혹은 바이너리버퍼)", + "auto": "자동판정(문자열혹은 바이너리버퍼)", + "auto-detect": "자동판정(JSON오브젝트, 문자열혹은 바이너리버퍼)", "json": "JSON오브젝트" }, "true": "한다", diff --git a/packages/node_modules/@node-red/nodes/locales/ru/messages.json b/packages/node_modules/@node-red/nodes/locales/ru/messages.json index 33d750950..32364b458 100755 --- a/packages/node_modules/@node-red/nodes/locales/ru/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/ru/messages.json @@ -384,7 +384,8 @@ "buffer": "буфер", "string": "строка", "base64": "строка в кодировке Base64", - "auto": "автоопределение (разобрать объект JSON, строка или буфер)", + "auto": "автоопределение (строка или буфер)", + "auto-detect": "автоопределение (разобрать объект JSON, строка или буфер)", "json": "объект JSON" }, "true": "да", diff --git a/packages/node_modules/@node-red/nodes/locales/zh-CN/messages.json b/packages/node_modules/@node-red/nodes/locales/zh-CN/messages.json index 9b972a3bb..b649163b7 100644 --- a/packages/node_modules/@node-red/nodes/locales/zh-CN/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/zh-CN/messages.json @@ -381,7 +381,8 @@ "buffer": "Buffer", "string": "字符串", "base64": "Base64编码字符串", - "auto": "自动检测 (已解析的JSON对象、字符串或buffer)", + "auto": "自动检测 (字符串或buffer)", + "auto-detect": "自动检测 (已解析的JSON对象、字符串或buffer)", "json": "解析的JSON对象" }, "true": "是", diff --git a/packages/node_modules/@node-red/nodes/locales/zh-TW/messages.json b/packages/node_modules/@node-red/nodes/locales/zh-TW/messages.json index 03db9d0c4..d63c47865 100644 --- a/packages/node_modules/@node-red/nodes/locales/zh-TW/messages.json +++ b/packages/node_modules/@node-red/nodes/locales/zh-TW/messages.json @@ -385,7 +385,8 @@ "buffer": "Buffer", "string": "字串", "base64": "Base64編碼字串", - "auto": "自动检测 (已解析的JSON对象、字符串或buffer)", + "auto": "自動檢測 (字符串或buffer)", + "auto-detect": "自动检测 (已解析的JSON对象、字符串或buffer)", "json": "解析的JSON對象" }, "true": "是", From 4b10b9ffc35677c0357756c0b4f6d1b7895f34a4 Mon Sep 17 00:00:00 2001 From: Steve-Mcl Date: Thu, 21 Apr 2022 21:32:24 +0100 Subject: [PATCH 7/7] update tests --- test/nodes/core/network/21-mqtt_spec.js | 58 +++++++++++++++++-------- 1 file changed, 40 insertions(+), 18 deletions(-) diff --git a/test/nodes/core/network/21-mqtt_spec.js b/test/nodes/core/network/21-mqtt_spec.js index 28321d115..8d2ef3ed9 100644 --- a/test/nodes/core/network/21-mqtt_spec.js +++ b/test/nodes/core/network/21-mqtt_spec.js @@ -94,20 +94,20 @@ describe('MQTT Nodes', function () { testSendRecv({}, { datatype: "auto", topicType: "static" }, {}, options, { done: done }); }); //Prior to V3, "auto" mode would only parse to string or buffer. - // itConditional('should send JSON and receive string (auto mode)', function (done) { - // if (skipTests) { return this.skip() } - // this.timeout = 2000; - // const options = {} - // options.sendMsg = { - // topic: nextTopic(), - // payload: '{"prop":"value1", "num":1}', - // qos: 1 - // } - // options.expectMsg = Object.assign({}, options.sendMsg); - // testSendRecv({}, { datatype: "auto", topicType: "static" }, {}, options, { done: done }); - // }) + itConditional('should send JSON and receive string (auto mode)', function (done) { + if (skipTests) { return this.skip() } + this.timeout = 2000; + const options = {} + options.sendMsg = { + topic: nextTopic(), + payload: '{"prop":"value1", "num":1}', + qos: 1 + } + options.expectMsg = Object.assign({}, options.sendMsg); + testSendRecv({}, { datatype: "auto", topicType: "static" }, {}, options, { done: done }); + }) //In V3, "auto" mode should try to parse JSON, then string and fall back to buffer - itConditional('should send JSON and receive object (auto mode)', function (done) { + itConditional('should send JSON and receive object (auto-detect mode)', function (done) { if (skipTests) { return this.skip() } this.timeout = 2000; const options = {} @@ -118,7 +118,7 @@ describe('MQTT Nodes', function () { } options.expectMsg = Object.assign({}, options.sendMsg); options.expectMsg.payload = JSON.parse(options.sendMsg.payload); - testSendRecv({}, { datatype: "auto", topicType: "static" }, {}, options, { done: done }); + testSendRecv({}, { datatype: "auto-detect", topicType: "static" }, {}, options, { done: done }); }) itConditional('should send invalid JSON and receive string (auto mode)', function (done) { if (skipTests) { return this.skip() } @@ -131,6 +131,17 @@ describe('MQTT Nodes', function () { options.expectMsg = Object.assign({}, options.sendMsg);//expect same payload testSendRecv({}, { datatype: "auto", topicType: "static" }, {}, options, { done: done }); }); + itConditional('should send invalid JSON and receive string (auto-detect mode)', function (done) { + if (skipTests) { return this.skip() } + this.timeout = 2000; + const options = {} + options.sendMsg = { + topic: nextTopic(), + payload: '{prop:"value3", "num":3}'// send invalid JSON ... + } + options.expectMsg = Object.assign({}, options.sendMsg);//expect same payload + testSendRecv({}, { datatype: "auto-detect", topicType: "static" }, {}, options, { done: done }); + }); itConditional('should send JSON and receive string (utf8 mode)', function (done) { if (skipTests) { return this.skip() } @@ -259,7 +270,18 @@ describe('MQTT Nodes', function () { options.expectMsg = Object.assign({}, options.sendMsg); testSendRecv({ protocolVersion: 5 }, { datatype: "auto", topicType: "static" }, {}, options, hooks); }); - itConditional('should send JSON with v5 media type "application/json" and receive an object (auto mode)', function (done) { + itConditional('should send JSON with v5 media type "text/plain" and receive a string (auto-detect mode)', function (done) { + if (skipTests) { return this.skip() } + this.timeout = 2000; + const options = {} + const hooks = { done: done, beforeLoad: null, afterLoad: null, afterConnect: null } + options.sendMsg = { + topic: nextTopic(), payload: '{"prop":"val"}', contentType: "text/plain" + } + options.expectMsg = Object.assign({}, options.sendMsg); + testSendRecv({ protocolVersion: 5 }, { datatype: "auto-detect", topicType: "static" }, {}, options, hooks); + }); + itConditional('should send JSON with v5 media type "application/json" and receive an object (auto-detect mode)', function (done) { if (skipTests) { return this.skip() } this.timeout = 2000; const options = {} @@ -268,7 +290,7 @@ describe('MQTT Nodes', function () { topic: nextTopic(), payload: '{"prop":"val"}', contentType: "application/json", } options.expectMsg = Object.assign({}, options.sendMsg, { payload: JSON.parse(options.sendMsg.payload)}); - testSendRecv({ protocolVersion: 5 }, { datatype: "auto", topicType: "static" }, {}, options, hooks); + testSendRecv({ protocolVersion: 5 }, { datatype: "auto-detect", topicType: "static" }, {}, options, hooks); }); itConditional('should send invalid JSON with v5 media type "application/json" and raise an error (auto mode)', function (done) { if (skipTests) { return this.skip() } @@ -295,7 +317,7 @@ describe('MQTT Nodes', function () { testSendRecv({ protocolVersion: 5 }, { datatype: "auto", topicType: "static" }, {}, options, hooks); }); - itConditional('should send buffer with v5 media type "application/json" and receive an object (auto mode)', function (done) { + itConditional('should send buffer with v5 media type "application/json" and receive an object (auto-detect mode)', function (done) { if (skipTests) { return this.skip() } this.timeout = 2000; const options = {} @@ -304,7 +326,7 @@ describe('MQTT Nodes', function () { topic: nextTopic(), payload: Buffer.from([0x7b,0x22,0x70,0x72,0x6f,0x70,0x22,0x3a,0x22,0x76,0x61,0x6c,0x22,0x7d]), contentType: "application/json", } options.expectMsg = Object.assign({}, options.sendMsg, { payload: {"prop":"val"}}); - testSendRecv({ protocolVersion: 5 }, { datatype: "auto", topicType: "static" }, {}, options, hooks); + testSendRecv({ protocolVersion: 5 }, { datatype: "auto-detect", topicType: "static" }, {}, options, hooks); }); itConditional('should send buffer with v5 media type "text/plain" and receive a string (auto mode)', function (done) { if (skipTests) { return this.skip() }