d03d16cbdf
This also introduces some 'reflection' support that will eventually unify message generation/parting and zone file format input/output
146 lines
3.6 KiB
C++
146 lines
3.6 KiB
C++
#pragma once
|
|
#include "dns.hh"
|
|
#include "dns-storage.hh"
|
|
#include "record-types.hh"
|
|
#include <arpa/inet.h>
|
|
#include <vector>
|
|
|
|
class DNSMessageReader
|
|
{
|
|
public:
|
|
DNSMessageReader(const char* input, uint16_t length);
|
|
DNSMessageReader(const std::string& str) : DNSMessageReader(str.c_str(), str.size()) {}
|
|
struct dnsheader dh=dnsheader{};
|
|
std::vector<uint8_t> payload;
|
|
uint16_t payloadpos{0};
|
|
|
|
void getQuestion(DNSName& name, DNSType& type) const;
|
|
bool getEDNS(uint16_t* newsize, bool* doBit) const;
|
|
|
|
bool getRR(DNSSection& section, DNSName& name, DNSType& type, uint32_t& ttl, std::unique_ptr<RRGen>& content);
|
|
|
|
uint8_t d_ednsVersion{0};
|
|
|
|
void xfrName(DNSName& ret, uint16_t* pos=0);
|
|
DNSName getName(uint16_t* pos=0) { DNSName res; xfrName(res, pos); return res;}
|
|
void xfrUInt8(uint8_t&res, uint16_t* pos = 0)
|
|
{
|
|
if(!pos) pos = &payloadpos;
|
|
res=payload.at((*pos)++);
|
|
}
|
|
uint8_t getUInt8(uint16_t* pos=0)
|
|
{ uint8_t ret; xfrUInt8(ret, pos); return ret; }
|
|
|
|
void xfrUInt16(uint16_t& res)
|
|
{
|
|
memcpy(&res, &payload.at(payloadpos+1)-1, 2);
|
|
payloadpos+=2;
|
|
res=htons(res);
|
|
}
|
|
uint16_t getUInt16()
|
|
{ uint16_t ret; xfrUInt16(ret); return ret; }
|
|
|
|
void xfrUInt32(uint32_t& res)
|
|
{
|
|
memcpy(&res, &payload.at(payloadpos+3)-3, 4);
|
|
payloadpos+=4;
|
|
res=ntohl(res);
|
|
}
|
|
|
|
void xfrBlob(std::string& blob, int size, uint16_t* pos = 0)
|
|
{
|
|
if(!pos) pos = &payloadpos;
|
|
if(!size) {
|
|
blob.clear();
|
|
return;
|
|
}
|
|
blob.assign(&payload.at(*pos), &payload.at(*pos+size-1)+1);
|
|
(*pos) += size;
|
|
}
|
|
|
|
std::string getBlob(int size, uint16_t* pos = 0)
|
|
{
|
|
std::string res;
|
|
xfrBlob(res, size, pos);
|
|
return res;
|
|
}
|
|
|
|
DNSName d_qname;
|
|
DNSType d_qtype{(DNSType)0};
|
|
DNSClass d_qclass{(DNSClass)0};
|
|
uint16_t d_bufsize;
|
|
bool d_doBit{false};
|
|
|
|
bool d_haveEDNS{false};
|
|
};
|
|
|
|
class DNSMessageWriter
|
|
{
|
|
public:
|
|
struct dnsheader dh=dnsheader{};
|
|
std::vector<uint8_t> payload;
|
|
uint16_t payloadpos=0;
|
|
DNSName d_qname;
|
|
DNSType d_qtype;
|
|
DNSClass d_qclass;
|
|
bool haveEDNS{false};
|
|
bool d_doBit;
|
|
bool d_nocompress{false}; // if set, never compress. For AXFR/IXFR
|
|
RCode d_ercode{(RCode)0};
|
|
|
|
DNSMessageWriter(const DNSName& name, DNSType type, int maxsize=500);
|
|
~DNSMessageWriter();
|
|
DNSMessageWriter(const DNSMessageWriter&) = delete;
|
|
DNSMessageWriter& operator=(const DNSMessageWriter&) = delete;
|
|
void clearRRs();
|
|
void putRR(DNSSection section, const DNSName& name, DNSType type, uint32_t ttl, const std::unique_ptr<RRGen>& rr);
|
|
void setEDNS(uint16_t bufsize, bool doBit, RCode ercode = (RCode)0);
|
|
std::string serialize();
|
|
|
|
void xfrUInt8(uint8_t val)
|
|
{
|
|
payload.at(payloadpos++)=val;
|
|
}
|
|
|
|
uint16_t xfrUInt16(uint16_t val)
|
|
{
|
|
val = htons(val);
|
|
memcpy(&payload.at(payloadpos+2)-2, &val, 2);
|
|
payloadpos+=2;
|
|
return payloadpos - 2;
|
|
}
|
|
|
|
void xfrUInt16At(uint16_t pos, uint16_t val)
|
|
{
|
|
val = htons(val);
|
|
memcpy(&payload.at(pos+2)-2, &val, 2);
|
|
}
|
|
|
|
void xfrUInt32(uint32_t val)
|
|
{
|
|
val = htonl(val);
|
|
memcpy(&payload.at(payloadpos+sizeof(val)) - sizeof(val), &val, sizeof(val));
|
|
payloadpos += sizeof(val);
|
|
}
|
|
|
|
void xfrBlob(const std::string& blob)
|
|
{
|
|
memcpy(&payload.at(payloadpos+blob.size()) - blob.size(), blob.c_str(), blob.size());
|
|
payloadpos += blob.size();;
|
|
}
|
|
|
|
void xfrBlob(const unsigned char* blob, int size)
|
|
{
|
|
memcpy(&payload.at(payloadpos+size) - size, blob, size);
|
|
payloadpos += size;
|
|
}
|
|
|
|
void xfrName(const DNSName& name, bool compress=true);
|
|
|
|
|
|
private:
|
|
std::unique_ptr<DNSNode> d_comptree;
|
|
void putEDNS(uint16_t bufsize, RCode ercode, bool doBit);
|
|
};
|
|
|