mirror of
				https://github.com/Ylianst/MeshCentral.git
				synced 2025-03-09 15:40:18 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			669 lines
		
	
	
	
		
			38 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			669 lines
		
	
	
	
		
			38 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /**
 | |
| * @description Meshcentral Intel AMT MPS server
 | |
| * @author Ylian Saint-Hilaire
 | |
| * @version v0.0.1
 | |
| */
 | |
| 
 | |
| // Construct a Intel AMT MPS server object
 | |
| module.exports.CreateMpsServer = function (parent, db, args, certificates) {
 | |
|     var obj = {};
 | |
|     obj.parent = parent;
 | |
|     obj.db = db;
 | |
|     obj.args = args;
 | |
|     obj.certificates = certificates;
 | |
|     obj.ciraConnections = {};
 | |
|     var common = require('./common.js');
 | |
|     var net = require('net');
 | |
|     var tls = require('tls');
 | |
| 
 | |
|     if (obj.args.tlsoffload) {
 | |
|         obj.server = net.createServer(onConnection);
 | |
|     } else {
 | |
|         obj.server = tls.createServer({ key: certificates.mps.key, cert: certificates.mps.cert, requestCert: true }, onConnection);
 | |
|     }
 | |
| 
 | |
|     obj.server.listen(args.mpsport, function () { console.log('MeshCentral Intel(R) AMT server running on ' + certificates.CommonName + ':' + args.mpsport + '.'); }).on('error', function (err) { console.error('ERROR: MeshCentral Intel(R) AMT server port ' + args.mpsport + ' is not available.'); if (args.exactports) { process.exit(); } });
 | |
|     
 | |
|     var APFProtocol = {
 | |
|         UNKNOWN: 0,
 | |
|         DISCONNECT: 1,
 | |
|         SERVICE_REQUEST: 5,
 | |
|         SERVICE_ACCEPT: 6,
 | |
|         USERAUTH_REQUEST: 50,
 | |
|         USERAUTH_FAILURE: 51,
 | |
|         USERAUTH_SUCCESS: 52,
 | |
|         GLOBAL_REQUEST: 80,
 | |
|         REQUEST_SUCCESS: 81,
 | |
|         REQUEST_FAILURE: 82,
 | |
|         CHANNEL_OPEN: 90,
 | |
|         CHANNEL_OPEN_CONFIRMATION: 91,
 | |
|         CHANNEL_OPEN_FAILURE: 92,
 | |
|         CHANNEL_WINDOW_ADJUST: 93,
 | |
|         CHANNEL_DATA: 94,
 | |
|         CHANNEL_CLOSE: 97,
 | |
|         PROTOCOLVERSION: 192,
 | |
|         KEEPALIVE_REQUEST: 208,
 | |
|         KEEPALIVE_REPLY: 209,
 | |
|         KEEPALIVE_OPTIONS_REQUEST: 210,
 | |
|         KEEPALIVE_OPTIONS_REPLY: 211
 | |
|     }
 | |
|     
 | |
|     var APFDisconnectCode = {
 | |
|         HOST_NOT_ALLOWED_TO_CONNECT: 1,
 | |
|         PROTOCOL_ERROR: 2,
 | |
|         KEY_EXCHANGE_FAILED: 3,
 | |
|         RESERVED: 4,
 | |
|         MAC_ERROR: 5,
 | |
|         COMPRESSION_ERROR: 6,
 | |
|         SERVICE_NOT_AVAILABLE: 7,
 | |
|         PROTOCOL_VERSION_NOT_SUPPORTED: 8,
 | |
|         HOST_KEY_NOT_VERIFIABLE: 9,
 | |
|         CONNECTION_LOST: 10,
 | |
|         BY_APPLICATION: 11,
 | |
|         TOO_MANY_CONNECTIONS: 12,
 | |
|         AUTH_CANCELLED_BY_USER: 13,
 | |
|         NO_MORE_AUTH_METHODS_AVAILABLE: 14,
 | |
|         INVALID_CREDENTIALS: 15,
 | |
|         CONNECTION_TIMED_OUT: 16,
 | |
|         BY_POLICY: 17,
 | |
|         TEMPORARILY_UNAVAILABLE: 18
 | |
|     }
 | |
|     
 | |
|     var APFChannelOpenFailCodes = {
 | |
|         ADMINISTRATIVELY_PROHIBITED: 1,
 | |
|         CONNECT_FAILED: 2,
 | |
|         UNKNOWN_CHANNEL_TYPE: 3,
 | |
|         RESOURCE_SHORTAGE: 4,
 | |
|     }
 | |
|     
 | |
|     var APFChannelOpenFailureReasonCode = {
 | |
|         AdministrativelyProhibited: 1,
 | |
|         ConnectFailed: 2,
 | |
|         UnknownChannelType: 3,
 | |
|         ResourceShortage: 4,
 | |
|     }
 | |
|     
 | |
|     function onConnection(socket) {
 | |
|         if (obj.args.tlsoffload) {
 | |
|             socket.tag = { first: true, clientCert: null, accumulator: "", activetunnels: 0, boundPorts: [], socket: socket, host: null, nextchannelid: 4, channels: {}, nextsourceport: 0 };
 | |
|         } else {
 | |
|             socket.tag = { first: true, clientCert: socket.getPeerCertificate(true), accumulator: "", activetunnels: 0, boundPorts: [], socket: socket, host: null, nextchannelid: 4, channels: {}, nextsourceport: 0 };
 | |
|         }
 | |
|         socket.setEncoding('binary');
 | |
|         Debug(1, 'MPS:New CIRA connection');
 | |
|         
 | |
|         socket.addListener("data", function (data) {
 | |
|             if (args.mpsdebug) { var buf = new Buffer(data, "binary"); console.log('MPS <-- (' + buf.length + '):' + buf.toString('hex')); } // Print out received bytes
 | |
|             socket.tag.accumulator += data;
 | |
|             
 | |
|             // Detect if this is an HTTPS request, if it is, return a simple answer and disconnect. This is useful for debugging access to the MPS port.
 | |
|             if (socket.tag.first == true) {
 | |
|                 if (socket.tag.accumulator.length < 3) return;
 | |
|                 //if (!socket.tag.clientCert.subject) { console.log("MPS Connection, no client cert: " + socket.remoteAddress); socket.write('HTTP/1.1 200 OK\r\nContent-Type: text/plain\r\nConnection: close\r\n\r\nMeshCentral2 MPS server.\r\nNo client certificate given.'); socket.end(); return; }
 | |
|                 if (socket.tag.accumulator.substring(0, 3) == 'GET') { console.log("MPS Connection, HTTP GET detected: " + socket.remoteAddress); socket.write('HTTP/1.1 200 OK\r\nContent-Type: text/html\r\nConnection: close\r\n\r\n<!DOCTYPE html><html><head><meta charset="UTF-8"></head><body>MeshCentral2 MPS server.<br />Intel® AMT computers should connect here.</body></html>'); socket.end(); return; }
 | |
|                 socket.tag.first = false;
 | |
|                 
 | |
|                 // Setup this node with certificate authentication
 | |
|                 if (socket.tag.clientCert && socket.tag.clientCert.subject && socket.tag.clientCert.subject.O && socket.tag.clientCert.subject.O.length == 64) {
 | |
|                     // This is a node where the MeshID is indicated within the CIRA certificate
 | |
|                     var domainid = '', meshid;
 | |
|                     var xx = socket.tag.clientCert.subject.O.split('/');
 | |
|                     if (xx.length == 1) { meshid = xx[0]; } else { domainid = xx[0].toLowerCase(); meshid = xx[1]; }
 | |
| 
 | |
|                     socket.tag.domainid = domainid;
 | |
|                     socket.tag.meshid = 'mesh/' + domainid + '/' + meshid;
 | |
|                     socket.tag.nodeid = 'node/' + domainid + '/' + require('crypto').createHash('sha384').update(common.hex2rstr(socket.tag.clientCert.modulus, 'binary')).digest('base64').replace(/\+/g, '@').replace(/\//g, '$');
 | |
|                     socket.tag.name = socket.tag.clientCert.subject.CN;
 | |
|                     socket.tag.connectTime = Date.now();
 | |
|                     socket.tag.host = '';
 | |
| 
 | |
|                     // Fetch the mesh
 | |
|                     obj.db.Get(socket.tag.meshid, function (err, meshes) {
 | |
|                         if (meshes.length == 1) {
 | |
|                             var mesh = meshes[0];
 | |
|                             obj.db.Get(socket.tag.nodeid, function (err, nodes) {
 | |
|                                 if (nodes.length == 0) {
 | |
|                                     if (mesh.mtype == 1) {
 | |
|                                         // Node is not in the database, add it. Credentials will be empty until added by the user.
 | |
|                                         var device = { type: 'node', mtype: 1, _id: socket.tag.nodeid, meshid: socket.tag.meshid, name: socket.tag.name, host: null, domain: domainid, intelamt: { user: '', pass: '', tls: 0 } };
 | |
|                                         obj.db.Set(device);
 | |
| 
 | |
|                                         // Event the new node
 | |
|                                         var device2 = common.Clone(device);
 | |
|                                         if (device2.intelamt.pass != undefined) delete device2.intelamt.pass; // Remove the Intel AMT password before eventing this.
 | |
|                                         var change = 'CIRA added device ' + socket.tag.name + ' to mesh ' + mesh.name;
 | |
|                                         obj.parent.DispatchEvent(['*', socket.tag.meshid], obj, { etype: 'node', action: 'addnode', node: device2, msg: change, domain: domainid })
 | |
|                                     } else {
 | |
|                                         // New CIRA connection for unknown node, disconnect.
 | |
|                                         console.log('CIRA connection for unknown node with incorrect mesh type. meshid: ' + socket.tag.meshid);
 | |
|                                         socket.end();
 | |
|                                         return;
 | |
|                                     }
 | |
|                                 } else {
 | |
|                                     // Node is already present
 | |
|                                     var node = nodes[0];
 | |
|                                     if (node.intelamt != undefined) { socket.tag.host = node.intelamt.host; }
 | |
|                                 }
 | |
| 
 | |
|                                 // Add the connection to the MPS connection list
 | |
|                                 obj.ciraConnections[socket.tag.nodeid] = socket;
 | |
|                                 obj.parent.SetConnectivityState(socket.tag.meshid, socket.tag.nodeid, socket.tag.connectTime, 2, 7); // TODO: Right now report power state as "present" (7) until we can poll.
 | |
|                             });
 | |
|                         } else {
 | |
|                             console.log('ERROR: Intel AMT CIRA connected with unknown meshid: ' + socket.tag.meshid);
 | |
|                             socket.end();
 | |
|                             return;
 | |
|                         }
 | |
|                     });
 | |
|                 } else {
 | |
|                     // This node connected without certificate authentication, use password auth
 | |
|                     //console.log('Intel AMT CIRA connected without certificate authentication');
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             try {
 | |
|                 // Parse all of the APF data we can
 | |
|                 var l = 0;
 | |
|                 do { l = ProcessCommand(socket); if (l > 0) { socket.tag.accumulator = socket.tag.accumulator.substring(l); } } while (l > 0);
 | |
|                 if (l < 0) { socket.end(); }
 | |
|             } catch (e) {
 | |
|                 console.log(e);
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         // Process one AFP command
 | |
|         function ProcessCommand(socket) {
 | |
|             var cmd = socket.tag.accumulator.charCodeAt(0);
 | |
|             var len = socket.tag.accumulator.length;
 | |
|             var data = socket.tag.accumulator;
 | |
|             if (len == 0) { return 0; }
 | |
| 
 | |
|             switch (cmd) {
 | |
|                 case APFProtocol.KEEPALIVE_REQUEST: {
 | |
|                     if (len < 5) return 0;
 | |
|                     Debug(3, 'MPS:KEEPALIVE_REQUEST');
 | |
|                     SendKeepAliveReply(socket, common.ReadInt(data, 1));
 | |
|                     return 5;
 | |
|                 }
 | |
|                 case APFProtocol.KEEPALIVE_REPLY: {
 | |
|                     if (len < 5) return 0;
 | |
|                     Debug(3, 'MPS:KEEPALIVE_REPLY');
 | |
|                     return 5;
 | |
|                 }
 | |
|                 case APFProtocol.PROTOCOLVERSION: {
 | |
|                     if (len < 93) return 0;
 | |
|                     socket.tag.MajorVersion = common.ReadInt(data, 1);
 | |
|                     socket.tag.MinorVersion = common.ReadInt(data, 5);
 | |
|                     socket.tag.SystemId = guidToStr(common.rstr2hex(data.substring(13, 29))).toLowerCase();
 | |
|                     Debug(3, 'MPS:PROTOCOLVERSION', socket.tag.MajorVersion, socket.tag.MinorVersion, socket.tag.SystemId);
 | |
|                     return 93;
 | |
|                 }
 | |
|                 case APFProtocol.USERAUTH_REQUEST: {
 | |
|                     if (len < 13) return 0;
 | |
|                     var usernameLen = common.ReadInt(data, 1);
 | |
|                     var username = data.substring(5, 5 + usernameLen);
 | |
|                     var serviceNameLen = common.ReadInt(data, 5 + usernameLen);
 | |
|                     var serviceName = data.substring(9 + usernameLen, 9 + usernameLen + serviceNameLen);
 | |
|                     var methodNameLen = common.ReadInt(data, 9 + usernameLen + serviceNameLen);
 | |
|                     var methodName = data.substring(13 + usernameLen + serviceNameLen, 13 + usernameLen + serviceNameLen + methodNameLen);
 | |
|                     var passwordLen = 0, password = null;
 | |
|                     if (methodName == 'password') {
 | |
|                         passwordLen = common.ReadInt(data, 14 + usernameLen + serviceNameLen + methodNameLen);
 | |
|                         password = data.substring(18 + usernameLen + serviceNameLen + methodNameLen, 18 + usernameLen + serviceNameLen + methodNameLen + passwordLen);
 | |
|                     }
 | |
|                     Debug(3, 'MPS:USERAUTH_REQUEST user=' + username + ', service=' + serviceName + ', method=' + methodName + ', password=' + password);
 | |
| 
 | |
|                     // Check the CIRA password
 | |
|                     if ((args.mpspass != undefined) && (password != args.mpspass)) { Debug(1, 'MPS:Incorrect password', username, password); SendUserAuthFail(socket); return -1; }
 | |
| 
 | |
|                     // Check the CIRA username, which should be the start of the MeshID.
 | |
|                     if (usernameLen != 16) { Debug(1, 'MPS:Username length not 16', username, password); SendUserAuthFail(socket); return -1; }
 | |
|                     var meshIdStart = '/' + username;
 | |
|                     obj.db.GetAllType('mesh', function (err, docs) {
 | |
|                         var mesh = null;
 | |
|                         for (var i in docs) { if (docs[i]._id.indexOf(meshIdStart) > 0) { mesh = docs[i]; break; } }
 | |
|                         if (mesh == null) { Debug(1, 'MPS:Mesh not found', username, password);SendUserAuthFail(socket); return -1; }
 | |
| 
 | |
|                         // Intel AMT GUID (socket.tag.SystemId) will be used at NodeID
 | |
|                         var systemid = socket.tag.SystemId.split('-').join('');
 | |
|                         var nodeid = new Buffer(systemid + systemid + systemid, 'hex').toString('base64').replace(/\+/g, '@').replace(/\//g, '$');
 | |
|                         socket.tag.name = '';
 | |
|                         socket.tag.nodeid = 'node/' + mesh.domain + '/' + nodeid; // Turn 16bit systemid guid into 48bit nodeid that is base64 encoded
 | |
|                         socket.tag.meshid = mesh._id;
 | |
| 
 | |
|                         obj.db.Get(socket.tag.nodeid, function (err, nodes) {
 | |
|                             if (nodes.length == 0) {
 | |
|                                 if (mesh.mtype == 1) {
 | |
|                                     // Node is not in the database, add it. Credentials will be empty until added by the user.
 | |
|                                     var device = { type: 'node', mtype: 1, _id: socket.tag.nodeid, meshid: socket.tag.meshid, name: socket.tag.name, host: null, domain: mesh.domain, intelamt: { user: '', pass: '', tls: 0 } };
 | |
|                                     obj.db.Set(device);
 | |
| 
 | |
|                                     // Event the new node
 | |
|                                     var device2 = common.Clone(device);
 | |
|                                     if (device2.intelamt.pass != undefined) delete device2.intelamt.pass; // Remove the Intel AMT password before eventing this.
 | |
|                                     var change = 'CIRA added device ' + socket.tag.name + ' to mesh ' + mesh.name;
 | |
|                                     obj.parent.DispatchEvent(['*', socket.tag.meshid], obj, { etype: 'node', action: 'addnode', node: device2, msg: change, domain: mesh.domain })
 | |
|                                 } else {
 | |
|                                     // New CIRA connection for unknown node, disconnect.
 | |
|                                     console.log('CIRA connection for unknown node with incorrect mesh type. meshid: ' + socket.tag.meshid);
 | |
|                                     socket.end();
 | |
|                                     return;
 | |
|                                 }
 | |
|                             } else {
 | |
|                                 // Node is already present
 | |
|                                 var node = nodes[0];
 | |
|                                 if (node.intelamt != undefined) { socket.tag.host = node.intelamt.host; }
 | |
|                             }
 | |
| 
 | |
|                             // Add the connection to the MPS connection list
 | |
|                             obj.ciraConnections[socket.tag.nodeid] = socket;
 | |
|                             obj.parent.SetConnectivityState(socket.tag.meshid, socket.tag.nodeid, socket.tag.connectTime, 2, 7); // TODO: Right now report power state as "present" (7) until we can poll.
 | |
|                             SendUserAuthSuccess(socket); // Notify the auth success on the CIRA connection
 | |
|                         });
 | |
|                     });
 | |
|                     return 18 + usernameLen + serviceNameLen + methodNameLen + passwordLen;
 | |
|                 }
 | |
|                 case APFProtocol.SERVICE_REQUEST: {
 | |
|                     if (len < 5) return 0;
 | |
|                     var serviceNameLen = common.ReadInt(data, 1);
 | |
|                     if (len < 5 + serviceNameLen) return 0;
 | |
|                     var serviceName = data.substring(5, 5 + serviceNameLen);
 | |
|                     Debug(3, 'MPS:SERVICE_REQUEST', serviceName);
 | |
|                     if (serviceName == "pfwd@amt.intel.com") { SendServiceAccept(socket, "pfwd@amt.intel.com"); }
 | |
|                     if (serviceName == "auth@amt.intel.com") { SendServiceAccept(socket, "auth@amt.intel.com"); }
 | |
|                     return 5 + serviceNameLen;
 | |
|                 }
 | |
|                 case APFProtocol.GLOBAL_REQUEST: {
 | |
|                     if (len < 14) return 0;
 | |
|                     var requestLen = common.ReadInt(data, 1);
 | |
|                     if (len < 14 + requestLen) return 0;
 | |
|                     var request = data.substring(5, 5 + requestLen);
 | |
|                     var wantResponse = data.charCodeAt(5 + requestLen);
 | |
| 
 | |
|                     if (request == "tcpip-forward") {
 | |
|                         var addrLen = common.ReadInt(data, 6 + requestLen);
 | |
|                         if (len < 14 + requestLen + addrLen) return 0;
 | |
|                         var addr = data.substring(10 + requestLen, 10 + requestLen + addrLen);
 | |
|                         var port = common.ReadInt(data, 10 + requestLen + addrLen);
 | |
|                         if (addr == '') addr = undefined;
 | |
|                         Debug(2, 'MPS:GLOBAL_REQUEST', request, addr + ':' + port);
 | |
|                         ChangeHostname(socket, addr);
 | |
|                         if (socket.tag.boundPorts.indexOf(port) == -1) { socket.tag.boundPorts.push(port); }
 | |
|                         SendTcpForwardSuccessReply(socket, port);
 | |
|                         return 14 + requestLen + addrLen;
 | |
|                     }
 | |
| 
 | |
|                     if (request == "cancel-tcpip-forward") {
 | |
|                         var addrLen = common.ReadInt(data, 6 + requestLen);
 | |
|                         if (len < 14 + requestLen + addrLen) return 0;
 | |
|                         var addr = data.substring(10 + requestLen, 10 + requestLen + addrLen);
 | |
|                         var port = common.ReadInt(data, 10 + requestLen + addrLen);
 | |
|                         Debug(2, 'MPS:GLOBAL_REQUEST', request, addr + ':' + port);
 | |
|                         var portindex = socket.tag.boundPorts.indexOf(port);
 | |
|                         if (portindex >= 0) { socket.tag.boundPorts.splice(portindex, 1); }
 | |
|                         SendTcpForwardCancelReply(socket);
 | |
|                         return 14 + requestLen + addrLen;
 | |
|                     }
 | |
| 
 | |
|                     if (request == "udp-send-to@amt.intel.com") {
 | |
|                         var addrLen = common.ReadInt(data, 6 + requestLen);
 | |
|                         if (len < 26 + requestLen + addrLen) return 0;
 | |
|                         var addr = data.substring(10 + requestLen, 10 + requestLen + addrLen);
 | |
|                         var port = common.ReadInt(data, 10 + requestLen + addrLen);
 | |
|                         var oaddrLen = common.ReadInt(data, 14 + requestLen + addrLen);
 | |
|                         if (len < 26 + requestLen + addrLen + oaddrLen) return 0;
 | |
|                         var oaddr = data.substring(18 + requestLen, 18 + requestLen + addrLen);
 | |
|                         var oport = common.ReadInt(data, 18 + requestLen + addrLen + oaddrLen);
 | |
|                         var datalen = common.ReadInt(data, 22 + requestLen + addrLen + oaddrLen);
 | |
|                         if (len < 26 + requestLen + addrLen + oaddrLen + datalen) return 0;
 | |
|                         Debug(2, 'MPS:GLOBAL_REQUEST', request, addr + ':' + port, oaddr + ':' + oport, datalen);
 | |
|                         // TODO
 | |
|                         return 26 + requestLen + addrLen + oaddrLen + datalen;
 | |
|                     }
 | |
| 
 | |
|                     return 6 + requestLen;
 | |
|                 }
 | |
|                 case APFProtocol.CHANNEL_OPEN: {
 | |
|                     if (len < 33) return 0;
 | |
|                     var ChannelTypeLength = common.ReadInt(data, 1);
 | |
|                     if (len < (33 + ChannelTypeLength)) return 0;
 | |
| 
 | |
|                     // Decode channel identifiers and window size
 | |
|                     var ChannelType = data.substring(5, 5 + ChannelTypeLength);
 | |
|                     var SenderChannel = common.ReadInt(data, 5 + ChannelTypeLength);
 | |
|                     var WindowSize = common.ReadInt(data, 9 + ChannelTypeLength);
 | |
| 
 | |
|                     // Decode the target
 | |
|                     var TargetLen = common.ReadInt(data, 17 + ChannelTypeLength);
 | |
|                     if (len < (33 + ChannelTypeLength + TargetLen)) return 0;
 | |
|                     var Target = data.substring(21 + ChannelTypeLength, 21 + ChannelTypeLength + TargetLen);
 | |
|                     var TargetPort = common.ReadInt(data, 21 + ChannelTypeLength + TargetLen);
 | |
| 
 | |
|                     // Decode the source
 | |
|                     var SourceLen = common.ReadInt(data, 25 + ChannelTypeLength + TargetLen);
 | |
|                     if (len < (33 + ChannelTypeLength + TargetLen + SourceLen)) return 0;
 | |
|                     var Source = data.substring(29 + ChannelTypeLength + TargetLen, 29 + ChannelTypeLength + TargetLen + SourceLen);
 | |
|                     var SourcePort = common.ReadInt(data, 29 + ChannelTypeLength + TargetLen + SourceLen);
 | |
|                     
 | |
|                     Debug(3, 'MPS:CHANNEL_OPEN', ChannelType, SenderChannel, WindowSize, Target + ':' + TargetPort, Source + ':' + SourcePort);
 | |
| 
 | |
|                     // Check if we understand this channel type
 | |
|                     //if (ChannelType.toLowerCase() == "direct-tcpip")
 | |
|                     {
 | |
|                         // We don't understand this channel type, send an error back
 | |
|                         SendChannelOpenFailure(socket, SenderChannel, APFChannelOpenFailureReasonCode.UnknownChannelType);
 | |
|                         return 33 + ChannelTypeLength + TargetLen + SourceLen;
 | |
|                     }
 | |
|                     
 | |
|                     /*
 | |
|                     // This is a correct connection. Lets get it setup
 | |
|                     var MeshAmtEventEndpoint = { ServerChannel: GetNextBindId(), AmtChannel: SenderChannel, MaxWindowSize: 2048, CurrentWindowSize:2048, SendWindow: WindowSize, InfoHeader: "Target: " + Target + ":" + TargetPort + ", Source: " + Source + ":" + SourcePort};
 | |
|                     // TODO: Connect this socket for a WSMAN event
 | |
|                     SendChannelOpenConfirmation(socket, SenderChannel, MeshAmtEventEndpoint.ServerChannel, MeshAmtEventEndpoint.MaxWindowSize);
 | |
|                     */
 | |
| 
 | |
|                     return 33 + ChannelTypeLength + TargetLen + SourceLen;
 | |
|                 }
 | |
|             case APFProtocol.CHANNEL_OPEN_CONFIRMATION:
 | |
|                 {
 | |
|                     if (len < 17) return 0;
 | |
|                     var RecipientChannel = common.ReadInt(data, 1);
 | |
|                     var SenderChannel = common.ReadInt(data, 5);
 | |
|                     var WindowSize = common.ReadInt(data, 9);
 | |
|                     socket.tag.activetunnels++;
 | |
|                     var cirachannel = socket.tag.channels[RecipientChannel];
 | |
|                     if (cirachannel == undefined) { console.log("MPS Error in CHANNEL_OPEN_CONFIRMATION: Unable to find channelid " + RecipientChannel); return; }
 | |
|                     cirachannel.amtchannelid = SenderChannel;
 | |
|                     cirachannel.sendcredits = cirachannel.amtCiraWindow = WindowSize;
 | |
|                     Debug(3, 'MPS:CHANNEL_OPEN_CONFIRMATION', RecipientChannel, SenderChannel, WindowSize);
 | |
|                     if (cirachannel.closing == 1) {
 | |
|                         // Close this channel
 | |
|                         SendChannelClose(cirachannel.socket, cirachannel.amtchannelid);
 | |
|                     } else {
 | |
|                         cirachannel.state = 2;
 | |
|                         // Send any pending data
 | |
|                         if (cirachannel.sendBuffer != undefined) {
 | |
|                             if (cirachannel.sendBuffer.length <= cirachannel.sendcredits) {
 | |
|                                 // Send the entire pending buffer
 | |
|                                 SendChannelData(cirachannel.socket, cirachannel.amtchannelid, cirachannel.sendBuffer);
 | |
|                                 cirachannel.sendcredits -= cirachannel.sendBuffer.length;
 | |
|                                 delete cirachannel.sendBuffer;
 | |
|                                 if (cirachannel.onSendOk) { cirachannel.onSendOk(cirachannel); }
 | |
|                             } else {
 | |
|                                 // Send a part of the pending buffer
 | |
|                                 SendChannelData(cirachannel.socket, cirachannel.amtchannelid, cirachannel.sendBuffer.substring(0, cirachannel.sendcredits));
 | |
|                                 cirachannel.sendBuffer = cirachannel.sendBuffer.substring(cirachannel.sendcredits);
 | |
|                                 cirachannel.sendcredits = 0;
 | |
|                             }
 | |
|                         }
 | |
|                         // Indicate the channel is open
 | |
|                         if (cirachannel.onStateChange) { cirachannel.onStateChange(cirachannel, cirachannel.state); }
 | |
|                     }
 | |
|                     return 17;
 | |
|                 }
 | |
|             case APFProtocol.CHANNEL_OPEN_FAILURE:
 | |
|                 {
 | |
|                     if (len < 17) return 0;
 | |
|                     var RecipientChannel = common.ReadInt(data, 1);
 | |
|                     var ReasonCode = common.ReadInt(data, 5);
 | |
|                     Debug(3, 'MPS:CHANNEL_OPEN_FAILURE', RecipientChannel, ReasonCode);
 | |
|                     var cirachannel = socket.tag.channels[RecipientChannel];
 | |
|                     if (cirachannel == undefined) { console.log("MPS Error in CHANNEL_OPEN_FAILURE: Unable to find channelid " + RecipientChannel); return; }
 | |
|                     if (cirachannel.state > 0) {
 | |
|                         cirachannel.state = 0;
 | |
|                         if (cirachannel.onStateChange) { cirachannel.onStateChange(cirachannel, cirachannel.state); }
 | |
|                         delete socket.tag.channels[RecipientChannel];
 | |
|                     }
 | |
|                     return 17;
 | |
|                 }
 | |
|             case APFProtocol.CHANNEL_CLOSE:
 | |
|                 {
 | |
|                     if (len < 5) return 0;
 | |
|                     var RecipientChannel = common.ReadInt(data, 1);
 | |
|                     Debug(3, 'MPS:CHANNEL_CLOSE', RecipientChannel);
 | |
|                     var cirachannel = socket.tag.channels[RecipientChannel];
 | |
|                     if (cirachannel == undefined) { console.log("MPS Error in CHANNEL_CLOSE: Unable to find channelid " + RecipientChannel); return; }
 | |
|                     socket.tag.activetunnels--;
 | |
|                     if (cirachannel.state > 0) {
 | |
|                         cirachannel.state = 0;
 | |
|                         if (cirachannel.onStateChange) { cirachannel.onStateChange(cirachannel, cirachannel.state); }
 | |
|                         delete socket.tag.channels[RecipientChannel];
 | |
|                     }
 | |
|                     return 5;
 | |
|                 }
 | |
|             case APFProtocol.CHANNEL_WINDOW_ADJUST:
 | |
|                 {
 | |
|                     if (len < 9) return 0;
 | |
|                     var RecipientChannel = common.ReadInt(data, 1);
 | |
|                     var ByteToAdd = common.ReadInt(data, 5);
 | |
|                     Debug(3, 'MPS:CHANNEL_WINDOW_ADJUST', RecipientChannel, ByteToAdd);
 | |
|                     var cirachannel = socket.tag.channels[RecipientChannel];
 | |
|                     if (cirachannel == undefined) { console.log("MPS Error in CHANNEL_WINDOW_ADJUST: Unable to find channelid " + RecipientChannel); return; }
 | |
|                     cirachannel.sendcredits += ByteToAdd;
 | |
|                     if (cirachannel.state == 2 && cirachannel.sendBuffer != undefined) {
 | |
|                         // Compute how much data we can send                
 | |
|                         if (cirachannel.sendBuffer.length <= cirachannel.sendcredits) {
 | |
|                             // Send the entire pending buffer
 | |
|                             SendChannelData(cirachannel.socket, cirachannel.amtchannelid, cirachannel.sendBuffer);
 | |
|                             cirachannel.sendcredits -= cirachannel.sendBuffer.length;
 | |
|                             delete cirachannel.sendBuffer;
 | |
|                             if (cirachannel.onSendOk) { cirachannel.onSendOk(cirachannel); }
 | |
|                         } else {
 | |
|                             // Send a part of the pending buffer
 | |
|                             SendChannelData(cirachannel.socket, cirachannel.amtchannelid, cirachannel.sendBuffer.substring(0, cirachannel.sendcredits));
 | |
|                             cirachannel.sendBuffer = cirachannel.sendBuffer.substring(cirachannel.sendcredits);
 | |
|                             cirachannel.sendcredits = 0;
 | |
|                         }
 | |
|                     }
 | |
|                     return 9;
 | |
|                 }
 | |
|             case APFProtocol.CHANNEL_DATA:
 | |
|                 {
 | |
|                     if (len < 9) return 0;
 | |
|                     var RecipientChannel = common.ReadInt(data, 1);
 | |
|                     var LengthOfData = common.ReadInt(data, 5);
 | |
|                     if (len < (9 + LengthOfData)) return 0;
 | |
|                     Debug(4, 'MPS:CHANNEL_DATA', RecipientChannel, LengthOfData);
 | |
|                     var cirachannel = socket.tag.channels[RecipientChannel];
 | |
|                     if (cirachannel == undefined) { console.log("MPS Error in CHANNEL_DATA: Unable to find channelid " + RecipientChannel); return; }
 | |
|                     cirachannel.amtpendingcredits += LengthOfData;
 | |
|                     if (cirachannel.onData) cirachannel.onData(cirachannel, data.substring(9, 9 + LengthOfData));
 | |
|                     if (cirachannel.amtpendingcredits > (cirachannel.ciraWindow / 2)) {
 | |
|                         SendChannelWindowAdjust(cirachannel.socket, cirachannel.amtchannelid, cirachannel.amtpendingcredits); // Adjust the buffer window
 | |
|                         cirachannel.amtpendingcredits = 0;
 | |
|                     }
 | |
|                     return 9 + LengthOfData;
 | |
|                 }
 | |
|             case APFProtocol.DISCONNECT:
 | |
|                 {
 | |
|                     if (len < 7) return 0;
 | |
|                     var ReasonCode = common.ReadInt(data, 1);
 | |
|                     Debug(3, 'MPS:DISCONNECT', ReasonCode);
 | |
|                     try { delete obj.ciraConnections[socket.tag.nodeid]; } catch (e) { }
 | |
|                     obj.parent.ClearConnectivityState(socket.tag.meshid, socket.tag.nodeid, 2);
 | |
|                     return 7;
 | |
|                 }
 | |
|             default:
 | |
|                 {
 | |
|                     Debug(1, 'MPS:Unknown CIRA command: ' + cmd);
 | |
|                     return -1;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         socket.addListener("close", function () {
 | |
|             Debug(1, 'MPS:CIRA connection closed');
 | |
|             try { delete obj.ciraConnections[socket.tag.nodeid]; } catch (e) { }
 | |
|             obj.parent.ClearConnectivityState(socket.tag.meshid, socket.tag.nodeid, 2);
 | |
|         });
 | |
|         
 | |
|         socket.addListener("error", function () {
 | |
|             //console.log("MPS Error: " + socket.remoteAddress);
 | |
|         });
 | |
| 
 | |
|     }
 | |
|     
 | |
|     // Disconnect CIRA tunnel
 | |
|     obj.close = function (socket) {
 | |
|         try { socket.close(); } catch (e) { }
 | |
|         try { delete obj.ciraConnections[socket.tag.nodeid]; } catch (e) { }
 | |
|         obj.parent.ClearConnectivityState(socket.tag.meshid, socket.tag.nodeid, 2);
 | |
|     }
 | |
| 
 | |
|     function SendServiceAccept(socket, service) {
 | |
|         Write(socket, String.fromCharCode(APFProtocol.SERVICE_ACCEPT) + common.IntToStr(service.length) + service);
 | |
|     }
 | |
|     
 | |
|     function SendTcpForwardSuccessReply(socket, port) {
 | |
|         Write(socket, String.fromCharCode(APFProtocol.REQUEST_SUCCESS) + common.IntToStr(port));
 | |
|     }
 | |
|     
 | |
|     function SendTcpForwardCancelReply(socket) {
 | |
|         Write(socket, String.fromCharCode(APFProtocol.REQUEST_SUCCESS));
 | |
|     }
 | |
|     
 | |
|     function SendKeepAliveRequest(socket, cookie) {
 | |
|         Write(socket, String.fromCharCode(APFProtocol.KEEPALIVE_REQUEST) + common.IntToStr(cookie));
 | |
|     }
 | |
| 
 | |
|     function SendKeepAliveReply(socket, cookie) {
 | |
|         Write(socket, String.fromCharCode(APFProtocol.KEEPALIVE_REPLY) + common.IntToStr(cookie));
 | |
|     }
 | |
|     
 | |
|     function SendChannelOpenFailure(socket, senderChannel, reasonCode) {
 | |
|         Write(socket, String.fromCharCode(APFProtocol.CHANNEL_OPEN_FAILURE) + common.IntToStr(senderChannel) + common.IntToStr(reasonCode) + common.IntToStr(0) + common.IntToStr(0));
 | |
|     }
 | |
|     
 | |
|     function SendChannelOpenConfirmation(socket, recipientChannelId, senderChannelId, initialWindowSize) {
 | |
|         Write(socket, String.fromCharCode(APFProtocol.CHANNEL_OPEN_CONFIRMATION) + common.IntToStr(recipientChannelId) + common.IntToStr(senderChannelId) + common.IntToStr(initialWindowSize) + common.IntToStr(-1));
 | |
|     }
 | |
|     
 | |
|     function SendChannelOpen(socket, direct, channelid, windowsize, target, targetport, source, sourceport) {
 | |
|         var connectionType = ((direct == true) ? "direct-tcpip" : "forwarded-tcpip");
 | |
|         if ((target == null) || (target == undefined)) target = ''; // TODO: Reports of target being undefined that causes target.length to fail. This is a hack.
 | |
|         Write(socket, String.fromCharCode(APFProtocol.CHANNEL_OPEN) + common.IntToStr(connectionType.length) + connectionType + common.IntToStr(channelid) + common.IntToStr(windowsize) + common.IntToStr(-1) + common.IntToStr(target.length) + target + common.IntToStr(targetport) + common.IntToStr(source.length) + source + common.IntToStr(sourceport));
 | |
|     }
 | |
|     
 | |
|     function SendChannelClose(socket, channelid) {
 | |
|         Write(socket, String.fromCharCode(APFProtocol.CHANNEL_CLOSE) + common.IntToStr(channelid));
 | |
|     }
 | |
|     
 | |
|     function SendChannelData(socket, channelid, data) {
 | |
|         Write(socket, String.fromCharCode(APFProtocol.CHANNEL_DATA) + common.IntToStr(channelid) + common.IntToStr(data.length) + data);
 | |
|     }
 | |
|     
 | |
|     function SendChannelWindowAdjust(socket, channelid, bytestoadd) {
 | |
|         Debug(3, 'MPS:SendChannelWindowAdjust', channelid, bytestoadd);
 | |
|         Write(socket, String.fromCharCode(APFProtocol.CHANNEL_WINDOW_ADJUST) + common.IntToStr(channelid) + common.IntToStr(bytestoadd));
 | |
|     }
 | |
|     
 | |
|     function SendDisconnect(socket, reasonCode) {
 | |
|         Write(socket, String.fromCharCode(APFProtocol.DISCONNECT) + common.IntToStr(ReasonCode) + common.ShortToStr(0));
 | |
|     }
 | |
| 
 | |
|     function SendUserAuthFail(socket) {
 | |
|         Write(socket, String.fromCharCode(APFProtocol.USERAUTH_FAILURE) + common.IntToStr(8) + 'password' + common.ShortToStr(0));
 | |
|     }
 | |
| 
 | |
|     function SendUserAuthSuccess(socket) {
 | |
|         Write(socket, String.fromCharCode(APFProtocol.USERAUTH_SUCCESS));
 | |
|     }
 | |
| 
 | |
|     function Write(socket, data) {
 | |
|         if (args.mpsdebug) {
 | |
|             // Print out sent bytes
 | |
|             var buf = new Buffer(data, "binary");
 | |
|             console.log('MPS --> (' + buf.length + '):' + buf.toString('hex'));
 | |
|             socket.write(buf);
 | |
|         } else {
 | |
|             socket.write(new Buffer(data, "binary"));
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     obj.SetupCiraChannel = function (socket, targetport) {
 | |
|         var sourceport = (socket.tag.nextsourceport++ % 30000) + 1024;
 | |
|         var cirachannel = { targetport: targetport, channelid: socket.tag.nextchannelid++, socket: socket, state: 1, sendcredits: 0, amtpendingcredits: 0, amtCiraWindow: 0, ciraWindow: 32768 };
 | |
|         SendChannelOpen(socket, false, cirachannel.channelid, cirachannel.ciraWindow, socket.tag.host, targetport, "1.2.3.4", sourceport);
 | |
|         
 | |
|         // This function writes data to this CIRA channel
 | |
|         cirachannel.write = function (data) {
 | |
|             if (cirachannel.state == 0) return false;
 | |
|             if (cirachannel.state == 1 || cirachannel.sendcredits == 0 || cirachannel.sendBuffer != undefined) { if (cirachannel.sendBuffer == undefined) { cirachannel.sendBuffer = data; } else { cirachannel.sendBuffer += data; } return; }
 | |
|             // Compute how much data we can send                
 | |
|             if (data.length <= cirachannel.sendcredits) {
 | |
|                 // Send the entire message
 | |
|                 SendChannelData(cirachannel.socket, cirachannel.amtchannelid, data);
 | |
|                 cirachannel.sendcredits -= data.length;
 | |
|                 return true;
 | |
|             }
 | |
|             // Send a part of the message
 | |
|             cirachannel.sendBuffer = data.substring(cirachannel.sendcredits);
 | |
|             SendChannelData(cirachannel.socket, cirachannel.amtchannelid, data.substring(0, cirachannel.sendcredits));
 | |
|             cirachannel.sendcredits = 0;
 | |
|             return false;
 | |
|         }
 | |
|         
 | |
|         // This function closes this CIRA channel
 | |
|         cirachannel.close = function () {
 | |
|             if (cirachannel.state == 0 || cirachannel.closing == 1) return;
 | |
|             if (cirachannel.state == 1) { cirachannel.closing = 1; cirachannel.state = 0; if (cirachannel.onStateChange) { cirachannel.onStateChange(cirachannel, cirachannel.state); } return; }
 | |
|             cirachannel.state = 0;
 | |
|             cirachannel.closing = 1;
 | |
|             SendChannelClose(cirachannel.socket, cirachannel.amtchannelid);
 | |
|             if (cirachannel.onStateChange) { cirachannel.onStateChange(cirachannel, cirachannel.state); }
 | |
|         }
 | |
|         
 | |
|         socket.tag.channels[cirachannel.channelid] = cirachannel;
 | |
|         return cirachannel;
 | |
|     }
 | |
| 
 | |
|     function ChangeHostname(socket, host) {
 | |
|         if (socket.tag.host == host) return; // Nothing to change
 | |
|         socket.tag.host = host;
 | |
|         
 | |
|         // Change the device
 | |
|         obj.db.Get(socket.tag.nodeid, function (err, nodes) {
 | |
|             if (nodes.length != 1) return;
 | |
|             var node = nodes[0];
 | |
|             if ((node.intelamt != undefined) && (node.intelamt.host == host)) return;
 | |
|             
 | |
|             // Get the mesh for this device
 | |
|             obj.db.Get(node.meshid, function (err, meshes) {
 | |
|                 if (meshes.length != 1) return;
 | |
|                 var mesh = meshes[0];
 | |
| 
 | |
|                 // Ready the node change event
 | |
|                 var changes = ['host'], event = { etype: 'node', action: 'changenode', nodeid: node._id };
 | |
|                 event.msg = +": ";
 | |
|                 
 | |
|                 // Make the change & save
 | |
|                 if (node.intelamt == undefined) node.intelamt = {};
 | |
|                 node.intelamt.host = host;
 | |
|                 if (node.name == '') { node.name = host.split('.')[0]; }
 | |
|                 obj.db.Set(node);
 | |
|                     
 | |
|                 // Event the node change
 | |
|                 event.msg = 'CIRA changed device ' + node.name + ' from mesh ' + mesh.name + ': ' + changes.join(', ');
 | |
|                 var node2 = common.Clone(node);
 | |
|                 if (node2.intelamt && node2.intelamt.pass) delete node2.intelamt.pass; // Remove the Intel AMT password before eventing this.
 | |
|                 event.node = node2;
 | |
|                 obj.parent.DispatchEvent(['*', node.meshid], obj, event);
 | |
|             });
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     function guidToStr(g) { return g.substring(6, 8) + g.substring(4, 6) + g.substring(2, 4) + g.substring(0, 2) + "-" + g.substring(10, 12) + g.substring(8, 10) + "-" + g.substring(14, 16) + g.substring(12, 14) + "-" + g.substring(16, 20) + "-" + g.substring(20); }
 | |
|     
 | |
|     // Debug
 | |
|     function Debug(lvl) {
 | |
|         if (lvl > obj.parent.debugLevel) return;
 | |
|         if (arguments.length == 2) { console.log(arguments[1]); }
 | |
|         else if (arguments.length == 3) { console.log(arguments[1], arguments[2]); }
 | |
|         else if (arguments.length == 4) { console.log(arguments[1], arguments[2], arguments[3]); }
 | |
|         else if (arguments.length == 5) { console.log(arguments[1], arguments[2], arguments[3], arguments[4]); }
 | |
|         else if (arguments.length == 6) { console.log(arguments[1], arguments[2], arguments[3], arguments[4], arguments[5]); }
 | |
|         else if (arguments.length == 7) { console.log(arguments[1], arguments[2], arguments[3], arguments[4], arguments[5], arguments[6]); }
 | |
|     }
 | |
| 
 | |
|     return obj;
 | |
| }
 |