mirror of
				https://github.com/node-red/node-red.git
				synced 2025-03-01 10:36:34 +00:00 
			
		
		
		
	Compare commits
	
		
			26 Commits
		
	
	
		
			3.1.12
			...
			envvar-aut
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 54e6d60fe5 | ||
|  | c2710f4f6f | ||
|  | 20187b51b1 | ||
|  | 4be6d57d98 | ||
|  | a77f8cc3e9 | ||
|  | ea4c0cdbee | ||
|  | 7197153fd5 | ||
|  | b9c1dedab3 | ||
|  | 918943816f | ||
|  | 33cf34f7c7 | ||
|  | 5b5b06cc06 | ||
|  | f49f692ffa | ||
|  | 08c6ea94cb | ||
|  | fea1da5542 | ||
|  | 32e8f4eac6 | ||
|  | bfe5a8a986 | ||
|  | f2cb5ea44e | ||
|  | c7335ed25b | ||
|  | eb940d6d57 | ||
|  | 9091935d77 | ||
|  | 34e8d2b051 | ||
|  | 0c2ab13c48 | ||
|  | 9489953a8f | ||
|  | 54d4079457 | ||
|  | cef3a01042 | ||
|  | 0c042abcab | 
| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|     "name": "node-red", | ||||
|     "version": "3.1.3", | ||||
|     "version": "4.0.0-dev", | ||||
|     "description": "Low-code programming for event-driven applications", | ||||
|     "homepage": "https://nodered.org", | ||||
|     "license": "Apache-2.0", | ||||
|   | ||||
| @@ -33,6 +33,9 @@ module.exports = { | ||||
|             store: req.query['store'], | ||||
|             req: apiUtils.getRequestLogObject(req) | ||||
|         } | ||||
|         if (req.query['keysOnly'] !== undefined) { | ||||
|             opts.keysOnly = true | ||||
|         } | ||||
|         runtimeAPI.context.getValue(opts).then(function(result) { | ||||
|             res.json(result); | ||||
|         }).catch(function(err) { | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|     "name": "@node-red/editor-api", | ||||
|     "version": "3.1.3", | ||||
|     "version": "4.0.0-dev", | ||||
|     "license": "Apache-2.0", | ||||
|     "main": "./lib/index.js", | ||||
|     "repository": { | ||||
| @@ -16,8 +16,8 @@ | ||||
|         } | ||||
|     ], | ||||
|     "dependencies": { | ||||
|         "@node-red/util": "3.1.3", | ||||
|         "@node-red/editor-client": "3.1.3", | ||||
|         "@node-red/util": "4.0.0-dev", | ||||
|         "@node-red/editor-client": "4.0.0-dev", | ||||
|         "bcryptjs": "2.4.3", | ||||
|         "body-parser": "1.20.2", | ||||
|         "clone": "2.1.2", | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|     "name": "@node-red/editor-client", | ||||
|     "version": "3.1.3", | ||||
|     "version": "4.0.0-dev", | ||||
|     "license": "Apache-2.0", | ||||
|     "repository": { | ||||
|         "type": "git", | ||||
|   | ||||
| @@ -46,6 +46,12 @@ | ||||
|             opacity: 0.3 | ||||
|         }).appendTo(container); | ||||
|         this.elementDiv.show(); | ||||
|         if (!this.input.hasClass('red-ui-autoComplete')) { | ||||
|             this.input.autoComplete({ | ||||
|                 search: contextAutoComplete({ input: that }), | ||||
|                 minLength: 0 | ||||
|             }) | ||||
|         } | ||||
|     } | ||||
|     var mapDeprecatedIcon = function(icon) { | ||||
|         if (/^red\/images\/typedInput\/.+\.png$/.test(icon)) { | ||||
| @@ -54,25 +60,26 @@ | ||||
|         return icon; | ||||
|     } | ||||
|  | ||||
|     var autoComplete = function(options) { | ||||
|         function getMatch(value, searchValue) { | ||||
|             const idx = value.toLowerCase().indexOf(searchValue.toLowerCase()); | ||||
|             const len = idx > -1 ? searchValue.length : 0; | ||||
|             return { | ||||
|                 index: idx, | ||||
|                 found: idx > -1, | ||||
|                 pre: value.substring(0,idx), | ||||
|                 match: value.substring(idx,idx+len), | ||||
|                 post: value.substring(idx+len), | ||||
|             } | ||||
|         } | ||||
|         function generateSpans(match) { | ||||
|             const els = []; | ||||
|             if(match.pre) { els.push($('<span/>').text(match.pre)); } | ||||
|             if(match.match) { els.push($('<span/>',{style:"font-weight: bold; color: var(--red-ui-text-color-link);"}).text(match.match)); } | ||||
|             if(match.post) { els.push($('<span/>').text(match.post)); } | ||||
|             return els; | ||||
|     function getMatch(value, searchValue) { | ||||
|         const idx = value.toLowerCase().indexOf(searchValue.toLowerCase()); | ||||
|         const len = idx > -1 ? searchValue.length : 0; | ||||
|         return { | ||||
|             index: idx, | ||||
|             found: idx > -1, | ||||
|             pre: value.substring(0,idx), | ||||
|             match: value.substring(idx,idx+len), | ||||
|             post: value.substring(idx+len), | ||||
|         } | ||||
|     } | ||||
|     function generateSpans(match) { | ||||
|         const els = []; | ||||
|         if(match.pre) { els.push($('<span/>').text(match.pre)); } | ||||
|         if(match.match) { els.push($('<span/>',{style:"font-weight: bold; color: var(--red-ui-text-color-link);"}).text(match.match)); } | ||||
|         if(match.post) { els.push($('<span/>').text(match.post)); } | ||||
|         return els; | ||||
|     } | ||||
|      | ||||
|     const msgAutoComplete = function(options) { | ||||
|         return function(val) { | ||||
|             var matches = []; | ||||
|             options.forEach(opt => { | ||||
| @@ -102,6 +109,196 @@ | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     function getEnvVars (obj, envVars = {}) { | ||||
|         contextKnownKeys.env = contextKnownKeys.env || {} | ||||
|         if (contextKnownKeys.env[obj.id]) { | ||||
|             return contextKnownKeys.env[obj.id] | ||||
|         } | ||||
|         let parent | ||||
|         if (obj.type === 'tab' || obj.type === 'subflow') { | ||||
|             RED.nodes.eachConfig(function (conf) { | ||||
|                 if (conf.type === "global-config") { | ||||
|                     parent = conf; | ||||
|                 } | ||||
|             }) | ||||
|         } else if (obj.g) { | ||||
|             parent = RED.nodes.group(obj.g) | ||||
|         } else if (obj.z) { | ||||
|             parent = RED.nodes.workspace(obj.z) || RED.nodes.subflow(obj.z) | ||||
|         } | ||||
|         if (parent) { | ||||
|             getEnvVars(parent, envVars) | ||||
|         } | ||||
|         if (obj.env) { | ||||
|             obj.env.forEach(env => { | ||||
|                 envVars[env.name] = obj | ||||
|             }) | ||||
|         } | ||||
|         contextKnownKeys.env[obj.id] = envVars | ||||
|         return envVars | ||||
|     } | ||||
|  | ||||
|     const envAutoComplete = function (val) { | ||||
|         const editStack = RED.editor.getEditStack() | ||||
|         if (editStack.length === 0) { | ||||
|             done([]) | ||||
|             return | ||||
|         } | ||||
|         const editingNode = editStack.pop() | ||||
|         if (!editingNode) { | ||||
|             return [] | ||||
|         } | ||||
|         const envVarsMap = getEnvVars(editingNode) | ||||
|         const envVars = Object.keys(envVarsMap) | ||||
|         const matches = [] | ||||
|         const i = val.lastIndexOf('${') | ||||
|         let searchKey = val | ||||
|         let isSubkey = false | ||||
|         if (i > -1) { | ||||
|             if (val.lastIndexOf('}') < i) { | ||||
|                 searchKey = val.substring(i+2) | ||||
|                 isSubkey = true | ||||
|             } | ||||
|         } | ||||
|         envVars.forEach(v => { | ||||
|             let valMatch = getMatch(v, searchKey); | ||||
|             if (valMatch.found) { | ||||
|                 const optSrc = envVarsMap[v] | ||||
|                 const element = $('<div>',{style: "display: flex"}); | ||||
|                 const valEl = $('<div/>',{style:"font-family: var(--red-ui-monospace-font); white-space:nowrap; overflow: hidden; flex-grow:1"}); | ||||
|                 valEl.append(generateSpans(valMatch)) | ||||
|                 valEl.appendTo(element) | ||||
|  | ||||
|                 if (optSrc) { | ||||
|                     const optEl = $('<div>').css({ "font-size": "0.8em" }); | ||||
|                     let label | ||||
|                     if (optSrc.type === 'global-config') { | ||||
|                         label = RED._('sidebar.context.global') | ||||
|                     } else if (optSrc.type === 'group') { | ||||
|                         label = RED.utils.getNodeLabel(optSrc) || (RED._('sidebar.info.group') + ': '+optSrc.id) | ||||
|                     } else { | ||||
|                         label = RED.utils.getNodeLabel(optSrc) || optSrc.id | ||||
|                     } | ||||
|  | ||||
|                     optEl.append(generateSpans({ match: label })); | ||||
|                     optEl.appendTo(element); | ||||
|                 } | ||||
|                 matches.push({ | ||||
|                     value: isSubkey ? val + v + '}' : v, | ||||
|                     label: element, | ||||
|                     i: valMatch.index | ||||
|                 }); | ||||
|             } | ||||
|         }) | ||||
|         matches.sort(function(A,B){return A.i-B.i}) | ||||
|         return matches | ||||
|     } | ||||
|  | ||||
|     let contextKnownKeys = {} | ||||
|     let contextCache = {} | ||||
|     if (RED.events) { | ||||
|         RED.events.on("editor:close", function () { | ||||
|             contextCache = {} | ||||
|             contextKnownKeys = {} | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     const contextAutoComplete = function(options) { | ||||
|         const getContextKeysFromRuntime = function(scope, store, searchKey, done) { | ||||
|             contextKnownKeys[scope] = contextKnownKeys[scope] || {} | ||||
|             contextKnownKeys[scope][store] = contextKnownKeys[scope][store] || new Set() | ||||
|             if (searchKey.length > 0) { | ||||
|                 try { | ||||
|                     RED.utils.normalisePropertyExpression(searchKey) | ||||
|                 } catch (err) { | ||||
|                     // Not a valid context key, so don't try looking up | ||||
|                     done() | ||||
|                     return | ||||
|                 } | ||||
|             } | ||||
|             const url = `context/${scope}/${encodeURIComponent(searchKey)}?store=${store}&keysOnly` | ||||
|             if (contextCache[url]) { | ||||
|                 // console.log('CACHED', url) | ||||
|                 done() | ||||
|             } else { | ||||
|                 // console.log('GET', url) | ||||
|                 $.getJSON(url, function(data) { | ||||
|                     // console.log(data) | ||||
|                     contextCache[url] = true | ||||
|                     const result = data[store] || {} | ||||
|                     const keys = result.keys || [] | ||||
|                     const keyPrefix = searchKey + (searchKey.length > 0 ? '.' : '') | ||||
|                     keys.forEach(key => { | ||||
|                         if (/^[a-zA-Z_$][0-9a-zA-Z_$]*$/.test(key)) { | ||||
|                             contextKnownKeys[scope][store].add(keyPrefix + key) | ||||
|                         } else { | ||||
|                             contextKnownKeys[scope][store].add(searchKey + "[\""+key.replace(/"/,"\\\"")+"\"]") | ||||
|                         }                         | ||||
|                     }) | ||||
|                     done() | ||||
|                 }) | ||||
|             } | ||||
|         } | ||||
|         const getContextKeys = function(key, done) { | ||||
|             const keyParts = key.split('.') | ||||
|             const partialKey = keyParts.pop() | ||||
|             let scope = options.input.propertyType | ||||
|             if (scope === 'flow') { | ||||
|                 // Get the flow id of the node we're editing | ||||
|                 const editStack = RED.editor.getEditStack() | ||||
|                 if (editStack.length === 0) { | ||||
|                     done([]) | ||||
|                     return | ||||
|                 } | ||||
|                 const editingNode = editStack.pop() | ||||
|                 if (editingNode.z) { | ||||
|                     scope = `${scope}/${editingNode.z}` | ||||
|                 } else { | ||||
|                     done([]) | ||||
|                     return | ||||
|                 } | ||||
|             } | ||||
|             const store = options.input.optionValue | ||||
|             const searchKey = keyParts.join('.') | ||||
|             | ||||
|             getContextKeysFromRuntime(scope, store, searchKey, function() { | ||||
|                 if (contextKnownKeys[scope][store].has(key) || key.endsWith(']')) { | ||||
|                     getContextKeysFromRuntime(scope, store, key, function() { | ||||
|                         done(contextKnownKeys[scope][store]) | ||||
|                     }) | ||||
|                 } | ||||
|                 done(contextKnownKeys[scope][store]) | ||||
|             }) | ||||
|         } | ||||
|  | ||||
|         return function(val, done) { | ||||
|             getContextKeys(val, function (keys) { | ||||
|                 const matches = [] | ||||
|                 keys.forEach(v => { | ||||
|                     let optVal = v | ||||
|                     let valMatch = getMatch(optVal, val); | ||||
|                     if (!valMatch.found && val.length > 0 && val.endsWith('.')) { | ||||
|                         // Search key ends in '.' - but doesn't match. Check again | ||||
|                         // with [" at the end instead so we match bracket notation | ||||
|                         valMatch = getMatch(optVal, val.substring(0, val.length - 1) + '["') | ||||
|                     } | ||||
|                     if (valMatch.found) { | ||||
|                         const element = $('<div>',{style: "display: flex"}); | ||||
|                         const valEl = $('<div/>',{style:"font-family: var(--red-ui-monospace-font); white-space:nowrap; overflow: hidden; flex-grow:1"}); | ||||
|                         valEl.append(generateSpans(valMatch)) | ||||
|                         valEl.appendTo(element) | ||||
|                         matches.push({ | ||||
|                             value: optVal, | ||||
|                             label: element, | ||||
|                         }); | ||||
|                     } | ||||
|                 }) | ||||
|                 matches.sort(function(a, b) { return a.value.localeCompare(b.value) }); | ||||
|                 done(matches); | ||||
|             }) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // This is a hand-generated list of completions for the core nodes (based on the node help html). | ||||
|     var msgCompletions = [ | ||||
|         { value: "payload" }, | ||||
| @@ -166,7 +363,7 @@ | ||||
|         { value: "_session", source: ["websocket out","tcp out"] }, | ||||
|     ] | ||||
|     var allOptions = { | ||||
|         msg: {value:"msg",label:"msg.",validate:RED.utils.validatePropertyExpression, autoComplete: autoComplete(msgCompletions)}, | ||||
|         msg: {value:"msg",label:"msg.",validate:RED.utils.validatePropertyExpression, autoComplete: msgAutoComplete(msgCompletions)}, | ||||
|         flow: {value:"flow",label:"flow.",hasValue:true, | ||||
|             options:[], | ||||
|             validate:RED.utils.validatePropertyExpression, | ||||
| @@ -251,7 +448,8 @@ | ||||
|         env: { | ||||
|             value: "env", | ||||
|             label: "env variable", | ||||
|             icon: "red/images/typedInput/env.svg" | ||||
|             icon: "red/images/typedInput/env.svg", | ||||
|             autoComplete: envAutoComplete | ||||
|         }, | ||||
|         node: { | ||||
|             value: "node", | ||||
| @@ -544,7 +742,7 @@ | ||||
|                 that.element.trigger('paste',evt); | ||||
|             }); | ||||
|             this.input.on('keydown', function(evt) { | ||||
|                 if (that.typeMap[that.propertyType].autoComplete) { | ||||
|                 if (that.typeMap[that.propertyType].autoComplete || that.input.hasClass('red-ui-autoComplete')) { | ||||
|                     return | ||||
|                 } | ||||
|                 if (evt.keyCode >= 37 && evt.keyCode <= 40) { | ||||
| @@ -967,6 +1165,9 @@ | ||||
|                     // If previousType is !null, then this is a change of the type, rather than the initialisation | ||||
|                     var previousType = this.typeMap[this.propertyType]; | ||||
|                     previousValue = this.input.val(); | ||||
|                     if (this.input.hasClass('red-ui-autoComplete')) { | ||||
|                         this.input.autoComplete("destroy"); | ||||
|                     } | ||||
|  | ||||
|                     if (previousType && this.typeChanged) { | ||||
|                         if (this.options.debug) { console.log(this.identifier,"typeChanged",{previousType,previousValue}) } | ||||
| @@ -1013,7 +1214,9 @@ | ||||
|                             this.input.val(this.oldValues.hasOwnProperty("_")?this.oldValues["_"]:(opt.default||"")) | ||||
|                         } | ||||
|                         if (previousType.autoComplete) { | ||||
|                             this.input.autoComplete("destroy"); | ||||
|                             if (this.input.hasClass('red-ui-autoComplete')) { | ||||
|                                 this.input.autoComplete("destroy"); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                     this.propertyType = type; | ||||
|   | ||||
| @@ -2082,6 +2082,7 @@ RED.editor = (function() { | ||||
|             } | ||||
|         }, | ||||
|         editBuffer: function(options) { showTypeEditor("_buffer", options) }, | ||||
|         getEditStack: function () { return [...editStack] }, | ||||
|         buildEditForm: buildEditForm, | ||||
|         validateNode: validateNode, | ||||
|         updateNodeProperties: updateNodeProperties, | ||||
|   | ||||
| @@ -5,6 +5,7 @@ module.exports = function(RED) { | ||||
|     const fs = require("fs-extra"); | ||||
|     const path = require("path"); | ||||
|     var debuglength = RED.settings.debugMaxLength || 1000; | ||||
|     var statuslength = RED.settings.debugStatusLength || 32; | ||||
|     var useColors = RED.settings.debugUseColors || false; | ||||
|     util.inspect.styles.boolean = "red"; | ||||
|     const { hasOwnProperty } = Object.prototype; | ||||
| @@ -164,7 +165,7 @@ module.exports = function(RED) { | ||||
|                             } | ||||
|                         } | ||||
|  | ||||
|                         if (st.length > 32) { st = st.substr(0,32) + "..."; } | ||||
|                         if (st.length > statuslength) { st = st.substr(0,statuslength) + "..."; } | ||||
|  | ||||
|                         var newStatus = {fill:fill, shape:shape, text:st}; | ||||
|                         if (JSON.stringify(newStatus) !== node.oldState) { // only send if we have to | ||||
|   | ||||
| @@ -411,23 +411,33 @@ module.exports = function(RED) { | ||||
|                 if (msg._session && msg._session.type == "tcp") { | ||||
|                     var client = connectionPool[msg._session.id]; | ||||
|                     if (client) { | ||||
|                         if (Buffer.isBuffer(msg.payload)) { | ||||
|                             client.write(msg.payload); | ||||
|                         } else if (typeof msg.payload === "string" && node.base64) { | ||||
|                             client.write(Buffer.from(msg.payload,'base64')); | ||||
|                         } else { | ||||
|                             client.write(Buffer.from(""+msg.payload)); | ||||
|                         if (msg?.reset === true) { | ||||
|                             client.destroy(); | ||||
|                         } | ||||
|                         else { | ||||
|                             if (Buffer.isBuffer(msg.payload)) { | ||||
|                                 client.write(msg.payload); | ||||
|                             } else if (typeof msg.payload === "string" && node.base64) { | ||||
|                                 client.write(Buffer.from(msg.payload,'base64')); | ||||
|                             } else { | ||||
|                                 client.write(Buffer.from(""+msg.payload)); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|                 else { | ||||
|                     for (var i in connectionPool) { | ||||
|                         if (Buffer.isBuffer(msg.payload)) { | ||||
|                             connectionPool[i].write(msg.payload); | ||||
|                         } else if (typeof msg.payload === "string" && node.base64) { | ||||
|                             connectionPool[i].write(Buffer.from(msg.payload,'base64')); | ||||
|                         } else { | ||||
|                             connectionPool[i].write(Buffer.from(""+msg.payload)); | ||||
|                         if (msg?.reset === true) { | ||||
|                             connectionPool[i].destroy(); | ||||
|                         } | ||||
|                         else { | ||||
|                             if (Buffer.isBuffer(msg.payload)) { | ||||
|                                 connectionPool[i].write(msg.payload); | ||||
|                             } else if (typeof msg.payload === "string" && node.base64) { | ||||
|                                 connectionPool[i].write(Buffer.from(msg.payload,'base64')); | ||||
|                             } else { | ||||
|                                 connectionPool[i].write(Buffer.from(""+msg.payload)); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
| @@ -547,13 +557,33 @@ module.exports = function(RED) { | ||||
|  | ||||
|         this.on("input", function(msg, nodeSend, nodeDone) { | ||||
|             var i = 0; | ||||
|             if ((!Buffer.isBuffer(msg.payload)) && (typeof msg.payload !== "string")) { | ||||
|             if (msg.payload !== undefined && (!Buffer.isBuffer(msg.payload)) && (typeof msg.payload !== "string")) { | ||||
|                 msg.payload = msg.payload.toString(); | ||||
|             } | ||||
|  | ||||
|             var host = node.server || msg.host; | ||||
|             var port = node.port || msg.port; | ||||
|  | ||||
|             if (node.out === "sit" && msg?.reset) { | ||||
|                 if (msg.reset === true) { // kill all connections | ||||
|                     for (var cl in clients) { | ||||
|                         if (clients[cl].hasOwnProperty("client")) { | ||||
|                             clients[cl].client.destroy(); | ||||
|                             delete clients[cl]; | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|                 if (typeof(msg.reset) === "string" && msg.reset.includes(":")) {  // just kill connection host:port | ||||
|                     if (clients.hasOwnProperty(msg.reset) && clients[msg.reset].hasOwnProperty("client")) { | ||||
|                         clients[msg.reset].client.destroy(); | ||||
|                         delete clients[msg.reset]; | ||||
|                     } | ||||
|                 } | ||||
|                 const cc = Object.keys(clients).length; | ||||
|                 node.status({fill:"green",shape:cc===0?"ring":"dot",text:RED._("tcpin.status.connections",{count:cc})}); | ||||
|                 if ((host === undefined || port === undefined) && !msg.hasOwnProperty("payload")) { return; } | ||||
|             } | ||||
|  | ||||
|             // Store client information independently | ||||
|             // the clients object will have: | ||||
|             // clients[id].client, clients[id].msg, clients[id].timeout | ||||
| @@ -621,13 +651,16 @@ module.exports = function(RED) { | ||||
|                     clients[connection_id].connecting = true; | ||||
|                     clients[connection_id].client.connect(connOpts, function() { | ||||
|                         //node.log(RED._("tcpin.errors.client-connected")); | ||||
|                         node.status({fill:"green",shape:"dot",text:"common.status.connected"}); | ||||
|                         // node.status({fill:"green",shape:"dot",text:"common.status.connected"}); | ||||
|                         node.status({fill:"green",shape:"dot",text:RED._("tcpin.status.connections",{count:Object.keys(clients).length})}); | ||||
|                         if (clients[connection_id] && clients[connection_id].client) { | ||||
|                             clients[connection_id].connected = true; | ||||
|                             clients[connection_id].connecting = false; | ||||
|                             let event; | ||||
|                             while (event = dequeue(clients[connection_id].msgQueue)) { | ||||
|                                 clients[connection_id].client.write(event.msg.payload); | ||||
|                                 if (event.msg.payload !== undefined) { | ||||
|                                     clients[connection_id].client.write(event.msg.payload); | ||||
|                                 } | ||||
|                                 event.nodeDone(); | ||||
|                             } | ||||
|                             if (node.out === "time" && node.splitc < 0) { | ||||
| @@ -823,7 +856,9 @@ module.exports = function(RED) { | ||||
|             else if (!clients[connection_id].connecting && clients[connection_id].connected) { | ||||
|                 if (clients[connection_id] && clients[connection_id].client) { | ||||
|                     let event = dequeue(clients[connection_id].msgQueue) | ||||
|                     clients[connection_id].client.write(event.msg.payload); | ||||
|                     if (event.msg.payload !== undefined ) { | ||||
|                         clients[connection_id].client.write(event.msg.payload); | ||||
|                     } | ||||
|                     event.nodeDone(); | ||||
|                 } | ||||
|             } | ||||
|   | ||||
| @@ -30,6 +30,8 @@ | ||||
|     before being sent.</p> | ||||
|     <p>If <code>msg._session</code> is not present the payload is | ||||
|     sent to <b>all</b> connected clients.</p> | ||||
|     <p>In Reply-to mode, setting <code>msg.reset = true</code> will reset the connection | ||||
|         specified by _session.id, or all connections if no _session.id is specified.</p> | ||||
|     <p><b>Note: </b>On some systems you may need root or administrator access | ||||
|     to access ports below 1024.</p> | ||||
| </script> | ||||
| @@ -40,6 +42,8 @@ | ||||
|     returned characters into a fixed buffer, match a specified character before returning, | ||||
|     wait a fixed timeout from first reply and then return, sit and wait for data, or send then close the connection | ||||
|     immediately, without waiting for a reply.</p> | ||||
|     <p>If in sit and wait mode (remain connected) you can send <code>msg.reset = true</code> or <code>msg.reset = "host:port"</code> to force a break in | ||||
|     the connection and an automatic reconnection.</p> | ||||
|     <p>The response will be output in <code>msg.payload</code> as a buffer, so you may want to .toString() it.</p> | ||||
|     <p>If you leave tcp host or port blank they must be set by using the <code>msg.host</code> and <code>msg.port</code> properties in every message sent to the node.</p> | ||||
| </script> | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|     "name": "@node-red/nodes", | ||||
|     "version": "3.1.3", | ||||
|     "version": "4.0.0-dev", | ||||
|     "license": "Apache-2.0", | ||||
|     "repository": { | ||||
|         "type": "git", | ||||
|   | ||||
| @@ -264,7 +264,7 @@ async function installModule(moduleDetails) { | ||||
|             "module": moduleDetails.module, | ||||
|             "version": moduleDetails.version, | ||||
|             "dir": installDir, | ||||
|             "args": ["--production","--engine-strict"] | ||||
|             "args": ["--omit=dev","--engine-strict"] | ||||
|         } | ||||
|         return hooks.trigger("preInstall", triggerPayload).then((result) => { | ||||
|             // preInstall passed | ||||
|   | ||||
| @@ -215,7 +215,7 @@ async function installModule(module,version,url) { | ||||
|             "dir": installDir, | ||||
|             "isExisting": isExisting, | ||||
|             "isUpgrade": isUpgrade, | ||||
|             "args": ['--no-audit','--no-update-notifier','--no-fund','--save','--save-prefix=~','--production','--engine-strict'] | ||||
|             "args": ['--no-audit','--no-update-notifier','--no-fund','--save','--save-prefix=~','--omit=dev','--engine-strict'] | ||||
|         } | ||||
|  | ||||
|         return hooks.trigger("preInstall", triggerPayload).then((result) => { | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|     "name": "@node-red/registry", | ||||
|     "version": "3.1.3", | ||||
|     "version": "4.0.0-dev", | ||||
|     "license": "Apache-2.0", | ||||
|     "main": "./lib/index.js", | ||||
|     "repository": { | ||||
| @@ -16,7 +16,7 @@ | ||||
|         } | ||||
|     ], | ||||
|     "dependencies": { | ||||
|         "@node-red/util": "3.1.3", | ||||
|         "@node-red/util": "4.0.0-dev", | ||||
|         "clone": "2.1.2", | ||||
|         "fs-extra": "11.1.1", | ||||
|         "semver": "7.5.4", | ||||
|   | ||||
| @@ -68,6 +68,7 @@ var api = module.exports = { | ||||
|     * @param {String} opts.store - the context store | ||||
|     * @param {String} opts.key - the context key | ||||
|     * @param {Object} opts.req - the request to log (optional) | ||||
|     * @param {Boolean} opts.keysOnly - whether to return keys only | ||||
|     * @return {Promise} - the node information | ||||
|     * @memberof @node-red/runtime_context | ||||
|     */ | ||||
| @@ -102,6 +103,15 @@ var api = module.exports = { | ||||
|                 if (key) { | ||||
|                     store = store || availableStores.default; | ||||
|                     ctx.get(key,store,function(err, v) { | ||||
|                         if (opts.keysOnly) { | ||||
|                             if (Array.isArray(v)) { | ||||
|                                 resolve({ [store]: { format: `array[${v.length}]`}}) | ||||
|                             } else if (typeof v === 'object') { | ||||
|                                 resolve({ [store]: { keys: Object.keys(v), format: 'Object' } }) | ||||
|                             } else { | ||||
|                                 resolve({ [store]: { keys: [] }}) | ||||
|                             } | ||||
|                         } | ||||
|                         var encoded = util.encodeObject({msg:v}); | ||||
|                         if (store !== availableStores.default) { | ||||
|                             encoded.store = store; | ||||
| @@ -118,32 +128,58 @@ var api = module.exports = { | ||||
|                         stores = [store]; | ||||
|                     } | ||||
|  | ||||
|  | ||||
|                     var result = {}; | ||||
|                     var c = stores.length; | ||||
|                     var errorReported = false; | ||||
|                     stores.forEach(function(store) { | ||||
|                         exportContextStore(scope,ctx,store,result,function(err) { | ||||
|                             if (err) { | ||||
|                                 // TODO: proper error reporting | ||||
|                                 if (!errorReported) { | ||||
|                                     errorReported = true; | ||||
|                                     runtime.log.audit({event: "context.get",scope:scope,id:id,store:store,key:key,error:"unexpected_error"}, opts.req); | ||||
|                                     var err = new Error(); | ||||
|                                     err.code = "unexpected_error"; | ||||
|                                     err.status = 400; | ||||
|                                     return reject(err); | ||||
|                         if (opts.keysOnly) { | ||||
|                             ctx.keys(store,function(err, keys) { | ||||
|                                 if (err) { | ||||
|                                     // TODO: proper error reporting | ||||
|                                     if (!errorReported) { | ||||
|                                         errorReported = true; | ||||
|                                         runtime.log.audit({event: "context.get",scope:scope,id:id,store:store,key:key,error:"unexpected_error"}, opts.req); | ||||
|                                         var err = new Error(); | ||||
|                                         err.code = "unexpected_error"; | ||||
|                                         err.status = 400; | ||||
|                                         return reject(err); | ||||
|                                     } | ||||
|                                     return | ||||
|                                 } | ||||
|                                 result[store] = { keys } | ||||
|                                 c--; | ||||
|                                 if (c === 0) { | ||||
|                                     if (!errorReported) { | ||||
|                                         runtime.log.audit({event: "context.get",scope:scope,id:id,store:store,key:key},opts.req); | ||||
|                                         resolve(result); | ||||
|                                     } | ||||
|                                 } | ||||
|                             }) | ||||
|                         } else { | ||||
|                             exportContextStore(scope,ctx,store,result,function(err) { | ||||
|                                 if (err) { | ||||
|                                     // TODO: proper error reporting | ||||
|                                     if (!errorReported) { | ||||
|                                         errorReported = true; | ||||
|                                         runtime.log.audit({event: "context.get",scope:scope,id:id,store:store,key:key,error:"unexpected_error"}, opts.req); | ||||
|                                         var err = new Error(); | ||||
|                                         err.code = "unexpected_error"; | ||||
|                                         err.status = 400; | ||||
|                                         return reject(err); | ||||
|                                     } | ||||
|  | ||||
|                                 return; | ||||
|                             } | ||||
|                             c--; | ||||
|                             if (c === 0) { | ||||
|                                 if (!errorReported) { | ||||
|                                     runtime.log.audit({event: "context.get",scope:scope,id:id,store:store,key:key},opts.req); | ||||
|                                     resolve(result); | ||||
|                                     return; | ||||
|                                 } | ||||
|                             } | ||||
|                         }); | ||||
|                                 c--; | ||||
|                                 if (c === 0) { | ||||
|                                     if (!errorReported) { | ||||
|                                         runtime.log.audit({event: "context.get",scope:scope,id:id,store:store,key:key},opts.req); | ||||
|                                         resolve(result); | ||||
|                                     } | ||||
|                                 } | ||||
|                             }); | ||||
|                         } | ||||
|                     }) | ||||
|                 } | ||||
|             } else { | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|     "name": "@node-red/runtime", | ||||
|     "version": "3.1.3", | ||||
|     "version": "4.0.0-dev", | ||||
|     "license": "Apache-2.0", | ||||
|     "main": "./lib/index.js", | ||||
|     "repository": { | ||||
| @@ -16,8 +16,8 @@ | ||||
|         } | ||||
|     ], | ||||
|     "dependencies": { | ||||
|         "@node-red/registry": "3.1.3", | ||||
|         "@node-red/util": "3.1.3", | ||||
|         "@node-red/registry": "4.0.0-dev", | ||||
|         "@node-red/util": "4.0.0-dev", | ||||
|         "async-mutex": "0.4.0", | ||||
|         "clone": "2.1.2", | ||||
|         "express": "4.18.2", | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|     "name": "@node-red/util", | ||||
|     "version": "3.1.3", | ||||
|     "version": "4.0.0-dev", | ||||
|     "license": "Apache-2.0", | ||||
|     "repository": { | ||||
|         "type": "git", | ||||
|   | ||||
							
								
								
									
										10
									
								
								packages/node_modules/node-red/package.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								packages/node_modules/node-red/package.json
									
									
									
									
										vendored
									
									
								
							| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|     "name": "node-red", | ||||
|     "version": "3.1.3", | ||||
|     "version": "4.0.0-dev", | ||||
|     "description": "Low-code programming for event-driven applications", | ||||
|     "homepage": "https://nodered.org", | ||||
|     "license": "Apache-2.0", | ||||
| @@ -31,10 +31,10 @@ | ||||
|         "flow" | ||||
|     ], | ||||
|     "dependencies": { | ||||
|         "@node-red/editor-api": "3.1.3", | ||||
|         "@node-red/runtime": "3.1.3", | ||||
|         "@node-red/util": "3.1.3", | ||||
|         "@node-red/nodes": "3.1.3", | ||||
|         "@node-red/editor-api": "4.0.0-dev", | ||||
|         "@node-red/runtime": "4.0.0-dev", | ||||
|         "@node-red/util": "4.0.0-dev", | ||||
|         "@node-red/nodes": "4.0.0-dev", | ||||
|         "basic-auth": "2.0.1", | ||||
|         "bcryptjs": "2.4.3", | ||||
|         "express": "4.18.2", | ||||
|   | ||||
							
								
								
									
										4
									
								
								packages/node_modules/node-red/settings.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								packages/node_modules/node-red/settings.js
									
									
									
									
										vendored
									
									
								
							| @@ -449,6 +449,7 @@ module.exports = { | ||||
|  *  - ui (for use with Node-RED Dashboard) | ||||
|  *  - debugUseColors | ||||
|  *  - debugMaxLength | ||||
|  *  - debugStatusLength | ||||
|  *  - execMaxBufferSize | ||||
|  *  - httpRequestTimeout | ||||
|  *  - mqttReconnectTime | ||||
| @@ -504,6 +505,9 @@ module.exports = { | ||||
|     /** The maximum length, in characters, of any message sent to the debug sidebar tab */ | ||||
|     debugMaxLength: 1000, | ||||
|  | ||||
|     /** The maximum length, in characters, of status messages under the debug node */ | ||||
|     //debugStatusLength: 32, | ||||
|  | ||||
|     /** Maximum buffer size for the exec node. Defaults to 10Mb */ | ||||
|     //execMaxBufferSize: 10000000, | ||||
|  | ||||
|   | ||||
| @@ -1718,9 +1718,13 @@ describe('function node', function() { | ||||
|     describe("init function", function() { | ||||
|  | ||||
|         it('should delay handling messages until init completes', function(done) { | ||||
|             const timeoutMS = 200; | ||||
|             // Since helper.load uses process.nextTick timers might occasionally finish | ||||
|             // a couple of milliseconds too early, so give some leeway to the check. | ||||
|             const timeoutCheckMargin = 5; | ||||
|             var flow = [{id:"n1",type:"function",wires:[["n2"]],initialize: ` | ||||
|                 return new Promise((resolve,reject) => { | ||||
|                     setTimeout(resolve,200) | ||||
|                     setTimeout(resolve, ${timeoutMS}); | ||||
|                 })`, | ||||
|                 func:"return msg;" | ||||
|             }, | ||||
| @@ -1733,9 +1737,10 @@ describe('function node', function() { | ||||
|                     msg.delta = Date.now() - msg.payload; | ||||
|                     receivedMsgs.push(msg) | ||||
|                     if (receivedMsgs.length === 5) { | ||||
|                         var errors = receivedMsgs.filter(msg => msg.delta < 200) | ||||
|                         let deltas = receivedMsgs.map(msg => msg.delta); | ||||
|                         var errors = deltas.filter(delta => delta < (timeoutMS - timeoutCheckMargin)) | ||||
|                         if (errors.length > 0) { | ||||
|                             done(new Error(`Message received before init completed - was ${msg.delta} expected >300`)) | ||||
|                             done(new Error(`Message received before init completed - delta values ${JSON.stringify(deltas)} expected to be > ${timeoutMS - timeoutCheckMargin}`)) | ||||
|                         } else { | ||||
|                             done(); | ||||
|                         } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user