/*
 * Copyright (c) 2024, Ali Mohammad Pur <mpfard@serenityos.org>
 *
 * SPDX-License-Identifier: BSD-2-Clause
 */

#include <AK/CountingStream.h>
#include <AK/MemoryStream.h>
#include <AK/Stream.h>
#include <AK/UFixedBigInt.h>
#include <LibDNS/Message.h>

namespace DNS::Messages {

String Options::to_string() const
{
    StringBuilder builder;
    builder.appendff("QR: {}, Opcode: {}, AA: {}, TC: {}, RD: {}, RA: {}, AD: {}, CD: {}, RCODE: {}",
        is_question() ? "Q" : "R",
        Messages::to_string(op_code()),
        is_authoritative_answer(),
        is_truncated(),
        recursion_desired(),
        recursion_available(),
        authenticated_data(),
        checking_disabled(),
        Messages::to_string(response_code()));
    return MUST(builder.to_string());
}

StringView to_string(Options::ResponseCode code)
{
    switch (code) {
    case Options::ResponseCode::NoError:
        return "NoError"sv;
    case Options::ResponseCode::FormatError:
        return "FormatError"sv;
    case Options::ResponseCode::ServerFailure:
        return "ServerFailure"sv;
    case Options::ResponseCode::NameError:
        return "NameError"sv;
    case Options::ResponseCode::NotImplemented:
        return "NotImplemented"sv;
    case Options::ResponseCode::Refused:
        return "Refused"sv;
    default:
        return "UNKNOWN"sv;
    }
}

ErrorOr<Message> Message::from_raw(AK::Stream& stream)
{
    CountingStream counting_stream { MaybeOwned(stream) };
    auto context = ParseContext { counting_stream, make<RedBlackTree<u16, DomainName>>() };
    return from_raw(context);
}

ErrorOr<Message> Message::from_raw(ParseContext& ctx)
{
    // RFC 1035, 4.1. (Messages) Format.
    // | Header      |
    // | Question    | the question for the name server
    // | Answer      | RRs answering the question
    // | Authority   | RRs pointing toward an authority
    // | Additional  | RRs holding additional information
    //
    // The header section is always present.  The header includes fields that
    // specify which of the remaining sections are present, and also specify
    // whether the message is a query or a response, a standard query or some
    // other opcode, etc.

    Header header;
    Bytes header_bytes { &header, sizeof(Header) };
    TRY(ctx.stream.read_until_filled(header_bytes));

    Message message {};
    message.header = header;

    for (size_t i = 0; i < header.question_count; ++i) {
        auto question = TRY(Question::from_raw(ctx));
        message.questions.append(move(question));
    }

    for (size_t i = 0; i < header.answer_count; ++i) {
        auto answer = TRY(ResourceRecord::from_raw(ctx));
        message.answers.append(move(answer));
    }

    for (size_t i = 0; i < header.authority_count; ++i) {
        auto authority = TRY(ResourceRecord::from_raw(ctx));
        message.authorities.append(move(authority));
    }

    for (size_t i = 0; i < header.additional_count; ++i) {
        auto additional = TRY(ResourceRecord::from_raw(ctx));
        message.additional_records.append(move(additional));
    }

    return message;
}

ErrorOr<size_t> Message::to_raw(ByteBuffer& out) const
{
    // NOTE: This is minimally implemented to allow for sending queries,
    //       server-side responses are not implemented yet.
    VERIFY(header.answer_count == 0);
    VERIFY(header.authority_count == 0);

    auto start_size = out.size();

    auto header_bytes = TRY(out.get_bytes_for_writing(sizeof(Header)));
    memcpy(header_bytes.data(), &header, sizeof(Header));

    for (size_t i = 0; i < header.question_count; i++)
        TRY(questions[i].to_raw(out));

    for (size_t i = 0; i < header.additional_count; i++)
        TRY(additional_records[i].to_raw(out));

    return out.size() - start_size;
}

ErrorOr<String> Message::format_for_log() const
{
    StringBuilder builder;
    builder.appendff("ID: {}\n", header.id);
    builder.appendff("Flags: {} ({:x})\n", header.options.to_string(), header.options.raw);
    builder.appendff("qdcount: {}, ancount: {}, nscount: {}, arcount: {}\n", header.question_count, header.answer_count, header.authority_count, header.additional_count);

    if (header.question_count > 0) {
        builder.appendff("Questions:\n");
        for (auto& q : questions)
            builder.appendff("    {} {} {}\n", q.name.to_string(), to_string(q.class_), to_string(q.type));
    }

    if (header.answer_count > 0) {
        builder.appendff("Answers:\n");
        for (auto& a : answers) {
            builder.appendff("    {} {} {}\n", a.name.to_string(), to_string(a.class_), to_string(a.type));
            a.record.visit(
                [&](auto const& record) { builder.appendff("        {}\n", MUST(record.to_string())); },
                [&](ByteBuffer const& raw) {
                    builder.appendff("        {:hex-dump}\n", raw.bytes());
                });
        }
    }

    if (header.authority_count > 0) {
        builder.appendff("Authorities:\n");
        for (auto& a : authorities) {
            builder.appendff("    {} {} {}\n", a.name.to_string(), to_string(a.class_), to_string(a.type));
            a.record.visit(
                [&](auto const& record) { builder.appendff("        {}\n", MUST(record.to_string())); },
                [&](ByteBuffer const& raw) {
                    builder.appendff("        {:hex-dump}\n", raw.bytes());
                });
        }
    }

    if (header.additional_count > 0) {
        builder.appendff("Additional:\n");
        for (auto& a : additional_records) {
            builder.appendff("    {} {} {}\n", a.name.to_string(), to_string(a.type), to_string(a.class_));
            a.record.visit(
                [&](auto const& record) { builder.appendff("        {}\n", MUST(record.to_string())); },
                [&](ByteBuffer const& raw) {
                    builder.appendff("        {:hex-dump}\n", raw.bytes());
                });
        }
    }

    return builder.to_string();
}

ErrorOr<Question> Question::from_raw(ParseContext& ctx)
{
    // RFC 1035, 4.1.2. Question section format.
    // +        +
    // | QNAME  | a domain name represented as a sequence of labels
    // +        +
    // | QTYPE  | a two octet code which specifies the type of the query
    // | QCLASS | a two octet code that specifies the class of the query

    auto name = TRY(DomainName::from_raw(ctx));
    auto type = static_cast<ResourceType>(static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>())));
    auto class_ = static_cast<Class>(static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>())));

    return Question { move(name), type, class_ };
}

