/** * Copyright JS Foundation and other contributors, http://js.foundation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. **/ module.exports = function(RED) { "use strict"; function ChangeNode(n) { RED.nodes.createNode(this, n); var node = this; this.rules = n.rules; var rule; if (!this.rules) { rule = { t:(n.action=="replace"?"set":n.action), p:n.property||"" } if ((rule.t === "set")||(rule.t === "move")) { rule.to = n.to||""; } else if (rule.t === "change") { rule.from = n.from||""; rule.to = n.to||""; rule.re = (n.reg===null||n.reg); } this.rules = [rule]; } var valid = true; for (var i=0;i { RED.util.evaluateNodeProperty(rule.to, rule.tot, node, msg, (err,value) => { if (err) { resolve(undefined); } else { resolve(value); } }); }); } else if (rule.tot === 'date') { value = Date.now(); } else if (rule.tot === 'jsonata') { return new Promise((resolve,reject) => { RED.util.evaluateJSONataExpression(rule.to,msg, (err, value) => { if (err) { reject(RED._("change.errors.invalid-expr",{error:err.message})) } else { resolve(value); } }); }); } return Promise.resolve(value); } function getFromValue(msg,rule) { var fromValue; var fromType; var fromRE; if (rule.t === 'change') { if (rule.fromt === 'msg' || rule.fromt === 'flow' || rule.fromt === 'global') { return new Promise((resolve,reject) => { if (rule.fromt === "msg") { resolve(RED.util.getMessageProperty(msg,rule.from)); } else if (rule.fromt === 'flow' || rule.fromt === 'global') { var contextKey = RED.util.parseContextStore(rule.from); node.context()[rule.fromt].get(contextKey.key, contextKey.store, (err,fromValue) => { if (err) { reject(err); } else { resolve(fromValue); } }); } }).then(fromValue => { if (typeof fromValue === 'number' || fromValue instanceof Number) { fromType = 'num'; } else if (typeof fromValue === 'boolean') { fromType = 'bool' } else if (fromValue instanceof RegExp) { fromType = 're'; fromRE = fromValue; } else if (typeof fromValue === 'string') { fromType = 'str'; fromRE = fromValue.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, "\\$&"); try { fromRE = new RegExp(fromRE, "g"); } catch (e) { throw new Error(RED._("change.errors.invalid-from",{error:e.message})); } } else { throw new Error(RED._("change.errors.invalid-from",{error:"unsupported type: "+(typeof fromValue)})); } return { fromType, fromValue, fromRE } }); } else { fromType = rule.fromt; fromValue = rule.from; fromRE = rule.fromRE; } } return Promise.resolve({ fromType, fromValue, fromRE }); } function applyRule(msg,rule) { var property = rule.p; var current; var fromValue; var fromType; var fromRE; try { return getToValue(msg,rule).then(value => { return getFromValue(msg,rule).then(fromParts => { fromValue = fromParts.fromValue; fromType = fromParts.fromType; fromRE = fromParts.fromRE; if (rule.pt === 'msg') { try { if (rule.t === 'delete') { RED.util.setMessageProperty(msg,property,undefined); } else if (rule.t === 'set') { RED.util.setMessageProperty(msg,property,value); } else if (rule.t === 'change') { current = RED.util.getMessageProperty(msg,property); if (typeof current === 'string') { if ((fromType === 'num' || fromType === 'bool' || fromType === 'str') && current === fromValue) { // str representation of exact from number/boolean // only replace if they match exactly RED.util.setMessageProperty(msg,property,value); } else { current = current.replace(fromRE,value); RED.util.setMessageProperty(msg,property,current); } } else if ((typeof current === 'number' || current instanceof Number) && fromType === 'num') { if (current == Number(fromValue)) { RED.util.setMessageProperty(msg,property,value); } } else if (typeof current === 'boolean' && fromType === 'bool') { if (current.toString() === fromValue) { RED.util.setMessageProperty(msg,property,value); } } } } catch(err) {} return msg; } else if (rule.pt === 'flow' || rule.pt === 'global') { var contextKey = RED.util.parseContextStore(property); return new Promise((resolve,reject) => { var target = node.context()[rule.pt]; var callback = err => { if (err) { reject(err); } else { resolve(msg); } } if (rule.t === 'delete') { target.set(contextKey.key,undefined,contextKey.store,callback); } else if (rule.t === 'set') { target.set(contextKey.key,value,contextKey.store,callback); } else if (rule.t === 'change') { target.get(contextKey.key,contextKey.store,(err,current) => { if (err) { reject(err); return; } if (typeof current === 'string') { if ((fromType === 'num' || fromType === 'bool' || fromType === 'str') && current === fromValue) { // str representation of exact from number/boolean // only replace if they match exactly target.set(contextKey.key,value,contextKey.store,callback); } else { current = current.replace(fromRE,value); target.set(contextKey.key,current,contextKey.store,callback); } } else if ((typeof current === 'number' || current instanceof Number) && fromType === 'num') { if (current == Number(fromValue)) { target.set(contextKey.key,value,contextKey.store,callback); } } else if (typeof current === 'boolean' && fromType === 'bool') { if (current.toString() === fromValue) { target.set(contextKey.key,value,contextKey.store,callback); } } }); } }); } }); }).catch(err => { node.error(err, msg); return null; }); } catch(err) { return Promise.resolve(msg); } } function applyRules(msg, currentRule) { if (currentRule >= node.rules.length) { return Promise.resolve(msg); } var r = node.rules[currentRule]; var rulePromise; if (r.t === "move") { if ((r.tot !== r.pt) || (r.p.indexOf(r.to) !== -1)) { rulePromise = applyRule(msg,{t:"set", p:r.to, pt:r.tot, to:r.p, tot:r.pt}).then( msg => applyRule(msg,{t:"delete", p:r.p, pt:r.pt}) ); } else { // 2 step move if we are moving from a child rulePromise = applyRule(msg,{t:"set", p:"_temp_move", pt:r.tot, to:r.p, tot:r.pt}).then( msg => applyRule(msg,{t:"delete", p:r.p, pt:r.pt}) ).then( msg => applyRule(msg,{t:"set", p:r.to, pt:r.tot, to:"_temp_move", tot:r.pt}) ).then( msg => applyRule(msg,{t:"delete", p:"_temp_move", pt:r.pt}) ) } } else { rulePromise = applyRule(msg,r); } return rulePromise.then( msg => { if (!msg) { return } else if (currentRule === node.rules.length - 1) { return msg; } else { return applyRules(msg, currentRule+1); } } ); } if (valid) { this.on('input', function(msg) { applyRules(msg, 0) .then( msg => { if (msg) { node.send(msg) }} ) .catch( err => node.error(err, msg)) }); } } RED.nodes.registerType("change", ChangeNode); };