ladybird/Userland/Libraries/LibCrypto/ASN1/ASN1.cpp
Shannon Booth e2e7c4d574 Everywhere: Use to_number<T> instead of to_{int,uint,float,double}
In a bunch of cases, this actually ends up simplifying the code as
to_number will handle something such as:

```
Optional<I> opt;
if constexpr (IsSigned<I>)
    opt = view.to_int<I>();
else
    opt = view.to_uint<I>();
```

For us.

The main goal here however is to have a single generic number conversion
API between all of the String classes.
2023-12-23 20:41:07 +01:00

239 lines
7 KiB
C++

/*
* Copyright (c) 2021, the SerenityOS developers.
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <AK/GenericLexer.h>
#include <LibCrypto/ASN1/ASN1.h>
namespace Crypto::ASN1 {
ByteString kind_name(Kind kind)
{
switch (kind) {
case Kind::Eol:
return "EndOfList";
case Kind::Boolean:
return "Boolean";
case Kind::Integer:
return "Integer";
case Kind::BitString:
return "BitString";
case Kind::OctetString:
return "OctetString";
case Kind::Null:
return "Null";
case Kind::ObjectIdentifier:
return "ObjectIdentifier";
case Kind::ObjectDescriptor:
return "ObjectDescriptor";
case Kind::External:
return "External";
case Kind::Real:
return "Real";
case Kind::Enumerated:
return "Enumerated";
case Kind::EmbeddedPdv:
return "EmbeddedPdv";
case Kind::Utf8String:
return "Utf8String";
case Kind::RelativeOid:
return "RelativeOid";
case Kind::Time:
return "Time";
case Kind::Reserved:
return "Reserved";
case Kind::Sequence:
return "Sequence";
case Kind::Set:
return "Set";
case Kind::NumericString:
return "NumericString";
case Kind::PrintableString:
return "PrintableString";
case Kind::T61String:
return "T61String";
case Kind::VideotexString:
return "VideotexString";
case Kind::IA5String:
return "IA5String";
case Kind::UTCTime:
return "UTCTime";
case Kind::GeneralizedTime:
return "GeneralizedTime";
case Kind::GraphicString:
return "GraphicString";
case Kind::VisibleString:
return "VisibleString";
case Kind::GeneralString:
return "GeneralString";
case Kind::UniversalString:
return "UniversalString";
case Kind::CharacterString:
return "CharacterString";
case Kind::BMPString:
return "BMPString";
case Kind::Date:
return "Date";
case Kind::TimeOfDay:
return "TimeOfDay";
case Kind::DateTime:
return "DateTime";
case Kind::Duration:
return "Duration";
case Kind::OidIri:
return "OidIri";
case Kind::RelativeOidIri:
return "RelativeOidIri";
}
return "InvalidKind";
}
ByteString class_name(Class class_)
{
switch (class_) {
case Class::Application:
return "Application";
case Class::Context:
return "Context";
case Class::Private:
return "Private";
case Class::Universal:
return "Universal";
}
return "InvalidClass";
}
ByteString type_name(Type type)
{
switch (type) {
case Type::Constructed:
return "Constructed";
case Type::Primitive:
return "Primitive";
}
return "InvalidType";
}
Optional<Core::DateTime> parse_utc_time(StringView time)
{
// YYMMDDhhmm[ss]Z or YYMMDDhhmm[ss](+|-)hhmm
GenericLexer lexer(time);
auto year_in_century = lexer.consume(2).to_number<unsigned>();
auto month = lexer.consume(2).to_number<unsigned>();
auto day = lexer.consume(2).to_number<unsigned>();
auto hour = lexer.consume(2).to_number<unsigned>();
auto minute = lexer.consume(2).to_number<unsigned>();
Optional<unsigned> seconds, offset_hours, offset_minutes;
[[maybe_unused]] bool negative_offset = false;
if (lexer.next_is(is_any_of("0123456789"sv))) {
seconds = lexer.consume(2).to_number<unsigned>();
if (!seconds.has_value()) {
return {};
}
}
if (lexer.next_is('Z')) {
lexer.consume();
} else if (lexer.next_is(is_any_of("+-"sv))) {
negative_offset = lexer.consume() == '-';
offset_hours = lexer.consume(2).to_number<unsigned>();
offset_minutes = lexer.consume(2).to_number<unsigned>();
if (!offset_hours.has_value() || !offset_minutes.has_value()) {
return {};
}
} else {
return {};
}
if (!year_in_century.has_value() || !month.has_value() || !day.has_value() || !hour.has_value() || !minute.has_value()) {
return {};
}
// RFC5280 section 4.1.2.5.1.
auto full_year = year_in_century.value();
full_year += (full_year < 50) ? 2000 : 1900;
auto full_seconds = seconds.value_or(0);
// FIXME: Handle offsets!
if (offset_hours.has_value() || offset_minutes.has_value())
dbgln("FIXME: Implement UTCTime with offset!");
return Core::DateTime::create(full_year, month.value(), day.value(), hour.value(), minute.value(), full_seconds);
}
Optional<Core::DateTime> parse_generalized_time(StringView time)
{
// YYYYMMDDhh[mm[ss[.fff]]] or YYYYMMDDhh[mm[ss[.fff]]]Z or YYYYMMDDhh[mm[ss[.fff]]](+|-)hhmm
GenericLexer lexer(time);
auto year = lexer.consume(4).to_number<unsigned>();
auto month = lexer.consume(2).to_number<unsigned>();
auto day = lexer.consume(2).to_number<unsigned>();
auto hour = lexer.consume(2).to_number<unsigned>();
Optional<unsigned> minute, seconds, milliseconds, offset_hours, offset_minutes;
[[maybe_unused]] bool negative_offset = false;
if (!lexer.is_eof()) {
if (lexer.consume_specific('Z'))
goto done_parsing;
if (!lexer.next_is(is_any_of("+-"sv))) {
minute = lexer.consume(2).to_number<unsigned>();
if (!minute.has_value()) {
return {};
}
if (lexer.is_eof() || lexer.consume_specific('Z'))
goto done_parsing;
}
if (!lexer.next_is(is_any_of("+-"sv))) {
seconds = lexer.consume(2).to_number<unsigned>();
if (!seconds.has_value()) {
return {};
}
if (lexer.is_eof() || lexer.consume_specific('Z'))
goto done_parsing;
}
if (lexer.consume_specific('.')) {
milliseconds = lexer.consume(3).to_number<unsigned>();
if (!milliseconds.has_value()) {
return {};
}
if (lexer.is_eof() || lexer.consume_specific('Z'))
goto done_parsing;
}
if (lexer.next_is(is_any_of("+-"sv))) {
negative_offset = lexer.consume() == '-';
offset_hours = lexer.consume(2).to_number<unsigned>();
offset_minutes = lexer.consume(2).to_number<unsigned>();
if (!offset_hours.has_value() || !offset_minutes.has_value()) {
return {};
}
}
// Any character would be garbage.
if (!lexer.is_eof()) {
return {};
}
}
done_parsing:;
if (!year.has_value() || !month.has_value() || !day.has_value() || !hour.has_value()) {
return {};
}
// FIXME: Handle offsets!
if (offset_hours.has_value() || offset_minutes.has_value())
dbgln("FIXME: Implement GeneralizedTime with offset!");
// Unceremoniously drop the milliseconds on the floor.
return Core::DateTime::create(year.value(), month.value(), day.value(), hour.value(), minute.value_or(0), seconds.value_or(0));
}
}