From 06340ca6749224edac56415ffe0d3db4765c6635 Mon Sep 17 00:00:00 2001 From: Fabian Dellwing Date: Mon, 13 Mar 2023 12:18:14 +0100 Subject: [PATCH] LibCrypto: Add multiple PEM parser This adds a function to parse multiple PEMs out of a single input. This allows us to load certificates from a cacert.pem file without need for preprocessing. --- Userland/Libraries/LibCrypto/ASN1/PEM.cpp | 37 +++++++++++++++++++++++ Userland/Libraries/LibCrypto/ASN1/PEM.h | 1 + 2 files changed, 38 insertions(+) diff --git a/Userland/Libraries/LibCrypto/ASN1/PEM.cpp b/Userland/Libraries/LibCrypto/ASN1/PEM.cpp index a23bed08bfa..f682493f56f 100644 --- a/Userland/Libraries/LibCrypto/ASN1/PEM.cpp +++ b/Userland/Libraries/LibCrypto/ASN1/PEM.cpp @@ -56,4 +56,41 @@ ByteBuffer decode_pem(ReadonlyBytes data) return decoded; } +ErrorOr> decode_pems(ReadonlyBytes data) +{ + GenericLexer lexer { data }; + ByteBuffer decoded; + Vector pems; + + enum { + Junk, + Parsing, + } state { Junk }; + while (!lexer.is_eof()) { + switch (state) { + case Junk: + if (lexer.consume_specific("-----BEGIN")) + state = Parsing; + lexer.consume_line(); + break; + case Parsing: { + if (lexer.consume_specific("-----END")) { + state = Junk; + lexer.consume_line(); + TRY(pems.try_append(decoded)); + decoded.clear(); + break; + } + auto b64decoded = TRY(decode_base64(lexer.consume_line().trim_whitespace(TrimMode::Right))); + TRY(decoded.try_append(b64decoded.data(), b64decoded.size())); + break; + } + default: + VERIFY_NOT_REACHED(); + } + } + + return pems; +} + } diff --git a/Userland/Libraries/LibCrypto/ASN1/PEM.h b/Userland/Libraries/LibCrypto/ASN1/PEM.h index c95617e5077..396a41fd05f 100644 --- a/Userland/Libraries/LibCrypto/ASN1/PEM.h +++ b/Userland/Libraries/LibCrypto/ASN1/PEM.h @@ -13,5 +13,6 @@ namespace Crypto { ByteBuffer decode_pem(ReadonlyBytes); +ErrorOr> decode_pems(ReadonlyBytes); }