ErrorOr<void> Question::to_raw(ByteBuffer& out) const
{
    TRY(name.to_raw(out));

    auto type_bytes = TRY(out.get_bytes_for_writing(2));
    auto net_type = static_cast<NetworkOrdered<u16>>(to_underlying(type));
    memcpy(type_bytes.data(), &net_type, 2);

    auto class_bytes = TRY(out.get_bytes_for_writing(2));
    auto net_class = static_cast<NetworkOrdered<u16>>(to_underlying(class_));
    memcpy(class_bytes.data(), &net_class, 2);

    return {};
}

StringView to_string(ResourceType type)
{
    switch (type) {
    case ResourceType::Reserved:
        return "Reserved"sv;
    case ResourceType::A:
        return "A"sv;
    case ResourceType::NS:
        return "NS"sv;
    case ResourceType::MD:
        return "MD"sv;
    case ResourceType::MF:
        return "MF"sv;
    case ResourceType::CNAME:
        return "CNAME"sv;
    case ResourceType::SOA:
        return "SOA"sv;
    case ResourceType::MB:
        return "MB"sv;
    case ResourceType::MG:
        return "MG"sv;
    case ResourceType::MR:
        return "MR"sv;
    case ResourceType::NULL_:
        return "NULL_"sv;
    case ResourceType::WKS:
        return "WKS"sv;
    case ResourceType::PTR:
        return "PTR"sv;
    case ResourceType::HINFO:
        return "HINFO"sv;
    case ResourceType::MINFO:
        return "MINFO"sv;
    case ResourceType::MX:
        return "MX"sv;
    case ResourceType::TXT:
        return "TXT"sv;
    case ResourceType::RP:
        return "RP"sv;
    case ResourceType::AFSDB:
        return "AFSDB"sv;
    case ResourceType::X25:
        return "X25"sv;
    case ResourceType::ISDN:
        return "ISDN"sv;
    case ResourceType::RT:
        return "RT"sv;
    case ResourceType::NSAP:
        return "NSAP"sv;
    case ResourceType::NSAP_PTR:
        return "NSAP_PTR"sv;
    case ResourceType::SIG:
        return "SIG"sv;
    case ResourceType::KEY:
        return "KEY"sv;
    case ResourceType::PX:
        return "PX"sv;
    case ResourceType::GPOS:
        return "GPOS"sv;
    case ResourceType::AAAA:
        return "AAAA"sv;
    case ResourceType::LOC:
        return "LOC"sv;
    case ResourceType::NXT:
        return "NXT"sv;
    case ResourceType::EID:
        return "EID"sv;
    case ResourceType::NIMLOC:
        return "NIMLOC"sv;
    case ResourceType::SRV:
        return "SRV"sv;
    case ResourceType::ATMA:
        return "ATMA"sv;
    case ResourceType::NAPTR:
        return "NAPTR"sv;
    case ResourceType::KX:
        return "KX"sv;
    case ResourceType::CERT:
        return "CERT"sv;
    case ResourceType::A6:
        return "A6"sv;
    case ResourceType::DNAME:
        return "DNAME"sv;
    case ResourceType::SINK:
        return "SINK"sv;
    case ResourceType::OPT:
        return "OPT"sv;
    case ResourceType::APL:
        return "APL"sv;
    case ResourceType::DS:
        return "DS"sv;
    case ResourceType::SSHFP:
        return "SSHFP"sv;
    case ResourceType::IPSECKEY:
        return "IPSECKEY"sv;
    case ResourceType::RRSIG:
        return "RRSIG"sv;
    case ResourceType::NSEC:
        return "NSEC"sv;
    case ResourceType::DNSKEY:
        return "DNSKEY"sv;
    case ResourceType::DHCID:
        return "DHCID"sv;
    case ResourceType::NSEC3:
        return "NSEC3"sv;
    case ResourceType::NSEC3PARAM:
        return "NSEC3PARAM"sv;
    case ResourceType::TLSA:
        return "TLSA"sv;
    case ResourceType::SMIMEA:
        return "SMIMEA"sv;
    case ResourceType::HIP:
        return "HIP"sv;
    case ResourceType::NINFO:
        return "NINFO"sv;
    case ResourceType::RKEY:
        return "RKEY"sv;
    case ResourceType::TALINK:
        return "TALINK"sv;
    case ResourceType::CDS:
        return "CDS"sv;
    case ResourceType::CDNSKEY:
        return "CDNSKEY"sv;
    case ResourceType::OPENPGPKEY:
        return "OPENPGPKEY"sv;
    case ResourceType::CSYNC:
        return "CSYNC"sv;
    case ResourceType::ZONEMD:
        return "ZONEMD"sv;
    case ResourceType::SVCB:
        return "SVCB"sv;
    case ResourceType::HTTPS:
        return "HTTPS"sv;
    case ResourceType::SPF:
        return "SPF"sv;
    case ResourceType::UINFO:
        return "UINFO"sv;
    case ResourceType::UID:
        return "UID"sv;
    case ResourceType::GID:
        return "GID"sv;
    case ResourceType::UNSPEC:
        return "UNSPEC"sv;
    case ResourceType::NID:
        return "NID"sv;
    case ResourceType::L32:
        return "L32"sv;
    case ResourceType::L64:
        return "L64"sv;
    case ResourceType::LP:
        return "LP"sv;
    case ResourceType::EUI48:
        return "EUI48"sv;
    case ResourceType::EUI64:
        return "EUI64"sv;
    case ResourceType::NXNAME:
        return "NXNAME"sv;
    case ResourceType::TKEY:
        return "TKEY"sv;
    case ResourceType::TSIG:
        return "TSIG"sv;
    case ResourceType::IXFR:
        return "IXFR"sv;
    case ResourceType::AXFR:
        return "AXFR"sv;
    case ResourceType::MAILB:
        return "MAILB"sv;
    case ResourceType::MAILA:
        return "MAILA"sv;
    case ResourceType::ANY:
        return "ANY"sv;
    case ResourceType::URI:
        return "URI"sv;
    case ResourceType::CAA:
        return "CAA"sv;
    case ResourceType::AVC:
        return "AVC"sv;
    case ResourceType::DOA:
        return "DOA"sv;
    case ResourceType::AMTRELAY:
        return "AMTRELAY"sv;
    case ResourceType::RESINFO:
        return "RESINFO"sv;
    case ResourceType::WALLET:
        return "WALLET"sv;
    case ResourceType::CLA:
        return "CLA"sv;
    case ResourceType::IPN:
        return "IPN"sv;
    case ResourceType::TA:
        return "TA"sv;
    case ResourceType::DLV:
        return "DLV"sv;
    default:
        return "UNKNOWN"sv;
    }
}

