MQL5Book/Web/wspubsub.js
super.admin 1c8e83ce31 convert
2025-05-30 16:09:41 +02:00

238 lines
7.3 KiB
JavaScript

//+------------------------------------------------------------------+
//| wspubsub.js |
//| Copyright 2022, MetaQuotes Ltd. |
//| https://www.mql5.com |
//+------------------------------------------------------------------+
// from command line arguments drop off two first of them (node.exe and current js-file)
const args = process.argv.slice(2);
const secure = args.length > 0 ? 'https' : 'http'; // secure connection flag
// attach dependencies
const fs = require('fs');
const http1 = require(secure);
const WebSocket = require('ws');
const crypto = require('crypto');
// attach key files for secure connections (if https used)
const options = args.length > 0 ?
{
key : fs.readFileSync(`${args[0]}.key`),
cert : fs.readFileSync(`${args[0]}.crt`)
} : null;
// create a http-server object for start page
http1.createServer(options, function (req, res)
{
console.log(req.method, req.url);
console.log(req.headers);
if(req.url == '/')
{
req.url = "wspubsub.htm";
}
fs.readFile('./' + req.url, (err, data) =>
{
if(!err)
{
var dotoffset = req.url.lastIndexOf('.');
var mimetype = dotoffset == -1 ? 'text/plain' :
{
'.htm' : 'text/html',
'.html' : 'text/html',
'.css' : 'text/css',
'.js' : 'text/javascript',
'.jpg' : 'image/jpeg',
'.png' : 'image/png',
'.ico' : 'image/x-icon',
'.gif' : 'image/gif'
}[ req.url.substr(dotoffset) ];
res.setHeader('Content-Type', mimetype == undefined ? 'text/plain' : mimetype);
res.end(data);
}
else
{
console.log('File not fount:' + req.url);
res.writeHead(404, "Not Found");
res.end();
}
});
}).listen(secure == 'https' ? 443 : 80);
// create a http-server object for websocket-server handshaking
const server = new http1.createServer(options).listen(9000);
server.on('upgrade', function(req, socket, head)
{
console.log(req.headers); // TODO: aux authorization
});
// keep track of users
const publishers = new Map();
const subscribers = new Map();
var count = 0;
// create a websocket-server object
const wsServer = new WebSocket.Server({ server });
wsServer.on('connection', function onConnect(client)
{
console.log('New user:', ++count, client.protocol);
if(client.protocol.startsWith('X-MQL5-publisher'))
{
const parts = client.protocol.split('-');
if(parts.length != 5)
{
console.log('Incomplete publisher ID');
client.send('{"origin":"Server", "msg":"Incomplete publisher ID"}');
client.close();
return;
}
client.id = parts[3];
client.key = parts[4];
if(client.id == "Server")
{
console.log('Login "Server" is reserved');
client.send('{"origin":"Server", "msg":"Login \'Server\' is reserved"}');
return;
}
if(publishers.get(client.id))
{
console.log('Publisher is already connected');
client.send('{"origin":"Server", "msg":"Publisher is already connected: ' + client.id + '"}');
return;
}
publishers.set(client.id, client);
client.send('{"origin":"Server", "msg":"Hello, publisher ' + client.id + '"}');
client.on('message', function(message)
{
try
{
console.log('%s : %s', client.id, message);
if(subscribers.get(client.id))
subscribers.get(client.id).forEach(function(elem)
{
try
{
elem.send('{"origin":"publisher ' + client.id + '", "msg":' + message + '}');
}
catch(e)
{
console.log('Disconnected:', elem.id, e);
}
});
}
catch(error)
{
console.log('Error', error);
}
});
client.on('close', function()
{
console.log('Publisher disconnected:', client.id);
if(subscribers.get(client.id))
subscribers.get(client.id).forEach(function(elem)
{
elem.close();
});
publishers.delete(client.id);
});
}
else if(client.protocol.startsWith('X-MQL5-subscriber'))
{
const parts = client.protocol.split('-');
if(parts.length != 6)
{
console.log('Incomplete subscriber ID');
client.send('{"origin":"Server", "msg":"Incomplete subscriber ID"}');
client.close();
return;
}
client.id = parts[3];
client.pub_id = parts[4];
client.access = parts[5];
if(client.id == "Server")
{
console.log('Login "Server" is reserved');
client.send('{"origin":"Server", "msg":"Login \'Server\' is reserved"}');
client.close();
return;
}
const id = client.pub_id;
var p = publishers.get(id);
if(p)
{
const check = crypto.createHash('sha256').update(id + ':' + p.key + ':' + client.id).digest('hex');
if(check != client.access)
{
console.log(`Bad credentials: '${client.access}' vs '${check}'`);
client.send('{"origin":"Server", "msg":"Bad credentials, subscriber ' + client.id + '"}');
client.close();
return;
}
var list = subscribers.get(id);
if(list == undefined)
{
list = [];
}
else
{
const found = list.find(function(el) { return el.id === client.id; });
if(found)
{
console.log(`Subscriber '${client.id}' is already connected to '${id}'`);
client.send('{"origin":"Server", "msg":"Subscriber is already connected: ' + client.id + '"}');
client.close();
return;
}
}
list.push(client);
subscribers.set(id, list);
client.send('{"origin":"Server", "msg":"Hello, subscriber ' + client.id + '"}');
p.send('{"origin":"Server", "msg":"New subscriber ' + client.id + '"}');
}
else
{
console.log('Auto-closing subscriber', client.id);
client.send('{"origin":"Server", "msg":"Publisher not found"}');
client.close();
return;
}
client.on('close', function()
{
console.log('Subscriber disconnected:', client.id);
const list = subscribers.get(client.pub_id);
if(list)
{
if(list.length > 1)
{
const filtered = list.filter(function(el) { return el !== client; });
subscribers.set(client.pub_id, filtered);
}
else
{
subscribers.delete(client.pub_id);
}
}
});
}
else
{
console.log('No X-MQL5-headers, disconnecting');
client.close();
}
});
//+------------------------------------------------------------------+