mirror of
				https://github.com/node-red/node-red.git
				synced 2025-03-01 10:36:34 +00:00 
			
		
		
		
	Merge branch 'master' into dev
This commit is contained in:
		
							
								
								
									
										23
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -1,3 +1,26 @@ | |||||||
|  | #### 3.1.7: Maintenance Release | ||||||
|  |  | ||||||
|  |  - Add Japanese translation for v3.1.6 (#4603) @kazuhitoyokoi | ||||||
|  |  - Update jsonata version (#4593) @hardillb | ||||||
|  |  | ||||||
|  | #### 3.1.6: Maintenance Release | ||||||
|  |  | ||||||
|  | Editor | ||||||
|  |  | ||||||
|  |  - Do not flag env var in num typedInput as error (#4582) @knolleary | ||||||
|  |  - Fix example flow name in import dialog (#4578) @kazuhitoyokoi | ||||||
|  |  - Fix missing node icons in workspace (#4570) @knolleary | ||||||
|  |  | ||||||
|  | Runtime | ||||||
|  |  | ||||||
|  |  - Handle undefined env vars (#4581) @knolleary | ||||||
|  |  - fix: Removed offending MD5 crypto hash and replaced with SHA1 and SHA256 … (#4568) @JaysonHurst | ||||||
|  |  - chore: remove never use import code (#4580) @giscafer | ||||||
|  |  | ||||||
|  | Nodes | ||||||
|  |  | ||||||
|  |  - fix: template node zh-CN translation (#4575) @giscafer | ||||||
|  |  | ||||||
| #### 3.1.5: Maintenance Release | #### 3.1.5: Maintenance Release | ||||||
|  |  | ||||||
| Runtime | Runtime | ||||||
|   | |||||||
| @@ -13,7 +13,6 @@ | |||||||
|  * See the License for the specific language governing permissions and |  * See the License for the specific language governing permissions and | ||||||
|  * limitations under the License. |  * limitations under the License. | ||||||
|  **/ |  **/ | ||||||
| var apiUtils = require("../util"); |  | ||||||
| var runtimeAPI; | var runtimeAPI; | ||||||
| var settings; | var settings; | ||||||
| var theme = require("../editor/theme"); | var theme = require("../editor/theme"); | ||||||
|   | |||||||
| @@ -18,7 +18,6 @@ var BearerStrategy = require('passport-http-bearer').Strategy; | |||||||
| var ClientPasswordStrategy = require('passport-oauth2-client-password').Strategy; | var ClientPasswordStrategy = require('passport-oauth2-client-password').Strategy; | ||||||
|  |  | ||||||
| var passport = require("passport"); | var passport = require("passport"); | ||||||
| var crypto = require("crypto"); |  | ||||||
| var util = require("util"); | var util = require("util"); | ||||||
|  |  | ||||||
| var Tokens = require("./tokens"); | var Tokens = require("./tokens"); | ||||||
|   | |||||||
| @@ -14,11 +14,9 @@ | |||||||
|  * limitations under the License. |  * limitations under the License. | ||||||
|  **/ |  **/ | ||||||
|  |  | ||||||
| var express = require("express"); |  | ||||||
| var path = require('path'); | var path = require('path'); | ||||||
|  |  | ||||||
| var comms = require("./comms"); | var comms = require("./comms"); | ||||||
| var library = require("./library"); |  | ||||||
| var info = require("./settings"); | var info = require("./settings"); | ||||||
|  |  | ||||||
| var auth = require("../auth"); | var auth = require("../auth"); | ||||||
|   | |||||||
| @@ -15,8 +15,6 @@ | |||||||
|  **/ |  **/ | ||||||
|  |  | ||||||
| var apiUtils = require("../util"); | var apiUtils = require("../util"); | ||||||
| var fs = require('fs'); |  | ||||||
| var fspath = require('path'); |  | ||||||
|  |  | ||||||
| var runtimeAPI; | var runtimeAPI; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -13,9 +13,6 @@ | |||||||
|  * See the License for the specific language governing permissions and |  * See the License for the specific language governing permissions and | ||||||
|  * limitations under the License. |  * limitations under the License. | ||||||
|  **/ |  **/ | ||||||
| var fs = require('fs'); |  | ||||||
| var path = require('path'); |  | ||||||
| // var apiUtil = require('../util'); |  | ||||||
|  |  | ||||||
| var i18n = require("@node-red/util").i18n; // TODO: separate module | var i18n = require("@node-red/util").i18n; // TODO: separate module | ||||||
|  |  | ||||||
|   | |||||||
| @@ -15,7 +15,6 @@ | |||||||
|  **/ |  **/ | ||||||
|  |  | ||||||
| var apiUtils = require("../util"); | var apiUtils = require("../util"); | ||||||
| var express = require("express"); |  | ||||||
| var runtimeAPI; | var runtimeAPI; | ||||||
| var settings; | var settings; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -14,7 +14,6 @@ | |||||||
|  * limitations under the License. |  * limitations under the License. | ||||||
|  **/ |  **/ | ||||||
|  |  | ||||||
| var express = require("express"); |  | ||||||
| var util = require("util"); | var util = require("util"); | ||||||
| var path = require("path"); | var path = require("path"); | ||||||
| var fs = require("fs"); | var fs = require("fs"); | ||||||
|   | |||||||
| @@ -99,7 +99,7 @@ module.exports = { | |||||||
|             // settings.instanceId is set asynchronously to the editor-api |             // settings.instanceId is set asynchronously to the editor-api | ||||||
|             // being initiaised. So we defer calculating the cacheBuster hash |             // being initiaised. So we defer calculating the cacheBuster hash | ||||||
|             // until the first load of the editor |             // until the first load of the editor | ||||||
|             cacheBuster = crypto.createHash('md5').update(`${settings.version || 'version'}-${settings.instanceId || 'instanceId'}`).digest("hex").substring(0,12)     |             cacheBuster = crypto.createHash('sha1').update(`${settings.version || 'version'}-${settings.instanceId || 'instanceId'}`).digest("hex").substring(0,12)     | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         let sessionMessages; |         let sessionMessages; | ||||||
|   | |||||||
| @@ -24,11 +24,8 @@ | |||||||
|   * @namespace @node-red/editor-api |   * @namespace @node-red/editor-api | ||||||
|   */ |   */ | ||||||
|  |  | ||||||
| var express = require("express"); |  | ||||||
| var bodyParser = require("body-parser"); | var bodyParser = require("body-parser"); | ||||||
| var util = require('util'); |  | ||||||
| var passport = require('passport'); | var passport = require('passport'); | ||||||
| var cors = require('cors'); |  | ||||||
|  |  | ||||||
| var auth = require("./auth"); | var auth = require("./auth"); | ||||||
| var apiUtil = require("./util"); | var apiUtil = require("./util"); | ||||||
|   | |||||||
| @@ -303,7 +303,8 @@ | |||||||
|                 "missingType": "不正なフロー - __index__ 番目の要素に'type'プロパティがありません" |                 "missingType": "不正なフロー - __index__ 番目の要素に'type'プロパティがありません" | ||||||
|             }, |             }, | ||||||
|             "conflictNotification1": "読み込もうとしているノードのいくつかは、既にワークスペース内に存在しています。", |             "conflictNotification1": "読み込もうとしているノードのいくつかは、既にワークスペース内に存在しています。", | ||||||
|             "conflictNotification2": "読み込むノードを選択し、また既存のノードを置き換えるか、もしくはそれらのコピーを読み込むかも選択してください。" |             "conflictNotification2": "読み込むノードを選択し、また既存のノードを置き換えるか、もしくはそれらのコピーを読み込むかも選択してください。", | ||||||
|  |             "alreadyExists": "本ノードは既に存在" | ||||||
|         }, |         }, | ||||||
|         "copyMessagePath": "パスをコピーしました", |         "copyMessagePath": "パスをコピーしました", | ||||||
|         "copyMessageValue": "値をコピーしました", |         "copyMessageValue": "値をコピーしました", | ||||||
|   | |||||||
| @@ -919,7 +919,10 @@ RED.utils = (function() { | |||||||
|      * @returns true if valid, String if invalid |      * @returns true if valid, String if invalid | ||||||
|      */ |      */ | ||||||
|     function validateTypedProperty(propertyValue, propertyType, opt) { |     function validateTypedProperty(propertyValue, propertyType, opt) { | ||||||
|  |         if (propertyValue && /^\${[^}]+}$/.test(propertyValue)) { | ||||||
|  |             // Allow ${ENV_VAR} value | ||||||
|  |             return true | ||||||
|  |         } | ||||||
|         let error |         let error | ||||||
|         if (propertyType === 'json') { |         if (propertyType === 'json') { | ||||||
|             try { |             try { | ||||||
|   | |||||||
| @@ -4156,7 +4156,7 @@ RED.view = (function() { | |||||||
|                     } |                     } | ||||||
|                     var width = img.width * scaleFactor; |                     var width = img.width * scaleFactor; | ||||||
|                     if (width > 20) { |                     if (width > 20) { | ||||||
|                         scalefactor *= 20/width; |                         scaleFactor *= 20/width; | ||||||
|                         width = 20; |                         width = 20; | ||||||
|                     } |                     } | ||||||
|                     var height = img.height * scaleFactor; |                     var height = img.height * scaleFactor; | ||||||
|   | |||||||
| @@ -16,8 +16,20 @@ | |||||||
| RED.validators = { | RED.validators = { | ||||||
|     number: function(blankAllowed,mopt){ |     number: function(blankAllowed,mopt){ | ||||||
|         return function(v, opt) { |         return function(v, opt) { | ||||||
|             if ((blankAllowed&&(v===''||v===undefined)) || (v!=='' && !isNaN(v))) { |             if (blankAllowed && (v === '' || v === undefined)) { | ||||||
|                 return true; |                 return true | ||||||
|  |             } | ||||||
|  |             if (v !== '') { | ||||||
|  |                 if (/^NaN$|^[+-]?[0-9]*\.?[0-9]*([eE][-+]?[0-9]+)?$|^[+-]?(0b|0B)[01]+$|^[+-]?(0o|0O)[0-7]+$|^[+-]?(0x|0X)[0-9a-fA-F]+$/.test(v)) { | ||||||
|  |                     return true | ||||||
|  |                 } | ||||||
|  |                 if (/^\${[^}]+}$/.test(v)) { | ||||||
|  |                     // Allow ${ENV_VAR} value | ||||||
|  |                     return true | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             if (!isNaN(v)) { | ||||||
|  |                 return true | ||||||
|             } |             } | ||||||
|             if (opt && opt.label) { |             if (opt && opt.label) { | ||||||
|                 return RED._("validator.errors.invalid-num-prop", { |                 return RED._("validator.errors.invalid-num-prop", { | ||||||
|   | |||||||
| @@ -227,34 +227,42 @@ | |||||||
|             name: {value:""}, |             name: {value:""}, | ||||||
|             props:{value:[{p:"payload"},{p:"topic",vt:"str"}], validate:function(v, opt) { |             props:{value:[{p:"payload"},{p:"topic",vt:"str"}], validate:function(v, opt) { | ||||||
|                     if (!v || v.length === 0) { return true } |                     if (!v || v.length === 0) { return true } | ||||||
|  |                     const errors = [] | ||||||
|                     for (var i=0;i<v.length;i++) { |                     for (var i=0;i<v.length;i++) { | ||||||
|  |                         if (/^\${[^}]+}$/.test(v[i].v)) { | ||||||
|  |                             // Allow ${ENV_VAR} value | ||||||
|  |                             continue | ||||||
|  |                         } | ||||||
|                         if (/msg|flow|global/.test(v[i].vt)) { |                         if (/msg|flow|global/.test(v[i].vt)) { | ||||||
|                             if (!RED.utils.validatePropertyExpression(v[i].v)) { |                             if (!RED.utils.validatePropertyExpression(v[i].v)) { | ||||||
|                                 return RED._("node-red:inject.errors.invalid-prop", { prop: 'msg.'+v[i].p, error: v[i].v }); |                                 errors.push(RED._("node-red:inject.errors.invalid-prop", { prop: 'msg.'+v[i].p, error: v[i].v })) | ||||||
|                             } |                             } | ||||||
|                         } else if (v[i].vt === "jsonata") { |                         } else if (v[i].vt === "jsonata") { | ||||||
|                             try{ jsonata(v[i].v); } |                             try{ jsonata(v[i].v); } | ||||||
|                             catch(e){ |                             catch(e){ | ||||||
|                                 return RED._("node-red:inject.errors.invalid-jsonata", { prop: 'msg.'+v[i].p, error: e.message }); |                                 errors.push(RED._("node-red:inject.errors.invalid-jsonata", { prop: 'msg.'+v[i].p, error: e.message })) | ||||||
|                             } |                             } | ||||||
|                         } else if (v[i].vt === "json") { |                         } else if (v[i].vt === "json") { | ||||||
|                             try{ JSON.parse(v[i].v); } |                             try{ JSON.parse(v[i].v); } | ||||||
|                             catch(e){ |                             catch(e){ | ||||||
|                                 return RED._("node-red:inject.errors.invalid-json", { prop: 'msg.'+v[i].p, error: e.message }); |                                 errors.push(RED._("node-red:inject.errors.invalid-json", { prop: 'msg.'+v[i].p, error: e.message })) | ||||||
|                             } |                             } | ||||||
|                         } else if (v[i].vt === "num"){ |                         } else if (v[i].vt === "num"){ | ||||||
|                             if (!/^[+-]?[0-9]*\.?[0-9]*([eE][-+]?[0-9]+)?$/.test(v[i].v)) { |                             if (!/^[+-]?[0-9]*\.?[0-9]*([eE][-+]?[0-9]+)?$/.test(v[i].v)) { | ||||||
|                                 return RED._("node-red:inject.errors.invalid-prop", { prop: 'msg.'+v[i].p, error: v[i].v }); |                                 errors.push(RED._("node-red:inject.errors.invalid-prop", { prop: 'msg.'+v[i].p, error: v[i].v })) | ||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|  |                     if (errors.length > 0) { | ||||||
|  |                         return errors | ||||||
|  |                     } | ||||||
|                     return true; |                     return true; | ||||||
|                 } |                 } | ||||||
|             }, |             }, | ||||||
|             repeat: { |             repeat: { | ||||||
|                 value:"", validate: function(v, opt) { |                 value:"", validate: function(v, opt) { | ||||||
|                     if ((v === "") || |                     if ((v === "") || | ||||||
|                         (RED.validators.number(v) && |                         (RED.validators.number()(v) && | ||||||
|                          (v >= 0) && (v <= 2147483))) { |                          (v >= 0) && (v <= 2147483))) { | ||||||
|                         return true; |                         return true; | ||||||
|                     } |                     } | ||||||
| @@ -263,7 +271,7 @@ | |||||||
|             }, |             }, | ||||||
|             crontab: {value:""}, |             crontab: {value:""}, | ||||||
|             once: {value:false}, |             once: {value:false}, | ||||||
|             onceDelay: {value:0.1}, |             onceDelay: {value:0.1, validate: RED.validators.number(true)}, | ||||||
|             topic: {value:""}, |             topic: {value:""}, | ||||||
|             payload: {value:"", validate: RED.validators.typedInput("payloadType", false) }, |             payload: {value:"", validate: RED.validators.typedInput("payloadType", false) }, | ||||||
|             payloadType: {value:"date"}, |             payloadType: {value:"date"}, | ||||||
|   | |||||||
| @@ -23,7 +23,7 @@ | |||||||
|         <dt class="optional">template <span class="property-type">string</span></dt> |         <dt class="optional">template <span class="property-type">string</span></dt> | ||||||
|         <dd>由<code>msg.payload</code>填充的模板。如果未在编辑面板中配置,则可以将设为msg的属性。</dd> |         <dd>由<code>msg.payload</code>填充的模板。如果未在编辑面板中配置,则可以将设为msg的属性。</dd> | ||||||
|     </dl> |     </dl> | ||||||
|     <h3>Outputs</h3> |     <h3>输出</h3> | ||||||
|     <dl class="message-properties"> |     <dl class="message-properties"> | ||||||
|         <dt>msg <span class="property-type">object</span></dt> |         <dt>msg <span class="property-type">object</span></dt> | ||||||
|         <dd>由来自传入msg的属性来填充已配置的模板后输出的带有属性的msg。</dd> |         <dd>由来自传入msg的属性来填充已配置的模板后输出的带有属性的msg。</dd> | ||||||
| @@ -32,7 +32,7 @@ | |||||||
|     <p>默认情况下使用<i><a href="http://mustache.github.io/mustache.5.html" target="_blank">mustache</a></i>格式。如有需要也可以切换其他格式。</p> |     <p>默认情况下使用<i><a href="http://mustache.github.io/mustache.5.html" target="_blank">mustache</a></i>格式。如有需要也可以切换其他格式。</p> | ||||||
|     <p>例如: |     <p>例如: | ||||||
|     <pre>Hello {{payload.name}}. Today is {{date}}</pre> |     <pre>Hello {{payload.name}}. Today is {{date}}</pre> | ||||||
|     <p>receives a message containing: |     <p>接收一条消息,其中包含: | ||||||
|     <pre>{ |     <pre>{ | ||||||
|   date: "Monday", |   date: "Monday", | ||||||
|   payload: { |   payload: { | ||||||
|   | |||||||
| @@ -36,7 +36,7 @@ async function getFlowsFromPath(path) { | |||||||
|                     promises.push(getFlowsFromPath(fullPath)); |                     promises.push(getFlowsFromPath(fullPath)); | ||||||
|                 } else if (/\.json$/.test(file)){ |                 } else if (/\.json$/.test(file)){ | ||||||
|                     validFiles.push(file); |                     validFiles.push(file); | ||||||
|                     promises.push(Promise.resolve(file.split(".")[0])) |                     promises.push(Promise.resolve(file.replace(/\.json$/, ''))) | ||||||
|                 } |                 } | ||||||
|             }) |             }) | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -485,7 +485,7 @@ class Flow { | |||||||
|         } |         } | ||||||
|         if (!key.startsWith("$parent.")) { |         if (!key.startsWith("$parent.")) { | ||||||
|             if (this._env.hasOwnProperty(key)) { |             if (this._env.hasOwnProperty(key)) { | ||||||
|                 return (Object.hasOwn(this._env[key], 'value') && this._env[key].__clone__) ? clone(this._env[key].value) : this._env[key] |                 return (this._env[key] && Object.hasOwn(this._env[key], 'value') && this._env[key].__clone__) ? clone(this._env[key].value) : this._env[key] | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|                 key = key.substring(8); |                 key = key.substring(8); | ||||||
|   | |||||||
| @@ -41,7 +41,7 @@ class Group { | |||||||
|         } |         } | ||||||
|         if (!key.startsWith("$parent.")) { |         if (!key.startsWith("$parent.")) { | ||||||
|             if (this._env.hasOwnProperty(key)) { |             if (this._env.hasOwnProperty(key)) { | ||||||
|                 return (Object.hasOwn(this._env[key], 'value') && this._env[key].__clone__) ? clone(this._env[key].value) : this._env[key] |                 return (this._env[key] && Object.hasOwn(this._env[key], 'value') && this._env[key].__clone__) ? clone(this._env[key].value) : this._env[key] | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             key = key.substring(8); |             key = key.substring(8); | ||||||
|   | |||||||
| @@ -376,7 +376,7 @@ class Subflow extends Flow { | |||||||
|         } |         } | ||||||
|         if (!key.startsWith("$parent.")) { |         if (!key.startsWith("$parent.")) { | ||||||
|             if (this._env.hasOwnProperty(key)) { |             if (this._env.hasOwnProperty(key)) { | ||||||
|                 return (Object.hasOwn(this._env[key], 'value') && this._env[key].__clone__) ? clone(this._env[key].value) : this._env[key] |                 return (this._env[key] && Object.hasOwn(this._env[key], 'value') && this._env[key].__clone__) ? clone(this._env[key].value) : this._env[key] | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             key = key.substring(8); |             key = key.substring(8); | ||||||
|   | |||||||
| @@ -77,7 +77,7 @@ var storageModuleInterface = { | |||||||
|                         flows: flows, |                         flows: flows, | ||||||
|                         credentials: creds |                         credentials: creds | ||||||
|                     }; |                     }; | ||||||
|                     result.rev = crypto.createHash('md5').update(JSON.stringify(result.flows)).digest("hex"); |                     result.rev = crypto.createHash('sha256').update(JSON.stringify(result.flows)).digest("hex"); | ||||||
|                     return result; |                     return result; | ||||||
|                 }) |                 }) | ||||||
|             }); |             }); | ||||||
| @@ -95,7 +95,7 @@ var storageModuleInterface = { | |||||||
|  |  | ||||||
|             return credentialSavePromise.then(function() { |             return credentialSavePromise.then(function() { | ||||||
|                 return storageModule.saveFlows(flows, user).then(function() { |                 return storageModule.saveFlows(flows, user).then(function() { | ||||||
|                     return crypto.createHash('md5').update(JSON.stringify(config.flows)).digest("hex"); |                     return crypto.createHash('sha256').update(JSON.stringify(config.flows)).digest("hex"); | ||||||
|                 }) |                 }) | ||||||
|             }); |             }); | ||||||
|         }, |         }, | ||||||
|   | |||||||
| @@ -33,16 +33,15 @@ describe("library api", function() { | |||||||
|         should.not.exist(library.getExampleFlowPath('foo','bar')); |         should.not.exist(library.getExampleFlowPath('foo','bar')); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     it('returns a valid example path', function(done) { |     it('returns valid example paths', function(done) { | ||||||
|         library.init(); |         library.init(); | ||||||
|         library.addExamplesDir("test-module",path.resolve(__dirname+'/resources/examples')).then(function() { |         library.addExamplesDir("test-module",path.resolve(__dirname+'/resources/examples')).then(function() { | ||||||
|             try { |             try { | ||||||
|                 var flows = library.getExampleFlows(); |                 var flows = library.getExampleFlows(); | ||||||
|                 flows.should.deepEqual({"test-module":{"f":["one"]}}); |                 flows.should.deepEqual({"test-module":{"f":["1.2.3","one"]}}); | ||||||
|  |  | ||||||
|                 var examplePath = library.getExampleFlowPath('test-module','one'); |                 var examplePath = library.getExampleFlowPath('test-module','one'); | ||||||
|                 examplePath.should.eql(path.resolve(__dirname+'/resources/examples/one.json')) |                 examplePath.should.eql(path.resolve(__dirname+'/resources/examples/one.json')); | ||||||
|  |  | ||||||
|  |  | ||||||
|                 library.removeExamplesDir('test-module'); |                 library.removeExamplesDir('test-module'); | ||||||
|  |  | ||||||
| @@ -57,6 +56,5 @@ describe("library api", function() { | |||||||
|                 done(err); |                 done(err); | ||||||
|             } |             } | ||||||
|         }); |         }); | ||||||
|  |     }); | ||||||
|     }) |  | ||||||
| }); | }); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user