plugins/experimental/ja4_fingerprint/ja4.cc (135 lines of code) (raw):
/** @file ja3_fingerprint.cc
*
JA4 fingerprint calculation.
@section license License
Licensed to the Apache Software Foundation (ASF) under one
or more contributor license agreements. See the NOTICE file
distributed with this work for additional information
regarding copyright ownership. The ASF licenses this file
to you under the Apache License, Version 2.0 (the
"License"); you may not use this file except in compliance
with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
#include "ja4.h"
#include <algorithm>
#include <cctype>
#include <cstddef>
#include <cstdint>
#include <cstdio>
#include <iterator>
#include <string>
#include <string_view>
static char convert_protocol_to_char(JA4::Protocol protocol);
static std::string convert_TLS_version_to_string(std::uint16_t TLS_version);
static char convert_SNI_to_char(JA4::SNI SNI_type);
static std::string convert_count_to_two_digit_string(std::size_t count);
static std::string convert_ALPN_to_two_char_string(std::string_view ALPN);
static void remove_trailing_character(std::string &s);
static std::string hexify(std::uint16_t n);
namespace
{
constexpr std::size_t U16_HEX_BUF_SIZE{4};
} // end anonymous namespace
std::string
JA4::make_JA4_a_raw(TLSClientHelloSummary const &TLS_summary)
{
std::string result;
result.reserve(9);
result.push_back(convert_protocol_to_char(TLS_summary.protocol));
result.append(convert_TLS_version_to_string(TLS_summary.TLS_version));
result.push_back(convert_SNI_to_char(TLS_summary.get_SNI_type()));
result.append(convert_count_to_two_digit_string(TLS_summary.get_cipher_count()));
result.append(convert_count_to_two_digit_string(TLS_summary.get_extension_count()));
result.append(convert_ALPN_to_two_char_string(TLS_summary.ALPN));
return result;
}
static char
convert_protocol_to_char(JA4::Protocol protocol)
{
return static_cast<char>(protocol);
}
static std::string
convert_TLS_version_to_string(std::uint16_t TLS_version)
{
switch (TLS_version) {
case 0x304:
return "13";
case 0x303:
return "12";
case 0x302:
return "11";
case 0x301:
return "10";
case 0x300:
return "s3";
case 0x200:
return "s2";
case 0x100:
return "s1";
case 0xfeff:
return "d1";
case 0xfefd:
return "d2";
case 0xfefc:
return "d3";
default:
return "00";
}
}
static char
convert_SNI_to_char(JA4::SNI SNI_type)
{
return static_cast<char>(SNI_type);
}
static std::string
convert_count_to_two_digit_string(std::size_t count)
{
std::string result;
if (count <= 9) {
result.push_back('0');
}
// We could also clamp the lower bound to 1 since there must be at least 1
// cipher, but 0 is more helpful for debugging if the cipher list is empty.
result.append(std::to_string(std::clamp(count, std::size_t{0}, std::size_t{99})));
return result;
}
std::string
convert_ALPN_to_two_char_string(std::string_view ALPN)
{
std::string result;
if (ALPN.empty()) {
result = "00";
} else {
result.push_back(ALPN.front());
result.push_back(ALPN.back());
}
return result;
}
std::string
JA4::make_JA4_b_raw(TLSClientHelloSummary const &TLS_summary)
{
std::string result;
result.reserve(12);
std::vector temp = TLS_summary.get_ciphers();
std::sort(temp.begin(), temp.end());
for (auto cipher : temp) {
result.append(hexify(cipher));
result.push_back(',');
}
remove_trailing_character(result);
return result;
}
std::string
JA4::make_JA4_c_raw(TLSClientHelloSummary const &TLS_summary)
{
std::string result;
result.reserve(12);
std::vector temp = TLS_summary.get_extensions();
std::sort(temp.begin(), temp.end());
for (auto extension : temp) {
result.append(hexify(extension));
result.push_back(',');
}
remove_trailing_character(result);
return result;
}
void
remove_trailing_character(std::string &s)
{
if (!s.empty()) {
s.pop_back();
}
}
std::string
hexify(std::uint16_t n)
{
char result[U16_HEX_BUF_SIZE + 1]{};
std::snprintf(result, sizeof(result), "%.4x", n);
return result;
}