Optional<ResourceType> resource_type_from_string(StringView name)
{
    if (name == "Reserved"sv)
        return ResourceType::Reserved;
    if (name == "A"sv)
        return ResourceType::A;
    if (name == "NS"sv)
        return ResourceType::NS;
    if (name == "MD"sv)
        return ResourceType::MD;
    if (name == "MF"sv)
        return ResourceType::MF;
    if (name == "CNAME"sv)
        return ResourceType::CNAME;
    if (name == "SOA"sv)
        return ResourceType::SOA;
    if (name == "MB"sv)
        return ResourceType::MB;
    if (name == "MG"sv)
        return ResourceType::MG;
    if (name == "MR"sv)
        return ResourceType::MR;
    if (name == "NULL_"sv)
        return ResourceType::NULL_;
    if (name == "WKS"sv)
        return ResourceType::WKS;
    if (name == "PTR"sv)
        return ResourceType::PTR;
    if (name == "HINFO"sv)
        return ResourceType::HINFO;
    if (name == "MINFO"sv)
        return ResourceType::MINFO;
    if (name == "MX"sv)
        return ResourceType::MX;
    if (name == "TXT"sv)
        return ResourceType::TXT;
    if (name == "RP"sv)
        return ResourceType::RP;
    if (name == "AFSDB"sv)
        return ResourceType::AFSDB;
    if (name == "X25"sv)
        return ResourceType::X25;
    if (name == "ISDN"sv)
        return ResourceType::ISDN;
    if (name == "RT"sv)
        return ResourceType::RT;
    if (name == "NSAP"sv)
        return ResourceType::NSAP;
    if (name == "NSAP_PTR"sv)
        return ResourceType::NSAP_PTR;
    if (name == "SIG"sv)
        return ResourceType::SIG;
    if (name == "KEY"sv)
        return ResourceType::KEY;
    if (name == "PX"sv)
        return ResourceType::PX;
    if (name == "GPOS"sv)
        return ResourceType::GPOS;
    if (name == "AAAA"sv)
        return ResourceType::AAAA;
    if (name == "LOC"sv)
        return ResourceType::LOC;
    if (name == "NXT"sv)
        return ResourceType::NXT;
    if (name == "EID"sv)
        return ResourceType::EID;
    if (name == "NIMLOC"sv)
        return ResourceType::NIMLOC;
    if (name == "SRV"sv)
        return ResourceType::SRV;
    if (name == "ATMA"sv)
        return ResourceType::ATMA;
    if (name == "NAPTR"sv)
        return ResourceType::NAPTR;
    if (name == "KX"sv)
        return ResourceType::KX;
    if (name == "CERT"sv)
        return ResourceType::CERT;
    if (name == "A6"sv)
        return ResourceType::A6;
    if (name == "DNAME"sv)
        return ResourceType::DNAME;
    if (name == "SINK"sv)
        return ResourceType::SINK;
    if (name == "OPT"sv)
        return ResourceType::OPT;
    if (name == "APL"sv)
        return ResourceType::APL;
    if (name == "DS"sv)
        return ResourceType::DS;
    if (name == "SSHFP"sv)
        return ResourceType::SSHFP;
    if (name == "IPSECKEY"sv)
        return ResourceType::IPSECKEY;
    if (name == "RRSIG"sv)
        return ResourceType::RRSIG;
    if (name == "NSEC"sv)
        return ResourceType::NSEC;
    if (name == "DNSKEY"sv)
        return ResourceType::DNSKEY;
    if (name == "DHCID"sv)
        return ResourceType::DHCID;
    if (name == "NSEC3"sv)
        return ResourceType::NSEC3;
    if (name == "NSEC3PARAM"sv)
        return ResourceType::NSEC3PARAM;
    if (name == "TLSA"sv)
        return ResourceType::TLSA;
    if (name == "SMIMEA"sv)
        return ResourceType::SMIMEA;
    if (name == "HIP"sv)
        return ResourceType::HIP;
    if (name == "NINFO"sv)
        return ResourceType::NINFO;
    if (name == "RKEY"sv)
        return ResourceType::RKEY;
    if (name == "TALINK"sv)
        return ResourceType::TALINK;
    if (name == "CDS"sv)
        return ResourceType::CDS;
    if (name == "CDNSKEY"sv)
        return ResourceType::CDNSKEY;
    if (name == "OPENPGPKEY"sv)
        return ResourceType::OPENPGPKEY;
    if (name == "CSYNC"sv)
        return ResourceType::CSYNC;
    if (name == "ZONEMD"sv)
        return ResourceType::ZONEMD;
    if (name == "SVCB"sv)
        return ResourceType::SVCB;
    if (name == "HTTPS"sv)
        return ResourceType::HTTPS;
    if (name == "SPF"sv)
        return ResourceType::SPF;
    if (name == "UINFO"sv)
        return ResourceType::UINFO;
    if (name == "UID"sv)
        return ResourceType::UID;
    if (name == "GID"sv)
        return ResourceType::GID;
    if (name == "UNSPEC"sv)
        return ResourceType::UNSPEC;
    if (name == "NID"sv)
        return ResourceType::NID;
    if (name == "L32"sv)
        return ResourceType::L32;
    if (name == "L64"sv)
        return ResourceType::L64;
    if (name == "LP"sv)
        return ResourceType::LP;
    if (name == "EUI48"sv)
        return ResourceType::EUI48;
    if (name == "EUI64"sv)
        return ResourceType::EUI64;
    if (name == "NXNAME"sv)
        return ResourceType::NXNAME;
    if (name == "TKEY"sv)
        return ResourceType::TKEY;
    if (name == "TSIG"sv)
        return ResourceType::TSIG;
    if (name == "IXFR"sv)
        return ResourceType::IXFR;
    if (name == "AXFR"sv)
        return ResourceType::AXFR;
    if (name == "MAILB"sv)
        return ResourceType::MAILB;
    if (name == "MAILA"sv)
        return ResourceType::MAILA;
    if (name == "ANY"sv)
        return ResourceType::ANY;
    if (name == "URI"sv)
        return ResourceType::URI;
    if (name == "CAA"sv)
        return ResourceType::CAA;
    if (name == "AVC"sv)
        return ResourceType::AVC;
    if (name == "DOA"sv)
        return ResourceType::DOA;
    if (name == "AMTRELAY"sv)
        return ResourceType::AMTRELAY;
    if (name == "RESINFO"sv)
        return ResourceType::RESINFO;
    if (name == "WALLET"sv)
        return ResourceType::WALLET;
    if (name == "CLA"sv)
        return ResourceType::CLA;
    if (name == "IPN"sv)
        return ResourceType::IPN;
    if (name == "TA"sv)
        return ResourceType::TA;
    if (name == "DLV"sv)
        return ResourceType::DLV;
    return {};
}

