mirror of
https://github.com/gchq/CyberChef
synced 2025-01-12 04:28:53 +00:00
Add Parse TLS record operation
This commit is contained in:
parent
3822c6c520
commit
1fcc365d9e
5 changed files with 2934 additions and 0 deletions
|
@ -235,6 +235,7 @@
|
||||||
"Parse IPv6 address",
|
"Parse IPv6 address",
|
||||||
"Parse IPv4 header",
|
"Parse IPv4 header",
|
||||||
"Parse TCP",
|
"Parse TCP",
|
||||||
|
"Parse TLS record",
|
||||||
"Parse UDP",
|
"Parse UDP",
|
||||||
"Parse SSH Host Key",
|
"Parse SSH Host Key",
|
||||||
"Parse URI",
|
"Parse URI",
|
||||||
|
|
|
@ -26,6 +26,9 @@ export function objToTable(obj, nested=false) {
|
||||||
</tr>`;
|
</tr>`;
|
||||||
|
|
||||||
for (const key in obj) {
|
for (const key in obj) {
|
||||||
|
if (typeof obj[key] === "function")
|
||||||
|
continue;
|
||||||
|
|
||||||
html += `<tr><td style='word-wrap: break-word'>${key}</td>`;
|
html += `<tr><td style='word-wrap: break-word'>${key}</td>`;
|
||||||
if (typeof obj[key] === "object")
|
if (typeof obj[key] === "object")
|
||||||
html += `<td style='padding: 0'>${objToTable(obj[key], true)}</td>`;
|
html += `<td style='padding: 0'>${objToTable(obj[key], true)}</td>`;
|
||||||
|
|
884
src/core/operations/ParseTLSRecord.mjs
Normal file
884
src/core/operations/ParseTLSRecord.mjs
Normal file
|
@ -0,0 +1,884 @@
|
||||||
|
/**
|
||||||
|
* @author c65722 []
|
||||||
|
* @copyright Crown Copyright 2024
|
||||||
|
* @license Apache-2.0
|
||||||
|
*/
|
||||||
|
|
||||||
|
import Operation from "../Operation.mjs";
|
||||||
|
import {toHexFast} from "../lib/Hex.mjs";
|
||||||
|
import {objToTable} from "../lib/Protocol.mjs";
|
||||||
|
import Stream from "../lib/Stream.mjs";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parse TLS record operation.
|
||||||
|
*/
|
||||||
|
class ParseTLSRecord extends Operation {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ParseTLSRecord constructor.
|
||||||
|
*/
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
|
||||||
|
this.name = "Parse TLS record";
|
||||||
|
this.module = "Default";
|
||||||
|
this.description = "Parses one or more TLS records";
|
||||||
|
this.infoURL = "https://wikipedia.org/wiki/Transport_Layer_Security";
|
||||||
|
this.inputType = "ArrayBuffer";
|
||||||
|
this.outputType = "json";
|
||||||
|
this.presentType = "html";
|
||||||
|
this.args = [];
|
||||||
|
this._handshakeParser = new HandshakeParser();
|
||||||
|
this._contentTypes = new Map();
|
||||||
|
|
||||||
|
for (const key in ContentType) {
|
||||||
|
this._contentTypes[ContentType[key]] = key.toString().toLocaleLowerCase();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {ArrayBuffer} input - Stream, containing one or more raw TLS Records.
|
||||||
|
* @param {Object[]} args
|
||||||
|
* @returns {Object[]} Array of Object representations of TLS Records contained within input.
|
||||||
|
*/
|
||||||
|
run(input, args) {
|
||||||
|
const s = new Stream(new Uint8Array(input));
|
||||||
|
|
||||||
|
const output = [];
|
||||||
|
|
||||||
|
while (s.hasMore()) {
|
||||||
|
const record = this._readRecord(s);
|
||||||
|
if (record) {
|
||||||
|
output.push(record);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads a TLS Record from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw TLS Record.
|
||||||
|
* @returns {Object} Object representation of TLS Record.
|
||||||
|
*/
|
||||||
|
_readRecord(input) {
|
||||||
|
const RECORD_HEADER_LEN = 5;
|
||||||
|
|
||||||
|
if (input.position + RECORD_HEADER_LEN > input.length) {
|
||||||
|
input.moveTo(input.length);
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const type = input.readInt(1);
|
||||||
|
const typeString = this._contentTypes[type] ?? type.toString();
|
||||||
|
const version = "0x" + toHexFast(input.getBytes(2));
|
||||||
|
const length = input.readInt(2);
|
||||||
|
const content = input.getBytes(length);
|
||||||
|
const truncated = content.length < length;
|
||||||
|
|
||||||
|
const recordHeader = new RecordHeader(typeString, version, length, truncated);
|
||||||
|
|
||||||
|
if (!content.length) {
|
||||||
|
return {...recordHeader};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type === ContentType.HANDSHAKE) {
|
||||||
|
return this._handshakeParser.parse(new Stream(content), recordHeader);
|
||||||
|
}
|
||||||
|
|
||||||
|
const record = {...recordHeader};
|
||||||
|
record.value = "0x" + toHexFast(content);
|
||||||
|
|
||||||
|
return record;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Displays the parsed TLS Records in a tabular style.
|
||||||
|
*
|
||||||
|
* @param {Object[]} data - Array of Object representations of the TLS Records.
|
||||||
|
* @returns {html} HTML representation of TLS Records contained within data.
|
||||||
|
*/
|
||||||
|
present(data) {
|
||||||
|
return data.map(r => objToTable(r)).join("\n\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default ParseTLSRecord;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Repesents the known values of type field of a TLS Record header.
|
||||||
|
*/
|
||||||
|
const ContentType = Object.freeze({
|
||||||
|
CHANGE_CIPHER_SPEC: 20,
|
||||||
|
ALERT: 21,
|
||||||
|
HANDSHAKE: 22,
|
||||||
|
APPLICATION_DATA: 23,
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents a TLS Record header
|
||||||
|
*/
|
||||||
|
class RecordHeader {
|
||||||
|
/**
|
||||||
|
* RecordHeader cosntructor.
|
||||||
|
*
|
||||||
|
* @param {string} type - String representation of TLS Record type field.
|
||||||
|
* @param {string} version - Hex representation of TLS Record version field.
|
||||||
|
* @param {int} length - Length of TLS Record.
|
||||||
|
* @param {bool} truncated - Is TLS Record truncated.
|
||||||
|
*/
|
||||||
|
constructor(type, version, length, truncated) {
|
||||||
|
this.type = type;
|
||||||
|
this.version = version;
|
||||||
|
this.length = length;
|
||||||
|
|
||||||
|
if (truncated) {
|
||||||
|
this.truncated = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses TLS Handshake messages.
|
||||||
|
*/
|
||||||
|
class HandshakeParser {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* HandshakeParser constructor.
|
||||||
|
*/
|
||||||
|
constructor() {
|
||||||
|
this._clientHelloParser = new ClientHelloParser();
|
||||||
|
this._serverHelloParser = new ServerHelloParser();
|
||||||
|
this._newSessionTicketParser = new NewSessionTicketParser();
|
||||||
|
this._certificateParser = new CertificateParser();
|
||||||
|
this._certificateRequestParser = new CertificateRequestParser();
|
||||||
|
this._certificateVerifyParser = new CertificateVerifyParser();
|
||||||
|
this._handshakeTypes = new Map();
|
||||||
|
|
||||||
|
for (const key in HandshakeType) {
|
||||||
|
this._handshakeTypes[HandshakeType[key]] = key.toString().toLowerCase();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses a single TLS handshake message.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw Handshake message.
|
||||||
|
* @param {RecordHeader} recordHeader - TLS Record header.
|
||||||
|
* @returns {Object} Object representation of Handshake.
|
||||||
|
*/
|
||||||
|
parse(input, recordHeader) {
|
||||||
|
const output = {...recordHeader};
|
||||||
|
|
||||||
|
if (!input.hasMore()) {
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
const handshakeType = input.readInt(1);
|
||||||
|
output.handshakeType = this._handshakeTypes[handshakeType] ?? handshakeType.toString();
|
||||||
|
|
||||||
|
if (input.position + 3 > input.length) {
|
||||||
|
input.moveTo(input.length);
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
const handshakeLength = input.readInt(3);
|
||||||
|
|
||||||
|
if (handshakeLength + 4 !== recordHeader.length) {
|
||||||
|
input.moveTo(0);
|
||||||
|
|
||||||
|
output.handshakeType = this._handshakeTypes[HandshakeType.FINISHED];
|
||||||
|
output.handshakeValue = "0x" + toHexFast(input.bytes);
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
const content = input.getBytes(handshakeLength);
|
||||||
|
if (!content.length) {
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (handshakeType) {
|
||||||
|
case HandshakeType.CLIENT_HELLO:
|
||||||
|
return {...output, ...this._clientHelloParser.parse(new Stream(content))};
|
||||||
|
case HandshakeType.SERVER_HELLO:
|
||||||
|
return {...output, ...this._serverHelloParser.parse(new Stream(content))};
|
||||||
|
case HandshakeType.NEW_SESSION_TICKET:
|
||||||
|
return {...output, ...this._newSessionTicketParser.parse(new Stream(content))};
|
||||||
|
case HandshakeType.CERTIFICATE:
|
||||||
|
return {...output, ...this._certificateParser.parse(new Stream(content))};
|
||||||
|
case HandshakeType.CERTIFICATE_REQUEST:
|
||||||
|
return {...output, ...this._certificateRequestParser.parse(new Stream(content))};
|
||||||
|
case HandshakeType.CERTIFICATE_VERIFY:
|
||||||
|
return {...output, ...this._certificateVerifyParser.parse(new Stream(content))};
|
||||||
|
default:
|
||||||
|
output.handshakeValue = "0x" + toHexFast(content);
|
||||||
|
}
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents the known values of the msg_type field of a TLS Handshake message.
|
||||||
|
*/
|
||||||
|
const HandshakeType = Object.freeze({
|
||||||
|
HELLO_REQUEST: 0,
|
||||||
|
CLIENT_HELLO: 1,
|
||||||
|
SERVER_HELLO: 2,
|
||||||
|
NEW_SESSION_TICKET: 4,
|
||||||
|
CERTIFICATE: 11,
|
||||||
|
SERVER_KEY_EXCHANGE: 12,
|
||||||
|
CERTIFICATE_REQUEST: 13,
|
||||||
|
SERVER_HELLO_DONE: 14,
|
||||||
|
CERTIFICATE_VERIFY: 15,
|
||||||
|
CLIENT_KEY_EXCHANGE: 16,
|
||||||
|
FINISHED: 20,
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses TLS Handshake ClientHello messages.
|
||||||
|
*/
|
||||||
|
class ClientHelloParser {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ClientHelloParser constructor.
|
||||||
|
*/
|
||||||
|
constructor() {
|
||||||
|
this._extensionsParser = new ExtensionsParser();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses a single TLS Handshake ClientHello message.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ClientHello message.
|
||||||
|
* @returns {Object} Object representation of ClientHello.
|
||||||
|
*/
|
||||||
|
parse(input) {
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
output.clientVersion = this._readClientVersion(input);
|
||||||
|
output.random = this._readRandom(input);
|
||||||
|
|
||||||
|
const sessionID = this._readSessionID(input);
|
||||||
|
if (sessionID) {
|
||||||
|
output.sessionID = sessionID;
|
||||||
|
}
|
||||||
|
|
||||||
|
output.cipherSuites = this._readCipherSuites(input);
|
||||||
|
output.compressionMethods = this._readCompressionMethods(input);
|
||||||
|
output.extensions = this._readExtensions(input);
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the client_version field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ClientHello message, with position before client_version field.
|
||||||
|
* @returns {string} Hex representation of client_version.
|
||||||
|
*/
|
||||||
|
_readClientVersion(input) {
|
||||||
|
return readBytesAsHex(input, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the random field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ClientHello message, with position before random field.
|
||||||
|
* @returns {string} Hex representation of random.
|
||||||
|
*/
|
||||||
|
_readRandom(input) {
|
||||||
|
return readBytesAsHex(input, 32);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the session_id field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ClientHello message, with position before session_id length field.
|
||||||
|
* @returns {string} Hex representation of session_id, or empty string if session_id not present.
|
||||||
|
*/
|
||||||
|
_readSessionID(input) {
|
||||||
|
return readSizePrefixedBytesAsHex(input, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the cipher_suites field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ClientHello message, with position before cipher_suites length field.
|
||||||
|
* @returns {Object} Object represention of cipher_suites field.
|
||||||
|
*/
|
||||||
|
_readCipherSuites(input) {
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
output.length = input.readInt(2);
|
||||||
|
if (!output.length) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
const cipherSuites = new Stream(input.getBytes(output.length));
|
||||||
|
if (cipherSuites.length < output.length) {
|
||||||
|
output.truncated = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
output.values = [];
|
||||||
|
|
||||||
|
while (cipherSuites.hasMore()) {
|
||||||
|
const cipherSuite = readBytesAsHex(cipherSuites, 2);
|
||||||
|
if (cipherSuite) {
|
||||||
|
output.values.push(cipherSuite);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the compression_methods field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ClientHello message, with position before compression_methods length field.
|
||||||
|
* @returns {Object} Object representation of compression_methods field.
|
||||||
|
*/
|
||||||
|
_readCompressionMethods(input) {
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
output.length = input.readInt(1);
|
||||||
|
if (!output.length) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
const compressionMethods = new Stream(input.getBytes(output.length));
|
||||||
|
if (compressionMethods.length < output.length) {
|
||||||
|
output.truncated = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
output.values = [];
|
||||||
|
|
||||||
|
while (compressionMethods.hasMore()) {
|
||||||
|
const compressionMethod = readBytesAsHex(compressionMethods, 1);
|
||||||
|
if (compressionMethod) {
|
||||||
|
output.values.push(compressionMethod);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the extensions field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ClientHello message, with position before extensions length field.
|
||||||
|
* @returns {Object} Object representations of extensions field.
|
||||||
|
*/
|
||||||
|
_readExtensions(input) {
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
output.length = input.readInt(2);
|
||||||
|
if (!output.length) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
const extensions = new Stream(input.getBytes(output.length));
|
||||||
|
if (extensions.length < output.length) {
|
||||||
|
output.truncated = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
output.values = this._extensionsParser.parse(extensions);
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses TLS Handshake ServeHello messages.
|
||||||
|
*/
|
||||||
|
class ServerHelloParser {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ServerHelloParser constructor.
|
||||||
|
*/
|
||||||
|
constructor() {
|
||||||
|
this._extensionsParser = new ExtensionsParser();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses a single TLS Handshake ServerHello message.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ServerHello message.
|
||||||
|
* @return {Object} Object representation of ServerHello.
|
||||||
|
*/
|
||||||
|
parse(input) {
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
output.serverVersion = this._readServerVersion(input);
|
||||||
|
output.random = this._readRandom(input);
|
||||||
|
|
||||||
|
const sessionID = this._readSessionID(input);
|
||||||
|
if (sessionID) {
|
||||||
|
output.sessionID = sessionID;
|
||||||
|
}
|
||||||
|
|
||||||
|
output.cipherSuite = this._readCipherSuite(input);
|
||||||
|
output.compressionMethod = this._readCompressionMethod(input);
|
||||||
|
output.extensions = this._readExtensions(input);
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the server_version field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ServerHello message, with position before server_version field.
|
||||||
|
* @returns {string} Hex representation of server_version.
|
||||||
|
*/
|
||||||
|
_readServerVersion(input) {
|
||||||
|
return readBytesAsHex(input, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the random field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ServerHello message, with position before random field.
|
||||||
|
* @returns {string} Hex representation of random.
|
||||||
|
*/
|
||||||
|
_readRandom(input) {
|
||||||
|
return readBytesAsHex(input, 32);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the session_id field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ServertHello message, with position before session_id length field.
|
||||||
|
* @returns {string} Hex representation of session_id, or empty string if session_id not present.
|
||||||
|
*/
|
||||||
|
_readSessionID(input) {
|
||||||
|
return readSizePrefixedBytesAsHex(input, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the cipher_suite field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ServerHello message, with position before cipher_suite field.
|
||||||
|
* @returns {string} Hex represention of cipher_suite.
|
||||||
|
*/
|
||||||
|
_readCipherSuite(input) {
|
||||||
|
return readBytesAsHex(input, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the compression_method field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ServerHello message, with position before compression_method field.
|
||||||
|
* @returns {string} Hex represention of compression_method.
|
||||||
|
*/
|
||||||
|
_readCompressionMethod(input) {
|
||||||
|
return readBytesAsHex(input, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the extensions field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw ServerHello message, with position before extensions length field.
|
||||||
|
* @returns {Object} Object representation of extensions field.
|
||||||
|
*/
|
||||||
|
_readExtensions(input) {
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
output.length = input.readInt(2);
|
||||||
|
if (!output.length) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
const extensions = new Stream(input.getBytes(output.length));
|
||||||
|
if (extensions.length < output.length) {
|
||||||
|
output.truncated = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
output.values = this._extensionsParser.parse(extensions);
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses TLS Handshake Hello Extensions.
|
||||||
|
*/
|
||||||
|
class ExtensionsParser {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses a stream of TLS Handshake Hello Extensions.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing multiple raw Extensions, with position before first extension length field.
|
||||||
|
* @returns {Object[]} Array of Object representations of Extensions contained within input.
|
||||||
|
*/
|
||||||
|
parse(input) {
|
||||||
|
const output = [];
|
||||||
|
|
||||||
|
while (input.hasMore()) {
|
||||||
|
const extension = this._readExtension(input);
|
||||||
|
if (extension) {
|
||||||
|
output.push(extension);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads a single Extension from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a list of Extensions, with position before the length field of the next Extension.
|
||||||
|
* @returns {Object} Object representation of Extension.
|
||||||
|
*/
|
||||||
|
_readExtension(input) {
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
if (input.position + 4 > input.length) {
|
||||||
|
input.moveTo(input.length);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
output.type = "0x" + toHexFast(input.getBytes(2));
|
||||||
|
output.length = input.readInt(2);
|
||||||
|
if (!output.length) {
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
const value = input.getBytes(output.length);
|
||||||
|
if (!value || value.length !== output.length) {
|
||||||
|
output.truncated = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (value && value.length) {
|
||||||
|
output.value = "0x" + toHexFast(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses TLS Handshake NewSessionTicket messages.
|
||||||
|
*/
|
||||||
|
class NewSessionTicketParser {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses a single TLS Handshake NewSessionTicket message.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw NewSessionTicket message.
|
||||||
|
* @returns {Object} Object representation of NewSessionTicket.
|
||||||
|
*/
|
||||||
|
parse(input) {
|
||||||
|
return {
|
||||||
|
ticketLifetimeHint: this._readTicketLifetimeHint(input),
|
||||||
|
ticket: this._readTicket(input),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the ticket_lifetime_hint field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw NewSessionTicket message, with position before ticket_lifetime_hint field.
|
||||||
|
* @returns {string} Lifetime hint, in seconds.
|
||||||
|
*/
|
||||||
|
_readTicketLifetimeHint(input) {
|
||||||
|
if (input.position + 4 > input.length) {
|
||||||
|
input.moveTo(input.length);
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
return input.readInt(4) + "s";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the ticket field fromt the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw NewSessionTicket message, with position before ticket length field.
|
||||||
|
* @returns {string} Hex representation of ticket.
|
||||||
|
*/
|
||||||
|
_readTicket(input) {
|
||||||
|
return readSizePrefixedBytesAsHex(input, 2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses TLS Handshake Certificate messages.
|
||||||
|
*/
|
||||||
|
class CertificateParser {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses a single TLS Handshake Certificate message.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw Certificate message.
|
||||||
|
* @returns {Object} Object representation of Certificate.
|
||||||
|
*/
|
||||||
|
parse(input) {
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
output.certificateList = this._readCertificateList(input);
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the certificate_list field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw Certificate message, with position before certificate_list length field.
|
||||||
|
* @returns {string[]} Array of strings, each containing a hex representation of a value within the certificate_list field.
|
||||||
|
*/
|
||||||
|
_readCertificateList(input) {
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
if (input.position + 3 > input.length) {
|
||||||
|
input.moveTo(input.length);
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
output.length = input.readInt(3);
|
||||||
|
if (!output.length) {
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
const certificates = new Stream(input.getBytes(output.length));
|
||||||
|
if (certificates.length < output.length) {
|
||||||
|
output.truncated = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
output.values = [];
|
||||||
|
|
||||||
|
while (certificates.hasMore()) {
|
||||||
|
const certificate = this._readCertificate(certificates);
|
||||||
|
if (certificate) {
|
||||||
|
output.values.push(certificate);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads a single certificate from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a list of certificicates, with position before the length field of the next certificate.
|
||||||
|
* @returns {string} Hex representation of certificate.
|
||||||
|
*/
|
||||||
|
_readCertificate(input) {
|
||||||
|
return readSizePrefixedBytesAsHex(input, 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses TLS Handshake CertificateRequest messages.
|
||||||
|
*/
|
||||||
|
class CertificateRequestParser {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses a single TLS Handshake CertificateRequest message.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw CertificateRequest message.
|
||||||
|
* @return {Object} Object representation of CertificateRequest.
|
||||||
|
*/
|
||||||
|
parse(input) {
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
output.certificateTypes = this._readCertificateTypes(input);
|
||||||
|
output.supportedSignatureAlgorithms = this._readSupportedSignatureAlgorithms(input);
|
||||||
|
|
||||||
|
const certificateAuthorities = this._readCertificateAuthorities(input);
|
||||||
|
if (certificateAuthorities.length) {
|
||||||
|
output.certificateAuthorities = certificateAuthorities;
|
||||||
|
}
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the certificate_types field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw CertificateRequest message, with position before certificate_types length field.
|
||||||
|
* @return {string[]} Array of strings, each containing a hex representation of a value within the certificate_types field.
|
||||||
|
*/
|
||||||
|
_readCertificateTypes(input) {
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
output.length = input.readInt(1);
|
||||||
|
if (!output.length) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
const certificateTypes = new Stream(input.getBytes(output.length));
|
||||||
|
if (certificateTypes.length < output.length) {
|
||||||
|
output.truncated = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
output.values = [];
|
||||||
|
|
||||||
|
while (certificateTypes.hasMore()) {
|
||||||
|
const certificateType = readBytesAsHex(certificateTypes, 1);
|
||||||
|
if (certificateType) {
|
||||||
|
output.values.push(certificateType);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the supported_signature_algorithms field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw CertificateRequest message, with position before supported_signature_algorithms length field.
|
||||||
|
* @returns {string[]} Array of strings, each containing a hex representation of a value within the supported_signature_algorithms field.
|
||||||
|
*/
|
||||||
|
_readSupportedSignatureAlgorithms(input) {
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
output.length = input.readInt(2);
|
||||||
|
if (!output.length) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
const signatureAlgorithms = new Stream(input.getBytes(output.length));
|
||||||
|
if (signatureAlgorithms.length < output.length) {
|
||||||
|
output.truncated = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
output.values = [];
|
||||||
|
|
||||||
|
while (signatureAlgorithms.hasMore()) {
|
||||||
|
const signatureAlgorithm = readBytesAsHex(signatureAlgorithms, 2);
|
||||||
|
if (signatureAlgorithm) {
|
||||||
|
output.values.push(signatureAlgorithm);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the certificate_authorities field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw CertificateRequest message, with position before certificate_authorities length field.
|
||||||
|
* @returns {string[]} Array of strings, each containing a hex representation of a value within the certificate_authorities field.
|
||||||
|
*/
|
||||||
|
_readCertificateAuthorities(input) {
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
output.length = input.readInt(2);
|
||||||
|
if (!output.length) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
const certificateAuthorities = new Stream(input.getBytes(output.length));
|
||||||
|
if (certificateAuthorities.length < output.length) {
|
||||||
|
output.truncated = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
output.values = [];
|
||||||
|
|
||||||
|
while (certificateAuthorities.hasMore()) {
|
||||||
|
const certificateAuthority = this._readCertificateAuthority(certificateAuthorities);
|
||||||
|
if (certificateAuthority) {
|
||||||
|
output.values.push(certificateAuthority);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads a single certificate authority from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a list of raw certificate authorities, with position before the length field of the next certificate authority.
|
||||||
|
* @returns {string} Hex representation of certificate authority.
|
||||||
|
*/
|
||||||
|
_readCertificateAuthority(input) {
|
||||||
|
return readSizePrefixedBytesAsHex(input, 2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses TLS Handshake CertificateVerify messages.
|
||||||
|
*/
|
||||||
|
class CertificateVerifyParser {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses a single CertificateVerify Message.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw CertificateVerify message.
|
||||||
|
* @returns {Object} Object representation of CertificateVerify.
|
||||||
|
*/
|
||||||
|
parse(input) {
|
||||||
|
return {
|
||||||
|
algorithmHash: this._readAlgorithmHash(input),
|
||||||
|
algorithmSignature: this._readAlgorithmSignature(input),
|
||||||
|
signature: this._readSignature(input),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the algorithm.hash field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw CertificateVerify message, with position before algorithm.hash field.
|
||||||
|
* @return {string} Hex representation of hash algorithm.
|
||||||
|
*/
|
||||||
|
_readAlgorithmHash(input) {
|
||||||
|
return readBytesAsHex(input, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the algorithm.signature field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw CertificateVerify message, with position before algorithm.signature field.
|
||||||
|
* @return {string} Hex representation of signature algorithm.
|
||||||
|
*/
|
||||||
|
_readAlgorithmSignature(input) {
|
||||||
|
return readBytesAsHex(input, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the signature field from the following bytes in the provided Stream.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream, containing a raw CertificateVerify message, with position before signature field.
|
||||||
|
* @return {string} Hex representation of signature.
|
||||||
|
*/
|
||||||
|
_readSignature(input) {
|
||||||
|
return readSizePrefixedBytesAsHex(input, 2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read the following size prefixed bytes from the provided Stream, and reuturn as a hex string.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream to read from.
|
||||||
|
* @param {int} sizePrefixLength - Length of the size prefix field.
|
||||||
|
* @returns {string} Hex representation of bytes read from Stream, empty string is returned if
|
||||||
|
* field cannot be read in full.
|
||||||
|
*/
|
||||||
|
function readSizePrefixedBytesAsHex(input, sizePrefixLength) {
|
||||||
|
const length = input.readInt(sizePrefixLength);
|
||||||
|
if (!length) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
return readBytesAsHex(input, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read n bytes from the provided Stream, and return as a hex string.
|
||||||
|
*
|
||||||
|
* @param {Stream} input - Stream to read from.
|
||||||
|
* @param {int} n - Number of bytes to read.
|
||||||
|
* @returns {string} Hex representation of bytes read from Stream, or empty string if field cannot
|
||||||
|
* be read in full.
|
||||||
|
*/
|
||||||
|
function readBytesAsHex(input, n) {
|
||||||
|
const bytes = input.getBytes(n);
|
||||||
|
if (!bytes || bytes.length !== n) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
return "0x" + toHexFast(bytes);
|
||||||
|
}
|
|
@ -115,6 +115,7 @@ import "./tests/ParseObjectIDTimestamp.mjs";
|
||||||
import "./tests/ParseQRCode.mjs";
|
import "./tests/ParseQRCode.mjs";
|
||||||
import "./tests/ParseSSHHostKey.mjs";
|
import "./tests/ParseSSHHostKey.mjs";
|
||||||
import "./tests/ParseTCP.mjs";
|
import "./tests/ParseTCP.mjs";
|
||||||
|
import "./tests/ParseTLSRecord.mjs";
|
||||||
import "./tests/ParseTLV.mjs";
|
import "./tests/ParseTLV.mjs";
|
||||||
import "./tests/ParseUDP.mjs";
|
import "./tests/ParseUDP.mjs";
|
||||||
import "./tests/PEMtoHex.mjs";
|
import "./tests/PEMtoHex.mjs";
|
||||||
|
|
2045
tests/operations/tests/ParseTLSRecord.mjs
Normal file
2045
tests/operations/tests/ParseTLSRecord.mjs
Normal file
File diff suppressed because one or more lines are too long
Loading…
Reference in a new issue