From fb48cb39459432548e0f83637e1924bec7c91b6e Mon Sep 17 00:00:00 2001 From: Ylian Saint-Hilaire Date: Sat, 1 Dec 2018 23:41:57 -0800 Subject: [PATCH] Integrated MeshMessenger for user-to-user. --- meshcentral.js | 3 ++- meshrelay.js | 7 ++++++ meshuser.js | 19 ++++++++++++++ public/images/icon-chat.png | Bin 0 -> 578 bytes public/messenger.htm | 46 ++++++++++++++++++---------------- views/default.handlebars | 23 ++++++++++++++--- views/login-mobile.handlebars | 2 +- views/login.handlebars | 2 +- webserver.js | 9 ++++--- 9 files changed, 80 insertions(+), 31 deletions(-) create mode 100644 public/images/icon-chat.png diff --git a/meshcentral.js b/meshcentral.js index 13ae02a2..af600a4a 100644 --- a/meshcentral.js +++ b/meshcentral.js @@ -183,7 +183,8 @@ function CreateMeshCentralServer(config, args) { xprocess.stdout.on('data', function (data) { if (data[data.length - 1] == '\n') { data = data.substring(0, data.length - 1); } if (data.indexOf('Updating settings folder...') >= 0) { xprocess.xrestart = 1; } else if (data.indexOf('Updating server certificates...') >= 0) { xprocess.xrestart = 1; } else if (data.indexOf('Server Ctrl-C exit...') >= 0) { xprocess.xrestart = 2; } else if (data.indexOf('Starting self upgrade...') >= 0) { xprocess.xrestart = 3; } console.log(data); }); xprocess.stderr.on('data', function (data) { if (data.startsWith('le.challenges[tls-sni-01].loopback')) { return; } // Ignore this error output from GreenLock - if (data[data.length - 1] == '\n') { data = data.substring(0, data.length - 1); } obj.fs.appendFileSync(obj.getConfigFilePath('mesherrors.txt'), '-------- ' + new Date().toLocaleString() + ' ---- ' + obj.currentVer + ' --------\r\n\r\n' + data + '\r\n\r\n\r\n'); + if (data[data.length - 1] == '\n') { data = data.substring(0, data.length - 1); } + try { obj.fs.appendFileSync(obj.getConfigFilePath('mesherrors.txt'), '-------- ' + new Date().toLocaleString() + ' ---- ' + obj.currentVer + ' --------\r\n\r\n' + data + '\r\n\r\n\r\n'); } catch (ex) { console.log('ERROR: Unable to write to mesherrors.txt.'); } }); xprocess.on('close', function (code) { if ((code != 0) && (code != 123)) { /* console.log("Exited with code " + code); */ } }); }; diff --git a/meshrelay.js b/meshrelay.js index f4f59d50..16f21a81 100644 --- a/meshrelay.js +++ b/meshrelay.js @@ -94,6 +94,13 @@ module.exports.CreateMeshRelay = function (parent, ws, req, domain, user, cookie if (obj.id == null) { try { obj.close(); } catch (e) { } return null; } // Attempt to connect without id, drop this. ws._socket.setKeepAlive(true, 240000); // Set TCP keep alive + // If this is a MeshMessenger session, the ID is the two userid's and authentication must match one of them. + if (obj.id.startsWith('meshmessenger/')) { + if (obj.user == null) { try { obj.close(); } catch (e) { } return null; } + var x = obj.id.split('/'), user1 = x[1] + '/' + x[2] + '/' + x[3], user2 = x[4] + '/' + x[5] + '/' + x[6]; + if ((obj.user._id != user1) && (obj.user._id != user2)) { try { obj.close(); } catch (e) { } return null; } + } + // Validate that the id is valid, we only need to do this on non-authenticated sessions. // TODO: Figure out when this needs to be done. /* diff --git a/meshuser.js b/meshuser.js index 6ef37359..c5a7e3f4 100644 --- a/meshuser.js +++ b/meshuser.js @@ -592,6 +592,25 @@ module.exports.CreateMeshUser = function (parent, db, ws, req, args, domain, use var sessions = obj.parent.wssessions[command.userid]; if (sessions != null) { for (i in sessions) { try { sessions[i].send(JSON.stringify(notification)); } catch (ex) { } } } + if (obj.parent.parent.multiServer != null) { + // TODO: Add multi-server support + } + break; + } + case 'meshmessenger': + { + // Send a notification message to a user + if ((user.siteadmin & 2) == 0) break; + if (obj.common.validateString(command.userid, 1, 2048) == false) break; + + // Create the notification message + var notification = { + "action": "msg", "type": "notify", "value": "" + user.name + ": Chat Request, Click here to accept.", "userid": user._id, "username": user.name, "tag": 'meshmessenger/' + encodeURIComponent(command.userid) + '/' + encodeURIComponent(user._id) }; + + // Get the list of sessions for this user + var sessions = obj.parent.wssessions[command.userid]; + if (sessions != null) { for (i in sessions) { try { sessions[i].send(JSON.stringify(notification)); } catch (ex) { } } } + if (obj.parent.parent.multiServer != null) { // TODO: Add multi-server support } diff --git a/public/images/icon-chat.png b/public/images/icon-chat.png new file mode 100644 index 0000000000000000000000000000000000000000..f156df7d7479134c21ceb48f03e030b0171014b4 GIT binary patch literal 578 zcmV-I0=@l-P)Px#1ZP1_K>z@;j|==^1poj532;bRa{vGi!vFvd!vV){sAK>D02p*dSaefwW^{L9 za%BK;VQFr3E^cLXAT%y8E;(#7eog=Y0kcU&K~y+Tom0O{0$~_F^*lX2Q~UvLE>7a) z=uk%7)}}&he?nG+6lg=6(Bh&<5UoK?f{h0*4J{%{wAi4FP5l7H+?MKzTf+uFhe*T4#~27DoIkyZnuxX3Hab}IL`22gv-p~_xn|w&Gwu~B&b|2lcs6Z zY&NM@t5GVIBCFN<>U27HxO^Ef$6$Uh7K^0oI+>wbt&%8;6NIdD+Z<@~DxFSq>1yP1 zIfC{s7K=&TIt6Basn_dVwi@krn=BR!u~A(v*Fm9Buo?kqZNz5&3B6v=2$&JRb0Yw4 zY59Df%T}XOsVoIBqLXdG?RIZrefoD@bUGahg+gBf;JY-sz{4I+n7Z9Aw@fh@45(Bp zk8VyP&lZ5q}nih$GeIXi+uHj@`@pzn6Rb{6<#%6BK z42y*B5o0MrFc{pxO&g=|hHHC)Vw81-kVDi*W@dET$J^)E8#{G@BLDdb_6pF07*qoM6N<$f?EpsD*ylh literal 0 HcmV?d00001 diff --git a/public/messenger.htm b/public/messenger.htm index 28401d82..3ed66a14 100644 --- a/public/messenger.htm +++ b/public/messenger.htm @@ -1,6 +1,7 @@ + MeshMessenger @@ -8,7 +9,7 @@ -
Messenger
+
MeshMessenger
@@ -25,7 +26,7 @@ var state = 0; // Set the title - if (args.title) { QH('xtitle', ' - ' + args.title); } + if (args.title) { QH('xtitle', ' - ' + args.title); document.title = document.title + ' - ' + args.title; } // Trap document key presses document.onkeypress = function ondockeypress(e) { @@ -76,31 +77,34 @@ function haltEvent(e) { if (e.preventDefault) e.preventDefault(); if (e.stopPropagation) e.stopPropagation(); return false; } function parseUriArgs() { var name, r = {}, parsedUri = window.document.location.href.split(/[\?&|\=]/); parsedUri.splice(0, 1); for (x in parsedUri) { switch (x % 2) { case 0: { name = parsedUri[x]; break; } case 1: { r[name] = parsedUri[x]; var x = parseInt(r[name]); if (x == r[name]) { r[name] = x; } break; } } } return r; } - // Get started - enableControls(false); - if ((typeof args.id == 'string') && (args.id.length > 0)) { - xcontrol('Connecting...'); - socket = new WebSocket(window.location.protocol.replace("http", "ws") + "//" + window.location.host + window.location.pathname.substring(0, window.location.pathname.lastIndexOf('/')) + '/meshrelay.ashx?id=' + args.id); - socket.onopen = function () { xcontrol('Waiting for other user...'); } - socket.onerror = function (e) { console.error(e); } - socket.onclose = function () { enableControls(false); xcontrol('Connection closed.'); socket = null; } - socket.onmessage = function (msg) { - if ((state == 0) && (typeof msg.data == 'string')) { enableControls(true); xcontrol('Connected.'); state = 1; return; } - if (state == 1) { - if (typeof msg.data == 'string') { - var obj = JSON.parse(msg.data); - switch (obj.action) { - case 'chat': { xrecv(obj.msg); break; } + function start() { + // Get started + enableControls(false); + if ((typeof args.id == 'string') && (args.id.length > 0)) { + //xcontrol('Connecting...'); + socket = new WebSocket(window.location.protocol.replace("http", "ws") + "//" + window.location.host + window.location.pathname.substring(0, window.location.pathname.lastIndexOf('/')) + '/meshrelay.ashx?id=' + args.id); + socket.onopen = function () { state = 1; xcontrol('Waiting for other user...'); } + socket.onerror = function (e) { console.error(e); } + socket.onclose = function () { enableControls(false); if (state > 0) { xcontrol('Connection closed.'); } socket = null; if (state > 1) { setTimeout(start, 500); } state = 0; } + socket.onmessage = function (msg) { + if ((state < 2) && (typeof msg.data == 'string')) { enableControls(true); xcontrol('Connected.'); state = 2; return; } + if (state == 2) { + if (typeof msg.data == 'string') { + var obj = JSON.parse(msg.data); + switch (obj.action) { + case 'chat': { xrecv(obj.msg); break; } + } + } else { + //xrecv(JSON.stringify(msg)); } - } else { - //xrecv(JSON.stringify(msg)); } } + } else { + xcontrol('Error: No connection key specified.'); } - } else { - xcontrol('Error: No connection key specified.'); } + start(); diff --git a/views/default.handlebars b/views/default.handlebars index 31874954..d780e08d 100644 --- a/views/default.handlebars +++ b/views/default.handlebars @@ -1195,6 +1195,7 @@ var n = { text:message.value }; if (message.nodeid != null) { n.nodeid = message.nodeid; } if (message.tag != null) { n.tag = message.tag; } + if (message.username != null) { n.username = message.username; } addNotification(n); } else if (message.type == 'ps') { showDeskToolsProcesses(message); @@ -1204,6 +1205,7 @@ if (message.type == 'notify') { // This is a notification message. var n = { text:message.value }; if (message.tag != null) { n.tag = message.tag; } + if (message.username != null) { n.username = message.username; } addNotification(n); } } @@ -5720,10 +5722,13 @@ } function addUserHtml(user, sessions) { - var x = '', gray = ' gray', icon = 'm2', msg = '', self = (user.name != userinfo.name); + var x = '', gray = ' gray', icon = 'm2', msg = '', msg2 = '', self = (user.name != userinfo.name); if (sessions != null) { gray = ''; - if (self) { msg += ""; } + if (self) { + msg2 = ""; + msg += ""; + } if (sessions == 1) { msg += '1 active session'; } else { msg += sessions + ' active sessions'; } if (self) { msg += ""; } } else { @@ -5752,7 +5757,7 @@ x += '
'; x += '
'; x += '
'; - x += '
' + username + '' + msg + '
'; // + x += '
' + username + '' + msg2 + '' + msg + '
'; // return x; } @@ -5765,6 +5770,13 @@ element.children[0].children[0].style['background-color'] = ((over == 0) ? '#c9c9c9' : '#b9b9b9'); } + function userChat(e, userid, name) { + haltEvent(e); + window.open('/messenger.htm?id=meshmessenger/' + userid + '/' + encodeURIComponent(userinfo._id) + '&title=' + name, 'meshmessenger:' + userid); + meshserver.send({ action: 'meshmessenger', userid: decodeURIComponent(userid) }); + return false; + } + function showUserAlertDialog(e, userid) { if (xxdialogMode) return; haltEvent(e); @@ -6223,6 +6235,11 @@ else if (n.tag == 'intelamt') gotoDevice(n.nodeid, 14); // Intel AMT else if (n.tag == 'console') gotoDevice(n.nodeid, 15); // Files else gotoDevice(n.nodeid, 10); // General + } else { + if (n.tag.startsWith('meshmessenger/')) { + window.open('/messenger.htm?id=' + n.tag + '&title=' + encodeURIComponent(n.username), n.tag.split('/')[2]); + notificationDelete(id); + } } } } diff --git a/views/login-mobile.handlebars b/views/login-mobile.handlebars index 6c97d230..0abdc1da 100644 --- a/views/login-mobile.handlebars +++ b/views/login-mobile.handlebars @@ -193,7 +193,7 @@ validateLogin(); validateCreate(); if ('{{loginmode}}' != '') { go(parseInt('{{loginmode}}')); } else { go(1); } - QV('newAccountDiv', '{{{newAccount}}}' != '0' ); + QV('newAccountDiv', ('{{{newAccount}}}' != '0') && ('{{{newAccount}}}' != 'false')); // If new accounts are not allowed, don't display the new account link. if ((passhint != null) && (passhint.length > 0)) { QV("showPassHintLink", true); } QV("newAccountPass", (newAccountPass == 1)); QV("resetAccountDiv", (emailCheck == true)); diff --git a/views/login.handlebars b/views/login.handlebars index 836bcb64..57be4ab4 100644 --- a/views/login.handlebars +++ b/views/login.handlebars @@ -276,7 +276,7 @@ validateLogin(); validateCreate(); if ('{{loginmode}}' != '') { go(parseInt('{{loginmode}}')); } else { go(1); } - QV('newAccountDiv', '{{{newAccount}}}' != '0' ); + QV('newAccountDiv', ('{{{newAccount}}}' != '0') && ('{{{newAccount}}}' != 'false')); // If new accounts are not allowed, don't display the new account link. if ((passhint != null) && (passhint.length > 0)) { QV("showPassHintLink", true); } QV("newAccountPass", (newAccountPass == 1)); QV("resetAccountDiv", (emailCheck == true)); diff --git a/webserver.js b/webserver.js index 5c17c512..39bd6098 100644 --- a/webserver.js +++ b/webserver.js @@ -220,7 +220,8 @@ module.exports.CreateWebServer = function (parent, db, args, certificates) { for (i in docs) { var u = obj.users[docs[i]._id] = docs[i]; domainUserCount[u.domain]++; } for (i in parent.config.domains) { if (domainUserCount[i] == 0) { - if (parent.config.domains[i].newaccounts == 0) { parent.config.domains[i].newaccounts = 2; } + // If newaccounts is set to no new accounts, but no accounts exists, temporarly allow account creation. + if ((parent.config.domains[i].newaccounts === 0) || (parent.config.domains[i].newaccounts === false)) { parent.config.domains[i].newaccounts = 2; } console.log('Server ' + ((i == '') ? '' : (i + ' ')) + 'has no users, next new account will be site administrator.'); } } @@ -390,7 +391,7 @@ module.exports.CreateWebServer = function (parent, db, args, certificates) { function handleCreateAccountRequest(req, res) { var domain = checkUserIpAddress(req, res); if (domain == null) return; - if (domain.newaccounts == 0) { res.sendStatus(401); return; } + if ((domain.newaccounts === 0) || (domain.newaccounts === false)) { res.sendStatus(401); return; } if (!obj.common.validateUsername(req.body.username, 1, 64) || !obj.common.validateEmail(req.body.email, 1, 256) || !obj.common.validateString(req.body.password1, 1, 256) || !obj.common.validateString(req.body.password2, 1, 256) || (req.body.password1 != req.body.password2) || req.body.username == '~') { req.session.loginmode = 2; req.session.error = 'Unable to create account.'; @@ -420,7 +421,7 @@ module.exports.CreateWebServer = function (parent, db, args, certificates) { var user = { type: 'user', _id: 'user/' + domain.id + '/' + req.body.username.toLowerCase(), name: req.body.username, email: req.body.email, creation: Date.now(), login: Date.now(), domain: domain.id, passhint: hint }; var usercount = 0; for (var i in obj.users) { if (obj.users[i].domain == domain.id) { usercount++; } } - if (usercount == 0) { user.siteadmin = 0xFFFFFFFF; if (domain.newaccounts == 2) { domain.newaccounts = 0; } } // If this is the first user, give the account site admin. + if (usercount == 0) { user.siteadmin = 0xFFFFFFFF; if (domain.newaccounts === 2) { domain.newaccounts = 0; } } // If this is the first user, give the account site admin. obj.users[user._id] = user; req.session.userid = user._id; req.session.domainid = domain.id; @@ -444,7 +445,7 @@ module.exports.CreateWebServer = function (parent, db, args, certificates) { function handleResetAccountRequest(req, res) { var domain = checkUserIpAddress(req, res); if (domain == null) return; - if (domain.newaccounts == 0) { res.sendStatus(401); return; } + if ((domain.newaccounts === 0) || (domain.newaccounts === false)) { res.sendStatus(401); return; } if (!req.body.email || checkEmail(req.body.email) == false) { req.session.loginmode = 3; req.session.error = 'Invalid email.';