StringView to_string(Class class_)
{
    switch (class_) {
    case Class::IN:
        return "IN"sv;
    case Class::CH:
        return "CH"sv;
    case Class::HS:
        return "HS"sv;
    default:
        return "UNKNOWN"sv;
    }
}

StringView to_string(OpCode code)
{
    if ((to_underlying(code) & to_underlying(OpCode::Reserved)) != 0)
        return "Reserved"sv;

    switch (code) {
    case OpCode::Query:
        return "Query"sv;
    case OpCode::IQuery:
        return "IQuery"sv;
    case OpCode::Status:
        return "Status"sv;
    case OpCode::Notify:
        return "Notify"sv;
    case OpCode::Update:
        return "Update"sv;
    case OpCode::DSO:
        return "DSO"sv;
    default:
        return "UNKNOWN"sv;
    }
}

DomainName DomainName::from_string(StringView name)
{
    DomainName domain_name;
    name.for_each_split_view('.', SplitBehavior::Nothing, [&](StringView piece) {
        domain_name.labels.append(piece);
    });
    return domain_name;
}

ErrorOr<DomainName> DomainName::from_raw(ParseContext& ctx)
{
    // RFC 1035, 4.1.2. Question section format.
    // QNAME           a domain name represented as a sequence of labels, where
    //                each label consists of a length octet followed by that
    //                number of octets.  The domain name terminates with the
    //                zero length octet for the null label of the root.  Note
    //                that this field may be an odd number of octets; no
    //                padding is used.
    DomainName name;
    auto input_offset_marker = ctx.stream.read_bytes();
    while (true) {
        auto length = TRY(ctx.stream.read_value<u8>());
        if (length == 0)
            break;

        constexpr static u8 OffsetMarkerMask = 0b11000000;
        if ((length & OffsetMarkerMask) == OffsetMarkerMask) {
            // This is a pointer to a prior domain name.
            u16 offset = static_cast<u16>(length & ~OffsetMarkerMask) << 8 | TRY(ctx.stream.read_value<u8>());
            if (auto it = ctx.pointers->find_largest_not_above_iterator(offset); !it.is_end()) {
                auto labels = it->labels;
                size_t start_index = 0;
                size_t start_entry_offset = offset - it.key();
                while (start_entry_offset > 0 && start_index < labels.size()) {
                    start_entry_offset -= labels[start_index].length() + 1; // +1 for the length byte
                    start_index++;
                }
                for (size_t i = start_index; i < labels.size(); ++i)
                    name.labels.append(labels[i].substring_view(i == start_index ? start_entry_offset : 0));
                break;
            }
            dbgln("Invalid domain name pointer in label, no prior domain name found around offset {}", offset);
            return Error::from_string_literal("Invalid domain name pointer in label");
        }

        ByteBuffer content;
        TRY(ctx.stream.read_until_filled(TRY(content.get_bytes_for_writing(length))));
        name.labels.append(ByteString::copy(content));
    }

    ctx.pointers->insert(input_offset_marker, name);

    return name;
}

ErrorOr<void> DomainName::to_raw(ByteBuffer& out) const
{
    for (auto& label : labels) {
        VERIFY(label.length() <= 63);
        auto size_bytes = TRY(out.get_bytes_for_writing(1));
        u8 size = static_cast<u8>(label.length());
        memcpy(size_bytes.data(), &size, 1);

        auto content_bytes = TRY(out.get_bytes_for_writing(label.length()));
        memcpy(content_bytes.data(), label.characters(), label.length());
    }

    TRY(out.try_append(0));

    return {};
}

String DomainName::to_string() const
{
    if (labels.is_empty())
        return "."_string;

    StringBuilder builder;
    for (size_t i = 0; i < labels.size(); ++i) {
        builder.append(labels[i]);
        builder.append('.');
    }

    return MUST(builder.to_string());
}

String DomainName::to_canonical_string() const
{
    if (labels.is_empty())
        return "."_string;

    StringBuilder builder;
    for (size_t i = 0; i < labels.size(); ++i) {
        auto& label = labels[i];
        for (size_t j = 0; j < label.length(); ++j) {
            auto ch = label[j];
            if (ch >= 'A' && ch <= 'Z')
                ch = to_ascii_lowercase(ch);
            builder.append(ch);
        }
        builder.append('.');
    }

    return MUST(builder.to_string());
}

class RecordingStream final : public Stream {
public:
    explicit RecordingStream(Stream& stream)
        : m_stream(stream)
    {
    }

    ByteBuffer take_recorded_data() && { return move(m_recorded_data); }

    virtual ErrorOr<Bytes> read_some(Bytes bytes) override
    {
        auto result = TRY(m_stream->read_some(bytes));
        m_recorded_data.append(result.data(), result.size());
        return result;
    }
    virtual ErrorOr<void> discard(size_t discarded_bytes) override
    {
        auto space = TRY(m_recorded_data.get_bytes_for_writing(discarded_bytes));
        TRY(m_stream->read_until_filled(space));
        return {};
    }
    virtual ErrorOr<size_t> write_some(ReadonlyBytes bytes) override { return m_stream->write_some(bytes); }
    virtual bool is_eof() const override { return m_stream->is_eof(); }
    virtual bool is_open() const override { return m_stream->is_open(); }
    virtual void close() override { m_stream->close(); }

private:
    MaybeOwned<Stream> m_stream;
    ByteBuffer m_recorded_data;
};

