2013-09-05 15:02:48 +01:00
/ * *
* Copyright 2013 IBM Corp .
*
* 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 .
* * /
2014-08-08 00:01:35 +01:00
RED . nodes = ( function ( ) {
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
var node _defs = { } ;
var nodes = [ ] ;
var configNodes = { } ;
var links = [ ] ;
2013-10-25 21:34:00 +01:00
var defaultWorkspace ;
var workspaces = { } ;
2014-08-28 00:35:07 +01:00
var registry = ( function ( ) {
var nodeList = [ ] ;
var nodeSets = { } ;
var typeToId = { } ;
var nodeDefinitions = { } ;
var exports = {
getNodeList : function ( ) {
return nodeList ;
} ,
setNodeList : function ( list ) {
nodeList = [ ] ;
for ( var i = 0 ; i < list . length ; i ++ ) {
var ns = list [ i ] ;
exports . addNodeSet ( ns ) ;
}
} ,
addNodeSet : function ( ns ) {
ns . added = false ;
nodeSets [ ns . id ] = ns ;
for ( var j = 0 ; j < ns . types . length ; j ++ ) {
typeToId [ ns . types [ j ] ] = ns . id ;
}
nodeList . push ( ns ) ;
} ,
removeNodeSet : function ( id ) {
var ns = nodeSets [ id ] ;
for ( var j = 0 ; j < ns . types . length ; j ++ ) {
if ( ns . added ) {
// TODO: too tightly coupled into palette UI
RED . palette . remove ( ns . types [ j ] ) ;
var def = nodeDefinitions [ ns . types [ j ] ] ;
if ( def . onpaletteremove && typeof def . onpaletteremove === "function" ) {
def . onpaletteremove . call ( def ) ;
}
}
delete typeToId [ ns . types [ j ] ] ;
}
delete nodeSets [ id ] ;
for ( var i = 0 ; i < nodeList . length ; i ++ ) {
if ( nodeList [ i ] . id == id ) {
nodeList . splice ( i , 1 ) ;
break ;
}
}
return ns ;
} ,
getNodeSet : function ( id ) {
return nodeSets [ id ] ;
} ,
enableNodeSet : function ( id ) {
var ns = nodeSets [ id ] ;
ns . enabled = true ;
for ( var j = 0 ; j < ns . types . length ; j ++ ) {
// TODO: too tightly coupled into palette UI
RED . palette . show ( ns . types [ j ] ) ;
var def = nodeDefinitions [ ns . types [ j ] ] ;
if ( def . onpaletteadd && typeof def . onpaletteadd === "function" ) {
def . onpaletteadd . call ( def ) ;
}
}
} ,
disableNodeSet : function ( id ) {
var ns = nodeSets [ id ] ;
ns . enabled = false ;
for ( var j = 0 ; j < ns . types . length ; j ++ ) {
// TODO: too tightly coupled into palette UI
RED . palette . hide ( ns . types [ j ] ) ;
var def = nodeDefinitions [ ns . types [ j ] ] ;
if ( def . onpaletteremove && typeof def . onpaletteremove === "function" ) {
def . onpaletteremove . call ( def ) ;
}
}
} ,
registerNodeType : function ( nt , def ) {
nodeDefinitions [ nt ] = def ;
nodeSets [ typeToId [ nt ] ] . added = true ;
// TODO: too tightly coupled into palette UI
RED . palette . add ( nt , def ) ;
if ( def . onpaletteadd && typeof def . onpaletteadd === "function" ) {
def . onpaletteadd . call ( def ) ;
}
} ,
getNodeType : function ( nt ) {
return nodeDefinitions [ nt ] ;
}
}
return exports ;
} ) ( ) ;
2013-10-25 21:34:00 +01:00
function getID ( ) {
return ( 1 + Math . random ( ) * 4294967295 ) . toString ( 16 ) ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
function addNode ( n ) {
if ( n . _def . category == "config" ) {
configNodes [ n . id ] = n ;
2014-02-26 22:58:44 +00:00
RED . sidebar . config . refresh ( ) ;
2013-09-05 15:02:48 +01:00
} else {
n . dirty = true ;
nodes . push ( n ) ;
2014-01-25 22:31:43 +00:00
var updatedConfigNode = false ;
for ( var d in n . _def . defaults ) {
2014-08-08 00:01:35 +01:00
if ( n . _def . defaults . hasOwnProperty ( d ) ) {
var property = n . _def . defaults [ d ] ;
if ( property . type ) {
2014-08-28 00:35:07 +01:00
var type = registry . getNodeType ( property . type )
2014-08-08 00:01:35 +01:00
if ( type && type . category == "config" ) {
var configNode = configNodes [ n [ d ] ] ;
if ( configNode ) {
updatedConfigNode = true ;
configNode . users . push ( n ) ;
}
2014-01-25 22:31:43 +00:00
}
}
}
}
if ( updatedConfigNode ) {
2014-02-26 22:58:44 +00:00
RED . sidebar . config . refresh ( ) ;
2014-01-25 22:31:43 +00:00
}
2013-09-05 15:02:48 +01:00
}
}
function addLink ( l ) {
links . push ( l ) ;
}
function addConfig ( c ) {
configNodes [ c . id ] = c ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
function getNode ( id ) {
if ( id in configNodes ) {
return configNodes [ id ] ;
} else {
for ( var n in nodes ) {
if ( nodes [ n ] . id == id ) {
return nodes [ n ] ;
}
}
}
return null ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
function removeNode ( id ) {
var removedLinks = [ ] ;
if ( id in configNodes ) {
delete configNodes [ id ] ;
2014-02-27 16:47:28 +00:00
RED . sidebar . config . refresh ( ) ;
2013-09-05 15:02:48 +01:00
} else {
var node = getNode ( id ) ;
if ( node ) {
nodes . splice ( nodes . indexOf ( node ) , 1 ) ;
removedLinks = links . filter ( function ( l ) { return ( l . source === node ) || ( l . target === node ) ; } ) ;
removedLinks . map ( function ( l ) { links . splice ( links . indexOf ( l ) , 1 ) ; } ) ;
}
2014-01-25 22:31:43 +00:00
var updatedConfigNode = false ;
for ( var d in node . _def . defaults ) {
2014-08-08 00:01:35 +01:00
if ( node . _def . defaults . hasOwnProperty ( d ) ) {
var property = node . _def . defaults [ d ] ;
if ( property . type ) {
2014-08-28 00:35:07 +01:00
var type = registry . getNodeType ( property . type )
2014-08-08 00:01:35 +01:00
if ( type && type . category == "config" ) {
var configNode = configNodes [ node [ d ] ] ;
if ( configNode ) {
updatedConfigNode = true ;
var users = configNode . users ;
users . splice ( users . indexOf ( node ) , 1 ) ;
}
2014-01-25 22:31:43 +00:00
}
}
}
}
if ( updatedConfigNode ) {
2014-02-27 16:47:28 +00:00
RED . sidebar . config . refresh ( ) ;
2014-01-25 22:31:43 +00:00
}
2013-09-05 15:02:48 +01:00
}
return removedLinks ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
function removeLink ( l ) {
var index = links . indexOf ( l ) ;
if ( index != - 1 ) {
links . splice ( index , 1 ) ;
}
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
function refreshValidation ( ) {
2014-08-08 00:01:35 +01:00
for ( var n = 0 ; n < nodes . length ; n ++ ) {
2013-09-05 15:02:48 +01:00
RED . editor . validateNode ( nodes [ n ] ) ;
}
}
2013-12-08 21:55:34 +00:00
2013-10-25 21:34:00 +01:00
function addWorkspace ( ws ) {
workspaces [ ws . id ] = ws ;
}
2013-10-26 22:29:24 +01:00
function getWorkspace ( id ) {
return workspaces [ id ] ;
}
function removeWorkspace ( id ) {
delete workspaces [ id ] ;
var removedNodes = [ ] ;
var removedLinks = [ ] ;
2014-08-08 00:01:35 +01:00
var n ;
for ( n = 0 ; n < nodes . length ; n ++ ) {
2013-10-26 22:29:24 +01:00
var node = nodes [ n ] ;
if ( node . z == id ) {
removedNodes . push ( node ) ;
}
}
2014-08-08 00:01:35 +01:00
for ( n = 0 ; n < removedNodes . length ; n ++ ) {
2013-10-26 22:29:24 +01:00
var rmlinks = removeNode ( removedNodes [ n ] . id ) ;
removedLinks = removedLinks . concat ( rmlinks ) ;
}
return { nodes : removedNodes , links : removedLinks } ;
}
2013-12-11 22:22:33 +00:00
2013-09-05 15:02:48 +01:00
function getAllFlowNodes ( node ) {
var visited = { } ;
visited [ node . id ] = true ;
var nns = [ node ] ;
var stack = [ node ] ;
2014-08-08 00:01:35 +01:00
while ( stack . length !== 0 ) {
2013-09-05 15:02:48 +01:00
var n = stack . shift ( ) ;
var childLinks = links . filter ( function ( d ) { return ( d . source === n ) || ( d . target === n ) ; } ) ;
2014-08-08 00:01:35 +01:00
for ( var i = 0 ; i < childLinks . length ; i ++ ) {
2013-09-05 15:02:48 +01:00
var child = ( childLinks [ i ] . source === n ) ? childLinks [ i ] . target : childLinks [ i ] . source ;
if ( ! visited [ child . id ] ) {
visited [ child . id ] = true ;
nns . push ( child ) ;
stack . push ( child ) ;
}
}
}
return nns ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
/ * *
2013-12-08 21:55:34 +00:00
* Converts a node to an exportable JSON Object
2013-09-05 15:02:48 +01:00
* * /
2014-06-24 13:13:19 +03:00
function convertNode ( n , exportCreds ) {
exportCreds = exportCreds || false ;
2013-09-05 15:02:48 +01:00
var node = { } ;
node . id = n . id ;
node . type = n . type ;
for ( var d in n . _def . defaults ) {
2014-08-08 00:01:35 +01:00
if ( n . _def . defaults . hasOwnProperty ( d ) ) {
node [ d ] = n [ d ] ;
}
2013-09-05 15:02:48 +01:00
}
2014-07-19 00:16:21 +01:00
if ( exportCreds && n . credentials ) {
node . credentials = { } ;
for ( var cred in n . _def . credentials ) {
if ( n . _def . credentials . hasOwnProperty ( cred ) ) {
if ( n . credentials [ cred ] != null ) {
node . credentials [ cred ] = n . credentials [ cred ] ;
}
}
}
2014-06-24 13:13:19 +03:00
}
2013-09-05 15:02:48 +01:00
if ( n . _def . category != "config" ) {
node . x = n . x ;
node . y = n . y ;
2013-10-20 23:11:38 +01:00
node . z = n . z ;
2013-09-05 15:02:48 +01:00
node . wires = [ ] ;
for ( var i = 0 ; i < n . outputs ; i ++ ) {
node . wires . push ( [ ] ) ;
}
var wires = links . filter ( function ( d ) { return d . source === n ; } ) ;
2014-08-08 00:01:35 +01:00
for ( var j = 0 ; j < wires . length ; j ++ ) {
var w = wires [ j ] ;
2013-09-05 15:02:48 +01:00
node . wires [ w . sourcePort ] . push ( w . target . id ) ;
}
}
return node ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
/ * *
* Converts the current node selection to an exportable JSON Object
* * /
function createExportableNodeSet ( set ) {
var nns = [ ] ;
var exportedConfigNodes = { } ;
2014-08-08 00:01:35 +01:00
for ( var n = 0 ; n < set . length ; n ++ ) {
2013-09-05 15:02:48 +01:00
var node = set [ n ] . n ;
var convertedNode = RED . nodes . convertNode ( node ) ;
for ( var d in node . _def . defaults ) {
if ( node . _def . defaults [ d ] . type && node [ d ] in configNodes ) {
var confNode = configNodes [ node [ d ] ] ;
2014-08-28 00:35:07 +01:00
var exportable = registry . getNodeType ( node . _def . defaults [ d ] . type ) . exportable ;
2013-09-05 15:02:48 +01:00
if ( ( exportable == null || exportable ) ) {
if ( ! ( node [ d ] in exportedConfigNodes ) ) {
exportedConfigNodes [ node [ d ] ] = true ;
nns . unshift ( RED . nodes . convertNode ( confNode ) ) ;
}
} else {
convertedNode [ d ] = "" ;
}
}
}
nns . push ( convertedNode ) ;
}
return nns ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
//TODO: rename this (createCompleteNodeSet)
function createCompleteNodeSet ( ) {
var nns = [ ] ;
2014-08-08 00:01:35 +01:00
var i ;
for ( i in workspaces ) {
if ( workspaces . hasOwnProperty ( i ) ) {
nns . push ( workspaces [ i ] ) ;
}
2013-10-25 21:34:00 +01:00
}
2014-08-08 00:01:35 +01:00
for ( i in configNodes ) {
if ( configNodes . hasOwnProperty ( i ) ) {
nns . push ( convertNode ( configNodes [ i ] , true ) ) ;
}
2013-09-05 15:02:48 +01:00
}
2014-08-08 00:01:35 +01:00
for ( i = 0 ; i < nodes . length ; i ++ ) {
2013-09-05 15:02:48 +01:00
var node = nodes [ i ] ;
2014-06-24 13:13:19 +03:00
nns . push ( convertNode ( node , true ) ) ;
2013-09-05 15:02:48 +01:00
}
return nns ;
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
function importNodes ( newNodesObj , createNewIds ) {
try {
2014-08-08 00:01:35 +01:00
var i ;
var n ;
2013-09-05 15:02:48 +01:00
var newNodes ;
if ( typeof newNodesObj === "string" ) {
2014-08-08 00:01:35 +01:00
if ( newNodesObj === "" ) {
2013-09-05 15:02:48 +01:00
return ;
}
newNodes = JSON . parse ( newNodesObj ) ;
} else {
newNodes = newNodesObj ;
}
if ( ! $ . isArray ( newNodes ) ) {
newNodes = [ newNodes ] ;
}
2013-12-12 15:51:15 +00:00
var unknownTypes = [ ] ;
2014-08-08 00:01:35 +01:00
for ( i = 0 ; i < newNodes . length ; i ++ ) {
n = newNodes [ i ] ;
2013-10-30 21:45:45 +00:00
// TODO: remove workspace in next release+1
2014-08-28 00:35:07 +01:00
if ( n . type != "workspace" && n . type != "tab" && ! registry . getNodeType ( n . type ) ) {
2013-09-05 15:02:48 +01:00
// TODO: get this UI thing out of here! (see below as well)
2013-12-12 15:51:15 +00:00
n . name = n . type ;
n . type = "unknown" ;
if ( unknownTypes . indexOf ( n . name ) == - 1 ) {
unknownTypes . push ( n . name ) ;
}
2014-04-28 21:40:32 +01:00
if ( n . x == null && n . y == null ) {
// config node - remove it
newNodes . splice ( i , 1 ) ;
i -- ;
}
2013-12-12 15:51:15 +00:00
}
2013-09-05 15:02:48 +01:00
}
2013-12-12 15:51:15 +00:00
if ( unknownTypes . length > 0 ) {
var typeList = "<ul><li>" + unknownTypes . join ( "</li><li>" ) + "</li></ul>" ;
var type = "type" + ( unknownTypes . length > 1 ? "s" : "" ) ;
RED . notify ( "<strong>Imported unrecognised " + type + ":</strong>" + typeList , "error" , false , 10000 ) ;
//"DO NOT DEPLOY while in this state.<br/>Either, add missing types to Node-RED, restart and then reload page,<br/>or delete unknown "+n.name+", rewire as required, and then deploy.","error");
}
2014-09-08 10:53:18 +01:00
var new _workspaces = [ ] ;
var workspace _map = { } ;
2014-08-08 00:01:35 +01:00
for ( i = 0 ; i < newNodes . length ; i ++ ) {
n = newNodes [ i ] ;
2013-10-30 21:45:45 +00:00
// TODO: remove workspace in next release+1
if ( n . type === "workspace" || n . type === "tab" ) {
if ( n . type === "workspace" ) {
n . type = "tab" ;
}
2013-10-25 21:34:00 +01:00
if ( defaultWorkspace == null ) {
defaultWorkspace = n ;
}
2014-09-08 10:53:18 +01:00
if ( createNewIds ) {
var nid = getID ( ) ;
workspace _map [ n . id ] = nid ;
n . id = nid ;
}
2013-10-25 21:34:00 +01:00
addWorkspace ( n ) ;
RED . view . addWorkspace ( n ) ;
2014-09-08 10:53:18 +01:00
new _workspaces . push ( n ) ;
2013-10-25 21:34:00 +01:00
}
}
if ( defaultWorkspace == null ) {
2013-10-30 21:45:45 +00:00
defaultWorkspace = { type : "tab" , id : getID ( ) , label : "Sheet 1" } ;
2013-10-25 21:34:00 +01:00
addWorkspace ( defaultWorkspace ) ;
RED . view . addWorkspace ( defaultWorkspace ) ;
2014-09-08 10:53:18 +01:00
new _workspaces . push ( defaultWorkspace ) ;
2013-10-25 21:34:00 +01:00
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
var node _map = { } ;
var new _nodes = [ ] ;
var new _links = [ ] ;
2013-12-08 21:55:34 +00:00
2014-08-08 00:01:35 +01:00
for ( i = 0 ; i < newNodes . length ; i ++ ) {
n = newNodes [ i ] ;
2013-10-30 21:45:45 +00:00
// TODO: remove workspace in next release+1
if ( n . type !== "workspace" && n . type !== "tab" ) {
2014-08-28 00:35:07 +01:00
var def = registry . getNodeType ( n . type ) ;
2013-10-25 21:34:00 +01:00
if ( def && def . category == "config" ) {
if ( ! RED . nodes . node ( n . id ) ) {
var configNode = { id : n . id , type : n . type , users : [ ] } ;
for ( var d in def . defaults ) {
2014-08-08 00:01:35 +01:00
if ( def . defaults . hasOwnProperty ( d ) ) {
configNode [ d ] = n [ d ] ;
}
2013-10-25 21:34:00 +01:00
}
configNode . label = def . label ;
configNode . _def = def ;
RED . nodes . add ( configNode ) ;
2013-09-05 15:02:48 +01:00
}
} else {
2013-11-15 23:40:36 +00:00
var node = { x : n . x , y : n . y , z : n . z , type : 0 , wires : n . wires , changed : false } ;
2013-10-25 21:34:00 +01:00
if ( createNewIds ) {
2014-09-08 10:53:18 +01:00
node . z = workspace _map [ node . z ] ;
if ( ! workspaces [ node . z ] ) {
node . z = RED . view . getWorkspace ( ) ;
}
2013-10-25 21:34:00 +01:00
node . id = getID ( ) ;
} else {
node . id = n . id ;
if ( node . z == null || ! workspaces [ node . z ] ) {
node . z = RED . view . getWorkspace ( ) ;
}
2013-09-05 15:02:48 +01:00
}
2013-10-25 21:34:00 +01:00
node . type = n . type ;
node . _def = def ;
if ( ! node . _def ) {
node . _def = {
color : "#fee" ,
defaults : { } ,
label : "unknown: " + n . type ,
labelStyle : "node_label_italic" ,
outputs : n . outputs || n . wires . length
}
}
node . outputs = n . outputs || node . _def . outputs ;
2013-12-08 21:55:34 +00:00
2014-08-08 00:01:35 +01:00
for ( var d2 in node . _def . defaults ) {
if ( node . _def . defaults . hasOwnProperty ( d2 ) ) {
node [ d2 ] = n [ d2 ] ;
}
2013-09-05 15:02:48 +01:00
}
2013-12-08 21:55:34 +00:00
2013-10-25 21:34:00 +01:00
addNode ( node ) ;
RED . editor . validateNode ( node ) ;
node _map [ n . id ] = node ;
new _nodes . push ( node ) ;
2013-09-05 15:02:48 +01:00
}
}
}
2014-08-08 00:01:35 +01:00
for ( i = 0 ; i < new _nodes . length ; i ++ ) {
n = new _nodes [ i ] ;
for ( var w1 = 0 ; w1 < n . wires . length ; w1 ++ ) {
2013-09-05 15:02:48 +01:00
var wires = ( n . wires [ w1 ] instanceof Array ) ? n . wires [ w1 ] : [ n . wires [ w1 ] ] ;
2014-08-08 00:01:35 +01:00
for ( var w2 = 0 ; w2 < wires . length ; w2 ++ ) {
2013-09-05 15:02:48 +01:00
if ( wires [ w2 ] in node _map ) {
var link = { source : n , sourcePort : w1 , target : node _map [ wires [ w2 ] ] } ;
addLink ( link ) ;
new _links . push ( link ) ;
}
}
}
delete n . wires ;
}
2014-09-08 10:53:18 +01:00
return [ new _nodes , new _links , new _workspaces ] ;
2013-09-05 15:02:48 +01:00
} catch ( error ) {
//TODO: get this UI thing out of here! (see above as well)
RED . notify ( "<strong>Error</strong>: " + error , "error" ) ;
return null ;
}
}
2013-12-08 21:55:34 +00:00
2013-09-05 15:02:48 +01:00
return {
2014-08-28 00:35:07 +01:00
registry : registry ,
setNodeList : registry . setNodeList ,
getNodeSet : registry . getNodeSet ,
addNodeSet : registry . addNodeSet ,
removeNodeSet : registry . removeNodeSet ,
enableNodeSet : registry . enableNodeSet ,
disableNodeSet : registry . disableNodeSet ,
registerType : registry . registerNodeType ,
getType : registry . getNodeType ,
2013-09-05 15:02:48 +01:00
convertNode : convertNode ,
add : addNode ,
addLink : addLink ,
remove : removeNode ,
removeLink : removeLink ,
2013-10-25 21:34:00 +01:00
addWorkspace : addWorkspace ,
2013-10-26 22:29:24 +01:00
removeWorkspace : removeWorkspace ,
workspace : getWorkspace ,
2013-09-05 15:02:48 +01:00
eachNode : function ( cb ) {
2014-08-08 00:01:35 +01:00
for ( var n = 0 ; n < nodes . length ; n ++ ) {
2013-09-05 15:02:48 +01:00
cb ( nodes [ n ] ) ;
}
} ,
eachLink : function ( cb ) {
2014-08-08 00:01:35 +01:00
for ( var l = 0 ; l < links . length ; l ++ ) {
2013-09-05 15:02:48 +01:00
cb ( links [ l ] ) ;
}
} ,
eachConfig : function ( cb ) {
for ( var id in configNodes ) {
2014-08-08 00:01:35 +01:00
if ( configNodes . hasOwnProperty ( id ) ) {
cb ( configNodes [ id ] ) ;
}
2013-09-05 15:02:48 +01:00
}
} ,
node : getNode ,
import : importNodes ,
refreshValidation : refreshValidation ,
getAllFlowNodes : getAllFlowNodes ,
createExportableNodeSet : createExportableNodeSet ,
createCompleteNodeSet : createCompleteNodeSet ,
2013-10-25 21:34:00 +01:00
id : getID ,
2013-09-05 15:02:48 +01:00
nodes : nodes , // TODO: exposed for d3 vis
links : links // TODO: exposed for d3 vis
} ;
2014-08-08 00:01:35 +01:00
} ) ( ) ;