2022-10-22 07:23:55 -07:00
/ * *
* @ description MeshCentral user messaging communication module
* @ author Ylian Saint - Hilaire
* @ copyright Intel Corporation 2022
* @ license Apache - 2.0
* @ version v0 . 0.1
* /
/*xjslint node: true */
/*xjslint plusplus: true */
/*xjslint maxlen: 256 */
/*jshint node: true */
/*jshint strict: false */
/*jshint esversion: 6 */
"use strict" ;
/ *
2022-10-22 22:05:42 -07:00
// For Telegram user login, add this in config.json
2022-10-22 07:23:55 -07:00
"messaging" : {
2022-10-28 12:40:52 -07:00
"telegram" : {
"apiid" : 00000000 ,
"apihash" : "00000000000000000000000" ,
"session" : "aaaaaaaaaaaaaaaaaaaaaaa"
}
2022-10-22 07:23:55 -07:00
}
2022-10-22 22:05:42 -07:00
// For Telegram bot login, add this in config.json
"messaging" : {
2022-10-28 12:40:52 -07:00
"telegram" : {
"apiid" : 00000000 ,
"apihash" : "00000000000000000000000" ,
"bottoken" : "00000000:aaaaaaaaaaaaaaaaaaaaaaaa"
}
2022-10-22 22:05:42 -07:00
}
2022-10-26 13:19:40 -07:00
2022-11-01 14:51:41 -07:00
// For Telegram login with proxy settings, add this in config.json
{
"messaging" : {
"telegram" : {
"apiid" : 0 ,
"apihash" : "00000000000000000000000" ,
"session" : "aaaaaaaaaaaaaaaaaaaaaaa" ,
"useWSS" : false , // Important. Most proxies cannot use SSL.
"proxy" : {
"ip" : "123.123.123.123" , // Proxy host (IP or hostname)
"port" : 123 , // Proxy port
"MTProxy" : false , // Whether it's an MTProxy or a normal Socks one
"secret" : "00000000000000000000000000000000" , // If used MTProxy then you need to provide a secret (or zeros).
"socksType" : 5 , // If used Socks you can choose 4 or 5.
"timeout" : 2 // Timeout (in seconds) for connection,
}
}
}
}
2022-10-26 13:19:40 -07:00
// For Discord login, add this in config.json
"messaging" : {
2022-10-28 12:40:52 -07:00
"discord" : {
"inviteurl" : "https://discord.gg/xxxxxxxxx" ,
"token" : "xxxxxxxxxxxxxxxxxx.xxxxxxxxxxxxxxxxxx"
}
2022-10-26 13:19:40 -07:00
}
2022-10-28 12:34:29 -07:00
// For XMPP login, add this in config.json
2022-10-28 12:40:52 -07:00
{
"messaging" : {
2022-10-28 12:34:29 -07:00
"xmpp" : {
2022-10-28 12:40:52 -07:00
"service" : "xmppserver.com" ,
"credentials" : {
"username" : "username" ,
"password" : "password"
}
2022-10-28 12:34:29 -07:00
}
2022-10-28 12:40:52 -07:00
}
2022-10-28 12:34:29 -07:00
}
2022-10-28 17:44:06 -07:00
// For CallMeBot
// For Signal Messenger: https://www.callmebot.com/blog/free-api-signal-send-messages/
{
"messaging" : {
"callmebot" : true
}
}
2022-10-30 11:28:24 -07:00
// For Pushover
{
"messaging" : {
"pushover" : {
"token" : "xxxxxxx"
}
}
}
2022-10-31 12:17:13 -07:00
// For ntfy
{
"messaging" : {
"ntfy" : true
}
}
2022-11-01 12:27:06 -07:00
// For zulip
{
"messaging" : {
2022-11-01 13:00:07 -07:00
"site" : "https://api.zulip.com" ,
"email" : "your-bot@zulip.com" ,
"api_key" : "your_32_character_api_key"
2022-11-01 12:27:06 -07:00
}
}
2023-11-25 20:45:18 +00:00
// For Slack Webhook
{
"messaging" : {
"slack" : true
}
}
2022-10-22 07:23:55 -07:00
* /
2022-10-22 14:51:16 -07:00
// Construct a messaging server object
2022-10-22 07:23:55 -07:00
module . exports . CreateServer = function ( parent ) {
var obj = { } ;
obj . parent = parent ;
2023-11-25 20:45:18 +00:00
obj . providers = 0 ; // 1 = Telegram, 2 = Signal, 4 = Discord, 8 = XMPP, 16 = CallMeBot, 32 = Pushover, 64 = ntfy, 128 = Zulip, 256 = Slack
2022-10-22 07:23:55 -07:00
obj . telegramClient = null ;
2022-10-26 13:19:40 -07:00
obj . discordClient = null ;
2022-10-27 10:38:15 -07:00
obj . discordUrl = null ;
2022-10-28 12:34:29 -07:00
obj . xmppClient = null ;
var xmppXml = null ;
2022-10-28 17:44:06 -07:00
obj . callMeBotClient = null ;
2022-10-30 11:28:24 -07:00
obj . pushoverClient = null ;
2022-11-01 12:27:06 -07:00
obj . zulipClient = null ;
2023-11-25 20:45:18 +00:00
obj . slackClient = null ;
2022-11-01 22:39:59 -07:00
const sortCollator = new Intl . Collator ( undefined , { numeric : true , sensitivity : 'base' } )
2022-10-22 07:23:55 -07:00
2022-10-26 13:19:40 -07:00
// Telegram client setup
2022-10-22 07:23:55 -07:00
if ( parent . config . messaging . telegram ) {
// Validate Telegram configuration values
var telegramOK = true ;
if ( typeof parent . config . messaging . telegram . apiid != 'number' ) { console . log ( 'Invalid or missing Telegram apiid.' ) ; telegramOK = false ; }
if ( typeof parent . config . messaging . telegram . apihash != 'string' ) { console . log ( 'Invalid or missing Telegram apihash.' ) ; telegramOK = false ; }
2022-10-22 22:05:42 -07:00
if ( ( typeof parent . config . messaging . telegram . session != 'string' ) && ( typeof parent . config . messaging . telegram . bottoken != 'string' ) ) { console . log ( 'Invalid or missing Telegram session or bottoken.' ) ; telegramOK = false ; }
2022-10-22 07:23:55 -07:00
if ( telegramOK ) {
// Setup Telegram
async function setupTelegram ( ) {
const { TelegramClient } = require ( 'telegram' ) ;
const { StringSession } = require ( 'telegram/sessions' ) ;
2022-10-22 15:38:33 -07:00
const { Logger } = require ( 'telegram/extensions/Logger' ) ;
const logger = new Logger ( { LogLevel : 'none' } ) ;
2022-10-22 07:23:55 -07:00
const input = require ( 'input' ) ;
2022-10-22 22:05:42 -07:00
var client ;
2022-11-01 14:51:41 -07:00
var options = { connectionRetries : 5 , baseLogger : logger } ;
if ( parent . config . messaging . telegram . usewss == false ) { options . useWSS = false ; }
if ( typeof parent . config . messaging . telegram . connectionretries == 'number' ) { options . connectionRetries = parent . config . messaging . telegram . connectionretries ; }
if ( typeof parent . config . messaging . telegram . proxy == 'object' ) { options . proxy = parent . config . messaging . telegram . proxy ; }
2022-10-22 22:05:42 -07:00
if ( parent . config . messaging . telegram . bottoken == null ) {
// User login
var stringSession = new StringSession ( parent . config . messaging . telegram . session ) ;
2022-11-01 14:51:41 -07:00
const client = new TelegramClient ( stringSession , parent . config . messaging . telegram . apiid , parent . config . messaging . telegram . apihash , options ) ;
2022-10-22 22:05:42 -07:00
await client . start ( { onError : function ( err ) { console . log ( 'Telegram error' , err ) ; } } ) ;
obj . telegramClient = client ;
obj . providers += 1 ; // Enable Telegram messaging
console . log ( "MeshCentral Telegram client is user connected." ) ;
} else {
// Bot login
var stringSession = new StringSession ( '' ) ;
2022-11-01 14:51:41 -07:00
const client = new TelegramClient ( stringSession , parent . config . messaging . telegram . apiid , parent . config . messaging . telegram . apihash , options ) ;
2022-10-22 22:05:42 -07:00
await client . start ( { botAuthToken : parent . config . messaging . telegram . bottoken , onError : function ( err ) { console . log ( 'Telegram error' , err ) ; } } ) ;
obj . telegramClient = client ;
obj . providers += 1 ; // Enable Telegram messaging
console . log ( "MeshCentral Telegram client is bot connected." ) ;
}
2022-10-22 07:23:55 -07:00
}
setupTelegram ( ) ;
}
}
2022-10-26 13:19:40 -07:00
// Discord client setup
if ( parent . config . messaging . discord ) {
// Validate Discord configuration values
var discordOK = true ;
if ( typeof parent . config . messaging . discord . token != 'string' ) { console . log ( 'Invalid or missing Discord token.' ) ; discordOK = false ; }
if ( discordOK ) {
// Setup Discord
const { Client , GatewayIntentBits } = require ( 'discord.js' ) ;
var discordClient = new Client ( {
intents : [
GatewayIntentBits . Guilds ,
GatewayIntentBits . GuildMessages ,
GatewayIntentBits . MessageContent ,
GatewayIntentBits . GuildMembers ,
GatewayIntentBits . DirectMessages
]
} ) ;
// Called when Discord client is connected
discordClient . on ( 'ready' , function ( ) {
console . log ( ` MeshCentral Discord client is connected as ${ discordClient . user . tag } ! ` ) ;
obj . discordClient = discordClient ;
2022-10-27 10:38:15 -07:00
obj . discordUrl = parent . config . messaging . discord . serverurl ;
2022-10-26 13:19:40 -07:00
obj . providers += 4 ; // Enable Discord messaging
} ) ;
// Receives incoming messages, ignore for now
discordClient . on ( 'messageCreate' , function ( message ) {
if ( message . author . bot ) return false ;
console . log ( ` Discord message from ${ message . author . username } : ${ message . content } ` , message . channel . type ) ;
//message.channel.send("Channel Hello");
//message.author.send('Private Hello');
} ) ;
// Called when Discord client received an interaction
discordClient . on ( 'interactionCreate' , async function ( interaction ) {
console . log ( 'Discord interaction' , interaction ) ;
if ( ! interaction . isChatInputCommand ( ) ) return ;
if ( interaction . commandName === 'ping' ) { await interaction . reply ( 'Pong!' ) ; }
} ) ;
// Connect Discord client
discordClient . login ( parent . config . messaging . discord . token ) ;
}
}
2022-10-28 12:34:29 -07:00
// XMPP client setup
if ( parent . config . messaging . xmpp ) {
2022-10-28 17:44:06 -07:00
// Validate XMPP configuration values
2022-10-28 12:34:29 -07:00
var xmppOK = true ;
if ( typeof parent . config . messaging . xmpp . service != 'string' ) { console . log ( 'Invalid or missing XMPP service.' ) ; xmppOK = false ; }
if ( xmppOK ) {
// Setup XMPP
const { client , xml } = require ( '@xmpp/client' ) ;
const xmpp = client ( parent . config . messaging . xmpp ) ;
xmpp . on ( 'error' , function ( err ) { parent . debug ( 'email' , 'XMPP error: ' + err ) ; console . error ( 'XMPP error' , err ) ; } ) ;
xmpp . on ( 'offline' , function ( ) { parent . debug ( 'email' , 'XMPP client is offline.' ) ; console . log ( 'XMPP offline' ) ; } ) ;
//xmpp.on('stanza', async function (stanza) { if (stanza.is("message")) { await xmpp.send(xml('presence', { type: 'unavailable' })); await xmpp.stop(); } });
xmpp . on ( 'online' , async function ( address ) {
// await xmpp.send(xml("presence")); const message = xml("message", { type: "chat", to: "username@server.com" }, xml("body", {}, "hello world")); await xmpp.send(message);
xmppXml = xml ;
obj . xmppClient = xmpp ;
obj . providers += 8 ; // Enable XMPP messaging
console . log ( "MeshCentral XMPP client is connected." ) ;
} ) ;
xmpp . start ( ) . catch ( console . error ) ;
}
}
2022-10-31 12:17:13 -07:00
// CallMeBot client setup (https://www.callmebot.com/)
2022-10-28 17:44:06 -07:00
if ( parent . config . messaging . callmebot ) {
obj . callMeBotClient = true ;
obj . providers += 16 ; // Enable CallMeBot messaging
}
2022-10-30 11:28:24 -07:00
// Pushover client setup (https://pushover.net)
2023-09-11 09:50:15 -07:00
if ( typeof parent . config . messaging . pushover == 'object' ) {
2022-10-30 11:28:24 -07:00
// Validate Pushover configuration values
var pushoverOK = true ;
if ( typeof parent . config . messaging . pushover . token != 'string' ) { console . log ( 'Invalid or missing Pushover token.' ) ; pushoverOK = false ; }
if ( pushoverOK ) {
// Setup PushOver
obj . pushoverClient = true ;
obj . providers += 32 ; // Enable Pushover messaging
}
}
2022-10-31 12:17:13 -07:00
// ntfy client setup (https://ntfy.sh/)
if ( parent . config . messaging . ntfy ) {
obj . ntfyClient = true ;
obj . providers += 64 ; // Enable ntfy messaging
}
2022-11-01 12:27:06 -07:00
// Zulip client setup (https://zulip.com/)
if ( typeof parent . config . messaging . zulip == 'object' ) {
var zulip = require ( 'zulip' ) ;
obj . zulipClient = new zulip . Client ( parent . config . messaging . zulip ) ;
obj . providers += 128 ; // Enable zulip messaging
}
2023-11-25 20:45:18 +00:00
// Slack Webhook setup (https://slack.com)
if ( parent . config . messaging . slack ) {
obj . slackClient = true ;
obj . providers += 256 ; // Enable slack messaging
}
2022-10-26 13:19:40 -07:00
// Send a direct message to a specific userid
async function discordSendMsg ( userId , message ) {
const user = await obj . discordClient . users . fetch ( userId ) . catch ( function ( ) { return null ; } ) ;
if ( ! user ) return ;
await user . send ( message ) . catch ( function ( ex ) { console . log ( 'Discord Error' , ex ) ; } ) ;
}
// Convert a userTag to a userId. We need to query the Discord server to find this information.
// Example: findUserByTab('aaaa#0000', function (userid) { sendMsg(userid, 'message'); });
async function discordFindUserByTag ( userTag , func ) {
var username = userTag . split ( '#' ) [ 0 ] ;
const guilds = await obj . discordClient . guilds . fetch ( ) ;
guilds . forEach ( async function ( value , key ) {
var guild = await value . fetch ( ) ;
const guildMembers = await guild . members . search ( { query : username } ) ;
guildMembers . forEach ( async function ( value , key ) {
2023-09-14 20:34:11 +01:00
if ( ( value . user . username + ( value . user . discriminator != '0' ? '#' + value . user . discriminator : '' ) ) == userTag ) { func ( key ) ; return ; }
2022-10-26 13:19:40 -07:00
} ) ;
} ) ;
}
2022-10-28 12:34:29 -07:00
// Send an XMPP message
async function sendXmppMessage ( to , msg , func ) {
const message = xmppXml ( 'message' , { type : 'chat' , to : to . substring ( 5 ) } , xmppXml ( 'body' , { } , msg ) ) ;
await obj . xmppClient . send ( message ) ;
if ( func != null ) { func ( true ) ; }
}
2022-10-22 07:23:55 -07:00
// Send an user message
2022-10-30 11:28:24 -07:00
obj . sendMessage = function ( to , msg , domain , func ) {
2022-10-26 13:19:40 -07:00
if ( ( to . startsWith ( 'telegram:' ) ) && ( obj . telegramClient != null ) ) { // Telegram
2022-10-22 07:23:55 -07:00
async function sendTelegramMessage ( to , msg , func ) {
if ( obj . telegramClient == null ) return ;
parent . debug ( 'email' , 'Sending Telegram message to: ' + to . substring ( 9 ) + ': ' + msg ) ;
2022-10-22 12:59:12 -07:00
try { await obj . telegramClient . sendMessage ( to . substring ( 9 ) , { message : msg } ) ; if ( func != null ) { func ( true ) ; } } catch ( ex ) { if ( func != null ) { func ( false , ex ) ; } }
2022-10-22 07:23:55 -07:00
}
sendTelegramMessage ( to , msg , func ) ;
2022-10-26 13:19:40 -07:00
} else if ( ( to . startsWith ( 'discord:' ) ) && ( obj . discordClient != null ) ) { // Discord
2022-10-28 12:34:29 -07:00
discordFindUserByTag ( to . substring ( 8 ) , function ( userid ) {
parent . debug ( 'email' , 'Sending Discord message to: ' + to . substring ( 9 ) + ', ' + userid + ': ' + msg ) ;
discordSendMsg ( userid , msg ) ; if ( func != null ) { func ( true ) ; }
} ) ;
} else if ( ( to . startsWith ( 'xmpp:' ) ) && ( obj . xmppClient != null ) ) { // XMPP
parent . debug ( 'email' , 'Sending XMPP message to: ' + to . substring ( 5 ) + ': ' + msg ) ;
sendXmppMessage ( to , msg , func ) ;
2022-10-28 17:44:06 -07:00
} else if ( ( to . startsWith ( 'callmebot:' ) ) && ( obj . callMeBotClient != null ) ) { // CallMeBot
parent . debug ( 'email' , 'Sending CallMeBot message to: ' + to . substring ( 10 ) + ': ' + msg ) ;
console . log ( 'Sending CallMeBot message to: ' + to . substring ( 10 ) + ': ' + msg ) ;
var toData = to . substring ( 10 ) . split ( '|' ) ;
if ( ( toData [ 0 ] == 'signal' ) && ( toData . length == 3 ) ) {
var url = 'https://api.callmebot.com/signal/send.php?phone=' + encodeURIComponent ( toData [ 1 ] ) + '&apikey=' + encodeURIComponent ( toData [ 2 ] ) + '&text=' + encodeURIComponent ( msg ) ;
require ( 'https' ) . get ( url , function ( r ) { if ( func != null ) { func ( r . statusCode == 200 ) ; } } ) ;
} else if ( ( toData [ 0 ] == 'whatsapp' ) && ( toData . length == 3 ) ) {
var url = 'https://api.callmebot.com/whatsapp.php?phone=' + encodeURIComponent ( toData [ 1 ] ) + '&apikey=' + encodeURIComponent ( toData [ 2 ] ) + '&text=' + encodeURIComponent ( msg ) ;
require ( 'https' ) . get ( url , function ( r ) { if ( func != null ) { func ( r . statusCode == 200 ) ; } } ) ;
} else if ( ( toData [ 0 ] == 'facebook' ) && ( toData . length == 2 ) ) {
var url = 'https://api.callmebot.com/facebook/send.php?apikey=' + encodeURIComponent ( toData [ 1 ] ) + '&text=' + encodeURIComponent ( msg ) ;
require ( 'https' ) . get ( url , function ( r ) { if ( func != null ) { func ( r . statusCode == 200 ) ; } } ) ;
2023-10-28 19:27:11 +01:00
} else if ( ( toData [ 0 ] == 'telegram' ) && ( toData . length == 2 ) ) {
var url = 'https://api.callmebot.com/text.php?user=' + encodeURIComponent ( toData [ 1 ] ) + '&text=' + encodeURIComponent ( msg ) ;
require ( 'https' ) . get ( url , function ( r ) { if ( func != null ) { func ( r . statusCode == 200 ) ; } } ) ;
2022-10-28 17:44:06 -07:00
}
2022-10-30 11:28:24 -07:00
} else if ( ( to . startsWith ( 'pushover:' ) ) && ( obj . pushoverClient != null ) ) { // Pushover
const Pushover = require ( 'node-pushover' ) ;
const push = new Pushover ( { token : parent . config . messaging . pushover . token , user : to . substring ( 9 ) } ) ;
push . send ( domain . title ? domain . title : 'MeshCentral' , msg , function ( err , res ) { if ( func != null ) { func ( err == null ) ; } } ) ;
2022-10-31 12:17:13 -07:00
} else if ( ( to . startsWith ( 'ntfy:' ) ) && ( obj . ntfyClient != null ) ) { // ntfy
2022-10-31 14:28:10 -07:00
const url = 'https://' + ( ( ( typeof parent . config . messaging . ntfy == 'object' ) && ( typeof parent . config . messaging . ntfy . host == 'string' ) ) ? parent . config . messaging . ntfy . host : 'ntfy.sh' ) + '/' + encodeURIComponent ( to . substring ( 5 ) ) ;
2023-10-04 12:49:55 +00:00
const headers = ( typeof parent . config . messaging . ntfy . authorization == 'string' ) ? { 'Authorization' : parent . config . messaging . ntfy . authorization } : { } ;
const req = require ( 'https' ) . request ( new URL ( url ) , { method : 'POST' , headers : headers } , function ( res ) { if ( func != null ) { func ( true ) ; } } ) ;
2022-10-31 12:17:13 -07:00
req . on ( 'error' , function ( err ) { if ( func != null ) { func ( false ) ; } } ) ;
req . end ( msg ) ;
2022-11-01 12:27:06 -07:00
} else if ( ( to . startsWith ( 'zulip:' ) ) && ( obj . zulipClient != null ) ) { // zulip
obj . zulipClient . sendMessage ( {
type : 'private' ,
content : msg ,
to : [ to . substring ( 6 ) ] ,
subject : domain . title ? domain . title : 'MeshCentral'
} ) ;
if ( func != null ) { func ( true ) ; }
2023-11-25 20:45:18 +00:00
} else if ( ( to . startsWith ( 'slack:' ) ) && ( obj . slackClient != null ) ) { //slack
const req = require ( 'https' ) . request ( new URL ( to . substring ( 6 ) ) , { method : 'POST' } , function ( res ) { if ( func != null ) { func ( true ) ; } } ) ;
req . on ( 'error' , function ( err ) { if ( func != null ) { func ( false ) ; } } ) ;
req . write ( JSON . stringify ( { "text" : msg } ) ) ;
req . end ( ) ;
2022-10-22 07:23:55 -07:00
} else {
// No providers found
2022-11-01 22:39:59 -07:00
if ( func != null ) { func ( false , "No messaging providers found for this message." ) ; }
2022-10-22 07:23:55 -07:00
}
}
2022-10-28 17:44:06 -07:00
// Convert a CallMeBot URL into a handle
obj . callmebotUrlToHandle = function ( xurl ) {
var url = null ;
try { url = require ( 'url' ) . parse ( xurl ) ; } catch ( ex ) { return ; }
2023-10-28 19:27:11 +01:00
if ( ( url == null ) || ( url . host != 'api.callmebot.com' ) || ( url . query == null ) ) return ;
2022-10-28 17:44:06 -07:00
var urlArgs = { } , urlArgs2 = url . query . split ( '&' ) ;
for ( var i in urlArgs2 ) { var j = urlArgs2 [ i ] . indexOf ( '=' ) ; if ( j > 0 ) { urlArgs [ urlArgs2 [ i ] . substring ( 0 , j ) ] = urlArgs2 [ i ] . substring ( j + 1 ) ; } }
if ( ( urlArgs [ 'phone' ] != null ) && ( urlArgs [ 'phone' ] . indexOf ( '|' ) >= 0 ) ) return ;
if ( ( urlArgs [ 'apikey' ] != null ) && ( urlArgs [ 'apikey' ] . indexOf ( '|' ) >= 0 ) ) return ;
2023-10-28 19:27:11 +01:00
if ( ( urlArgs [ 'user' ] != null ) && ( urlArgs [ 'user' ] . indexOf ( '|' ) >= 0 ) ) return ;
// Signal Messenger, Whatapp, Facebook and Telegram
2022-10-28 17:44:06 -07:00
if ( url . path . startsWith ( '/signal' ) && ( urlArgs [ 'phone' ] != null ) && ( urlArgs [ 'apikey' ] != null ) ) { return 'callmebot:signal|' + urlArgs [ 'phone' ] + '|' + urlArgs [ 'apikey' ] ; }
if ( url . path . startsWith ( '/whatsapp' ) && ( urlArgs [ 'phone' ] != null ) && ( urlArgs [ 'apikey' ] != null ) ) { return 'callmebot:whatsapp|' + urlArgs [ 'phone' ] + '|' + urlArgs [ 'apikey' ] ; }
if ( url . path . startsWith ( '/facebook' ) && ( urlArgs [ 'apikey' ] != null ) ) { return 'callmebot:facebook|' + urlArgs [ 'apikey' ] ; }
2023-10-28 19:27:11 +01:00
if ( url . path . startsWith ( '/text' ) && ( urlArgs [ 'user' ] != null ) ) { return 'callmebot:telegram|' + urlArgs [ 'user' ] ; }
2022-10-28 17:44:06 -07:00
return null ;
}
2022-10-22 07:23:55 -07:00
// Get the correct SMS template
function getTemplate ( templateNumber , domain , lang ) {
parent . debug ( 'email' , 'Getting SMS template #' + templateNumber + ', lang: ' + lang ) ;
if ( Array . isArray ( lang ) ) { lang = lang [ 0 ] ; } // TODO: For now, we only use the first language given.
var r = { } , emailsPath = null ;
if ( ( domain != null ) && ( domain . webemailspath != null ) ) { emailsPath = domain . webemailspath ; }
else if ( obj . parent . webEmailsOverridePath != null ) { emailsPath = obj . parent . webEmailsOverridePath ; }
else if ( obj . parent . webEmailsPath != null ) { emailsPath = obj . parent . webEmailsPath ; }
if ( ( emailsPath == null ) || ( obj . parent . fs . existsSync ( emailsPath ) == false ) ) { return null }
// Get the non-english email if needed
var txtfile = null ;
if ( ( lang != null ) && ( lang != 'en' ) ) {
var translationsPath = obj . parent . path . join ( emailsPath , 'translations' ) ;
var translationsPathTxt = obj . parent . path . join ( emailsPath , 'translations' , 'sms-messages_' + lang + '.txt' ) ;
if ( obj . parent . fs . existsSync ( translationsPath ) && obj . parent . fs . existsSync ( translationsPathTxt ) ) {
txtfile = obj . parent . fs . readFileSync ( translationsPathTxt ) . toString ( ) ;
}
}
// Get the english email
if ( txtfile == null ) {
var pathTxt = obj . parent . path . join ( emailsPath , 'sms-messages.txt' ) ;
if ( obj . parent . fs . existsSync ( pathTxt ) ) {
txtfile = obj . parent . fs . readFileSync ( pathTxt ) . toString ( ) ;
}
}
// No email templates
if ( txtfile == null ) { return null ; }
// Decode the TXT file
var lines = txtfile . split ( '\r\n' ) . join ( '\n' ) . split ( '\n' )
if ( lines . length <= templateNumber ) return null ;
return lines [ templateNumber ] ;
}
2022-10-22 09:20:00 -07:00
// Send messaging account verification
obj . sendMessagingCheck = function ( domain , to , verificationCode , language , func ) {
2022-10-22 07:23:55 -07:00
parent . debug ( 'email' , "Sending verification message to " + to ) ;
var sms = getTemplate ( 0 , domain , language ) ;
if ( sms == null ) { parent . debug ( 'email' , "Error: Failed to get SMS template" ) ; return ; } // No SMS template found
// Setup the template
sms = sms . split ( '[[0]]' ) . join ( domain . title ? domain . title : 'MeshCentral' ) ;
sms = sms . split ( '[[1]]' ) . join ( verificationCode ) ;
2022-10-22 09:20:00 -07:00
// Send the message
2022-10-30 11:28:24 -07:00
obj . sendMessage ( to , sms , domain , func ) ;
2022-10-22 07:23:55 -07:00
} ;
2022-10-22 09:20:00 -07:00
// Send 2FA verification
2022-10-22 07:23:55 -07:00
obj . sendToken = function ( domain , to , verificationCode , language , func ) {
parent . debug ( 'email' , "Sending login token message to " + to ) ;
var sms = getTemplate ( 1 , domain , language ) ;
if ( sms == null ) { parent . debug ( 'email' , "Error: Failed to get SMS template" ) ; return ; } // No SMS template found
// Setup the template
sms = sms . split ( '[[0]]' ) . join ( domain . title ? domain . title : 'MeshCentral' ) ;
sms = sms . split ( '[[1]]' ) . join ( verificationCode ) ;
2022-10-22 09:20:00 -07:00
// Send the message
2022-10-30 11:28:24 -07:00
obj . sendMessage ( to , sms , domain , func ) ;
2022-10-22 07:23:55 -07:00
} ;
2022-11-01 22:39:59 -07:00
// Send device state change notification
obj . sendDeviceNotify = function ( domain , username , to , connections , disconnections , lang ) {
if ( to == null ) return ;
parent . debug ( 'email' , "Sending device state change message to " + to ) ;
2022-11-01 23:48:21 -07:00
// Format the message
2022-11-01 22:39:59 -07:00
var sms = [ ] ;
if ( connections . length > 0 ) { sms . push ( 'Connections: ' + connections . join ( ', ' ) ) ; } // TODO: Translate 'Connections: '
if ( disconnections . length > 0 ) { sms . push ( 'Disconnections: ' + disconnections . join ( ', ' ) ) ; } // TODO: Translate 'Disconnections: '
if ( sms . length == 0 ) return ;
sms = sms . join ( ' - ' ) ;
if ( sms . length > 1000 ) { sms = sms . substring ( 0 , 997 ) + '...' ; } // Limit messages to 1000 characters
// Send the message
obj . sendMessage ( to , sms , domain , null ) ;
} ;
2022-11-01 23:48:21 -07:00
// Send help request notification
obj . sendDeviceHelpRequest = function ( domain , username , to , devicename , nodeid , helpusername , helprequest , lang ) {
if ( to == null ) return ;
parent . debug ( 'email' , "Sending device help request message to " + to ) ;
// Format the message
var sms = "Help Request from " + devicename + ': ' + helprequest ; // TODO: Translate 'Help Request from {0}:'
if ( sms . length > 1000 ) { sms = sms . substring ( 0 , 997 ) + '...' ; } // Limit messages to 1000 characters
// Send the message
obj . sendMessage ( to , sms , domain , null ) ;
}
2022-11-01 22:39:59 -07:00
//
2022-12-16 09:03:45 -05:00
// Device connection and disconnection notifications
2022-11-01 22:39:59 -07:00
//
obj . deviceNotifications = { } ; // UserId --> { timer, nodes: nodeid --> connectType }
// A device connected and a user needs to be notified about it.
obj . notifyDeviceConnect = function ( user , meshid , nodeid , connectTime , connectType , powerState , serverid , extraInfo ) {
const mesh = parent . webserver . meshes [ meshid ] ;
if ( mesh == null ) return ;
// Add the user and start a timer
if ( obj . deviceNotifications [ user . _id ] == null ) {
obj . deviceNotifications [ user . _id ] = { nodes : { } } ;
obj . deviceNotifications [ user . _id ] . timer = setTimeout ( function ( ) { sendDeviceNotifications ( user . _id ) ; } , 1 * 60 * 1000 ) ; // 1 minute before message is sent
}
// Add the device
if ( obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] == null ) {
obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] = { c : connectType } ; // This device connection need to be added
} else {
const info = obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] ;
if ( ( info . d != null ) && ( ( info . d & connectType ) != 0 ) ) {
info . d -= connectType ; // This device disconnect cancels out a device connection
if ( ( ( info . c == null ) || ( info . c == 0 ) ) && ( ( info . d == null ) || ( info . d == 0 ) ) ) {
// This device no longer needs a notification
delete obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] ;
if ( Object . keys ( obj . deviceNotifications [ user . _id ] . nodes ) . length == 0 ) {
// This user no longer needs a notification
clearTimeout ( obj . deviceNotifications [ user . _id ] . timer ) ;
delete obj . deviceNotifications [ user . _id ] ;
}
return ;
}
} else {
if ( info . c != null ) {
info . c |= connectType ; // This device disconnect needs to be added
} else {
info . c = connectType ; // This device disconnect needs to be added
}
}
}
// Set the device group name
if ( ( extraInfo != null ) && ( extraInfo . name != null ) ) { obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] . nn = extraInfo . name ; }
obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] . mn = mesh . name ;
}
// Cancel a device disconnect notification
obj . cancelNotifyDeviceDisconnect = function ( user , meshid , nodeid , connectTime , connectType , powerState , serverid , extraInfo ) {
const mesh = parent . webserver . meshes [ meshid ] ;
if ( mesh == null ) return ;
if ( ( obj . deviceNotifications [ user . _id ] != null ) && ( obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] != null ) ) {
const info = obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] ;
if ( ( info . d != null ) && ( ( info . d & connectType ) != 0 ) ) {
info . d -= connectType ; // This device disconnect cancels out a device connection
if ( ( ( info . c == null ) || ( info . c == 0 ) ) && ( ( info . d == null ) || ( info . d == 0 ) ) ) {
// This device no longer needs a notification
delete obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] ;
if ( Object . keys ( obj . deviceNotifications [ user . _id ] . nodes ) . length == 0 ) {
// This user no longer needs a notification
clearTimeout ( obj . deviceNotifications [ user . _id ] . timer ) ;
delete obj . deviceNotifications [ user . _id ] ;
}
}
}
}
}
// A device disconnected and a user needs to be notified about it.
obj . notifyDeviceDisconnect = function ( user , meshid , nodeid , connectTime , connectType , powerState , serverid , extraInfo ) {
const mesh = parent . webserver . meshes [ meshid ] ;
if ( mesh == null ) return ;
// Add the user and start a timer
if ( obj . deviceNotifications [ user . _id ] == null ) {
obj . deviceNotifications [ user . _id ] = { nodes : { } } ;
obj . deviceNotifications [ user . _id ] . timer = setTimeout ( function ( ) { sendDeviceNotifications ( user . _id ) ; } , 1 * 60 * 1000 ) ; // 1 minute before message is sent
}
// Add the device
if ( obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] == null ) {
obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] = { d : connectType } ; // This device disconnect need to be added
} else {
const info = obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] ;
if ( ( info . c != null ) && ( ( info . c & connectType ) != 0 ) ) {
info . c -= connectType ; // This device disconnect cancels out a device connection
if ( ( ( info . d == null ) || ( info . d == 0 ) ) && ( ( info . c == null ) || ( info . c == 0 ) ) ) {
// This device no longer needs a notification
delete obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] ;
if ( Object . keys ( obj . deviceNotifications [ user . _id ] . nodes ) . length == 0 ) {
// This user no longer needs a notification
clearTimeout ( obj . deviceNotifications [ user . _id ] . timer ) ;
delete obj . deviceNotifications [ user . _id ] ;
}
return ;
}
} else {
if ( info . d != null ) {
info . d |= connectType ; // This device disconnect needs to be added
} else {
info . d = connectType ; // This device disconnect needs to be added
}
}
}
// Set the device group name
if ( ( extraInfo != null ) && ( extraInfo . name != null ) ) { obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] . nn = extraInfo . name ; }
obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] . mn = mesh . name ;
}
// Cancel a device connect notification
obj . cancelNotifyDeviceConnect = function ( user , meshid , nodeid , connectTime , connectType , powerState , serverid , extraInfo ) {
const mesh = parent . webserver . meshes [ meshid ] ;
if ( mesh == null ) return ;
if ( ( obj . deviceNotifications [ user . _id ] != null ) && ( obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] != null ) ) {
const info = obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] ;
if ( ( info . c != null ) && ( ( info . c & connectType ) != 0 ) ) {
info . c -= connectType ; // This device disconnect cancels out a device connection
if ( ( ( info . d == null ) || ( info . d == 0 ) ) && ( ( info . c == null ) || ( info . c == 0 ) ) ) {
// This device no longer needs a notification
delete obj . deviceNotifications [ user . _id ] . nodes [ nodeid ] ;
if ( Object . keys ( obj . deviceNotifications [ user . _id ] . nodes ) . length == 0 ) {
// This user no longer needs a notification
clearTimeout ( obj . deviceNotifications [ user . _id ] . timer ) ;
delete obj . deviceNotifications [ user . _id ] ;
}
}
}
}
}
// Send a notification about device connections and disconnections to a user
function sendDeviceNotifications ( userid ) {
if ( obj . deviceNotifications [ userid ] == null ) return ;
clearTimeout ( obj . deviceNotifications [ userid ] . timer ) ;
var connections = [ ] ;
var disconnections = [ ] ;
for ( var nodeid in obj . deviceNotifications [ userid ] . nodes ) {
var info = obj . deviceNotifications [ userid ] . nodes [ nodeid ] ;
if ( ( info . c != null ) && ( info . c > 0 ) && ( info . nn != null ) && ( info . mn != null ) ) {
/ *
var c = [ ] ;
if ( info . c & 1 ) { c . push ( "Agent" ) ; }
if ( info . c & 2 ) { c . push ( "CIRA" ) ; }
if ( info . c & 4 ) { c . push ( "AMT" ) ; }
if ( info . c & 8 ) { c . push ( "AMT-Relay" ) ; }
if ( info . c & 16 ) { c . push ( "MQTT" ) ; }
connections . push ( info . mn + ', ' + info . nn + ': ' + c . join ( ', ' ) ) ;
* /
if ( info . c & 1 ) { connections . push ( info . nn ) ; }
}
if ( ( info . d != null ) && ( info . d > 0 ) && ( info . nn != null ) && ( info . mn != null ) ) {
/ *
var d = [ ] ;
if ( info . d & 1 ) { d . push ( "Agent" ) ; }
if ( info . d & 2 ) { d . push ( "CIRA" ) ; }
if ( info . d & 4 ) { d . push ( "AMT" ) ; }
if ( info . d & 8 ) { d . push ( "AMT-Relay" ) ; }
if ( info . d & 16 ) { d . push ( "MQTT" ) ; }
disconnections . push ( info . mn + ', ' + info . nn + ': ' + d . join ( ', ' ) ) ;
* /
if ( info . d & 1 ) { disconnections . push ( info . nn ) ; }
}
}
// Sort the notifications
connections . sort ( sortCollator . compare ) ;
disconnections . sort ( sortCollator . compare ) ;
// Get the user and domain
const user = parent . webserver . users [ userid ] ;
2023-09-14 14:55:49 +01:00
if ( ( user == null ) || ( user . msghandle == null ) ) return ;
2022-11-01 22:39:59 -07:00
const domain = obj . parent . config . domains [ user . domain ] ;
if ( domain == null ) return ;
// Send the message
obj . sendDeviceNotify ( domain , user . name , user . msghandle , connections , disconnections , user . llang ) ;
// Clean up
delete obj . deviceNotifications [ userid ] ;
}
2022-10-22 07:23:55 -07:00
return obj ;
} ;
2022-10-22 14:51:16 -07:00
// Called to setup the Telegram session key
module . exports . SetupTelegram = async function ( parent ) {
// If basic telegram values are not setup, instruct the user on how to get them.
if ( ( typeof parent . config . messaging != 'object' ) || ( typeof parent . config . messaging . telegram != 'object' ) || ( typeof parent . config . messaging . telegram . apiid != 'number' ) || ( typeof parent . config . messaging . telegram . apihash != 'string' ) ) {
console . log ( 'Login to your Telegram account at this URL: https://my.telegram.org/.' ) ;
console . log ( 'Click "API development tools" and fill your application details (only app title and short name required).' ) ;
console . log ( 'Click "Create application"' ) ;
console . log ( 'Set this apiid and apihash values in the messaging section of the config.json like this:' ) ;
console . log ( '{' ) ;
console . log ( ' "messaging": {' ) ;
console . log ( ' "telegram": {' ) ;
console . log ( ' "apiid": 123456,' ) ;
console . log ( ' "apihash": "123456abcdfg"' ) ;
console . log ( ' }' ) ;
console . log ( ' }' ) ;
console . log ( '}' ) ;
console . log ( 'Then, run --setuptelegram again to continue.' ) ;
process . exit ( ) ;
return ;
}
// If the session value is missing, perform the process to get it
2022-10-22 22:05:42 -07:00
if ( ( ( parent . config . messaging . telegram . session == null ) || ( parent . config . messaging . telegram . session == '' ) || ( typeof parent . config . messaging . telegram . session != 'string' ) ) && ( ( parent . config . messaging . telegram . bottoken == null ) || ( parent . config . messaging . telegram . bottoken == '' ) || ( typeof parent . config . messaging . telegram . bottoken != 'string' ) ) ) {
if ( parent . args . setuptelegram == 'user' ) {
const { TelegramClient } = require ( 'telegram' ) ;
const { StringSession } = require ( 'telegram/sessions' ) ;
const { Logger } = require ( 'telegram/extensions/Logger' ) ;
const logger = new Logger ( { LogLevel : 'none' } ) ;
const input = require ( 'input' ) ;
const stringSession = new StringSession ( '' ) ;
const client = new TelegramClient ( stringSession , parent . config . messaging . telegram . apiid , parent . config . messaging . telegram . apihash , { connectionRetries : 5 , baseLogger : logger } ) ;
await client . start ( {
phoneNumber : async function ( ) { return await input . text ( "Please enter your number (+1-111-222-3333): " ) ; } ,
password : async function ( ) { return await input . text ( "Please enter your password: " ) ; } ,
phoneCode : async function ( ) { return await input . text ( "Please enter the code you received: " ) ; } ,
onError : function ( err ) { console . log ( 'Telegram error' , err ) ; }
} ) ;
console . log ( 'Set this session value in the messaging section of the config.json like this:' ) ;
console . log ( '{' ) ;
console . log ( ' "messaging": {' ) ;
console . log ( ' "telegram": {' ) ;
console . log ( ' "apiid": ' + parent . config . messaging . telegram . apiid + ',' ) ;
console . log ( ' "apihash": "' + parent . config . messaging . telegram . apihash + '",' ) ;
console . log ( ' "session": "' + client . session . save ( ) + '"' ) ;
console . log ( ' }' ) ;
console . log ( ' }' ) ;
console . log ( '}' ) ;
process . exit ( ) ;
} else if ( parent . args . setuptelegram == 'bot' ) {
console . log ( 'Login to your Telegram account, search for "BotFather", message him and create a bot.' ) ;
console . log ( 'Once you get the HTTP API token, add it in the config.json as "bottoken" like so:' ) ;
console . log ( '{' ) ;
console . log ( ' "messaging": {' ) ;
console . log ( ' "telegram": {' ) ;
console . log ( ' "apiid": ' + parent . config . messaging . telegram . apiid + ',' ) ;
console . log ( ' "apihash": "' + parent . config . messaging . telegram . apihash + '",' ) ;
console . log ( ' "bottoken": "00000000:aaaaaaaaaaaaaaaaaaaaaaaa"' ) ;
console . log ( ' }' ) ;
console . log ( ' }' ) ;
console . log ( '}' ) ;
process . exit ( ) ;
} else {
console . log ( 'run "--setuptelegram bot" to setup Telegram login as a bot (typical).' ) ;
console . log ( 'run "--setuptelegram user" to setup Telegram login as a user.' ) ;
process . exit ( ) ;
}
2022-10-22 14:51:16 -07:00
}
// All Telegram values seem ok
console . log ( 'Telegram seems to be configured correctly in the config.json, no need to run --setuptelegram.' ) ;
process . exit ( ) ;
} ;