ErrorOr<ResourceRecord> ResourceRecord::from_raw(ParseContext& ctx)
{
    // RFC 1035, 4.1.3. Resource record format.
    // +           +
    // | NAME      | a domain name to which this resource record pertains
    // +           +
    // | TYPE      | two octets containing one of the RR type codes
    // | CLASS     | two octets containing one of the RR class codes
    // | TTL       | a 32-bit unsigned integer that specifies the time interval
    // |           | that the resource record may be cached
    // | RDLENGTH  | an unsigned 16-bit integer that specifies the length in
    // |           | octets of the RDATA field
    // | RDATA     | a variable length string of octets that describes the resource

    ByteBuffer rdata;
    ByteBuffer rr_raw_data;
    DomainName name;
    ResourceType type;
    Class class_;
    u32 ttl;
    size_t original_offset = ctx.stream.read_bytes();
    {
        RecordingStream rr_stream { ctx.stream };
        CountingStream rr_counting_stream { MaybeOwned<Stream>(rr_stream), original_offset };
        ParseContext rr_ctx { rr_counting_stream, move(ctx.pointers) };
        ScopeGuard guard([&] { ctx.pointers = move(rr_ctx.pointers); });

        name = TRY(DomainName::from_raw(rr_ctx));
        type = static_cast<ResourceType>(static_cast<u16>(TRY(rr_ctx.stream.read_value<NetworkOrdered<u16>>())));
        if (type == ResourceType::OPT) {
            auto record = ResourceRecord {
                move(name),
                type,
                Class::IN,
                0,
                TRY(Records::OPT::from_raw(rr_ctx)),
                {},
            };
            record.raw = move(rr_stream).take_recorded_data();
            return record;
        }
        class_ = static_cast<Class>(static_cast<u16>(TRY(rr_ctx.stream.read_value<NetworkOrdered<u16>>())));
        ttl = static_cast<u32>(TRY(rr_ctx.stream.read_value<NetworkOrdered<u32>>()));
        auto rd_length = static_cast<u16>(TRY(rr_ctx.stream.read_value<NetworkOrdered<u16>>()));
        original_offset = rr_ctx.stream.read_bytes();
        TRY(rr_ctx.stream.read_until_filled(TRY(rdata.get_bytes_for_writing(rd_length))));

        rr_raw_data = move(rr_stream).take_recorded_data();
    }

    FixedMemoryStream stream { rdata.bytes() };
    CountingStream rdata_stream { MaybeOwned<Stream>(stream), original_offset };
    ParseContext rdata_ctx { rdata_stream, move(ctx.pointers) };
    ScopeGuard guard([&] { ctx.pointers = move(rdata_ctx.pointers); });

#define PARSE_AS_RR(TYPE)                                                                                                                                   \
    do {                                                                                                                                                    \
        auto rr = TRY(Records::TYPE::from_raw(rdata_ctx));                                                                                                  \
        if (!rdata_stream.is_eof()) {                                                                                                                       \
            dbgln("Extra data ({}) left in stream: {:hex-dump}", rdata.size() - rdata_stream.read_bytes(), rdata.bytes().slice(rdata_stream.read_bytes())); \
            return Error::from_string_literal("Extra data in " #TYPE " record content");                                                                    \
        }                                                                                                                                                   \
        return ResourceRecord { move(name), type, class_, ttl, rr, move(rr_raw_data) };                                                                     \
    } while (0)

    switch (type) {
    case ResourceType::A:
        PARSE_AS_RR(A);
    case ResourceType::AAAA:
        PARSE_AS_RR(AAAA);
    case ResourceType::TXT:
        PARSE_AS_RR(TXT);
    case ResourceType::CNAME:
        PARSE_AS_RR(CNAME);
    case ResourceType::NS:
        PARSE_AS_RR(NS);
    case ResourceType::SOA:
        PARSE_AS_RR(SOA);
    case ResourceType::MX:
        PARSE_AS_RR(MX);
    case ResourceType::PTR:
        PARSE_AS_RR(PTR);
    case ResourceType::SRV:
        PARSE_AS_RR(SRV);
    case ResourceType::DNSKEY:
        PARSE_AS_RR(DNSKEY);
    case ResourceType::CDNSKEY:
        PARSE_AS_RR(CDNSKEY);
    case ResourceType::DS:
        PARSE_AS_RR(DS);
    case ResourceType::CDS:
        PARSE_AS_RR(CDS);
    case ResourceType::RRSIG:
        PARSE_AS_RR(RRSIG);
    // case ResourceType::NSEC:
    //     PARSE_AS_RR(NSEC);
    // case ResourceType::NSEC3:
    //     PARSE_AS_RR(NSEC3);
    // case ResourceType::NSEC3PARAM:
    //     PARSE_AS_RR(NSEC3PARAM);
    // case ResourceType::TLSA:
    //     PARSE_AS_RR(TLSA);
    case ResourceType::HINFO:
        PARSE_AS_RR(HINFO);
    default:
        return ResourceRecord { move(name), type, class_, ttl, move(rdata), move(rr_raw_data) };
    }
#undef PARSE_AS_RR
}

ErrorOr<void> ResourceRecord::to_raw(ByteBuffer& buffer) const
{
    TRY(name.to_raw(buffer));

    auto type_bytes = TRY(buffer.get_bytes_for_writing(2));
    auto net_type = static_cast<NetworkOrdered<u16>>(to_underlying(type));
    memcpy(type_bytes.data(), &net_type, 2);

    if (type != ResourceType::OPT) {
        auto class_bytes = TRY(buffer.get_bytes_for_writing(2));
        auto net_class = static_cast<NetworkOrdered<u16>>(to_underlying(class_));
        memcpy(class_bytes.data(), &net_class, 2);

        auto ttl_bytes = TRY(buffer.get_bytes_for_writing(4));
        auto net_ttl = static_cast<NetworkOrdered<u32>>(ttl);
        memcpy(ttl_bytes.data(), &net_ttl, 4);
    }

    ByteBuffer rdata;
    TRY(record.visit(
        [&](auto const& record) { return record.to_raw(rdata); },
        [&](ByteBuffer const& raw) { return rdata.try_append(raw); }));

    if (type != ResourceType::OPT) {
        auto rdata_length_bytes = TRY(buffer.get_bytes_for_writing(2));
        auto net_rdata_length = static_cast<NetworkOrdered<u16>>(rdata.size());
        memcpy(rdata_length_bytes.data(), &net_rdata_length, 2);
    }

    TRY(buffer.try_append(rdata));

    return {};
}

ErrorOr<String> ResourceRecord::to_string() const
{
    StringBuilder builder;
    builder.appendff("[{} {} ", Messages::to_string(class_), Messages::to_string(type));
    record.visit(
        [&](auto const& record) { builder.appendff("{}", MUST(record.to_string())); },
        [&](ByteBuffer const& raw) { builder.appendff("{:hex-dump}", raw.bytes()); });
    builder.appendff(" | ttl={}, name={}]", ttl, name.to_string());
    return builder.to_string();
}

ErrorOr<Records::A> Records::A::from_raw(ParseContext& ctx)
{
    // RFC 1035, 3.4.1. A RDATA format.
    // | ADDRESS | a 32 bit Internet address.

    u32 const address = TRY(ctx.stream.read_value<LittleEndian<u32>>());
    return Records::A { IPv4Address { address } };
}

ErrorOr<void> Records::A::to_raw(ByteBuffer& buffer) const
{
    auto const address = this->address.to_u32();
    auto const net_address = bit_cast<NetworkOrdered<u32>>(address);
    auto bytes = TRY(buffer.get_bytes_for_writing(sizeof(net_address)));
    bytes.overwrite(0, &net_address, sizeof(net_address));
    return {};
}

ErrorOr<Records::AAAA> Records::AAAA::from_raw(ParseContext& ctx)
{
    // RFC 3596, 2.2. AAAA RDATA format.
    // | ADDRESS | a 128 bit Internet address.

    u128 const address = TRY(ctx.stream.read_value<LittleEndian<u128>>());
    return Records::AAAA { IPv6Address { bit_cast<Array<u8, 16>>(address) } };
}

ErrorOr<void> Records::AAAA::to_raw(ByteBuffer& buffer) const
{
    auto const* const address_bytes = this->address.to_in6_addr_t();
    u128 address {};
    memcpy(&address, address_bytes, sizeof(address));

    auto const net_address = bit_cast<NetworkOrdered<u128>>(address);
    auto bytes = TRY(buffer.get_bytes_for_writing(sizeof(net_address)));
    bytes.overwrite(0, &net_address, sizeof(net_address));
    return {};
}

ErrorOr<Records::TXT> Records::TXT::from_raw(ParseContext& ctx)
{
    // RFC 1035, 3.3.14. TXT RDATA format.
    // | TXT-DATA | a <character-string> which is used for human readability.

    auto length = TRY(ctx.stream.read_value<u8>());
    ByteBuffer content;
    TRY(ctx.stream.read_until_filled(TRY(content.get_bytes_for_writing(length))));
    return Records::TXT { ByteString::copy(content) };
}

ErrorOr<void> Records::TXT::to_raw(ByteBuffer& buffer) const
{
    auto const length = static_cast<u8>(content.length());
    auto length_bytes = TRY(buffer.get_bytes_for_writing(1));
    memcpy(length_bytes.data(), &length, 1);

    auto content_bytes = TRY(buffer.get_bytes_for_writing(length));
    memcpy(content_bytes.data(), content.characters(), length);

    return {};
}

ErrorOr<Records::CNAME> Records::CNAME::from_raw(ParseContext& ctx)
{
    // RFC 1035, 3.3.1. CNAME RDATA format.
    // | CNAME | a <domain-name> which specifies the canonical or primary name for the owner.

    auto name = TRY(DomainName::from_raw(ctx));
    return Records::CNAME { move(name) };
}

ErrorOr<void> Records::CNAME::to_raw(ByteBuffer& buffer) const
{
    return names.to_raw(buffer);
}

ErrorOr<Records::NS> Records::NS::from_raw(ParseContext& ctx)
{
    // RFC 1035, 3.3.11. NS RDATA format.
    // | NSDNAME | a <domain-name> which specifies a host which should be authoritative for the specified class and domain.

    auto name = TRY(DomainName::from_raw(ctx));
    return Records::NS { move(name) };
}

ErrorOr<Records::SOA> Records::SOA::from_raw(ParseContext& ctx)
{
    // RFC 1035, 3.3.13. SOA RDATA format.
    // | MNAME   | <domain-name> which specifies the name of the host where the master file for the zone is maintained.
    // | RNAME   | <domain-name> which specifies the mailbox of the person responsible for this zone.
    // | SERIAL  | a 32-bit unsigned integer that specifies the version number of the original copy of the zone.
    // | REFRESH | a 32-bit unsigned integer that specifies the time interval before the zone should be refreshed.
    // | RETRY   | a 32-bit unsigned integer that specifies the time interval that should elapse before a failed refresh should be retried.
    // | EXPIRE  | a 32-bit unsigned integer that specifies the time value that specifies the upper limit on the time interval that can elapse before the zone is no longer authoritative.
    // | MINIMUM | a 32-bit unsigned integer that specifies the minimum TTL field that should be exported with any RR from this zone.

    auto mname = TRY(DomainName::from_raw(ctx));
    auto rname = TRY(DomainName::from_raw(ctx));
    auto serial = static_cast<u32>(TRY(ctx.stream.read_value<NetworkOrdered<u32>>()));
    auto refresh = static_cast<u32>(TRY(ctx.stream.read_value<NetworkOrdered<u32>>()));
    auto retry = static_cast<u32>(TRY(ctx.stream.read_value<NetworkOrdered<u32>>()));
    auto expire = static_cast<u32>(TRY(ctx.stream.read_value<NetworkOrdered<u32>>()));
    auto minimum = static_cast<u32>(TRY(ctx.stream.read_value<NetworkOrdered<u32>>()));

    return Records::SOA { move(mname), move(rname), serial, refresh, retry, expire, minimum };
}

ErrorOr<void> Records::SOA::to_raw(ByteBuffer& buffer) const
{
    TRY(mname.to_raw(buffer));
    TRY(rname.to_raw(buffer));

    auto const output_size = 5 * sizeof(u32);
    FixedMemoryStream stream { TRY(buffer.get_bytes_for_writing(output_size)) };

    TRY(stream.write_value(static_cast<NetworkOrdered<u32>>(serial)));
    TRY(stream.write_value(static_cast<NetworkOrdered<u32>>(refresh)));
    TRY(stream.write_value(static_cast<NetworkOrdered<u32>>(retry)));
    TRY(stream.write_value(static_cast<NetworkOrdered<u32>>(expire)));
    TRY(stream.write_value(static_cast<NetworkOrdered<u32>>(minimum)));

    return {};
}

ErrorOr<Records::MX> Records::MX::from_raw(ParseContext& ctx)
{
    // RFC 1035, 3.3.9. MX RDATA format.
    // | PREFERENCE | a 16 bit integer which specifies the preference given to this RR among others at the same owner.
    // | EXCHANGE   | a <domain-name> which specifies a host willing to act as a mail exchange for the owner name.

    auto preference = static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>()));
    auto exchange = TRY(DomainName::from_raw(ctx));
    return Records::MX { preference, move(exchange) };
}

