blob: c7f20b50ebc97c75af645207893db3b249f90298 [file] [log] [blame]
/*
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.
*/
// Helper functions for certificate data extraction
static std::string
get_x509_name_string(X509_NAME *name)
{
if (!name) {
return "";
}
BIO *bio = BIO_new(BIO_s_mem());
if (!bio) {
return "";
}
if (X509_NAME_print_ex(bio, name, 0, XN_FLAG_RFC2253) <= 0) {
BIO_free(bio);
return "";
}
char *data = nullptr;
long length = BIO_get_mem_data(bio, &data);
std::string result;
if (data && length > 0) {
result.assign(data, length);
}
BIO_free(bio);
return result;
}
static std::string
get_x509_serial_string(X509 *cert)
{
if (!cert) {
return "";
}
ASN1_INTEGER *serial = X509_get_serialNumber(cert);
if (!serial) {
return "";
}
BIO *bio = BIO_new(BIO_s_mem());
if (!bio) {
return "";
}
if (i2a_ASN1_INTEGER(bio, serial) <= 0) {
BIO_free(bio);
return "";
}
char *data = nullptr;
long length = BIO_get_mem_data(bio, &data);
std::string result;
if (data && length > 0) {
result.assign(data, length);
}
BIO_free(bio);
return result;
}
static std::string
get_x509_time_string(ASN1_TIME *time)
{
if (!time) {
return "";
}
BIO *bio = BIO_new(BIO_s_mem());
if (!bio) {
return "";
}
if (ASN1_TIME_print(bio, time) <= 0) {
BIO_free(bio);
return "";
}
char *data = nullptr;
long length = BIO_get_mem_data(bio, &data);
std::string result;
if (data && length > 0) {
result.assign(data, length);
}
BIO_free(bio);
return result;
}
static std::string
get_x509_pem_string(X509 *cert)
{
if (!cert) {
return "";
}
BIO *bio = BIO_new(BIO_s_mem());
if (!bio) {
return "";
}
if (PEM_write_bio_X509(bio, cert) <= 0) {
BIO_free(bio);
return "";
}
char *data = nullptr;
long length = BIO_get_mem_data(bio, &data);
std::string result;
if (data && length > 0) {
result.assign(data, length);
}
BIO_free(bio);
return result;
}
static std::string
get_x509_signature_string(X509 *cert)
{
if (!cert) {
return "";
}
const ASN1_BIT_STRING *sig = nullptr;
X509_get0_signature(&sig, nullptr, cert);
if (!sig) {
return "";
}
BIO *bio = BIO_new(BIO_s_mem());
if (!bio) {
return "";
}
for (int i = 0; i < sig->length; i++) {
if (BIO_printf(bio, "%02x", sig->data[i]) <= 0) {
BIO_free(bio);
return "";
}
if (i < sig->length - 1) {
if (BIO_printf(bio, ":") <= 0) {
BIO_free(bio);
return "";
}
}
}
char *data = nullptr;
long length = BIO_get_mem_data(bio, &data);
std::string result;
if (data && length > 0) {
result.assign(data, length);
}
BIO_free(bio);
return result;
}
static std::vector<std::string>
get_x509_san_strings(X509 *cert, int san_type)
{
std::vector<std::string> results;
if (!cert) {
return results;
}
GENERAL_NAMES *names = static_cast<GENERAL_NAMES *>(X509_get_ext_d2i(cert, NID_subject_alt_name, nullptr, nullptr));
if (!names) {
return results;
}
int num_names = sk_GENERAL_NAME_num(names);
for (int i = 0; i < num_names; i++) {
GENERAL_NAME *name = sk_GENERAL_NAME_value(names, i);
if (!name || name->type != san_type) {
continue;
}
switch (san_type) {
case GEN_DNS:
case GEN_EMAIL:
case GEN_URI: {
ASN1_STRING *str = name->d.ia5;
if (str) {
const unsigned char *data = ASN1_STRING_get0_data(str);
int len = ASN1_STRING_length(str);
if (data && len > 0) {
results.emplace_back(reinterpret_cast<const char *>(data), len);
}
}
break;
}
case GEN_IPADD: {
ASN1_OCTET_STRING *ip = name->d.iPAddress;
if (ip) {
const unsigned char *data = ASN1_STRING_get0_data(ip);
int len = ASN1_STRING_length(ip);
char ip_str[INET6_ADDRSTRLEN];
if (len == 4) { // IPv4
if (inet_ntop(AF_INET, data, ip_str, sizeof(ip_str)) != nullptr) {
results.emplace_back(ip_str);
}
} else if (len == 16) { // IPv6
if (inet_ntop(AF_INET6, data, ip_str, sizeof(ip_str)) != nullptr) {
results.emplace_back(ip_str);
}
}
}
break;
}
}
}
GENERAL_NAMES_free(names);
return results;
}