pluto-src 0.1.1+0.10.4

Sources of Pluto (Lua 5.4 dialect) and logic to build it.
Documentation
#include "X509Certificate.hpp"

#include "Asn1Type.hpp"
#include "IpAddr.hpp"
#include "joaat.hpp"
#include "sha1.hpp"
#include "sha256.hpp"
#include "sha384.hpp"
#include "sha512.hpp"
#include "string.hpp"

NAMESPACE_SOUP
{
	bool X509Certificate::fromDer(const std::string& str) noexcept
	{
		return load(Asn1Sequence::fromDer(str));
	}

	bool X509Certificate::fromDer(const char* data, size_t size) noexcept
	{
		return load(Asn1Sequence::fromDer(data, size));
	}

	bool X509Certificate::load(const Asn1Sequence& cert) noexcept
	{
		SOUP_TRY
		{
			auto tbsCert = cert.getSeq(0);
			auto oid = cert.getSeq(1).getOid(0);
			if (oid == Oid::SHA1_WITH_RSA_ENCRYPTION) { sig_type = RSA_WITH_SHA1; }
			else if (oid == Oid::SHA256_WITH_RSA_ENCRYPTION) { sig_type = RSA_WITH_SHA256; }
			else if (oid == Oid::SHA384_WITH_RSA_ENCRYPTION) { sig_type = RSA_WITH_SHA384; }
			else if (oid == Oid::SHA512_WITH_RSA_ENCRYPTION) { sig_type = RSA_WITH_SHA512; }
			else if (oid == Oid::ECDSA_WITH_SHA256) { sig_type = ECDSA_WITH_SHA256; }
			else if (oid == Oid::ECDSA_WITH_SHA384) { sig_type = ECDSA_WITH_SHA384; }
			else { sig_type = UNK_WITH_UNK; }
			tbsCertDer = tbsCert.toDer();
			sig = cert.getString(2);
			sig.erase(0, 1); // trim leading zero

			hash = joaat::hash(cert.at(0).data);

			auto pubInfo = tbsCert.getSeq(6);
			auto pubType = pubInfo.getSeq(0);
			auto pubCrypto = pubType.getOid(0);

			std::string pubKeyStr = pubInfo.getString(1);
			pubKeyStr.erase(0, 1); // trim leading zero

			if (pubCrypto == Oid::RSA_ENCRYPTION)
			{
				is_ec = false;

				auto pubKey = Asn1Sequence::fromDer(pubKeyStr);
				setRsaPublicKey(
					pubKey.getInt(0),
					pubKey.getInt(1)
				);
			}
			else if (pubCrypto == Oid::EC_PUBLIC_KEY)
			{
				is_ec = true;

				auto pubCurve = pubType.getOid(1);
				if (pubCurve == Oid::PRIME256V1)
				{
					curve = &EccCurve::secp256r1();
				}
				else if (pubCurve == Oid::ANSIP384R1)
				{
					curve = &EccCurve::secp384r1();
				}
				else
				{
					curve = nullptr;
				}

				if (curve)
				{
					key = curve->decodePoint(pubKeyStr);
				}
			}
			else
			{
				return false;
			}

			issuer.read(tbsCert.getSeq(3));
			subject.read(tbsCert.getSeq(5));

			Asn1Sequence validityPeriod = tbsCert.getSeq(4);
			valid_from = validityPeriod.getUtctime(0);
			valid_to = validityPeriod.getUtctime(1);

			Asn1Sequence extensions = tbsCert.getSeq(7).getSeq(0);
			for (size_t i = 0; i != extensions.countChildren(); ++i)
			{
				Asn1Sequence ext = extensions.getSeq(i);
				const auto oid = ext.getOid(0);
				if (oid == Oid::CE_SUBJECTALTNAME)
				{
					size_t data_idx = ((ext.at(1).identifier.type == ASN1_BOOLEAN) ? 2 : 1);
					// RFC 2459, page 33
					Asn1Sequence data = Asn1Sequence::fromDer(ext.getString(data_idx));
					for (size_t j = 0; j != data.countChildren(); ++j)
					{
						if (data.getChildType(j).type == 2) // DNS Name
						{
							subject_alt_names.emplace_back(std::move(data.getString(j)));
						}
						else if (data.getChildType(j).type == 7) // IP Address
						{
							if (data.getString(j).size() == 4)
							{
								subject_alt_names.emplace_back(IpAddr(*(network_u32_t*)data.getString(j).data()).toString4());
							}
							else if (data.getString(j).size() == 16)
							{
								subject_alt_names.emplace_back(IpAddr((uint8_t*)data.getString(j).data()).toString6());
							}
						}
					}
				}
				else if (oid == Oid::CE_BASICCONSTRAINTS)
				{
					size_t data_idx = ((ext.at(1).identifier.type == ASN1_BOOLEAN) ? 2 : 1);
					auto data = Asn1Sequence::fromDer(ext.getString(data_idx));
					if (data.countChildren() > 0 && data.at(0).identifier.type == ASN1_BOOLEAN && data.at(0).data == "\xff") // Is CA?
					{
						if (data.countChildren() > 1) // Has chain size limit?
						{
							max_children = 1 + data.getInt(1).getChunk(0);
						}
						else
						{
							max_children = -1;
						}
					}
				}
			}

			return true;
		}
		SOUP_CATCH_ANY
		{
		}
		return false;
	}

	RsaPublicKey X509Certificate::getRsaPublicKey() const SOUP_EXCAL
	{
		return RsaPublicKey(key.x, key.y);
	}

	void X509Certificate::setRsaPublicKey(Bigint n, Bigint e) noexcept
	{
		key = EccPoint{ std::move(n), std::move(e) };
	}

	void X509Certificate::setRsaPublicKey(RsaPublicKey pub) noexcept
	{
		key = EccPoint{ std::move(pub.n), std::move(pub.e) };
	}

	bool X509Certificate::verify(const X509Certificate& issuer) const SOUP_EXCAL
	{
		switch (sig_type)
		{
		case RSA_WITH_SHA1:
			return issuer.isRsa()
				&& issuer.verifySignature<soup::sha1>(tbsCertDer, sig)
				;

		case RSA_WITH_SHA256:
			return issuer.isRsa()
				&& issuer.verifySignature<soup::sha256>(tbsCertDer, sig)
				;

		case ECDSA_WITH_SHA256:
			return issuer.isEc()
				&& issuer.verifySignature<soup::sha256>(tbsCertDer, sig)
				;

		case RSA_WITH_SHA384:
			return issuer.isRsa()
				&& issuer.verifySignature<soup::sha384>(tbsCertDer, sig)
				;

		case RSA_WITH_SHA512:
			return issuer.isRsa()
				&& issuer.verifySignature<soup::sha512>(tbsCertDer, sig)
				;

		case ECDSA_WITH_SHA384:
			return issuer.isEc()
				&& issuer.verifySignature<soup::sha384>(tbsCertDer, sig)
				;

		case UNK_WITH_UNK:;
		}
		return false;
	}

	bool X509Certificate::isValidForDomain(const std::string& domain) const SOUP_EXCAL
	{
		if (matchDomain(domain, subject.getCommonName()))
		{
			return true;
		}
		for (const auto& name : subject_alt_names)
		{
			if (matchDomain(domain, name))
			{
				return true;
			}
		}
		return false;
	}

	bool X509Certificate::matchDomain(const std::string& domain, const std::string& name) SOUP_EXCAL
	{
		auto domain_parts = string::explode(domain, '.');
		auto name_parts = string::explode(name, '.');
		if (domain_parts.size() != name_parts.size())
		{
			return false;
		}
		for (size_t i = 0; i != domain_parts.size(); ++i)
		{
			if (name_parts.at(i) != "*"
				&& name_parts.at(i) != domain_parts.at(i)
				)
			{
				return false;
			}
		}
		return true;
	}

	const Oid& X509Certificate::getAlgoOid() const noexcept
	{
		if (sig_type == RSA_WITH_SHA1) { return Oid::SHA1_WITH_RSA_ENCRYPTION; }
		else if (sig_type == RSA_WITH_SHA384) { return Oid::SHA384_WITH_RSA_ENCRYPTION; }
		else if (sig_type == RSA_WITH_SHA512) { return Oid::SHA512_WITH_RSA_ENCRYPTION; }
		else if (sig_type == ECDSA_WITH_SHA256) { return Oid::ECDSA_WITH_SHA256; }
		else if (sig_type == ECDSA_WITH_SHA384) { return Oid::ECDSA_WITH_SHA384; }
		return Oid::SHA256_WITH_RSA_ENCRYPTION;
	}

	Asn1Sequence X509Certificate::toAsn1() const SOUP_EXCAL
	{
		Asn1Sequence algo_seq;
		algo_seq.addOid(getAlgoOid());
		switch (sig_type)
		{
		default: // RSA_WITH_*
			algo_seq.addNull();
			break;

		case ECDSA_WITH_SHA256:
		case ECDSA_WITH_SHA384:
			break;
		}

		Asn1Sequence cert;
		cert.addSeq(Asn1Sequence::fromDer(tbsCertDer));
		cert.addSeq(algo_seq);
		cert.addBitString(sig);
		return cert;
	}

	std::string X509Certificate::toDer() const SOUP_EXCAL
	{
		return toAsn1().toDer();
	}

	std::string X509Certificate::getTbsCertDer() const SOUP_EXCAL
	{
		Asn1Sequence algo_seq;
		algo_seq.addOid(getAlgoOid());
		algo_seq.addNull();

		//auto tbsCert = Asn1Sequence::fromDer(tbsCertDer);
		Asn1Sequence tbsCert;
		tbsCert.addNull(); // [0] This should be "Version 3", respresented by "Constructed Context-specific 0: 020102"
		tbsCert.addInt({}); // [1] Serial number
		tbsCert.addSeq(algo_seq);
		tbsCert.addSet(issuer.toSet());
		{
			Asn1Sequence validityPeriod;
			validityPeriod.addUtctime(valid_from);
			validityPeriod.addUtctime(valid_to);
			tbsCert.addSeq(validityPeriod);
		}
		tbsCert.addSet(subject.toSet());
		{
			Asn1Sequence pubInfo;
			{
				Asn1Sequence pubType;
				pubType.addOid(Oid::RSA_ENCRYPTION);
				pubType.addNull();
				pubInfo.addSeq(pubType);
			}
			{
				auto key = getRsaPublicKey();

				Asn1Sequence pubKey;
				pubKey.addInt(key.n);
				pubKey.addInt(key.e);
				pubInfo.addBitString(pubKey.toDer());
			}
			tbsCert.addSeq(pubInfo);
		}
		{
			Asn1Sequence dummy;
			dummy.addSeq({});
			tbsCert.addSeq(dummy); // [7] Extensions, should be a context-specific 3.
		}

		return tbsCert.toDer();
	}
}