#include #include #include #include #include #include namespace llarp { namespace dns { Proxy::Proxy( llarp_ev_loop_ptr serverLoop, Logic_ptr serverLogic, llarp_ev_loop_ptr clientLoop, Logic_ptr clientLogic, IQueryHandler* h) : m_ServerLoop(std::move(serverLoop)) , m_ClientLoop(std::move(clientLoop)) , m_ServerLogic(std::move(serverLogic)) , m_ClientLogic(std::move(clientLogic)) , m_QueryHandler(h) { m_Client.user = this; m_Server.user = this; m_Client.tick = nullptr; m_Server.tick = nullptr; m_Client.recvfrom = &HandleUDPRecv_client; m_Server.recvfrom = &HandleUDPRecv_server; } void Proxy::Stop() { } bool Proxy::Start(const llarp::Addr addr, const std::vector& resolvers) { m_Resolvers.clear(); m_Resolvers = resolvers; const llarp::Addr any("0.0.0.0", 0); auto self = shared_from_this(); LogicCall(m_ClientLogic, [=]() { llarp_ev_add_udp(self->m_ClientLoop.get(), &self->m_Client, any); }); LogicCall(m_ServerLogic, [=]() { llarp_ev_add_udp(self->m_ServerLoop.get(), &self->m_Server, addr); }); return true; } static Proxy::Buffer_t CopyBuffer(const llarp_buffer_t& buf) { std::vector msgbuf(buf.sz); std::copy_n(buf.base, buf.sz, msgbuf.data()); return msgbuf; } void Proxy::HandleUDPRecv_server(llarp_udp_io* u, const sockaddr* from, ManagedBuffer buf) { const llarp::Addr addr(*from); Buffer_t msgbuf = CopyBuffer(buf.underlying); auto self = static_cast(u->user)->shared_from_this(); // yes we use the server loop here because if the server loop is not the // client loop we'll crash again LogicCall( self->m_ServerLogic, [self, addr, msgbuf]() { self->HandlePktServer(addr, msgbuf); }); } void Proxy::HandleUDPRecv_client(llarp_udp_io* u, const sockaddr* from, ManagedBuffer buf) { const llarp::Addr addr(*from); Buffer_t msgbuf = CopyBuffer(buf.underlying); auto self = static_cast(u->user)->shared_from_this(); LogicCall( self->m_ServerLogic, [self, addr, msgbuf]() { self->HandlePktClient(addr, msgbuf); }); } llarp::Addr Proxy::PickRandomResolver() const { const size_t sz = m_Resolvers.size(); if (sz <= 1) return m_Resolvers[0]; auto itr = m_Resolvers.begin(); std::advance(itr, llarp::randint() % sz); return *itr; } void Proxy::HandleTick(llarp_udp_io*) { } void Proxy::SendServerMessageTo(llarp::Addr to, Message msg) { auto self = shared_from_this(); LogicCall(m_ServerLogic, [to, msg, self]() { std::array tmp = {{0}}; llarp_buffer_t buf(tmp); if (msg.Encode(&buf)) { buf.sz = buf.cur - buf.base; buf.cur = buf.base; llarp_ev_udp_sendto(&self->m_Server, to, buf); } else llarp::LogWarn("failed to encode dns message when sending"); }); } void Proxy::SendClientMessageTo(llarp::Addr to, Message msg) { auto self = shared_from_this(); LogicCall(m_ClientLogic, [to, msg, self]() { std::array tmp = {{0}}; llarp_buffer_t buf(tmp); if (msg.Encode(&buf)) { buf.sz = buf.cur - buf.base; buf.cur = buf.base; llarp_ev_udp_sendto(&self->m_Client, to, buf); } else llarp::LogWarn("failed to encode dns message when sending"); }); } void Proxy::HandlePktClient(llarp::Addr from, Buffer_t buf) { llarp_buffer_t pkt(buf); MessageHeader hdr; if (!hdr.Decode(&pkt)) { llarp::LogWarn("failed to parse dns header from ", from); return; } TX tx = {hdr.id, from}; auto itr = m_Forwarded.find(tx); if (itr == m_Forwarded.end()) return; const Addr requester = itr->second; auto self = shared_from_this(); Message msg(hdr); if (msg.Decode(&pkt)) { if (m_QueryHandler && m_QueryHandler->ShouldHookDNSMessage(msg)) { msg.hdr_id = itr->first.txid; if (!m_QueryHandler->HandleHookedDNSMessage( std::move(msg), std::bind(&Proxy::SendServerMessageTo, self, requester, std::placeholders::_1))) { llarp::LogWarn("failed to handle hooked dns"); } return; } } LogicCall(m_ServerLogic, [=]() { // forward reply to requester via server const llarp_buffer_t tmpbuf(buf); llarp_ev_udp_sendto(&self->m_Server, requester, tmpbuf); }); // remove pending m_Forwarded.erase(itr); } void Proxy::HandlePktServer(llarp::Addr from, Buffer_t buf) { MessageHeader hdr; llarp_buffer_t pkt(buf); if (!hdr.Decode(&pkt)) { llarp::LogWarn("failed to parse dns header from ", from); return; } TX tx = {hdr.id, from}; auto itr = m_Forwarded.find(tx); Message msg(hdr); if (!msg.Decode(&pkt)) { llarp::LogWarn("failed to parse dns message from ", from); return; } // we don't provide a DoH resolver because it requires verified TLS // TLS needs X509/ASN.1-DER and opting into the Root CA Cabal // thankfully mozilla added a backdoor that allows ISPs to turn it off // so we disable DoH for firefox using mozilla's ISP backdoor // see: https://github.com/loki-project/loki-network/issues/832 for (const auto& q : msg.questions) { // is this firefox looking for their backdoor record? if (q.IsName("use-application-dns.net")) { // yea it is, let's turn off DoH because god is dead. msg.AddNXReply(); // press F to pay respects SendServerMessageTo(from, std::move(msg)); return; } } auto self = shared_from_this(); if (m_QueryHandler && m_QueryHandler->ShouldHookDNSMessage(msg)) { if (!m_QueryHandler->HandleHookedDNSMessage( std::move(msg), std::bind(&Proxy::SendServerMessageTo, self, from, std::placeholders::_1))) { llarp::LogWarn("failed to handle hooked dns"); } } else if (m_Resolvers.size() == 0) { // no upstream resolvers // let's serv fail it msg.AddServFail(); SendServerMessageTo(from, std::move(msg)); } else if (itr == m_Forwarded.end()) { // new forwarded query tx.from = PickRandomResolver(); m_Forwarded[tx] = from; LogicCall(m_ClientLogic, [=] { // do query const llarp_buffer_t tmpbuf(buf); llarp_ev_udp_sendto(&self->m_Client, tx.from, tmpbuf); }); } else { // send the query again because it's probably FEC from the requester const auto resolver = itr->first.from; LogicCall(m_ClientLogic, [=] { // send it const llarp_buffer_t tmpbuf(buf); llarp_ev_udp_sendto(&self->m_Client, resolver, tmpbuf); }); } } } // namespace dns } // namespace llarp