/* * Copyright (c) 2020, the SerenityOS developers. * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include namespace TLS { enum class CertificateKeyAlgorithm { Unsupported = 0x00, RSA_RSA = 0x01, RSA_MD5 = 0x04, RSA_SHA1 = 0x05, RSA_SHA256 = 0x0b, RSA_SHA384 = 0x0c, RSA_SHA512 = 0x0d, }; class Certificate { public: u16 version { 0 }; CertificateKeyAlgorithm algorithm { CertificateKeyAlgorithm::Unsupported }; CertificateKeyAlgorithm key_algorithm { CertificateKeyAlgorithm::Unsupported }; CertificateKeyAlgorithm ec_algorithm { CertificateKeyAlgorithm::Unsupported }; ByteBuffer exponent {}; Crypto::PK::RSAPublicKey public_key {}; Crypto::PK::RSAPrivateKey private_key {}; struct Name { String country; String state; String location; String entity; String subject; String unit; } issuer, subject; Core::DateTime not_before; Core::DateTime not_after; Vector SAN; u8* ocsp { nullptr }; Crypto::UnsignedBigInteger serial_number; ByteBuffer sign_key {}; ByteBuffer fingerprint {}; ByteBuffer der {}; ByteBuffer data {}; CertificateKeyAlgorithm signature_algorithm { CertificateKeyAlgorithm::Unsupported }; ByteBuffer signature_value {}; ByteBuffer original_asn1 {}; bool is_allowed_to_sign_certificate { false }; bool is_certificate_authority { false }; Optional path_length_constraint {}; static Optional parse_asn1(ReadonlyBytes, bool client_cert = false); bool is_valid() const; String subject_identifier_string() const { StringBuilder cert_name; if (!subject.country.is_empty()) { cert_name.append("/C="sv); cert_name.append(subject.country); } if (!subject.state.is_empty()) { cert_name.append("/ST="sv); cert_name.append(subject.state); } if (!subject.location.is_empty()) { cert_name.append("/L="sv); cert_name.append(subject.location); } if (!subject.entity.is_empty()) { cert_name.append("/O="sv); cert_name.append(subject.entity); } if (!subject.unit.is_empty()) { cert_name.append("/OU="sv); cert_name.append(subject.unit); } if (!subject.subject.is_empty()) { cert_name.append("/CN="sv); cert_name.append(subject.subject); } return cert_name.build(); } String issuer_identifier_string() const { StringBuilder cert_name; if (!issuer.country.is_empty()) { cert_name.append("/C="sv); cert_name.append(issuer.country); } if (!issuer.state.is_empty()) { cert_name.append("/ST="sv); cert_name.append(issuer.state); } if (!issuer.location.is_empty()) { cert_name.append("/L="sv); cert_name.append(issuer.location); } if (!issuer.entity.is_empty()) { cert_name.append("/O="sv); cert_name.append(issuer.entity); } if (!issuer.unit.is_empty()) { cert_name.append("/OU="sv); cert_name.append(issuer.unit); } if (!issuer.subject.is_empty()) { cert_name.append("/CN="sv); cert_name.append(issuer.subject); } return cert_name.build(); } }; class DefaultRootCACertificates { public: DefaultRootCACertificates(); Vector const& certificates() const { return m_ca_certificates; } static DefaultRootCACertificates& the() { return s_the; } private: static Singleton s_the; Vector m_ca_certificates; }; } using TLS::Certificate; using TLS::DefaultRootCACertificates;