ErrorOr<Records::PTR> Records::PTR::from_raw(ParseContext& ctx)
{
    // RFC 1035, 3.3.12. PTR RDATA format.
    // | PTRDNAME | a <domain-name> which points to some location in the domain name space.

    auto name = TRY(DomainName::from_raw(ctx));
    return Records::PTR { move(name) };
}

ErrorOr<Records::SRV> Records::SRV::from_raw(ParseContext& ctx)
{
    // RFC 2782, 2. Service location and priority.
    // | PRIORITY | a 16 bit integer that specifies the priority of this target host.
    // | WEIGHT   | a 16 bit integer that specifies a weight for this target host.
    // | PORT     | a 16 bit integer that specifies the port on this target host.
    // | TARGET   | a <domain-name> which specifies the target host.

    auto priority = static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>()));
    auto weight = static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>()));
    auto port = static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>()));
    auto target = TRY(DomainName::from_raw(ctx));
    return Records::SRV { priority, weight, port, move(target) };
}

ErrorOr<Records::DNSKEY> Records::DNSKEY::from_raw(ParseContext& ctx)
{
    // RFC 4034, 2.1. The DNSKEY Resource Record.
    // | FLAGS    | a 16-bit value that flags the key.
    // | PROTOCOL | an 8-bit value that specifies the protocol for this key.
    // | ALGORITHM| an 8-bit value that identifies the public key's cryptographic algorithm.
    // | PUBLICKEY| the public key material.

    u32 key_tag = 0;
    auto flags = static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>()));
    key_tag += (bit_cast<u16>(NetworkOrdered<u16>(flags)) & 0xff) << 8;
    key_tag += (bit_cast<u16>(NetworkOrdered<u16>(flags)) >> 8) & 0xff;
    auto protocol = TRY(ctx.stream.read_value<u8>());
    key_tag += static_cast<u16>(protocol) << 8;
    auto algorithm = static_cast<DNSSEC::Algorithm>(static_cast<u8>(TRY(ctx.stream.read_value<u8>())));
    key_tag += static_cast<u16>(algorithm);
    auto public_key = TRY(ctx.stream.read_until_eof());
    for (size_t i = 0; i < public_key.size(); ++i) {
        key_tag += (i & 1) ? static_cast<u16>(public_key[i]) : static_cast<u16>(public_key[i]) << 8;
    }
    key_tag += (key_tag >> 16) & 0xffff;

    if (public_key.is_empty())
        return Error::from_string_literal("Empty public key in DNSKEY record");

    return Records::DNSKEY { flags, protocol, algorithm, move(public_key), static_cast<u16>(key_tag & 0xffff) };
}

