lokinet/llarp/dns/name.cpp

134 lines
3.2 KiB
C++
Raw Normal View History

#define __USE_MINGW_ANSI_STDIO 1
2018-12-12 00:58:08 +00:00
#include <dns/name.hpp>
#include <net/net.hpp>
2019-06-12 13:48:14 +00:00
#include <net/ip.hpp>
2018-12-12 00:58:08 +00:00
2018-12-03 22:22:59 +00:00
#include <algorithm>
2018-12-12 00:58:08 +00:00
#include <sstream>
2018-12-03 22:22:59 +00:00
namespace llarp
{
namespace dns
{
bool
2019-04-26 12:11:34 +00:00
DecodeName(llarp_buffer_t* buf, Name_t& name, bool trimTrailingDot)
2018-12-03 22:22:59 +00:00
{
if(buf->size_left() < 1)
2018-12-03 22:22:59 +00:00
return false;
std::stringstream ss;
size_t l;
do
{
l = *buf->cur;
buf->cur++;
if(l)
2018-12-04 16:16:43 +00:00
{
if(l > 63)
{
llarp::LogError("decode name failed, field too big: ", l, " > 63");
llarp::DumpBuffer(*buf);
return false;
}
if(buf->size_left() < l)
2018-12-04 16:16:43 +00:00
return false;
2018-12-03 22:22:59 +00:00
ss << Name_t((const char*)buf->cur, l);
2018-12-04 16:16:43 +00:00
ss << ".";
}
buf->cur = buf->cur + l;
2018-12-03 22:22:59 +00:00
} while(l);
name = ss.str();
2019-04-26 12:11:34 +00:00
/// trim off last dot
if(trimTrailingDot)
name = name.substr(0, name.find_last_of('.'));
2018-12-03 22:22:59 +00:00
return true;
}
bool
EncodeName(llarp_buffer_t* buf, const Name_t& name)
{
2018-12-04 16:16:43 +00:00
std::stringstream ss;
if(name.size() && name[name.size() - 1] == '.')
ss << name.substr(0, name.size() - 1);
else
ss << name;
2018-12-03 22:22:59 +00:00
std::string part;
while(std::getline(ss, part, '.'))
{
2018-12-04 16:16:43 +00:00
size_t l = part.length();
if(l > 63)
2018-12-03 22:22:59 +00:00
return false;
2018-12-04 16:16:43 +00:00
*(buf->cur) = l;
2018-12-03 22:22:59 +00:00
buf->cur++;
if(buf->size_left() < l)
2018-12-03 22:22:59 +00:00
return false;
2018-12-04 16:16:43 +00:00
if(l)
{
memcpy(buf->cur, part.data(), l);
buf->cur += l;
}
else
break;
2018-12-03 22:22:59 +00:00
}
2018-12-04 16:16:43 +00:00
*buf->cur = 0;
buf->cur++;
2018-12-03 22:22:59 +00:00
return true;
}
bool
2019-06-12 13:48:14 +00:00
DecodePTR(const Name_t& name, huint128_t& ip)
2018-12-03 22:22:59 +00:00
{
2019-06-12 13:48:14 +00:00
bool isV6 = false;
auto pos = name.find(".in-addr.arpa");
if(pos == std::string::npos)
{
pos = name.find(".ip6.arpa");
isV6 = true;
}
2018-12-04 16:52:48 +00:00
if(pos == std::string::npos)
2018-12-03 22:22:59 +00:00
return false;
2019-06-12 13:48:14 +00:00
std::string sub = name.substr(0, pos + 1);
const auto numdots = std::count(sub.begin(), sub.end(), '.');
if(numdots == 4 && !isV6)
2018-12-03 22:22:59 +00:00
{
uint8_t a, b, c, d;
pos = sub.find('.');
d = atoi(sub.substr(0, pos).c_str());
2018-12-04 16:52:48 +00:00
sub = sub.substr(pos + 1);
2018-12-03 22:22:59 +00:00
pos = sub.find('.');
c = atoi(sub.substr(0, pos).c_str());
2018-12-04 16:52:48 +00:00
sub = sub.substr(pos + 1);
2018-12-03 22:22:59 +00:00
pos = sub.find('.');
b = atoi(sub.substr(0, pos).c_str());
2018-12-04 16:52:48 +00:00
sub = sub.substr(pos + 1);
2018-12-03 22:22:59 +00:00
pos = sub.find('.');
a = atoi(sub.substr(0, pos).c_str());
2019-06-12 13:48:14 +00:00
ip = net::IPPacket::ExpandV4(llarp::ipaddr_ipv4_bits(a, b, c, d));
return true;
}
else if(numdots == 32 && isV6)
{
size_t idx = 0;
uint8_t lo, hi;
uint8_t* ptr = (uint8_t*)&ip.h;
while(idx < 16)
{
pos = sub.find('.');
lo = (*sub.substr(0, pos).c_str()) - 'a';
sub = sub.substr(pos + 1);
pos = sub.find('.');
hi = (*sub.substr(0, pos).c_str()) - 'a';
sub = sub.substr(pos + 1);
ptr[idx] = lo | (hi << 4);
++idx;
}
2018-12-03 22:22:59 +00:00
return true;
}
2019-07-06 17:03:40 +00:00
return false;
2018-12-03 22:22:59 +00:00
}
} // namespace dns
} // namespace llarp