mirror of
https://github.com/Ylianst/MeshCentral.git
synced 2025-01-25 21:53:14 -05:00
Improved MongoDB change stream.
This commit is contained in:
parent
88b0b8601c
commit
bbd0f5efda
46
db.js
46
db.js
@ -231,12 +231,38 @@ module.exports.CreateDB = function (parent, func) {
|
|||||||
|
|
||||||
// Setup the changeStream on the MongoDB main collection if possible
|
// Setup the changeStream on the MongoDB main collection if possible
|
||||||
if (parent.args.mongodbchangestream == true) {
|
if (parent.args.mongodbchangestream == true) {
|
||||||
obj.fileChangeStream = obj.file.watch([{ $match: { 'fullDocument.type': { $in: ['node', 'mesh', 'user'] } } }], { fullDocument: 'updateLookup' });
|
obj.fileChangeStream = obj.file.watch( [ { $match: { $or: [{ 'fullDocument.type': { $in: ['node', 'mesh', 'user'] } }, { 'operationType': 'delete' }] } } ], { fullDocument: 'updateLookup' });
|
||||||
obj.fileChangeStream.on('change', function (change) {
|
obj.fileChangeStream.on('change', function (change) {
|
||||||
|
if (change.operationType == 'update') {
|
||||||
switch (change.fullDocument.type) {
|
switch (change.fullDocument.type) {
|
||||||
case 'node': { dbNodeChange(change); break; } // A node has changed
|
case 'node': { dbNodeChange(change, false); break; } // A node has changed
|
||||||
case 'mesh': { dbMeshChange(change); break; } // A device group has changed
|
case 'mesh': { dbMeshChange(change, false); break; } // A device group has changed
|
||||||
case 'user': { dbUserChange(change); break; } // A user account has changed
|
case 'user': { dbUserChange(change, false); break; } // A user account has changed
|
||||||
|
}
|
||||||
|
} else if (change.operationType == 'insert') {
|
||||||
|
switch (change.fullDocument.type) {
|
||||||
|
case 'node': { dbNodeChange(change, true); break; } // A node has added
|
||||||
|
case 'mesh': { dbMeshChange(change, true); break; } // A device group has created
|
||||||
|
case 'user': { dbUserChange(change, true); break; } // A user account has created
|
||||||
|
}
|
||||||
|
} else if (change.operationType == 'delete') {
|
||||||
|
var splitId = change.documentKey._id.split('/');
|
||||||
|
switch (splitId[0]) {
|
||||||
|
case 'node': {
|
||||||
|
//Not Good: Problem here is that we don't know what meshid the node belonged to before the delete.
|
||||||
|
//parent.DispatchEvent(['*', node.meshid], obj, { etype: 'node', action: 'removenode', nodeid: change.documentKey._id, domain: splitId[1] });
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'mesh': {
|
||||||
|
parent.DispatchEvent(['*', node.meshid], obj, { etype: 'mesh', action: 'deletemesh', meshid: change.documentKey._id, domain: splitId[1] });
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'user': {
|
||||||
|
//Not Good: This is not a perfect user removal because we don't know what groups the user was in.
|
||||||
|
//parent.DispatchEvent(['*', 'server-users'], obj, { etype: 'user', action: 'accountremove', userid: change.documentKey._id, domain: splitId[1], username: splitId[2] });
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
obj.changeStream = true;
|
obj.changeStream = true;
|
||||||
@ -773,16 +799,16 @@ module.exports.CreateDB = function (parent, func) {
|
|||||||
function padNumber(number, digits) { return Array(Math.max(digits - String(number).length + 1, 0)).join(0) + number; }
|
function padNumber(number, digits) { return Array(Math.max(digits - String(number).length + 1, 0)).join(0) + number; }
|
||||||
|
|
||||||
// Called when a node has changed
|
// Called when a node has changed
|
||||||
function dbNodeChange(nodeChange) {
|
function dbNodeChange(nodeChange, added) {
|
||||||
const node = nodeChange.fullDocument;
|
const node = nodeChange.fullDocument;
|
||||||
if (node.intelamt && node.intelamt.pass) { delete node.intelamt.pass; } // Remove the Intel AMT password before eventing this.
|
if (node.intelamt && node.intelamt.pass) { delete node.intelamt.pass; } // Remove the Intel AMT password before eventing this.
|
||||||
parent.DispatchEvent(['*', node.meshid], obj, { etype: 'node', action: 'changenode', node: node, nodeid: node._id, domain: node.domain, nolog: 1 });
|
parent.DispatchEvent(['*', node.meshid], obj, { etype: 'node', action: (added ? 'addnode' : 'changenode'), node: node, nodeid: node._id, domain: node.domain, nolog: 1 });
|
||||||
}
|
}
|
||||||
|
|
||||||
// Called when a device group has changed
|
// Called when a device group has changed
|
||||||
function dbMeshChange(meshChange) {
|
function dbMeshChange(meshChange, added) {
|
||||||
const mesh = meshChange.fullDocument;
|
const mesh = meshChange.fullDocument;
|
||||||
mesh.action = 'meshchange';
|
if (mesh.deleted) { mesh.action = 'deletemesh'; } else { mesh.action = (added ? 'createmesh' : 'meshchange'); }
|
||||||
mesh.meshid = mesh._id;
|
mesh.meshid = mesh._id;
|
||||||
mesh.nolog = 1;
|
mesh.nolog = 1;
|
||||||
delete mesh.type;
|
delete mesh.type;
|
||||||
@ -791,9 +817,9 @@ module.exports.CreateDB = function (parent, func) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Called when a user account has changed
|
// Called when a user account has changed
|
||||||
function dbUserChange(userChange) {
|
function dbUserChange(userChange, added) {
|
||||||
const user = userChange.fullDocument;
|
const user = userChange.fullDocument;
|
||||||
parent.DispatchEvent(['*', 'server-users', user._id], obj, { etype: 'user', username: user.name, account: parent.webserver.CloneSafeUser(user), action: 'accountchange', domain: user.domain, nolog: 1 });
|
parent.DispatchEvent(['*', 'server-users', user._id], obj, { etype: 'user', username: user.name, account: parent.webserver.CloneSafeUser(user), action: (added ? 'accountcreate' : 'accountchange'), domain: user.domain, nolog: 1 });
|
||||||
}
|
}
|
||||||
|
|
||||||
return obj;
|
return obj;
|
||||||
|
29
meshuser.js
29
meshuser.js
@ -1038,13 +1038,15 @@ module.exports.CreateMeshUser = function (parent, db, ws, req, args, domain, use
|
|||||||
newuser.hash = hash;
|
newuser.hash = hash;
|
||||||
db.SetUser(newuser);
|
db.SetUser(newuser);
|
||||||
|
|
||||||
var targets = ['*', 'server-users'];
|
var event, targets = ['*', 'server-users'];
|
||||||
if (newuser.groups) { for (var i in newuser.groups) { targets.push('server-users:' + i); } }
|
if (newuser.groups) { for (var i in newuser.groups) { targets.push('server-users:' + i); } }
|
||||||
if (newuser.email == null) {
|
if (newuser.email == null) {
|
||||||
parent.parent.DispatchEvent(targets, obj, { etype: 'user', username: newuser.name, account: parent.CloneSafeUser(newuser), action: 'accountcreate', msg: 'Account created, username is ' + newuser.name, domain: domain.id });
|
event = { etype: 'user', username: newuser.name, account: parent.CloneSafeUser(newuser), action: 'accountcreate', msg: 'Account created, username is ' + newuser.name, domain: domain.id };
|
||||||
} else {
|
} else {
|
||||||
parent.parent.DispatchEvent(targets, obj, { etype: 'user', username: newuser.name, account: parent.CloneSafeUser(newuser), action: 'accountcreate', msg: 'Account created, email is ' + newuser.email, domain: domain.id });
|
event = { etype: 'user', username: newuser.name, account: parent.CloneSafeUser(newuser), action: 'accountcreate', msg: 'Account created, email is ' + newuser.email, domain: domain.id };
|
||||||
}
|
}
|
||||||
|
if (parent.db.changeStream) { event.noact = 1; } // If DB change stream is active, don't use this event to create the user. Another event will come.
|
||||||
|
parent.parent.DispatchEvent(targets, obj, event);
|
||||||
}, newuser);
|
}, newuser);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1094,13 +1096,15 @@ module.exports.CreateMeshUser = function (parent, db, ws, req, args, domain, use
|
|||||||
newuser.hash = hash;
|
newuser.hash = hash;
|
||||||
db.SetUser(newuser);
|
db.SetUser(newuser);
|
||||||
|
|
||||||
var targets = ['*', 'server-users'];
|
var event, targets = ['*', 'server-users'];
|
||||||
if (newuser.groups) { for (var i in newuser.groups) { targets.push('server-users:' + i); } }
|
if (newuser.groups) { for (var i in newuser.groups) { targets.push('server-users:' + i); } }
|
||||||
if (command.email == null) {
|
if (command.email == null) {
|
||||||
parent.parent.DispatchEvent(targets, obj, { etype: 'user', username: newusername, account: parent.CloneSafeUser(newuser), action: 'accountcreate', msg: 'Account created, username is ' + command.user, domain: domain.id });
|
event = { etype: 'user', username: newusername, account: parent.CloneSafeUser(newuser), action: 'accountcreate', msg: 'Account created, username is ' + command.user, domain: domain.id };
|
||||||
} else {
|
} else {
|
||||||
parent.parent.DispatchEvent(targets, obj, { etype: 'user', username: newusername, account: parent.CloneSafeUser(newuser), action: 'accountcreate', msg: 'Account created, email is ' + command.email, domain: domain.id });
|
event = { etype: 'user', username: newusername, account: parent.CloneSafeUser(newuser), action: 'accountcreate', msg: 'Account created, email is ' + command.email, domain: domain.id };
|
||||||
}
|
}
|
||||||
|
if (parent.db.changeStream) { event.noact = 1; } // If DB change stream is active, don't use this event to create the user. Another event will come.
|
||||||
|
parent.parent.DispatchEvent(targets, obj, event);
|
||||||
}, 0);
|
}, 0);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -1390,7 +1394,9 @@ module.exports.CreateMeshUser = function (parent, db, ws, req, args, domain, use
|
|||||||
user.links[meshid] = { rights: 0xFFFFFFFF };
|
user.links[meshid] = { rights: 0xFFFFFFFF };
|
||||||
user.subscriptions = parent.subscribe(user._id, ws);
|
user.subscriptions = parent.subscribe(user._id, ws);
|
||||||
db.SetUser(user);
|
db.SetUser(user);
|
||||||
parent.parent.DispatchEvent(['*', meshid, user._id], obj, { etype: 'mesh', username: user.name, meshid: meshid, name: command.meshname, mtype: command.meshtype, desc: command.desc, action: 'createmesh', links: links, msg: 'Mesh created: ' + command.meshname, domain: domain.id });
|
var event = { etype: 'mesh', username: user.name, meshid: meshid, name: command.meshname, mtype: command.meshtype, desc: command.desc, action: 'createmesh', links: links, msg: 'Mesh created: ' + command.meshname, domain: domain.id };
|
||||||
|
if (db.changeStream) { event.noact = 1; } // If DB change stream is active, don't use this event to create the mesh. Another event will come.
|
||||||
|
parent.parent.DispatchEvent(['*', meshid, user._id], obj, event);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@ -1408,7 +1414,9 @@ module.exports.CreateMeshUser = function (parent, db, ws, req, args, domain, use
|
|||||||
if ((command.meshid.split('/').length != 3) || (command.meshid.split('/')[1] != domain.id)) return; // Invalid domain, operation only valid for current domain
|
if ((command.meshid.split('/').length != 3) || (command.meshid.split('/')[1] != domain.id)) return; // Invalid domain, operation only valid for current domain
|
||||||
|
|
||||||
// Fire the removal event first, because after this, the event will not route
|
// Fire the removal event first, because after this, the event will not route
|
||||||
parent.parent.DispatchEvent(['*', command.meshid], obj, { etype: 'mesh', username: user.name, meshid: command.meshid, name: command.meshname, action: 'deletemesh', msg: 'Mesh deleted: ' + command.meshname, domain: domain.id });
|
var event = { etype: 'mesh', username: user.name, meshid: command.meshid, name: command.meshname, action: 'deletemesh', msg: 'Mesh deleted: ' + command.meshname, domain: domain.id };
|
||||||
|
if (db.changeStream) { event.noact = 1; } // If DB change stream is active, don't use this event to remove the mesh. Another event will come.
|
||||||
|
parent.parent.DispatchEvent(['*', command.meshid], obj, event);
|
||||||
|
|
||||||
// Remove all user links to this mesh
|
// Remove all user links to this mesh
|
||||||
for (i in meshes) {
|
for (i in meshes) {
|
||||||
@ -1727,7 +1735,10 @@ module.exports.CreateMeshUser = function (parent, db, ws, req, args, domain, use
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Event node deletion
|
// Event node deletion
|
||||||
parent.parent.DispatchEvent(['*', node.meshid], obj, { etype: 'node', username: user.name, action: 'removenode', nodeid: node._id, msg: 'Removed device ' + node.name + ' from group ' + mesh.name, domain: domain.id });
|
var event = { etype: 'node', username: user.name, action: 'removenode', nodeid: node._id, msg: 'Removed device ' + node.name + ' from group ' + mesh.name, domain: domain.id };
|
||||||
|
// TODO: We can't use the changeStream for node delete because we will not know the meshid the device was in.
|
||||||
|
//if (db.changeStream) { event.noact = 1; } // If DB change stream is active, don't use this event to remove the node. Another event will come.
|
||||||
|
parent.parent.DispatchEvent(['*', node.meshid], obj, event);
|
||||||
|
|
||||||
// Disconnect all connections if needed
|
// Disconnect all connections if needed
|
||||||
var state = parent.parent.GetConnectivityState(nodeid);
|
var state = parent.parent.GetConnectivityState(nodeid);
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "meshcentral",
|
"name": "meshcentral",
|
||||||
"version": "0.3.5-j",
|
"version": "0.3.5-k",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"Remote Management",
|
"Remote Management",
|
||||||
"Intel AMT",
|
"Intel AMT",
|
||||||
|
@ -881,6 +881,7 @@
|
|||||||
case 'addnode': {
|
case 'addnode': {
|
||||||
var node = message.event.node;
|
var node = message.event.node;
|
||||||
if (!meshes[node.meshid]) break; // This is a node for a mesh we don't know. Happens when we are site administrator, we get all messages.
|
if (!meshes[node.meshid]) break; // This is a node for a mesh we don't know. Happens when we are site administrator, we get all messages.
|
||||||
|
if (getNodeFromId(node._id) != null) break; // This node is already known.
|
||||||
node.namel = node.name.toLowerCase();
|
node.namel = node.name.toLowerCase();
|
||||||
if (node.rname) { node.rnamel = node.rname.toLowerCase(); } else { node.rnamel = node.namel; }
|
if (node.rname) { node.rnamel = node.rname.toLowerCase(); } else { node.rnamel = node.namel; }
|
||||||
node.meshnamel = meshes[node.meshid].name.toLowerCase();
|
node.meshnamel = meshes[node.meshid].name.toLowerCase();
|
||||||
|
@ -1690,7 +1690,7 @@
|
|||||||
}
|
}
|
||||||
case 'createmesh': {
|
case 'createmesh': {
|
||||||
// A new mesh was created
|
// A new mesh was created
|
||||||
if (message.event.links[userinfo._id] != null) { // Check if this is a mesh create for a mesh we own. If site administrator, we get all messages so need to ignore some.
|
if ((meshes[message.event.meshid] == null) && (message.event.links[userinfo._id] != null)) { // Check if this is a mesh create for a mesh we own. If site administrator, we get all messages so need to ignore some.
|
||||||
meshes[message.event.meshid] = { _id: message.event.meshid, name: message.event.name, mtype: message.event.mtype, desc: message.event.desc, links: message.event.links };
|
meshes[message.event.meshid] = { _id: message.event.meshid, name: message.event.name, mtype: message.event.mtype, desc: message.event.desc, links: message.event.links };
|
||||||
masterUpdate(4 + 128);
|
masterUpdate(4 + 128);
|
||||||
meshserver.send({ action: 'files' });
|
meshserver.send({ action: 'files' });
|
||||||
@ -1758,6 +1758,7 @@
|
|||||||
case 'addnode': {
|
case 'addnode': {
|
||||||
var node = message.event.node;
|
var node = message.event.node;
|
||||||
if (!meshes[node.meshid]) break; // This is a node for a mesh we don't know. Happens when we are site administrator, we get all messages.
|
if (!meshes[node.meshid]) break; // This is a node for a mesh we don't know. Happens when we are site administrator, we get all messages.
|
||||||
|
if (getNodeFromId(node._id) != null) break; // This node is already known.
|
||||||
node.namel = node.name.toLowerCase();
|
node.namel = node.name.toLowerCase();
|
||||||
if (node.rname) { node.rnamel = node.rname.toLowerCase(); } else { node.rnamel = node.namel; }
|
if (node.rname) { node.rnamel = node.rname.toLowerCase(); } else { node.rnamel = node.namel; }
|
||||||
node.meshnamel = meshes[node.meshid].name.toLowerCase();
|
node.meshnamel = meshes[node.meshid].name.toLowerCase();
|
||||||
|
16
webserver.js
16
webserver.js
@ -309,7 +309,9 @@ module.exports.CreateWebServer = function (parent, db, args, certificates) {
|
|||||||
if (usercount == 0) { user.siteadmin = 0xFFFFFFFF; /*if (domain.newaccounts === 2) { delete domain.newaccounts; }*/ } // If this is the first user, give the account site admin.
|
if (usercount == 0) { user.siteadmin = 0xFFFFFFFF; /*if (domain.newaccounts === 2) { delete domain.newaccounts; }*/ } // If this is the first user, give the account site admin.
|
||||||
obj.users[user._id] = user;
|
obj.users[user._id] = user;
|
||||||
obj.db.SetUser(user);
|
obj.db.SetUser(user);
|
||||||
obj.parent.DispatchEvent(['*', 'server-users'], obj, { etype: 'user', userid: userid, username: username, account: obj.CloneSafeUser(user), action: 'accountcreate', msg: 'Account created, name is ' + name, domain: domain.id });
|
var event = { etype: 'user', userid: userid, username: username, account: obj.CloneSafeUser(user), action: 'accountcreate', msg: 'Account created, name is ' + name, domain: domain.id };
|
||||||
|
if (obj.db.changeStream) { event.noact = 1; } // If DB change stream is active, don't use this event to create the user. Another event will come.
|
||||||
|
obj.parent.DispatchEvent(['*', 'server-users'], obj, event);
|
||||||
return fn(null, user._id);
|
return fn(null, user._id);
|
||||||
} else {
|
} else {
|
||||||
// This is an existing user
|
// This is an existing user
|
||||||
@ -363,7 +365,9 @@ module.exports.CreateWebServer = function (parent, db, args, certificates) {
|
|||||||
if (usercount == 0) { user.siteadmin = 0xFFFFFFFF; /*if (domain.newaccounts === 2) { delete domain.newaccounts; }*/ } // If this is the first user, give the account site admin.
|
if (usercount == 0) { user.siteadmin = 0xFFFFFFFF; /*if (domain.newaccounts === 2) { delete domain.newaccounts; }*/ } // If this is the first user, give the account site admin.
|
||||||
obj.users[user._id] = user;
|
obj.users[user._id] = user;
|
||||||
obj.db.SetUser(user);
|
obj.db.SetUser(user);
|
||||||
obj.parent.DispatchEvent(['*', 'server-users'], obj, { etype: 'user', username: user.name, account: obj.CloneSafeUser(user), action: 'accountcreate', msg: 'Account created, name is ' + name, domain: domain.id });
|
var event = { etype: 'user', username: user.name, account: obj.CloneSafeUser(user), action: 'accountcreate', msg: 'Account created, name is ' + name, domain: domain.id };
|
||||||
|
if (obj.db.changeStream) { event.noact = 1; } // If DB change stream is active, don't use this event to create the user. Another event will come.
|
||||||
|
obj.parent.DispatchEvent(['*', 'server-users'], obj, event);
|
||||||
return fn(null, user._id);
|
return fn(null, user._id);
|
||||||
} else {
|
} else {
|
||||||
// This is an existing user
|
// This is an existing user
|
||||||
@ -814,7 +818,9 @@ module.exports.CreateWebServer = function (parent, db, args, certificates) {
|
|||||||
// Send the verification email
|
// Send the verification email
|
||||||
if ((obj.parent.mailserver != null) && (domain.auth != 'sspi') && (domain.auth != 'ldap') && (obj.common.validateEmail(user.email, 1, 256) == true)) { obj.parent.mailserver.sendAccountCheckMail(domain, user.name, user.email); }
|
if ((obj.parent.mailserver != null) && (domain.auth != 'sspi') && (domain.auth != 'ldap') && (obj.common.validateEmail(user.email, 1, 256) == true)) { obj.parent.mailserver.sendAccountCheckMail(domain, user.name, user.email); }
|
||||||
}, 0);
|
}, 0);
|
||||||
obj.parent.DispatchEvent(['*', 'server-users'], obj, { etype: 'user', username: user.name, account: obj.CloneSafeUser(user), action: 'accountcreate', msg: 'Account created, email is ' + req.body.email, domain: domain.id });
|
var event = { etype: 'user', username: user.name, account: obj.CloneSafeUser(user), action: 'accountcreate', msg: 'Account created, email is ' + req.body.email, domain: domain.id };
|
||||||
|
if (obj.db.changeStream) { event.noact = 1; } // If DB change stream is active, don't use this event to create the user. Another event will come.
|
||||||
|
obj.parent.DispatchEvent(['*', 'server-users'], obj, event);
|
||||||
}
|
}
|
||||||
res.redirect(domain.url);
|
res.redirect(domain.url);
|
||||||
}
|
}
|
||||||
@ -1239,7 +1245,9 @@ module.exports.CreateWebServer = function (parent, db, args, certificates) {
|
|||||||
if (usercount == 0) { user2.siteadmin = 0xFFFFFFFF; } // If this is the first user, give the account site admin.
|
if (usercount == 0) { user2.siteadmin = 0xFFFFFFFF; } // If this is the first user, give the account site admin.
|
||||||
obj.users[req.session.userid] = user2;
|
obj.users[req.session.userid] = user2;
|
||||||
obj.db.SetUser(user2);
|
obj.db.SetUser(user2);
|
||||||
obj.parent.DispatchEvent(['*', 'server-users'], obj, { etype: 'user', username: req.connection.user, account: obj.CloneSafeUser(user2), action: 'accountcreate', msg: 'Domain account created, user ' + req.connection.user, domain: domain.id });
|
var event = { etype: 'user', username: req.connection.user, account: obj.CloneSafeUser(user2), action: 'accountcreate', msg: 'Domain account created, user ' + req.connection.user, domain: domain.id };
|
||||||
|
if (obj.db.changeStream) { event.noact = 1; } // If DB change stream is active, don't use this event to create the user. Another event will come.
|
||||||
|
obj.parent.DispatchEvent(['*', 'server-users'], obj, event);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user