mirror of
https://github.com/Ylianst/MeshCentral.git
synced 2025-03-09 15:40:18 +00:00
parent
f88d3063fe
commit
db06ec1975
37 changed files with 28174 additions and 44 deletions
346
rdp/protocol/x224.js
Normal file
346
rdp/protocol/x224.js
Normal file
|
|
@ -0,0 +1,346 @@
|
|||
/*
|
||||
* Copyright (c) 2014-2015 Sylvain Peyrefitte
|
||||
*
|
||||
* This file is part of node-rdpjs.
|
||||
*
|
||||
* node-rdpjs is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
var inherits = require('util').inherits;
|
||||
var events = require('events');
|
||||
var type = require('../core').type;
|
||||
var log = require('../core').log;
|
||||
var error = require('../core').error;
|
||||
|
||||
/**
|
||||
* Message type present in X224 packet header
|
||||
*/
|
||||
var MessageType = {
|
||||
X224_TPDU_CONNECTION_REQUEST : 0xE0,
|
||||
X224_TPDU_CONNECTION_CONFIRM : 0xD0,
|
||||
X224_TPDU_DISCONNECT_REQUEST : 0x80,
|
||||
X224_TPDU_DATA : 0xF0,
|
||||
X224_TPDU_ERROR : 0x70
|
||||
};
|
||||
|
||||
/**
|
||||
* Type of negotiation present in negotiation packet
|
||||
*/
|
||||
var NegotiationType = {
|
||||
TYPE_RDP_NEG_REQ : 0x01,
|
||||
TYPE_RDP_NEG_RSP : 0x02,
|
||||
TYPE_RDP_NEG_FAILURE : 0x03
|
||||
};
|
||||
|
||||
/**
|
||||
* Protocols available for x224 layer
|
||||
*/
|
||||
var Protocols = {
|
||||
PROTOCOL_RDP : 0x00000000,
|
||||
PROTOCOL_SSL : 0x00000001,
|
||||
PROTOCOL_HYBRID : 0x00000002,
|
||||
PROTOCOL_HYBRID_EX : 0x00000008
|
||||
};
|
||||
|
||||
/**
|
||||
* Use to negotiate security layer of RDP stack
|
||||
* In node-rdpjs only ssl is available
|
||||
* @param opt {object} component type options
|
||||
* @see request -> http://msdn.microsoft.com/en-us/library/cc240500.aspx
|
||||
* @see response -> http://msdn.microsoft.com/en-us/library/cc240506.aspx
|
||||
* @see failure ->http://msdn.microsoft.com/en-us/library/cc240507.aspx
|
||||
*/
|
||||
function negotiation(opt) {
|
||||
var self = {
|
||||
type : new type.UInt8(),
|
||||
flag : new type.UInt8(),
|
||||
length : new type.UInt16Le(0x0008, { constant : true }),
|
||||
result : new type.UInt32Le()
|
||||
};
|
||||
return new type.Component(self, opt);
|
||||
}
|
||||
|
||||
/**
|
||||
* X224 client connection request
|
||||
* @param opt {object} component type options
|
||||
* @see http://msdn.microsoft.com/en-us/library/cc240470.aspx
|
||||
*/
|
||||
function clientConnectionRequestPDU(opt, cookie) {
|
||||
var self = {
|
||||
len : new type.UInt8(function() {
|
||||
return new type.Component(self).size() - 1;
|
||||
}),
|
||||
code : new type.UInt8(MessageType.X224_TPDU_CONNECTION_REQUEST, { constant : true }),
|
||||
padding : new type.Component([new type.UInt16Le(), new type.UInt16Le(), new type.UInt8()]),
|
||||
cookie : cookie || new type.Factory( function (s) {
|
||||
var offset = 0;
|
||||
while (true) {
|
||||
var token = s.buffer.readUInt16LE(s.offset + offset);
|
||||
if (token === 0x0a0d) {
|
||||
self.cookie = new type.BinaryString(null, { readLength : new type.CallableValue(offset + 2) }).read(s);
|
||||
return;
|
||||
}
|
||||
else {
|
||||
offset += 1;
|
||||
}
|
||||
}
|
||||
}, { conditional : function () {
|
||||
return self.len.value > 14;
|
||||
}}),
|
||||
protocolNeg : negotiation({ optional : true })
|
||||
};
|
||||
|
||||
return new type.Component(self, opt);
|
||||
}
|
||||
|
||||
/**
|
||||
* X224 Server connection confirm
|
||||
* @param opt {object} component type options
|
||||
* @see http://msdn.microsoft.com/en-us/library/cc240506.aspx
|
||||
*/
|
||||
function serverConnectionConfirm(opt) {
|
||||
var self = {
|
||||
len : new type.UInt8(function() {
|
||||
return new type.Component(self).size() - 1;
|
||||
}),
|
||||
code : new type.UInt8(MessageType.X224_TPDU_CONNECTION_CONFIRM, { constant : true }),
|
||||
padding : new type.Component([new type.UInt16Le(), new type.UInt16Le(), new type.UInt8()]),
|
||||
protocolNeg : negotiation({ optional : true })
|
||||
};
|
||||
|
||||
return new type.Component(self, opt);
|
||||
}
|
||||
|
||||
/**
|
||||
* Header of each data message from x224 layer
|
||||
* @returns {type.Component}
|
||||
*/
|
||||
function x224DataHeader() {
|
||||
var self = {
|
||||
header : new type.UInt8(2),
|
||||
messageType : new type.UInt8(MessageType.X224_TPDU_DATA, { constant : true }),
|
||||
separator : new type.UInt8(0x80, { constant : true })
|
||||
};
|
||||
return new type.Component(self);
|
||||
}
|
||||
|
||||
/**
|
||||
* Common X224 Automata
|
||||
* @param presentation {Layer} presentation layer
|
||||
*/
|
||||
function X224(transport) {
|
||||
this.transport = transport;
|
||||
this.requestedProtocol = Protocols.PROTOCOL_SSL | Protocols.PROTOCOL_HYBRID;
|
||||
this.selectedProtocol = Protocols.PROTOCOL_SSL | Protocols.PROTOCOL_HYBRID;
|
||||
|
||||
var self = this;
|
||||
this.transport.on('close', function() {
|
||||
self.emit('close');
|
||||
}).on('error', function (err) {
|
||||
self.emit('error', err);
|
||||
});
|
||||
}
|
||||
|
||||
//inherit from Layer
|
||||
inherits(X224, events.EventEmitter);
|
||||
|
||||
/**
|
||||
* Main data received function
|
||||
* after connection sequence
|
||||
* @param s {type.Stream} stream formated from transport layer
|
||||
*/
|
||||
X224.prototype.recvData = function(s) {
|
||||
// check header
|
||||
x224DataHeader().read(s);
|
||||
this.emit('data', s);
|
||||
};
|
||||
|
||||
/**
|
||||
* Format message from x224 layer to transport layer
|
||||
* @param message {type}
|
||||
* @returns {type.Component} x224 formated message
|
||||
*/
|
||||
X224.prototype.send = function(message) {
|
||||
this.transport.send(new type.Component([x224DataHeader(), message]));
|
||||
};
|
||||
|
||||
/**
|
||||
* Client x224 automata
|
||||
* @param transport {events.EventEmitter} (bind data events)
|
||||
*/
|
||||
function Client(transport, config) {
|
||||
this.config = config;
|
||||
X224.call(this, transport);
|
||||
}
|
||||
|
||||
//inherit from X224 automata
|
||||
inherits(Client, X224);
|
||||
|
||||
/**
|
||||
* Client automata connect event
|
||||
*/
|
||||
Client.prototype.connect = function() {
|
||||
var message = clientConnectionRequestPDU(null, new type.BinaryString());
|
||||
message.obj.protocolNeg.obj.type.value = NegotiationType.TYPE_RDP_NEG_REQ;
|
||||
message.obj.protocolNeg.obj.result.value = this.requestedProtocol;
|
||||
this.transport.send(message);
|
||||
|
||||
// next state wait connection confirm packet
|
||||
var self = this;
|
||||
this.transport.once('data', function(s) {
|
||||
self.recvConnectionConfirm(s);
|
||||
});
|
||||
};
|
||||
|
||||
/**
|
||||
* close stack
|
||||
*/
|
||||
Client.prototype.close = function() {
|
||||
this.transport.close();
|
||||
};
|
||||
|
||||
/**
|
||||
* Receive connection from server
|
||||
* @param s {Stream}
|
||||
*/
|
||||
Client.prototype.recvConnectionConfirm = function(s) {
|
||||
var message = serverConnectionConfirm().read(s);
|
||||
|
||||
if (message.obj.protocolNeg.obj.type.value == NegotiationType.TYPE_RDP_NEG_FAILURE) {
|
||||
throw new error.ProtocolError('NODE_RDP_PROTOCOL_X224_NEG_FAILURE',
|
||||
'Failure code:' + message.obj.protocolNeg.obj.result.value + " (see https://msdn.microsoft.com/en-us/library/cc240507.aspx)");
|
||||
}
|
||||
|
||||
if (message.obj.protocolNeg.obj.type.value == NegotiationType.TYPE_RDP_NEG_RSP) {
|
||||
this.selectedProtocol = message.obj.protocolNeg.obj.result.value;
|
||||
}
|
||||
|
||||
if ([Protocols.PROTOCOL_HYBRID_EX].indexOf(this.selectedProtocol) !== -1) {
|
||||
throw new error.ProtocolError('NODE_RDP_PROTOCOL_X224_NLA_NOT_SUPPORTED');
|
||||
}
|
||||
|
||||
if (this.selectedProtocol == Protocols.PROTOCOL_RDP) {
|
||||
log.debug("RDP standard security selected");
|
||||
return;
|
||||
}
|
||||
|
||||
if (this.selectedProtocol == Protocols.PROTOCOL_HYBRID) {
|
||||
log.debug("NLA security layer selected");
|
||||
var self = this;
|
||||
var transportEx = this.transport.transport;
|
||||
this.transport.transport.startTLS(function () {
|
||||
//console.log('TLS connected, start cssp_connect()');
|
||||
var NLA = require('./nla');
|
||||
self.nla = new NLA(transportEx, function () { self.nlaCompleted(); }, self.config.domain, self.config.userName, self.config.password);
|
||||
self.nla.sendNegotiateMessage();
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
// finish connection sequence
|
||||
var self = this;
|
||||
this.transport.on('data', function(s) {
|
||||
self.recvData(s);
|
||||
});
|
||||
|
||||
if (this.selectedProtocol == Protocols.PROTOCOL_SSL) {
|
||||
log.debug("SSL standard security selected");
|
||||
this.transport.transport.startTLS(function() {
|
||||
self.emit('connect', self.selectedProtocol);
|
||||
});
|
||||
return;
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* Called when NLA is completed
|
||||
*/
|
||||
Client.prototype.nlaCompleted = function () {
|
||||
const self = this;
|
||||
delete self.nla;
|
||||
this.transport.on('data', function (s) { self.recvData(s); });
|
||||
this.emit('connect', this.selectedProtocol);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Server x224 automata
|
||||
*/
|
||||
function Server(transport, keyFilePath, crtFilePath) {
|
||||
X224.call(this, transport);
|
||||
this.keyFilePath = keyFilePath;
|
||||
this.crtFilePath = crtFilePath;
|
||||
var self = this;
|
||||
this.transport.once('data', function (s) {
|
||||
self.recvConnectionRequest(s);
|
||||
});
|
||||
}
|
||||
|
||||
//inherit from X224 automata
|
||||
inherits(Server, X224);
|
||||
|
||||
/**
|
||||
* @see http://msdn.microsoft.com/en-us/library/cc240470.aspx
|
||||
* @param s {type.Stream}
|
||||
*/
|
||||
Server.prototype.recvConnectionRequest = function (s) {
|
||||
var request = clientConnectionRequestPDU().read(s);
|
||||
if (!request.obj.protocolNeg.isReaded) {
|
||||
throw new Error('NODE_RDP_PROTOCOL_X224_NO_BASIC_SECURITY_LAYER');
|
||||
}
|
||||
|
||||
this.requestedProtocol = request.obj.protocolNeg.obj.result.value;
|
||||
this.selectedProtocol = this.requestedProtocol & Protocols.PROTOCOL_SSL;
|
||||
|
||||
if (!(this.selectedProtocol & Protocols.PROTOCOL_SSL)) {
|
||||
var confirm = serverConnectionConfirm();
|
||||
confirm.obj.protocolNeg.obj.type.value = NegociationType.TYPE_RDP_NEG_FAILURE;
|
||||
confirm.obj.protocolNeg.obj.result.value = NegotiationFailureCode.SSL_REQUIRED_BY_SERVER;
|
||||
this.transport.send(confirm);
|
||||
this.close();
|
||||
}
|
||||
else {
|
||||
this.sendConnectionConfirm();
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* Start SSL connection if needed
|
||||
* @see http://msdn.microsoft.com/en-us/library/cc240501.aspx
|
||||
*/
|
||||
Server.prototype.sendConnectionConfirm = function () {
|
||||
var confirm = serverConnectionConfirm();
|
||||
confirm.obj.protocolNeg.obj.type.value = NegotiationType.TYPE_RDP_NEG_RSP;
|
||||
confirm.obj.protocolNeg.obj.result.value = this.selectedProtocol;
|
||||
this.transport.send(confirm);
|
||||
|
||||
// finish connection sequence
|
||||
var self = this;
|
||||
this.transport.on('data', function(s) {
|
||||
self.recvData(s);
|
||||
});
|
||||
|
||||
this.transport.transport.listenTLS(this.keyFilePath, this.crtFilePath, function() {
|
||||
log.debug('start SSL connection');
|
||||
self.emit('connect', self.requestedProtocol);
|
||||
});
|
||||
};
|
||||
|
||||
/**
|
||||
* Module exports
|
||||
*/
|
||||
module.exports = {
|
||||
Client : Client,
|
||||
Server : Server
|
||||
};
|
||||
Loading…
Add table
Add a link
Reference in a new issue