ErrorOr<void> Records::DNSKEY::to_raw(ByteBuffer& buffer) const
{
    auto const output_size = 2 + 1 + 1 + public_key.size();
    FixedMemoryStream stream { TRY(buffer.get_bytes_for_writing(output_size)) };

    TRY(stream.write_value(static_cast<u16>(bit_cast<NetworkOrdered<u16>>(flags))));
    TRY(stream.write_value(protocol));
    TRY(stream.write_value(to_underlying(algorithm)));
    TRY(stream.write_until_depleted(public_key.bytes()));
    return {};
}

ErrorOr<Records::DS> Records::DS::from_raw(ParseContext& ctx)
{
    // RFC 4034, 5.1. The DS Resource Record.
    // | KEYTAG      | a 16-bit value that identifies the DNSKEY RR.
    // | ALGORITHM   | an 8-bit value that identifies the DS's hash algorithm.
    // | DIGEST TYPE | an 8-bit value that identifies the DS's digest algorithm.
    // | DIGEST      | Digest of the DNSKEY RDATA (Flags | Protocol | Algorithm | Pubkey).

    auto key_tag = static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>()));
    auto algorithm = static_cast<DNSSEC::Algorithm>(static_cast<u8>(TRY(ctx.stream.read_value<u8>())));
    auto digest_type = static_cast<DNSSEC::DigestType>(static_cast<u8>(TRY(ctx.stream.read_value<u8>())));
    size_t digest_size;
    switch (digest_type) {
    case DNSSEC::DigestType::SHA1:
        digest_size = 20;
        break;
    case DNSSEC::DigestType::SHA256:
    case DNSSEC::DigestType::GOST3411:
        digest_size = 32;
        break;
    case DNSSEC::DigestType::SHA384:
        digest_size = 48;
        break;
    case DNSSEC::DigestType::SHA512:
        digest_size = 64;
        break;
    case DNSSEC::DigestType::SHA224:
        digest_size = 28;
        break;
    case DNSSEC::DigestType::Unknown:
    default:
        return Error::from_string_literal("Unknown digest type in DS record");
    }

    ByteBuffer digest;
    TRY(ctx.stream.read_until_filled(TRY(digest.get_bytes_for_writing(digest_size))));
    return Records::DS { key_tag, algorithm, digest_type, move(digest) };
}

ErrorOr<void> Records::DS::to_raw(ByteBuffer& buffer) const
{
    auto const output_size = 2 + 1 + 1 + digest.size();
    FixedMemoryStream stream { TRY(buffer.get_bytes_for_writing(output_size)) };

    TRY(stream.write_value(static_cast<NetworkOrdered<u16>>(key_tag)));
    TRY(stream.write_value(static_cast<u8>(algorithm)));
    TRY(stream.write_value(static_cast<u8>(digest_type)));
    TRY(stream.write_until_depleted(digest.bytes()));

    return {};
}

ErrorOr<Records::SIG> Records::SIG::from_raw(ParseContext& ctx)
{
    // RFC 4034, 2.2. The SIG Resource Record.
    // | TYPE-COVERED | a 16-bit value that specifies the type of the RRset that is covered by this SIG.
    // | ALGORITHM    | an 8-bit value that specifies the algorithm used to create the signature.
    // | LABELS       | an 8-bit value that specifies the number of labels in the original SIG RR owner name.
    // | ORIGINAL TTL | a 32-bit value that specifies the TTL of the covered RRset as it appears in the authoritative zone.
    // | SIGNATURE EXPIRATION | a 32-bit value that specifies the expiration date of the signature.
    // | SIGNATURE INCEPTION  | a 32-bit value that specifies the inception date of the signature.
    // | KEY TAG      | a 16-bit value that contains the key tag value of the DNSKEY RR that was used to create the signature.
    // | SIGNER'S NAME| a <domain-name> which specifies the domain name of the signer generating the SIG RR.
    // | SIGNATURE    | a <signature> that authenticates the RRs.

    auto type_covered = static_cast<ResourceType>(static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>())));
    auto algorithm = static_cast<DNSSEC::Algorithm>(static_cast<u8>(TRY(ctx.stream.read_value<u8>())));
    auto labels = TRY(ctx.stream.read_value<u8>());
    auto original_ttl = static_cast<u32>(TRY(ctx.stream.read_value<NetworkOrdered<u32>>()));
    auto signature_expiration = static_cast<u32>(TRY(ctx.stream.read_value<NetworkOrdered<u32>>()));
    auto signature_inception = static_cast<u32>(TRY(ctx.stream.read_value<NetworkOrdered<u32>>()));
    auto key_tag = static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>()));
    auto signer_name = TRY(DomainName::from_raw(ctx));
    auto signature = TRY(ctx.stream.read_until_eof());

    return Records::SIG { type_covered, algorithm, labels, original_ttl, UnixDateTime::from_seconds_since_epoch(signature_expiration), UnixDateTime::from_seconds_since_epoch(signature_inception), key_tag, move(signer_name), move(signature) };
}

