2015-10-11 20:37:11 +01:00
|
|
|
/**
|
2017-01-11 15:24:33 +00:00
|
|
|
* Copyright JS Foundation and other contributors, http://js.foundation
|
2015-10-11 20:37:11 +01:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
**/
|
2023-06-23 15:48:06 +01:00
|
|
|
const clone = require("clone");
|
|
|
|
const redUtil = require("@node-red/util").util;
|
|
|
|
const Log = require("@node-red/util").log;
|
|
|
|
const typeRegistry = require("@node-red/registry");
|
|
|
|
const subflowInstanceRE = /^subflow:(.+)$/;
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2022-11-04 18:42:51 +09:00
|
|
|
let _runtime = null;
|
2023-06-23 15:48:06 +01:00
|
|
|
let envVarExcludes = {};
|
2020-08-28 16:36:11 +01:00
|
|
|
|
2023-06-23 15:48:06 +01:00
|
|
|
function init(runtime) {
|
|
|
|
_runtime = runtime;
|
|
|
|
envVarExcludes = {};
|
|
|
|
if (runtime.settings.hasOwnProperty('envVarExcludes') && Array.isArray(runtime.settings.envVarExcludes)) {
|
|
|
|
runtime.settings.envVarExcludes.forEach(v => envVarExcludes[v] = true);
|
|
|
|
}
|
|
|
|
}
|
2019-03-07 22:54:20 +00:00
|
|
|
|
2015-10-11 20:37:11 +01:00
|
|
|
function diffNodes(oldNode,newNode) {
|
|
|
|
if (oldNode == null) {
|
|
|
|
return true;
|
|
|
|
}
|
2023-07-10 12:04:52 +01:00
|
|
|
const keyFilter = p => p != 'x' && p != 'y' && p != 'wires'
|
|
|
|
const groupKeyFilter = p => keyFilter(p) && p != 'nodes' && p != 'style' && p != 'w' && p != 'h'
|
|
|
|
var oldKeys = Object.keys(oldNode).filter(oldNode.type === 'group' ? groupKeyFilter : keyFilter);
|
|
|
|
var newKeys = Object.keys(newNode).filter(newNode.type === 'group' ? groupKeyFilter : keyFilter);
|
|
|
|
|
2015-10-11 20:37:11 +01:00
|
|
|
if (oldKeys.length != newKeys.length) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
for (var i=0;i<newKeys.length;i++) {
|
|
|
|
var p = newKeys[i];
|
|
|
|
if (!redUtil.compareObjects(oldNode[p],newNode[p])) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-05-21 15:19:50 +01:00
|
|
|
var EnvVarPropertyRE_old = /^\$\((\S+)\)$/;
|
|
|
|
var EnvVarPropertyRE = /^\${(\S+)}$/;
|
2016-11-17 13:56:17 +00:00
|
|
|
|
2021-08-19 21:15:13 +09:00
|
|
|
|
|
|
|
function mapEnvVarProperties(obj,prop,flow,config) {
|
2019-01-16 22:38:04 +00:00
|
|
|
var v = obj[prop];
|
|
|
|
if (Buffer.isBuffer(v)) {
|
2016-11-17 13:56:17 +00:00
|
|
|
return;
|
2019-01-16 22:38:04 +00:00
|
|
|
} else if (Array.isArray(v)) {
|
|
|
|
for (var i=0;i<v.length;i++) {
|
2021-08-19 21:15:13 +09:00
|
|
|
mapEnvVarProperties(v,i,flow,config);
|
2016-11-17 13:56:17 +00:00
|
|
|
}
|
|
|
|
} else if (typeof obj[prop] === 'string') {
|
2019-01-16 22:38:04 +00:00
|
|
|
if (obj[prop][0] === "$" && (EnvVarPropertyRE_old.test(v) || EnvVarPropertyRE.test(v)) ) {
|
|
|
|
var envVar = v.substring(2,v.length-1);
|
2021-08-19 21:15:13 +09:00
|
|
|
var r = redUtil.getSetting(config, envVar, flow);
|
|
|
|
obj[prop] = r ? r : obj[prop];
|
2016-11-17 13:56:17 +00:00
|
|
|
}
|
|
|
|
} else {
|
2019-01-16 22:38:04 +00:00
|
|
|
for (var p in v) {
|
|
|
|
if (v.hasOwnProperty(p)) {
|
2021-08-19 21:15:13 +09:00
|
|
|
mapEnvVarProperties(v,p,flow,config);
|
2016-11-17 13:56:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
async function evaluateEnvProperties(flow, env, credentials) {
|
2023-09-25 18:08:02 +01:00
|
|
|
credentials = credentials || {}
|
2023-06-23 02:11:57 +01:00
|
|
|
const pendingEvaluations = []
|
|
|
|
const evaluatedEnv = {}
|
|
|
|
const envTypes = []
|
|
|
|
for (let i = 0; i < env.length; i++) {
|
|
|
|
let { name, value, type } = env[i]
|
|
|
|
if (type === "env") {
|
|
|
|
// Do env types last as they may include references to other env vars
|
|
|
|
// at this level which need to be resolved before they can be looked-up
|
|
|
|
envTypes.push(env[i])
|
|
|
|
} else if (type === "bool") {
|
|
|
|
value = (value === "true") || (value === true);
|
|
|
|
} else if (type === "cred") {
|
|
|
|
if (credentials.hasOwnProperty(name)) {
|
|
|
|
value = credentials[name];
|
|
|
|
}
|
|
|
|
} else if (type ==='jsonata') {
|
|
|
|
pendingEvaluations.push(new Promise((resolve, _) => {
|
|
|
|
redUtil.evaluateNodeProperty(value, 'jsonata', {_flow: flow}, null, (err, result) => {
|
|
|
|
if (!err) {
|
|
|
|
evaluatedEnv[name] = result
|
|
|
|
}
|
|
|
|
resolve()
|
|
|
|
});
|
|
|
|
}))
|
|
|
|
} else {
|
|
|
|
value = redUtil.evaluateNodeProperty(value, type, {_flow: flow}, null, null);
|
|
|
|
}
|
|
|
|
evaluatedEnv[name] = value
|
|
|
|
}
|
|
|
|
if (pendingEvaluations.length > 0) {
|
|
|
|
await Promise.all(pendingEvaluations)
|
|
|
|
}
|
2023-09-22 13:49:54 +01:00
|
|
|
// Now loop over the env types and evaluate them properly
|
2023-06-23 02:11:57 +01:00
|
|
|
for (let i = 0; i < envTypes.length; i++) {
|
|
|
|
let { name, value, type } = envTypes[i]
|
|
|
|
// If an env-var wants to lookup itself, delegate straight to the parent
|
|
|
|
// https://github.com/node-red/node-red/issues/2099
|
|
|
|
if (value === name) {
|
|
|
|
value = `$parent.${name}`
|
|
|
|
}
|
|
|
|
if (evaluatedEnv.hasOwnProperty(value)) {
|
|
|
|
value = evaluatedEnv[value]
|
|
|
|
} else {
|
2023-09-22 13:49:54 +01:00
|
|
|
value = redUtil.evaluateNodeProperty(value, type, {_flow: {
|
|
|
|
// Provide a hook so when it tries to look up a flow setting,
|
|
|
|
// we can insert the just-evaluated value which hasn't yet
|
|
|
|
// been set on the flow object - otherwise delegate up to the flow
|
|
|
|
getSetting: function(name) {
|
|
|
|
if (evaluatedEnv.hasOwnProperty(name)){
|
|
|
|
return evaluatedEnv[name]
|
|
|
|
}
|
|
|
|
return flow.getSetting(name)
|
|
|
|
}
|
|
|
|
}}, null, null);
|
2023-06-23 02:11:57 +01:00
|
|
|
}
|
|
|
|
evaluatedEnv[name] = value
|
|
|
|
}
|
2020-08-28 16:36:11 +01:00
|
|
|
|
2023-06-23 02:11:57 +01:00
|
|
|
return evaluatedEnv
|
|
|
|
}
|
|
|
|
|
2023-06-23 15:48:06 +01:00
|
|
|
/**
|
|
|
|
* Create a new instance of a node
|
|
|
|
* @param {Flow} flow The containing flow
|
|
|
|
* @param {object} config The node configuration object
|
|
|
|
* @return {Node} The instance of the node
|
|
|
|
*/
|
2023-06-23 02:11:57 +01:00
|
|
|
async function createNode(flow,config) {
|
2020-08-28 16:36:11 +01:00
|
|
|
var newNode = null;
|
|
|
|
var type = config.type;
|
|
|
|
try {
|
|
|
|
var nodeTypeConstructor = typeRegistry.get(type);
|
|
|
|
if (typeof nodeTypeConstructor === "function") {
|
|
|
|
var conf = clone(config);
|
|
|
|
delete conf.credentials;
|
|
|
|
try {
|
2022-01-12 11:07:25 +00:00
|
|
|
Object.defineProperty(conf,'_module', {value: typeRegistry.getNodeInfo(type), enumerable: false, writable: true })
|
2020-08-28 16:36:11 +01:00
|
|
|
Object.defineProperty(conf,'_flow', {value: flow, enumerable: false, writable: true })
|
2022-01-25 21:32:28 +00:00
|
|
|
Object.defineProperty(conf,'_path', {value: `${flow.path}/${config._alias||config.id}`, enumerable: false, writable: true })
|
2022-02-15 14:23:01 +09:00
|
|
|
|
|
|
|
for (var p in conf) {
|
|
|
|
if (conf.hasOwnProperty(p)) {
|
|
|
|
mapEnvVarProperties(conf,p,flow,conf);
|
|
|
|
}
|
|
|
|
}
|
2020-08-28 16:36:11 +01:00
|
|
|
newNode = new nodeTypeConstructor(conf);
|
|
|
|
} catch (err) {
|
|
|
|
Log.log({
|
|
|
|
level: Log.ERROR,
|
|
|
|
id:conf.id,
|
|
|
|
type: type,
|
|
|
|
msg: err
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} else if (nodeTypeConstructor) {
|
|
|
|
// console.log(nodeTypeConstructor)
|
|
|
|
var subflowConfig = parseConfig([nodeTypeConstructor.subflow].concat(nodeTypeConstructor.subflow.flow));
|
2021-04-28 21:49:32 +01:00
|
|
|
var subflowInstanceConfig = subflowConfig.subflows[nodeTypeConstructor.subflow.id];
|
|
|
|
delete subflowConfig.subflows[nodeTypeConstructor.subflow.id];
|
|
|
|
subflowInstanceConfig.subflows = subflowConfig.subflows;
|
2020-08-28 16:36:11 +01:00
|
|
|
var instanceConfig = clone(config);
|
|
|
|
instanceConfig.env = clone(nodeTypeConstructor.subflow.env);
|
|
|
|
|
|
|
|
instanceConfig.env = nodeTypeConstructor.subflow.env.map(nodeProp => {
|
|
|
|
var nodePropType;
|
|
|
|
var nodePropValue = config[nodeProp.name];
|
|
|
|
if (nodeProp.type === "cred") {
|
|
|
|
nodePropType = "cred";
|
|
|
|
} else {
|
|
|
|
switch(typeof config[nodeProp.name]) {
|
|
|
|
case "string": nodePropType = "str"; break;
|
|
|
|
case "number": nodePropType = "num"; break;
|
2021-06-02 14:48:54 +01:00
|
|
|
case "boolean": nodePropType = "bool"; nodePropValue == nodeProp?"true":"false"; break;
|
2020-08-28 16:36:11 +01:00
|
|
|
default:
|
|
|
|
nodePropType = config[nodeProp.name].type;
|
|
|
|
nodePropValue = config[nodeProp.name].value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
name: nodeProp.name,
|
|
|
|
type: nodePropType,
|
|
|
|
value: nodePropValue
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2020-09-02 19:33:31 +01:00
|
|
|
var subflow = require("./Subflow").createModuleInstance(
|
|
|
|
nodeTypeConstructor.type,
|
2020-08-28 16:36:11 +01:00
|
|
|
flow,
|
|
|
|
flow.global,
|
2021-04-28 21:49:32 +01:00
|
|
|
subflowInstanceConfig,
|
2020-08-28 16:36:11 +01:00
|
|
|
instanceConfig
|
|
|
|
);
|
2022-08-02 00:05:21 +01:00
|
|
|
// Register this subflow as an instance node of the parent flow.
|
|
|
|
// This allows nodes inside the subflow to get ahold of each other
|
|
|
|
// such as a node accessing its config node
|
|
|
|
flow.subflowInstanceNodes[config.id] = subflow
|
2023-06-23 02:11:57 +01:00
|
|
|
await subflow.start();
|
2020-08-28 16:36:11 +01:00
|
|
|
return subflow.node;
|
|
|
|
}
|
|
|
|
} catch(err) {
|
|
|
|
Log.error(err);
|
|
|
|
}
|
|
|
|
return newNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
function parseConfig(config) {
|
2023-06-23 02:11:57 +01:00
|
|
|
var flow = {};
|
|
|
|
flow.allNodes = {};
|
|
|
|
flow.subflows = {};
|
|
|
|
flow.configs = {};
|
|
|
|
flow.flows = {};
|
|
|
|
flow.missingTypes = [];
|
|
|
|
|
|
|
|
config.forEach(function (n) {
|
|
|
|
flow.allNodes[n.id] = clone(n);
|
|
|
|
if (n.type === 'tab') {
|
|
|
|
flow.flows[n.id] = n;
|
|
|
|
flow.flows[n.id].subflows = {};
|
|
|
|
flow.flows[n.id].configs = {};
|
|
|
|
flow.flows[n.id].nodes = {};
|
|
|
|
flow.flows[n.id].groups = {};
|
|
|
|
} else if (n.type === 'subflow') {
|
|
|
|
flow.subflows[n.id] = n;
|
|
|
|
flow.subflows[n.id].configs = {};
|
|
|
|
flow.subflows[n.id].nodes = {};
|
|
|
|
flow.subflows[n.id].groups = {};
|
|
|
|
flow.subflows[n.id].instances = [];
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
var linkWires = {};
|
|
|
|
var linkOutNodes = [];
|
|
|
|
config.forEach(function (n) {
|
|
|
|
if (n.type !== 'subflow' && n.type !== 'tab' && n.type !== 'group') {
|
|
|
|
var subflowDetails = subflowInstanceRE.exec(n.type);
|
|
|
|
|
|
|
|
if ((subflowDetails && !flow.subflows[subflowDetails[1]]) || (!subflowDetails && !typeRegistry.get(n.type))) {
|
|
|
|
if (flow.missingTypes.indexOf(n.type) === -1) {
|
|
|
|
flow.missingTypes.push(n.type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var container = null;
|
|
|
|
if (flow.flows[n.z]) {
|
|
|
|
container = flow.flows[n.z];
|
|
|
|
} else if (flow.subflows[n.z]) {
|
|
|
|
container = flow.subflows[n.z];
|
|
|
|
}
|
|
|
|
if (n.hasOwnProperty('x') && n.hasOwnProperty('y')) {
|
|
|
|
if (subflowDetails) {
|
|
|
|
var subflowType = subflowDetails[1]
|
|
|
|
n.subflow = subflowType;
|
|
|
|
if (flow.subflows[subflowType]) {
|
|
|
|
flow.subflows[subflowType].instances.push(n)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (container) {
|
|
|
|
container.nodes[n.id] = n;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (container) {
|
|
|
|
container.configs[n.id] = n;
|
|
|
|
} else {
|
|
|
|
flow.configs[n.id] = n;
|
|
|
|
flow.configs[n.id]._users = [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (n.type === 'link in' && n.links) {
|
|
|
|
// Ensure wires are present in corresponding link out nodes
|
|
|
|
n.links.forEach(function (id) {
|
|
|
|
linkWires[id] = linkWires[id] || {};
|
|
|
|
linkWires[id][n.id] = true;
|
|
|
|
})
|
|
|
|
} else if (n.type === 'link out' && n.links) {
|
|
|
|
linkWires[n.id] = linkWires[n.id] || {};
|
|
|
|
n.links.forEach(function (id) {
|
|
|
|
linkWires[n.id][id] = true;
|
|
|
|
})
|
|
|
|
linkOutNodes.push(n);
|
|
|
|
}
|
|
|
|
} else if (n.type === 'group') {
|
|
|
|
const parentContainer = flow.flows[n.z] || flow.subflows[n.z]
|
|
|
|
if (parentContainer) {
|
|
|
|
parentContainer.groups[n.id] = n
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
linkOutNodes.forEach(function (n) {
|
|
|
|
var links = linkWires[n.id];
|
|
|
|
var targets = Object.keys(links);
|
|
|
|
n.wires = [targets];
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
var addedTabs = {};
|
|
|
|
config.forEach(function (n) {
|
|
|
|
if (n.type !== 'subflow' && n.type !== 'tab' && n.type !== 'group') {
|
|
|
|
for (var prop in n) {
|
|
|
|
if (n.hasOwnProperty(prop) && prop !== 'id' && prop !== 'wires' && prop !== 'type' && prop !== '_users' && flow.configs.hasOwnProperty(n[prop])) {
|
|
|
|
// This property references a global config node
|
|
|
|
flow.configs[n[prop]]._users.push(n.id)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (n.z && !flow.subflows[n.z]) {
|
|
|
|
|
|
|
|
if (!flow.flows[n.z]) {
|
|
|
|
flow.flows[n.z] = { type: 'tab', id: n.z };
|
|
|
|
flow.flows[n.z].subflows = {};
|
|
|
|
flow.flows[n.z].configs = {};
|
|
|
|
flow.flows[n.z].nodes = {};
|
|
|
|
addedTabs[n.z] = flow.flows[n.z];
|
|
|
|
}
|
|
|
|
if (addedTabs[n.z]) {
|
|
|
|
if (n.hasOwnProperty('x') && n.hasOwnProperty('y')) {
|
|
|
|
addedTabs[n.z].nodes[n.id] = n;
|
|
|
|
} else {
|
|
|
|
addedTabs[n.z].configs[n.id] = n;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return flow;
|
2020-08-28 16:36:11 +01:00
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
function getEnvVar(k) {
|
|
|
|
if (!envVarExcludes[k]) {
|
|
|
|
return process.env[k];
|
|
|
|
}
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
function diffConfigs(oldConfig, newConfig) {
|
|
|
|
var id;
|
|
|
|
var node;
|
|
|
|
var nn;
|
|
|
|
var wires;
|
|
|
|
var j,k;
|
|
|
|
|
|
|
|
if (!oldConfig) {
|
|
|
|
oldConfig = {
|
|
|
|
flows:{},
|
|
|
|
allNodes:{}
|
2017-04-21 23:36:21 +01:00
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
|
|
|
var changedSubflows = {};
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-06-23 15:48:06 +01:00
|
|
|
var added = {};
|
|
|
|
var removed = {};
|
|
|
|
var changed = {};
|
2023-07-10 12:30:36 +01:00
|
|
|
var flowChanged = {};
|
2023-06-23 15:48:06 +01:00
|
|
|
var wiringChanged = {};
|
2023-07-10 12:04:52 +01:00
|
|
|
var globalConfigChanged = false;
|
2023-06-23 15:48:06 +01:00
|
|
|
var linkMap = {};
|
2023-07-10 12:04:52 +01:00
|
|
|
var allNestedGroups = []
|
2018-04-26 12:32:05 +01:00
|
|
|
|
2023-06-23 15:48:06 +01:00
|
|
|
// Look for tabs that have been removed
|
|
|
|
for (id in oldConfig.flows) {
|
|
|
|
if (oldConfig.flows.hasOwnProperty(id) && (!newConfig.flows.hasOwnProperty(id))) {
|
|
|
|
removed[id] = oldConfig.allNodes[id];
|
2017-05-11 15:39:41 +01:00
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
2017-03-06 15:23:31 +00:00
|
|
|
|
2023-06-23 15:48:06 +01:00
|
|
|
// Look for tabs that have been disabled
|
|
|
|
for (id in oldConfig.flows) {
|
|
|
|
if (oldConfig.flows.hasOwnProperty(id) && newConfig.flows.hasOwnProperty(id)) {
|
|
|
|
var originalState = oldConfig.flows[id].disabled||false;
|
|
|
|
var newState = newConfig.flows[id].disabled||false;
|
|
|
|
if (originalState !== newState) {
|
|
|
|
if (originalState) {
|
|
|
|
added[id] = oldConfig.allNodes[id];
|
|
|
|
} else {
|
|
|
|
removed[id] = oldConfig.allNodes[id];
|
2017-03-06 15:23:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
2017-03-06 15:23:31 +00:00
|
|
|
|
2023-06-23 15:48:06 +01:00
|
|
|
for (id in oldConfig.allNodes) {
|
|
|
|
if (oldConfig.allNodes.hasOwnProperty(id)) {
|
|
|
|
node = oldConfig.allNodes[id];
|
|
|
|
if (node.type !== 'tab') {
|
|
|
|
// build the map of what this node was previously wired to
|
|
|
|
if (node.wires) {
|
|
|
|
linkMap[node.id] = linkMap[node.id] || [];
|
|
|
|
for (j=0;j<node.wires.length;j++) {
|
|
|
|
wires = node.wires[j];
|
|
|
|
for (k=0;k<wires.length;k++) {
|
|
|
|
linkMap[node.id].push(wires[k]);
|
|
|
|
nn = oldConfig.allNodes[wires[k]];
|
|
|
|
if (nn) {
|
|
|
|
linkMap[nn.id] = linkMap[nn.id] || [];
|
|
|
|
linkMap[nn.id].push(node.id);
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
|
|
|
// This node has been removed or its flow disabled
|
|
|
|
if (removed[node.z] || !newConfig.allNodes.hasOwnProperty(id)) {
|
|
|
|
removed[id] = node;
|
|
|
|
// Mark the container as changed
|
|
|
|
if (!removed[node.z] && newConfig.allNodes[removed[id].z]) {
|
|
|
|
changed[removed[id].z] = newConfig.allNodes[removed[id].z];
|
|
|
|
if (changed[removed[id].z].type === "subflow") {
|
|
|
|
changedSubflows[removed[id].z] = changed[removed[id].z];
|
|
|
|
//delete removed[id];
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (added[node.z]) {
|
|
|
|
added[id] = node;
|
2017-03-06 15:23:31 +00:00
|
|
|
} else {
|
2023-06-23 15:48:06 +01:00
|
|
|
var currentState = node.d;
|
|
|
|
var newState = newConfig.allNodes[id].d;
|
|
|
|
if (!currentState && newState) {
|
|
|
|
removed[id] = node;
|
|
|
|
}
|
|
|
|
// This node has a material configuration change
|
|
|
|
if (diffNodes(node,newConfig.allNodes[id]) || newConfig.allNodes[id].credentials) {
|
|
|
|
changed[id] = newConfig.allNodes[id];
|
|
|
|
if (changed[id].type === "subflow") {
|
|
|
|
changedSubflows[id] = changed[id];
|
2020-06-08 20:59:00 +01:00
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
// Mark the container as changed
|
|
|
|
if (newConfig.allNodes[changed[id].z]) {
|
|
|
|
changed[changed[id].z] = newConfig.allNodes[changed[id].z];
|
|
|
|
if (changed[changed[id].z].type === "subflow") {
|
|
|
|
changedSubflows[changed[id].z] = changed[changed[id].z];
|
|
|
|
delete changed[id];
|
2017-03-06 15:23:31 +00:00
|
|
|
}
|
|
|
|
}
|
2023-07-10 12:04:52 +01:00
|
|
|
if (newConfig.allNodes[id].type === 'global-config') {
|
|
|
|
globalConfigChanged = true
|
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
|
|
|
// This node's wiring has changed
|
|
|
|
if (!redUtil.compareObjects(node.wires,newConfig.allNodes[id].wires)) {
|
|
|
|
wiringChanged[id] = newConfig.allNodes[id];
|
|
|
|
// Mark the container as changed
|
|
|
|
if (newConfig.allNodes[wiringChanged[id].z]) {
|
|
|
|
changed[wiringChanged[id].z] = newConfig.allNodes[wiringChanged[id].z];
|
|
|
|
if (changed[wiringChanged[id].z].type === "subflow") {
|
|
|
|
changedSubflows[wiringChanged[id].z] = changed[wiringChanged[id].z];
|
|
|
|
delete wiringChanged[id];
|
2017-03-06 15:23:31 +00:00
|
|
|
}
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-07-10 12:30:36 +01:00
|
|
|
} else if (!removed[id]) {
|
2023-07-10 12:04:52 +01:00
|
|
|
if (JSON.stringify(node.env) !== JSON.stringify(newConfig.allNodes[id].env)) {
|
2023-07-10 12:30:36 +01:00
|
|
|
flowChanged[id] = newConfig.allNodes[id];
|
2023-07-10 12:04:52 +01:00
|
|
|
}
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
|
|
|
// Look for added nodes
|
|
|
|
for (id in newConfig.allNodes) {
|
|
|
|
if (newConfig.allNodes.hasOwnProperty(id)) {
|
|
|
|
node = newConfig.allNodes[id];
|
2023-07-10 12:04:52 +01:00
|
|
|
if (node.type === 'group') {
|
|
|
|
if (node.g) {
|
|
|
|
allNestedGroups.push(node)
|
|
|
|
}
|
|
|
|
if (changed[node.id]) {
|
|
|
|
if (node.nodes) {
|
|
|
|
node.nodes.forEach(nid => {
|
|
|
|
if (!changed[nid]) {
|
|
|
|
changed[nid] = true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
// build the map of what this node is now wired to
|
|
|
|
if (node.wires) {
|
|
|
|
linkMap[node.id] = linkMap[node.id] || [];
|
|
|
|
for (j=0;j<node.wires.length;j++) {
|
|
|
|
wires = node.wires[j];
|
|
|
|
for (k=0;k<wires.length;k++) {
|
|
|
|
if (linkMap[node.id].indexOf(wires[k]) === -1) {
|
|
|
|
linkMap[node.id].push(wires[k]);
|
|
|
|
}
|
|
|
|
nn = newConfig.allNodes[wires[k]];
|
|
|
|
if (nn) {
|
|
|
|
linkMap[nn.id] = linkMap[nn.id] || [];
|
|
|
|
if (linkMap[nn.id].indexOf(node.id) === -1) {
|
|
|
|
linkMap[nn.id].push(node.id);
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
|
|
|
// This node has been added
|
|
|
|
if (!oldConfig.allNodes.hasOwnProperty(id)) {
|
|
|
|
added[id] = node;
|
|
|
|
// Mark the container as changed
|
|
|
|
if (newConfig.allNodes[added[id].z]) {
|
|
|
|
changed[added[id].z] = newConfig.allNodes[added[id].z];
|
|
|
|
if (changed[added[id].z].type === "subflow") {
|
|
|
|
changedSubflows[added[id].z] = changed[added[id].z];
|
|
|
|
delete added[id];
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-06-23 15:48:06 +01:00
|
|
|
var madeChange;
|
|
|
|
// Loop through the nodes looking for references to changed config nodes
|
|
|
|
// Repeat the loop if anything is marked as changed as it may need to be
|
|
|
|
// propagated to parent nodes.
|
|
|
|
// TODO: looping through all nodes every time is a bit inefficient - could be more targeted
|
|
|
|
do {
|
|
|
|
madeChange = false;
|
|
|
|
for (id in newConfig.allNodes) {
|
|
|
|
if (newConfig.allNodes.hasOwnProperty(id)) {
|
|
|
|
node = newConfig.allNodes[id];
|
|
|
|
for (var prop in node) {
|
|
|
|
if (node.hasOwnProperty(prop) && prop != "z" && prop != "id" && prop != "wires") {
|
|
|
|
// This node has a property that references a changed/removed node
|
|
|
|
// Assume it is a config node change and mark this node as
|
|
|
|
// changed.
|
|
|
|
|
|
|
|
var changeOrigin = changed[node[prop]];
|
|
|
|
if (changeOrigin || removed[node[prop]]) {
|
|
|
|
if (!changed[node.id]) {
|
|
|
|
if (changeOrigin &&
|
|
|
|
(prop === "g") &&
|
|
|
|
(changeOrigin.type === "group")) {
|
|
|
|
var oldNode = oldConfig.allNodes[node.id];
|
|
|
|
// ignore change of group node
|
|
|
|
// if group of this node not changed
|
|
|
|
if (oldNode &&
|
|
|
|
(node.g === oldNode.g)) {
|
|
|
|
continue;
|
2021-02-14 10:06:46 +09:00
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
|
|
|
madeChange = true;
|
|
|
|
changed[node.id] = node;
|
|
|
|
// This node exists within subflow template
|
|
|
|
// Mark the template as having changed
|
|
|
|
if (newConfig.allNodes[node.z]) {
|
|
|
|
changed[node.z] = newConfig.allNodes[node.z];
|
|
|
|
if (changed[node.z].type === "subflow") {
|
|
|
|
changedSubflows[node.z] = changed[node.z];
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
|
|
|
} while (madeChange===true)
|
|
|
|
|
|
|
|
// Find any nodes that exist on a subflow template and remove from changed
|
|
|
|
// list as the parent subflow will now be marked as containing a change
|
|
|
|
for (id in newConfig.allNodes) {
|
|
|
|
if (newConfig.allNodes.hasOwnProperty(id)) {
|
|
|
|
node = newConfig.allNodes[id];
|
|
|
|
if (newConfig.allNodes[node.z] && newConfig.allNodes[node.z].type === "subflow") {
|
|
|
|
delete changed[node.id];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-07-10 12:04:52 +01:00
|
|
|
// Recursively mark all children of changed groups as changed
|
|
|
|
do {
|
|
|
|
madeChange = false
|
|
|
|
for (let i = 0; i < allNestedGroups.length; i++) {
|
|
|
|
const group = allNestedGroups[i]
|
|
|
|
if (!changed[group.id] && group.g && changed[group.g]) {
|
|
|
|
changed[group.id] = true
|
|
|
|
madeChange = true
|
|
|
|
}
|
|
|
|
if (changed[group.id] && group.nodes) {
|
|
|
|
group.nodes.forEach(nid => {
|
|
|
|
if (!changed[nid]) {
|
|
|
|
changed[nid] = true
|
|
|
|
madeChange = true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while(madeChange)
|
|
|
|
|
2023-06-23 15:48:06 +01:00
|
|
|
// Recursively mark all instances of changed subflows as changed
|
|
|
|
var changedSubflowStack = Object.keys(changedSubflows);
|
|
|
|
while (changedSubflowStack.length > 0) {
|
|
|
|
var subflowId = changedSubflowStack.pop();
|
2016-04-27 10:30:52 +01:00
|
|
|
for (id in newConfig.allNodes) {
|
|
|
|
if (newConfig.allNodes.hasOwnProperty(id)) {
|
|
|
|
node = newConfig.allNodes[id];
|
2023-06-23 15:48:06 +01:00
|
|
|
if (node.type === 'subflow:'+subflowId) {
|
|
|
|
if (!changed[node.id]) {
|
|
|
|
changed[node.id] = node;
|
|
|
|
if (!changed[changed[node.id].z] && newConfig.allNodes[changed[node.id].z]) {
|
|
|
|
changed[changed[node.id].z] = newConfig.allNodes[changed[node.id].z];
|
|
|
|
if (newConfig.allNodes[changed[node.id].z].type === "subflow") {
|
|
|
|
// This subflow instance is inside a subflow. Add the
|
|
|
|
// containing subflow to the stack to mark
|
|
|
|
changedSubflowStack.push(changed[node.id].z);
|
|
|
|
delete changed[node.id];
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-07-10 12:04:52 +01:00
|
|
|
|
|
|
|
|
2023-06-23 15:48:06 +01:00
|
|
|
var diff = {
|
|
|
|
added:Object.keys(added),
|
|
|
|
changed:Object.keys(changed),
|
|
|
|
removed:Object.keys(removed),
|
|
|
|
rewired:Object.keys(wiringChanged),
|
2023-07-10 12:04:52 +01:00
|
|
|
linked:[],
|
2023-07-10 12:30:36 +01:00
|
|
|
flowChanged: Object.keys(flowChanged),
|
2023-07-10 12:04:52 +01:00
|
|
|
globalConfigChanged
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
2015-10-11 20:37:11 +01:00
|
|
|
|
2023-06-23 15:48:06 +01:00
|
|
|
// Traverse the links of all modified nodes to mark the connected nodes
|
|
|
|
var modifiedNodes = diff.added.concat(diff.changed).concat(diff.removed).concat(diff.rewired);
|
|
|
|
var visited = {};
|
|
|
|
while (modifiedNodes.length > 0) {
|
|
|
|
node = modifiedNodes.pop();
|
|
|
|
if (!visited[node]) {
|
|
|
|
visited[node] = true;
|
|
|
|
if (linkMap[node]) {
|
|
|
|
if (!changed[node] && !added[node] && !removed[node] && !wiringChanged[node]) {
|
|
|
|
diff.linked.push(node);
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
modifiedNodes = modifiedNodes.concat(linkMap[node]);
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|
|
|
|
}
|
2023-06-23 15:48:06 +01:00
|
|
|
}
|
|
|
|
// console.log(diff);
|
|
|
|
// for (id in newConfig.allNodes) {
|
2023-07-10 12:04:52 +01:00
|
|
|
// if (added[id] || changed[id] || wiringChanged[id] || diff.linked.indexOf(id)!==-1) {
|
|
|
|
// console.log(
|
|
|
|
// (added[id]?"a":(changed[id]?"c":" "))+(wiringChanged[id]?"w":" ")+(diff.linked.indexOf(id)!==-1?"l":" "),
|
|
|
|
// newConfig.allNodes[id].type.padEnd(10),
|
|
|
|
// id.padEnd(16),
|
|
|
|
// (newConfig.allNodes[id].z||"").padEnd(16),
|
|
|
|
// newConfig.allNodes[id].name||newConfig.allNodes[id].label||""
|
|
|
|
// );
|
|
|
|
// }
|
2023-06-23 15:48:06 +01:00
|
|
|
// }
|
|
|
|
// for (id in removed) {
|
|
|
|
// console.log(
|
|
|
|
// "- "+(diff.linked.indexOf(id)!==-1?"~":" "),
|
|
|
|
// id,
|
|
|
|
// oldConfig.allNodes[id].type,
|
|
|
|
// oldConfig.allNodes[id].name||oldConfig.allNodes[id].label||""
|
|
|
|
// );
|
|
|
|
// }
|
|
|
|
|
|
|
|
return diff;
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
init,
|
|
|
|
createNode,
|
|
|
|
parseConfig,
|
|
|
|
diffConfigs,
|
|
|
|
diffNodes,
|
|
|
|
getEnvVar,
|
|
|
|
mapEnvVarProperties,
|
2023-06-23 02:11:57 +01:00
|
|
|
evaluateEnvProperties
|
2015-10-11 20:37:11 +01:00
|
|
|
}
|