This page hosts a formal specification of tls_client_hello using Kaitai Struct. This specification can be automatically translated into a variety of programming languages to get a parsing library.
All parsing code for C++11/STL generated by Kaitai Struct depends on the C++/STL runtime library. You have to install it before you can parse data.
For C++, the easiest way is to clone the runtime library sources and build them along with your project.
Using Kaitai Struct in C++/STL usually consists of 3 steps.
std::istream
). One can open local file for that, or use existing std::string
or char*
buffer.
#include <fstream>
std::ifstream is("path/to/local/file.bin", std::ifstream::binary);
#include <sstream>
std::istringstream is(str);
#include <sstream>
const char buf[] = { ... };
std::string str(buf, sizeof buf);
std::istringstream is(str);
#include "kaitai/kaitaistream.h"
kaitai::kstream ks(&is);
tls_client_hello_t data(&ks);
After that, one can get various attributes from the structure by invoking getter methods like:
data.version() // => get version
#pragma once
// This is a generated file! Please edit source .ksy file and use kaitai-struct-compiler to rebuild
#include "kaitai/kaitaistruct.h"
#include <stdint.h>
#include <memory>
#include <vector>
#if KAITAI_STRUCT_VERSION < 9000L
#error "Incompatible Kaitai Struct C++/STL API: version 0.9 or later is required"
#endif
class tls_client_hello_t : public kaitai::kstruct {
public:
class server_name_t;
class random_t;
class session_id_t;
class sni_t;
class cipher_suites_t;
class compression_methods_t;
class alpn_t;
class extensions_t;
class version_t;
class protocol_t;
class extension_t;
tls_client_hello_t(kaitai::kstream* p__io, kaitai::kstruct* p__parent = nullptr, tls_client_hello_t* p__root = nullptr);
private:
void _read();
void _clean_up();
public:
~tls_client_hello_t();
class server_name_t : public kaitai::kstruct {
public:
server_name_t(kaitai::kstream* p__io, tls_client_hello_t::sni_t* p__parent = nullptr, tls_client_hello_t* p__root = nullptr);
private:
void _read();
void _clean_up();
public:
~server_name_t();
private:
uint8_t m_name_type;
uint16_t m_length;
std::string m_host_name;
tls_client_hello_t* m__root;
tls_client_hello_t::sni_t* m__parent;
public:
uint8_t name_type() const { return m_name_type; }
uint16_t length() const { return m_length; }
std::string host_name() const { return m_host_name; }
tls_client_hello_t* _root() const { return m__root; }
tls_client_hello_t::sni_t* _parent() const { return m__parent; }
};
class random_t : public kaitai::kstruct {
public:
random_t(kaitai::kstream* p__io, tls_client_hello_t* p__parent = nullptr, tls_client_hello_t* p__root = nullptr);
private:
void _read();
void _clean_up();
public:
~random_t();
private:
uint32_t m_gmt_unix_time;
std::string m_random;
tls_client_hello_t* m__root;
tls_client_hello_t* m__parent;
public:
uint32_t gmt_unix_time() const { return m_gmt_unix_time; }
std::string random() const { return m_random; }
tls_client_hello_t* _root() const { return m__root; }
tls_client_hello_t* _parent() const { return m__parent; }
};
class session_id_t : public kaitai::kstruct {
public:
session_id_t(kaitai::kstream* p__io, tls_client_hello_t* p__parent = nullptr, tls_client_hello_t* p__root = nullptr);
private:
void _read();
void _clean_up();
public:
~session_id_t();
private:
uint8_t m_len;
std::string m_sid;
tls_client_hello_t* m__root;
tls_client_hello_t* m__parent;
public:
uint8_t len() const { return m_len; }
std::string sid() const { return m_sid; }
tls_client_hello_t* _root() const { return m__root; }
tls_client_hello_t* _parent() const { return m__parent; }
};
class sni_t : public kaitai::kstruct {
public:
sni_t(kaitai::kstream* p__io, tls_client_hello_t::extension_t* p__parent = nullptr, tls_client_hello_t* p__root = nullptr);
private:
void _read();
void _clean_up();
public:
~sni_t();
private:
uint16_t m_list_length;
std::unique_ptr<std::vector<std::unique_ptr<server_name_t>>> m_server_names;
tls_client_hello_t* m__root;
tls_client_hello_t::extension_t* m__parent;
public:
uint16_t list_length() const { return m_list_length; }
std::vector<std::unique_ptr<server_name_t>>* server_names() const { return m_server_names.get(); }
tls_client_hello_t* _root() const { return m__root; }
tls_client_hello_t::extension_t* _parent() const { return m__parent; }
};
class cipher_suites_t : public kaitai::kstruct {
public:
cipher_suites_t(kaitai::kstream* p__io, tls_client_hello_t* p__parent = nullptr, tls_client_hello_t* p__root = nullptr);
private:
void _read();
void _clean_up();
public:
~cipher_suites_t();
private:
uint16_t m_len;
std::unique_ptr<std::vector<uint16_t>> m_cipher_suites;
tls_client_hello_t* m__root;
tls_client_hello_t* m__parent;
public:
uint16_t len() const { return m_len; }
std::vector<uint16_t>* cipher_suites() const { return m_cipher_suites.get(); }
tls_client_hello_t* _root() const { return m__root; }
tls_client_hello_t* _parent() const { return m__parent; }
};
class compression_methods_t : public kaitai::kstruct {
public:
compression_methods_t(kaitai::kstream* p__io, tls_client_hello_t* p__parent = nullptr, tls_client_hello_t* p__root = nullptr);
private:
void _read();
void _clean_up();
public:
~compression_methods_t();
private:
uint8_t m_len;
std::string m_compression_methods;
tls_client_hello_t* m__root;
tls_client_hello_t* m__parent;
public:
uint8_t len() const { return m_len; }
std::string compression_methods() const { return m_compression_methods; }
tls_client_hello_t* _root() const { return m__root; }
tls_client_hello_t* _parent() const { return m__parent; }
};
class alpn_t : public kaitai::kstruct {
public:
alpn_t(kaitai::kstream* p__io, tls_client_hello_t::extension_t* p__parent = nullptr, tls_client_hello_t* p__root = nullptr);
private:
void _read();
void _clean_up();
public:
~alpn_t();
private:
uint16_t m_ext_len;
std::unique_ptr<std::vector<std::unique_ptr<protocol_t>>> m_alpn_protocols;
tls_client_hello_t* m__root;
tls_client_hello_t::extension_t* m__parent;
public:
uint16_t ext_len() const { return m_ext_len; }
std::vector<std::unique_ptr<protocol_t>>* alpn_protocols() const { return m_alpn_protocols.get(); }
tls_client_hello_t* _root() const { return m__root; }
tls_client_hello_t::extension_t* _parent() const { return m__parent; }
};
class extensions_t : public kaitai::kstruct {
public:
extensions_t(kaitai::kstream* p__io, tls_client_hello_t* p__parent = nullptr, tls_client_hello_t* p__root = nullptr);
private:
void _read();
void _clean_up();
public:
~extensions_t();
private:
uint16_t m_len;
std::unique_ptr<std::vector<std::unique_ptr<extension_t>>> m_extensions;
tls_client_hello_t* m__root;
tls_client_hello_t* m__parent;
public:
uint16_t len() const { return m_len; }
std::vector<std::unique_ptr<extension_t>>* extensions() const { return m_extensions.get(); }
tls_client_hello_t* _root() const { return m__root; }
tls_client_hello_t* _parent() const { return m__parent; }
};
class version_t : public kaitai::kstruct {
public:
version_t(kaitai::kstream* p__io, tls_client_hello_t* p__parent = nullptr, tls_client_hello_t* p__root = nullptr);
private:
void _read();
void _clean_up();
public:
~version_t();
private:
uint8_t m_major;
uint8_t m_minor;
tls_client_hello_t* m__root;
tls_client_hello_t* m__parent;
public:
uint8_t major() const { return m_major; }
uint8_t minor() const { return m_minor; }
tls_client_hello_t* _root() const { return m__root; }
tls_client_hello_t* _parent() const { return m__parent; }
};
class protocol_t : public kaitai::kstruct {
public:
protocol_t(kaitai::kstream* p__io, tls_client_hello_t::alpn_t* p__parent = nullptr, tls_client_hello_t* p__root = nullptr);
private:
void _read();
void _clean_up();
public:
~protocol_t();
private:
uint8_t m_strlen;
std::string m_name;
tls_client_hello_t* m__root;
tls_client_hello_t::alpn_t* m__parent;
public:
uint8_t strlen() const { return m_strlen; }
std::string name() const { return m_name; }
tls_client_hello_t* _root() const { return m__root; }
tls_client_hello_t::alpn_t* _parent() const { return m__parent; }
};
class extension_t : public kaitai::kstruct {
public:
extension_t(kaitai::kstream* p__io, tls_client_hello_t::extensions_t* p__parent = nullptr, tls_client_hello_t* p__root = nullptr);
private:
void _read();
void _clean_up();
public:
~extension_t();
private:
uint16_t m_type;
uint16_t m_len;
std::unique_ptr<kaitai::kstruct> m_body;
bool n_body;
public:
bool _is_null_body() { body(); return n_body; };
private:
tls_client_hello_t* m__root;
tls_client_hello_t::extensions_t* m__parent;
std::string m__raw_body;
std::unique_ptr<kaitai::kstream> m__io__raw_body;
public:
uint16_t type() const { return m_type; }
uint16_t len() const { return m_len; }
kaitai::kstruct* body() const { return m_body.get(); }
tls_client_hello_t* _root() const { return m__root; }
tls_client_hello_t::extensions_t* _parent() const { return m__parent; }
std::string _raw_body() const { return m__raw_body; }
kaitai::kstream* _io__raw_body() const { return m__io__raw_body.get(); }
};
private:
std::unique_ptr<version_t> m_version;
std::unique_ptr<random_t> m_random;
std::unique_ptr<session_id_t> m_session_id;
std::unique_ptr<cipher_suites_t> m_cipher_suites;
std::unique_ptr<compression_methods_t> m_compression_methods;
std::unique_ptr<extensions_t> m_extensions;
bool n_extensions;
public:
bool _is_null_extensions() { extensions(); return n_extensions; };
private:
tls_client_hello_t* m__root;
kaitai::kstruct* m__parent;
public:
version_t* version() const { return m_version.get(); }
random_t* random() const { return m_random.get(); }
session_id_t* session_id() const { return m_session_id.get(); }
cipher_suites_t* cipher_suites() const { return m_cipher_suites.get(); }
compression_methods_t* compression_methods() const { return m_compression_methods.get(); }
extensions_t* extensions() const { return m_extensions.get(); }
tls_client_hello_t* _root() const { return m__root; }
kaitai::kstruct* _parent() const { return m__parent; }
};
// This is a generated file! Please edit source .ksy file and use kaitai-struct-compiler to rebuild
#include "tls_client_hello.h"
tls_client_hello_t::tls_client_hello_t(kaitai::kstream* p__io, kaitai::kstruct* p__parent, tls_client_hello_t* p__root) : kaitai::kstruct(p__io) {
m__parent = p__parent;
m__root = this;
m_version = nullptr;
m_random = nullptr;
m_session_id = nullptr;
m_cipher_suites = nullptr;
m_compression_methods = nullptr;
m_extensions = nullptr;
_read();
}
void tls_client_hello_t::_read() {
m_version = std::unique_ptr<version_t>(new version_t(m__io, this, m__root));
m_random = std::unique_ptr<random_t>(new random_t(m__io, this, m__root));
m_session_id = std::unique_ptr<session_id_t>(new session_id_t(m__io, this, m__root));
m_cipher_suites = std::unique_ptr<cipher_suites_t>(new cipher_suites_t(m__io, this, m__root));
m_compression_methods = std::unique_ptr<compression_methods_t>(new compression_methods_t(m__io, this, m__root));
n_extensions = true;
if (_io()->is_eof() == false) {
n_extensions = false;
m_extensions = std::unique_ptr<extensions_t>(new extensions_t(m__io, this, m__root));
}
}
tls_client_hello_t::~tls_client_hello_t() {
_clean_up();
}
void tls_client_hello_t::_clean_up() {
if (!n_extensions) {
}
}
tls_client_hello_t::server_name_t::server_name_t(kaitai::kstream* p__io, tls_client_hello_t::sni_t* p__parent, tls_client_hello_t* p__root) : kaitai::kstruct(p__io) {
m__parent = p__parent;
m__root = p__root;
_read();
}
void tls_client_hello_t::server_name_t::_read() {
m_name_type = m__io->read_u1();
m_length = m__io->read_u2be();
m_host_name = m__io->read_bytes(length());
}
tls_client_hello_t::server_name_t::~server_name_t() {
_clean_up();
}
void tls_client_hello_t::server_name_t::_clean_up() {
}
tls_client_hello_t::random_t::random_t(kaitai::kstream* p__io, tls_client_hello_t* p__parent, tls_client_hello_t* p__root) : kaitai::kstruct(p__io) {
m__parent = p__parent;
m__root = p__root;
_read();
}
void tls_client_hello_t::random_t::_read() {
m_gmt_unix_time = m__io->read_u4be();
m_random = m__io->read_bytes(28);
}
tls_client_hello_t::random_t::~random_t() {
_clean_up();
}
void tls_client_hello_t::random_t::_clean_up() {
}
tls_client_hello_t::session_id_t::session_id_t(kaitai::kstream* p__io, tls_client_hello_t* p__parent, tls_client_hello_t* p__root) : kaitai::kstruct(p__io) {
m__parent = p__parent;
m__root = p__root;
_read();
}
void tls_client_hello_t::session_id_t::_read() {
m_len = m__io->read_u1();
m_sid = m__io->read_bytes(len());
}
tls_client_hello_t::session_id_t::~session_id_t() {
_clean_up();
}
void tls_client_hello_t::session_id_t::_clean_up() {
}
tls_client_hello_t::sni_t::sni_t(kaitai::kstream* p__io, tls_client_hello_t::extension_t* p__parent, tls_client_hello_t* p__root) : kaitai::kstruct(p__io) {
m__parent = p__parent;
m__root = p__root;
m_server_names = nullptr;
_read();
}
void tls_client_hello_t::sni_t::_read() {
m_list_length = m__io->read_u2be();
m_server_names = std::unique_ptr<std::vector<std::unique_ptr<server_name_t>>>(new std::vector<std::unique_ptr<server_name_t>>());
{
int i = 0;
while (!m__io->is_eof()) {
m_server_names->push_back(std::move(std::unique_ptr<server_name_t>(new server_name_t(m__io, this, m__root))));
i++;
}
}
}
tls_client_hello_t::sni_t::~sni_t() {
_clean_up();
}
void tls_client_hello_t::sni_t::_clean_up() {
}
tls_client_hello_t::cipher_suites_t::cipher_suites_t(kaitai::kstream* p__io, tls_client_hello_t* p__parent, tls_client_hello_t* p__root) : kaitai::kstruct(p__io) {
m__parent = p__parent;
m__root = p__root;
m_cipher_suites = nullptr;
_read();
}
void tls_client_hello_t::cipher_suites_t::_read() {
m_len = m__io->read_u2be();
m_cipher_suites = std::unique_ptr<std::vector<uint16_t>>(new std::vector<uint16_t>());
const int l_cipher_suites = (len() / 2);
for (int i = 0; i < l_cipher_suites; i++) {
m_cipher_suites->push_back(std::move(m__io->read_u2be()));
}
}
tls_client_hello_t::cipher_suites_t::~cipher_suites_t() {
_clean_up();
}
void tls_client_hello_t::cipher_suites_t::_clean_up() {
}
tls_client_hello_t::compression_methods_t::compression_methods_t(kaitai::kstream* p__io, tls_client_hello_t* p__parent, tls_client_hello_t* p__root) : kaitai::kstruct(p__io) {
m__parent = p__parent;
m__root = p__root;
_read();
}
void tls_client_hello_t::compression_methods_t::_read() {
m_len = m__io->read_u1();
m_compression_methods = m__io->read_bytes(len());
}
tls_client_hello_t::compression_methods_t::~compression_methods_t() {
_clean_up();
}
void tls_client_hello_t::compression_methods_t::_clean_up() {
}
tls_client_hello_t::alpn_t::alpn_t(kaitai::kstream* p__io, tls_client_hello_t::extension_t* p__parent, tls_client_hello_t* p__root) : kaitai::kstruct(p__io) {
m__parent = p__parent;
m__root = p__root;
m_alpn_protocols = nullptr;
_read();
}
void tls_client_hello_t::alpn_t::_read() {
m_ext_len = m__io->read_u2be();
m_alpn_protocols = std::unique_ptr<std::vector<std::unique_ptr<protocol_t>>>(new std::vector<std::unique_ptr<protocol_t>>());
{
int i = 0;
while (!m__io->is_eof()) {
m_alpn_protocols->push_back(std::move(std::unique_ptr<protocol_t>(new protocol_t(m__io, this, m__root))));
i++;
}
}
}
tls_client_hello_t::alpn_t::~alpn_t() {
_clean_up();
}
void tls_client_hello_t::alpn_t::_clean_up() {
}
tls_client_hello_t::extensions_t::extensions_t(kaitai::kstream* p__io, tls_client_hello_t* p__parent, tls_client_hello_t* p__root) : kaitai::kstruct(p__io) {
m__parent = p__parent;
m__root = p__root;
m_extensions = nullptr;
_read();
}
void tls_client_hello_t::extensions_t::_read() {
m_len = m__io->read_u2be();
m_extensions = std::unique_ptr<std::vector<std::unique_ptr<extension_t>>>(new std::vector<std::unique_ptr<extension_t>>());
{
int i = 0;
while (!m__io->is_eof()) {
m_extensions->push_back(std::move(std::unique_ptr<extension_t>(new extension_t(m__io, this, m__root))));
i++;
}
}
}
tls_client_hello_t::extensions_t::~extensions_t() {
_clean_up();
}
void tls_client_hello_t::extensions_t::_clean_up() {
}
tls_client_hello_t::version_t::version_t(kaitai::kstream* p__io, tls_client_hello_t* p__parent, tls_client_hello_t* p__root) : kaitai::kstruct(p__io) {
m__parent = p__parent;
m__root = p__root;
_read();
}
void tls_client_hello_t::version_t::_read() {
m_major = m__io->read_u1();
m_minor = m__io->read_u1();
}
tls_client_hello_t::version_t::~version_t() {
_clean_up();
}
void tls_client_hello_t::version_t::_clean_up() {
}
tls_client_hello_t::protocol_t::protocol_t(kaitai::kstream* p__io, tls_client_hello_t::alpn_t* p__parent, tls_client_hello_t* p__root) : kaitai::kstruct(p__io) {
m__parent = p__parent;
m__root = p__root;
_read();
}
void tls_client_hello_t::protocol_t::_read() {
m_strlen = m__io->read_u1();
m_name = m__io->read_bytes(strlen());
}
tls_client_hello_t::protocol_t::~protocol_t() {
_clean_up();
}
void tls_client_hello_t::protocol_t::_clean_up() {
}
tls_client_hello_t::extension_t::extension_t(kaitai::kstream* p__io, tls_client_hello_t::extensions_t* p__parent, tls_client_hello_t* p__root) : kaitai::kstruct(p__io) {
m__parent = p__parent;
m__root = p__root;
m__io__raw_body = nullptr;
_read();
}
void tls_client_hello_t::extension_t::_read() {
m_type = m__io->read_u2be();
m_len = m__io->read_u2be();
n_body = true;
switch (type()) {
case 0: {
n_body = false;
m__raw_body = m__io->read_bytes(len());
m__io__raw_body = std::unique_ptr<kaitai::kstream>(new kaitai::kstream(m__raw_body));
m_body = std::unique_ptr<sni_t>(new sni_t(m__io__raw_body.get(), this, m__root));
break;
}
case 16: {
n_body = false;
m__raw_body = m__io->read_bytes(len());
m__io__raw_body = std::unique_ptr<kaitai::kstream>(new kaitai::kstream(m__raw_body));
m_body = std::unique_ptr<alpn_t>(new alpn_t(m__io__raw_body.get(), this, m__root));
break;
}
default: {
m__raw_body = m__io->read_bytes(len());
break;
}
}
}
tls_client_hello_t::extension_t::~extension_t() {
_clean_up();
}
void tls_client_hello_t::extension_t::_clean_up() {
if (!n_body) {
}
}