hello-dns/tdns/dns-storage.hh

124 lines
3.2 KiB
C++
Raw Normal View History

2018-04-09 04:20:11 +07:00
#pragma once
#include <strings.h>
#include <string>
#include <map>
#include <vector>
#include <deque>
#include <iostream>
#include <cstdint>
#include <functional>
2018-04-10 01:49:37 +07:00
#include <memory>
2018-04-09 04:20:11 +07:00
#include "nenum.hh"
class dnslabel
{
public:
dnslabel() {}
dnslabel(const char* s) : dnslabel(std::string(s)) {}
dnslabel(const std::string& s) : d_s(s) {} // XXX check length here!
bool operator<(const dnslabel& rhs) const
{
return std::lexicographical_compare(d_s.begin(), d_s.end(), rhs.d_s.begin(), rhs.d_s.end(), charcomp);
}
bool operator==(const dnslabel &rhs) const
{
return !(*this < rhs) && !(rhs<*this);
}
auto size() const { return d_s.size(); }
std::string d_s;
private:
static bool charcomp(char a, char b)
{
if(a >= 0x61 && a <= 0x7A)
a -= 0x20;
if(b >= 0x61 && b <= 0x7A)
b -= 0x20;
return a < b;
}
};
std::ostream & operator<<(std::ostream &os, const dnslabel& d);
2018-04-09 04:20:11 +07:00
enum class RCode
{
Noerror = 0, Servfail = 2, Nxdomain = 3, Notimp = 4, Refused = 5
};
SMARTENUMSTART(RCode)
SENUM5(RCode, Noerror, Servfail, Nxdomain, Notimp, Refused)
SMARTENUMEND(RCode)
enum class DNSType : uint16_t
{
A = 1, NS = 2, CNAME = 5, SOA=6, PTR=12, MX=15, TXT=16, AAAA = 28, SRV=33, OPT=41, IXFR = 251, AXFR = 252, ANY = 255
2018-04-09 04:20:11 +07:00
};
SMARTENUMSTART(DNSType)
SENUM13(DNSType, A, NS, CNAME, SOA, PTR, MX, TXT, AAAA, IXFR, AAAA, SRV, OPT, IXFR)
SENUM2(DNSType, AXFR, ANY)
2018-04-09 04:20:11 +07:00
SMARTENUMEND(DNSType)
2018-04-10 04:04:13 +07:00
COMBOENUM4(DNSSection, Question, 0, Answer, 1, Authority, 2, Additional, 3)
2018-04-09 04:20:11 +07:00
struct dnsname
{
dnsname() {}
dnsname(std::initializer_list<dnslabel> dls) : d_name(dls) {}
void push_back(const dnslabel& l) { d_name.push_back(l); }
auto back() const { return d_name.back(); }
auto begin() const { return d_name.begin(); }
bool empty() const { return d_name.empty(); }
auto end() const { return d_name.end(); }
auto front() const { return d_name.front(); }
void pop_back() { d_name.pop_back(); }
auto push_front(const dnslabel& dn) { return d_name.push_front(dn); }
auto size() { return d_name.size(); }
2018-04-10 04:04:13 +07:00
void clear() { d_name.clear(); }
2018-04-10 01:49:37 +07:00
bool makeRelative(const dnsname& root);
2018-04-09 04:20:11 +07:00
std::deque<dnslabel> d_name;
};
std::ostream & operator<<(std::ostream &os, const dnsname& d);
dnsname operator+(const dnsname& a, const dnsname& b);
2018-04-09 17:54:27 +07:00
2018-04-09 22:46:31 +07:00
struct DNSMessageWriter;
2018-04-11 17:45:26 +07:00
struct RRGen
2018-04-09 22:46:31 +07:00
{
virtual void toMessage(DNSMessageWriter& dpw) = 0;
2018-04-11 17:45:26 +07:00
virtual DNSType getType() const = 0;
2018-04-09 22:46:31 +07:00
};
2018-04-09 04:20:11 +07:00
struct RRSet
{
2018-04-11 17:45:26 +07:00
std::vector<std::unique_ptr<RRGen>> contents;
void add(std::unique_ptr<RRGen>&& rr)
2018-04-10 01:49:37 +07:00
{
contents.emplace_back(std::move(rr));
}
2018-04-09 04:20:11 +07:00
uint32_t ttl{3600};
};
struct DNSNode
{
2018-04-10 05:05:41 +07:00
const DNSNode* find(dnsname& name, dnsname& last, const DNSNode** passedZonecut=0, dnsname* zonecutname=0) const;
2018-04-09 04:20:11 +07:00
DNSNode* add(dnsname name);
std::map<dnslabel, DNSNode> children;
2018-04-09 04:20:11 +07:00
std::map<DNSType, RRSet > rrsets;
2018-04-11 17:45:26 +07:00
void addRRs(std::unique_ptr<RRGen>&&a);
template<typename... Types>
void addRRs(std::unique_ptr<RRGen>&&a, Types&&... args)
{
addRRs(std::move(a));
addRRs(std::forward<Types>(args)...);
}
2018-04-09 04:20:11 +07:00
void visit(std::function<void(const dnsname& name, const DNSNode*)> visitor, dnsname name) const;
DNSNode* zone{0}; // if this is set, this node is a zone
};
2018-04-10 15:04:49 +07:00
void loadZones(DNSNode& zones);