ErrorOr<void> Records::SIG::to_raw_excluding_signature(ByteBuffer& buffer) const
{
    AllocatingMemoryStream stream;
    TRY(stream.write_value(static_cast<NetworkOrdered<u16>>(to_underlying(type_covered))));
    TRY(stream.write_value(static_cast<u8>(algorithm)));
    TRY(stream.write_value(label_count));
    TRY(stream.write_value(static_cast<NetworkOrdered<u32>>(original_ttl)));
    TRY(stream.write_value(static_cast<NetworkOrdered<u32>>(expiration.seconds_since_epoch())));
    TRY(stream.write_value(static_cast<NetworkOrdered<u32>>(inception.seconds_since_epoch())));
    TRY(stream.write_value(static_cast<NetworkOrdered<u16>>(key_tag)));

    TRY(stream.read_until_filled(TRY(buffer.get_bytes_for_writing(stream.used_buffer_size()))));
    TRY(signers_name.to_raw(buffer));
    return {};
}

ErrorOr<void> Records::SIG::to_raw(ByteBuffer& buffer) const
{
    TRY(to_raw_excluding_signature(buffer));
    TRY(buffer.try_append(signature));
    return {};
}

ErrorOr<String> Records::SIG::to_string() const
{
    // Single line:
    // SIG Type covered: <type>, Algorithm: <algorithm>, Labels: <labels>, Original TTL: <ttl>, Signature expiration: <expiration>, Signature inception: <inception>, Key tag: <key tag>, Signer's name: <signer>, Signature: <signature>
    StringBuilder builder;
    builder.append("SIG "sv);
    builder.appendff("Type covered: {}, ", Messages::to_string(type_covered));
    builder.appendff("Algorithm: {}, ", DNSSEC::to_string(algorithm));
    builder.appendff("Labels: {}, ", label_count);
    builder.appendff("Original TTL: {}, ", original_ttl);
    builder.appendff("Signature expiration: {}, ", expiration);
    builder.appendff("Signature inception: {}, ", inception);
    builder.appendff("Key tag: {}, ", key_tag);
    builder.appendff("Signer's name: '{}', ", signers_name.to_string());
    builder.appendff("Signature: {}", TRY(encode_base64(signature)));
    return builder.to_string();
}

ErrorOr<Records::HINFO> Records::HINFO::from_raw(ParseContext& ctx)
{
    // RFC 1035, 3.3.2. HINFO RDATA format.
    // | CPU    | a <character-string> which specifies the CPU type.
    // | OS     | a <character-string> which specifies the operating system type.

    auto cpu_length = TRY(ctx.stream.read_value<u8>());
    ByteBuffer cpu;
    TRY(ctx.stream.read_until_filled(TRY(cpu.get_bytes_for_writing(cpu_length))));
    auto os_length = TRY(ctx.stream.read_value<u8>());
    ByteBuffer os;
    TRY(ctx.stream.read_until_filled(TRY(os.get_bytes_for_writing(os_length))));
    return Records::HINFO { ByteString::copy(cpu), ByteString::copy(os) };
}

ErrorOr<void> Records::HINFO::to_raw(ByteBuffer& buffer) const
{
    auto allocated_length = cpu.length() + os.length() + 2;
    auto bytes = TRY(buffer.get_bytes_for_writing(allocated_length));
    FixedMemoryStream stream { bytes };
    TRY(stream.write_value(static_cast<u8>(cpu.length())));
    TRY(stream.write_until_depleted(cpu.bytes()));
    TRY(stream.write_value(static_cast<u8>(os.length())));
    TRY(stream.write_until_depleted(os.bytes()));
    return {};
}

ErrorOr<Records::OPT> Records::OPT::from_raw(ParseContext& ctx)
{
    // RFC 6891, 6.1. The OPT pseudo-RR.
    // This RR does *not* use the standard RDATA format, `ctx` starts right after 'TYPE'.
    // | NAME       | empty (root domain)
    // | TYPE       | OPT (41)
    // - we are here -
    // | UDP SIZE   | 16-bit max UDP payload size
    // | RCODE_AND_FLAGS | 32-bit flags and response code
    // | RDLENGTH   | 16-bit length of the RDATA field
    // | RDATA      | variable length, pairs of OPTION-CODE and OPTION-DATA { length(16), data(length) }

    auto udp_size = static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>()));
    auto rcode_and_flags = static_cast<u32>(TRY(ctx.stream.read_value<NetworkOrdered<u32>>()));
    auto rd_length = static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>()));
    Vector<OPT::Option> options;
    while (rd_length > 0 && !ctx.stream.is_eof()) {
        auto option_code = static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>()));
        auto option_length = static_cast<u16>(TRY(ctx.stream.read_value<NetworkOrdered<u16>>()));
        ByteBuffer option_data;
        TRY(ctx.stream.read_until_filled(TRY(option_data.get_bytes_for_writing(option_length))));
        rd_length -= 4 + option_length;
        options.append(OPT::Option { option_code, move(option_data) });
    }

    if (rd_length != 0)
        return Error::from_string_literal("Invalid OPT record");

    return Records::OPT { udp_size, rcode_and_flags, move(options) };
}

ErrorOr<void> Records::OPT::to_raw(ByteBuffer& buffer) const
{
    auto udp_size_bytes = TRY(buffer.get_bytes_for_writing(sizeof(udp_payload_size)));
    auto net_udp_size = static_cast<NetworkOrdered<u16>>(udp_payload_size);
    memcpy(udp_size_bytes.data(), &net_udp_size, 2);

    auto rcode_and_flags_bytes = TRY(buffer.get_bytes_for_writing(sizeof(extended_rcode_and_flags)));
    auto net_rcode_and_flags = static_cast<NetworkOrdered<u32>>(extended_rcode_and_flags);
    memcpy(rcode_and_flags_bytes.data(), &net_rcode_and_flags, 4);

    auto rd_length_bytes = TRY(buffer.get_bytes_for_writing(2));
    u16 rd_length = 0;
    for (auto const& option : options) {
        rd_length += 4 + option.data.size();
    }
    auto net_rd_length = static_cast<NetworkOrdered<u16>>(rd_length);
    memcpy(rd_length_bytes.data(), &net_rd_length, 2);

    for (auto& option : options) {
        auto option_code_bytes = TRY(buffer.get_bytes_for_writing(sizeof(option.code)));
        auto net_option_code = static_cast<NetworkOrdered<u16>>(option.code);
        memcpy(option_code_bytes.data(), &net_option_code, 2);

        auto option_length_bytes = TRY(buffer.get_bytes_for_writing(2));
        auto net_option_length = static_cast<NetworkOrdered<u16>>(option.data.size());
        memcpy(option_length_bytes.data(), &net_option_length, 2);

        TRY(buffer.try_append(option.data));
    }

    return {};
}

}
