mirror of
https://github.com/RGBCube/serenity
synced 2025-07-27 06:17:35 +00:00
LookupServer: Convert to Core::Stream::UDPSocket
This commit is contained in:
parent
17d3592cab
commit
4ca0669d1e
5 changed files with 43 additions and 45 deletions
|
@ -30,9 +30,12 @@ void ClientConnection::die()
|
||||||
|
|
||||||
Messages::LookupServer::LookupNameResponse ClientConnection::lookup_name(String const& name)
|
Messages::LookupServer::LookupNameResponse ClientConnection::lookup_name(String const& name)
|
||||||
{
|
{
|
||||||
auto answers = LookupServer::the().lookup(name, DNSRecordType::A);
|
auto maybe_answers = LookupServer::the().lookup(name, DNSRecordType::A);
|
||||||
if (answers.is_empty())
|
if (maybe_answers.is_error()) {
|
||||||
return { 1, Vector<String>() };
|
dbgln("LookupServer: Failed to lookup PTR record: {}", maybe_answers.error());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto answers = maybe_answers.release_value();
|
||||||
Vector<String> addresses;
|
Vector<String> addresses;
|
||||||
for (auto& answer : answers) {
|
for (auto& answer : answers) {
|
||||||
addresses.append(answer.record_data());
|
addresses.append(answer.record_data());
|
||||||
|
@ -50,7 +53,13 @@ Messages::LookupServer::LookupAddressResponse ClientConnection::lookup_address(S
|
||||||
ip_address[2],
|
ip_address[2],
|
||||||
ip_address[1],
|
ip_address[1],
|
||||||
ip_address[0]);
|
ip_address[0]);
|
||||||
auto answers = LookupServer::the().lookup(name, DNSRecordType::PTR);
|
|
||||||
|
auto maybe_answers = LookupServer::the().lookup(name, DNSRecordType::PTR);
|
||||||
|
if (maybe_answers.is_error()) {
|
||||||
|
dbgln("LookupServer: Failed to lookup PTR record: {}", maybe_answers.error());
|
||||||
|
}
|
||||||
|
|
||||||
|
auto answers = maybe_answers.release_value();
|
||||||
if (answers.is_empty())
|
if (answers.is_empty())
|
||||||
return { 1, String() };
|
return { 1, String() };
|
||||||
return { 0, answers[0].record_data() };
|
return { 0, answers[0].record_data() };
|
||||||
|
|
|
@ -16,24 +16,27 @@ DNSServer::DNSServer(Object* parent)
|
||||||
{
|
{
|
||||||
bind(IPv4Address(), 53);
|
bind(IPv4Address(), 53);
|
||||||
on_ready_to_receive = [this]() {
|
on_ready_to_receive = [this]() {
|
||||||
handle_client();
|
auto result = handle_client();
|
||||||
|
if (result.is_error()) {
|
||||||
|
dbgln("DNSServer: Failed to handle client: {}", result.error());
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
void DNSServer::handle_client()
|
ErrorOr<void> DNSServer::handle_client()
|
||||||
{
|
{
|
||||||
sockaddr_in client_address;
|
sockaddr_in client_address;
|
||||||
auto buffer = receive(1024, client_address);
|
auto buffer = receive(1024, client_address);
|
||||||
auto optional_request = DNSPacket::from_raw_packet(buffer.data(), buffer.size());
|
auto optional_request = DNSPacket::from_raw_packet(buffer.data(), buffer.size());
|
||||||
if (!optional_request.has_value()) {
|
if (!optional_request.has_value()) {
|
||||||
dbgln("Got an invalid DNS packet");
|
dbgln("Got an invalid DNS packet");
|
||||||
return;
|
return {};
|
||||||
}
|
}
|
||||||
auto& request = optional_request.value();
|
auto& request = optional_request.value();
|
||||||
|
|
||||||
if (!request.is_query()) {
|
if (!request.is_query()) {
|
||||||
dbgln("It's not a request");
|
dbgln("It's not a request");
|
||||||
return;
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
LookupServer& lookup_server = LookupServer::the();
|
LookupServer& lookup_server = LookupServer::the();
|
||||||
|
@ -46,7 +49,7 @@ void DNSServer::handle_client()
|
||||||
if (question.class_code() != DNSRecordClass::IN)
|
if (question.class_code() != DNSRecordClass::IN)
|
||||||
continue;
|
continue;
|
||||||
response.add_question(question);
|
response.add_question(question);
|
||||||
auto answers = lookup_server.lookup(question.name(), question.record_type());
|
auto answers = TRY(lookup_server.lookup(question.name(), question.record_type()));
|
||||||
for (auto& answer : answers) {
|
for (auto& answer : answers) {
|
||||||
response.add_answer(answer);
|
response.add_answer(answer);
|
||||||
}
|
}
|
||||||
|
@ -59,8 +62,8 @@ void DNSServer::handle_client()
|
||||||
|
|
||||||
buffer = response.to_byte_buffer();
|
buffer = response.to_byte_buffer();
|
||||||
|
|
||||||
// FIXME: We should be handling errors here.
|
TRY(send(buffer, client_address));
|
||||||
[[maybe_unused]] auto result = send(buffer, client_address);
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,7 +16,7 @@ class DNSServer : public Core::UDPServer {
|
||||||
private:
|
private:
|
||||||
explicit DNSServer(Object* parent = nullptr);
|
explicit DNSServer(Object* parent = nullptr);
|
||||||
|
|
||||||
void handle_client();
|
ErrorOr<void> handle_client();
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,7 +15,7 @@
|
||||||
#include <LibCore/ConfigFile.h>
|
#include <LibCore/ConfigFile.h>
|
||||||
#include <LibCore/File.h>
|
#include <LibCore/File.h>
|
||||||
#include <LibCore/LocalServer.h>
|
#include <LibCore/LocalServer.h>
|
||||||
#include <LibCore/UDPSocket.h>
|
#include <LibCore/Stream.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
@ -131,7 +131,7 @@ static String get_hostname()
|
||||||
return buffer;
|
return buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
Vector<DNSAnswer> LookupServer::lookup(const DNSName& name, DNSRecordType record_type)
|
ErrorOr<Vector<DNSAnswer>> LookupServer::lookup(const DNSName& name, DNSRecordType record_type)
|
||||||
{
|
{
|
||||||
dbgln_if(LOOKUPSERVER_DEBUG, "Got request for '{}'", name.as_string());
|
dbgln_if(LOOKUPSERVER_DEBUG, "Got request for '{}'", name.as_string());
|
||||||
|
|
||||||
|
@ -196,7 +196,7 @@ Vector<DNSAnswer> LookupServer::lookup(const DNSName& name, DNSRecordType record
|
||||||
int retries = 3;
|
int retries = 3;
|
||||||
Vector<DNSAnswer> upstream_answers;
|
Vector<DNSAnswer> upstream_answers;
|
||||||
do {
|
do {
|
||||||
upstream_answers = lookup(name, nameserver, did_get_response, record_type);
|
upstream_answers = TRY(lookup(name, nameserver, did_get_response, record_type));
|
||||||
if (did_get_response)
|
if (did_get_response)
|
||||||
break;
|
break;
|
||||||
} while (--retries);
|
} while (--retries);
|
||||||
|
@ -215,13 +215,13 @@ Vector<DNSAnswer> LookupServer::lookup(const DNSName& name, DNSRecordType record
|
||||||
// Sixth, fail.
|
// Sixth, fail.
|
||||||
if (answers.is_empty()) {
|
if (answers.is_empty()) {
|
||||||
dbgln("Tried all nameservers but never got a response :(");
|
dbgln("Tried all nameservers but never got a response :(");
|
||||||
return {};
|
return Vector<DNSAnswer> {};
|
||||||
}
|
}
|
||||||
|
|
||||||
return answers;
|
return answers;
|
||||||
}
|
}
|
||||||
|
|
||||||
Vector<DNSAnswer> LookupServer::lookup(const DNSName& name, const String& nameserver, bool& did_get_response, DNSRecordType record_type, ShouldRandomizeCase should_randomize_case)
|
ErrorOr<Vector<DNSAnswer>> LookupServer::lookup(const DNSName& name, const String& nameserver, bool& did_get_response, DNSRecordType record_type, ShouldRandomizeCase should_randomize_case)
|
||||||
{
|
{
|
||||||
DNSPacket request;
|
DNSPacket request;
|
||||||
request.set_is_query();
|
request.set_is_query();
|
||||||
|
@ -233,41 +233,27 @@ Vector<DNSAnswer> LookupServer::lookup(const DNSName& name, const String& namese
|
||||||
|
|
||||||
auto buffer = request.to_byte_buffer();
|
auto buffer = request.to_byte_buffer();
|
||||||
|
|
||||||
auto udp_socket = Core::UDPSocket::construct();
|
auto udp_socket = TRY(Core::Stream::UDPSocket::connect(nameserver, 53, Time::from_seconds(1)));
|
||||||
udp_socket->set_blocking(true);
|
TRY(udp_socket->set_blocking(true));
|
||||||
|
|
||||||
struct timeval timeout {
|
TRY(udp_socket->write(buffer));
|
||||||
1, 0
|
|
||||||
};
|
|
||||||
|
|
||||||
int rc = setsockopt(udp_socket->fd(), SOL_SOCKET, SO_RCVTIMEO, &timeout, sizeof(timeout));
|
|
||||||
if (rc < 0) {
|
|
||||||
perror("setsockopt(SOL_SOCKET, SO_RCVTIMEO)");
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!udp_socket->connect(nameserver, 53))
|
|
||||||
return {};
|
|
||||||
|
|
||||||
if (!udp_socket->write(buffer))
|
|
||||||
return {};
|
|
||||||
|
|
||||||
u8 response_buffer[4096];
|
u8 response_buffer[4096];
|
||||||
int nrecv = udp_socket->read(response_buffer, sizeof(response_buffer));
|
int nrecv = TRY(udp_socket->read({ response_buffer, sizeof(response_buffer) }));
|
||||||
if (nrecv == 0)
|
if (udp_socket->is_eof())
|
||||||
return {};
|
return Vector<DNSAnswer> {};
|
||||||
|
|
||||||
did_get_response = true;
|
did_get_response = true;
|
||||||
|
|
||||||
auto o_response = DNSPacket::from_raw_packet(response_buffer, nrecv);
|
auto o_response = DNSPacket::from_raw_packet(response_buffer, nrecv);
|
||||||
if (!o_response.has_value())
|
if (!o_response.has_value())
|
||||||
return {};
|
return Vector<DNSAnswer> {};
|
||||||
|
|
||||||
auto& response = o_response.value();
|
auto& response = o_response.value();
|
||||||
|
|
||||||
if (response.id() != request.id()) {
|
if (response.id() != request.id()) {
|
||||||
dbgln("LookupServer: ID mismatch ({} vs {}) :(", response.id(), request.id());
|
dbgln("LookupServer: ID mismatch ({} vs {}) :(", response.id(), request.id());
|
||||||
return {};
|
return Vector<DNSAnswer> {};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (response.code() == DNSPacket::Code::REFUSED) {
|
if (response.code() == DNSPacket::Code::REFUSED) {
|
||||||
|
@ -275,12 +261,12 @@ Vector<DNSAnswer> LookupServer::lookup(const DNSName& name, const String& namese
|
||||||
// Retry with 0x20 case randomization turned off.
|
// Retry with 0x20 case randomization turned off.
|
||||||
return lookup(name, nameserver, did_get_response, record_type, ShouldRandomizeCase::No);
|
return lookup(name, nameserver, did_get_response, record_type, ShouldRandomizeCase::No);
|
||||||
}
|
}
|
||||||
return {};
|
return Vector<DNSAnswer> {};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (response.question_count() != request.question_count()) {
|
if (response.question_count() != request.question_count()) {
|
||||||
dbgln("LookupServer: Question count ({} vs {}) :(", response.question_count(), request.question_count());
|
dbgln("LookupServer: Question count ({} vs {}) :(", response.question_count(), request.question_count());
|
||||||
return {};
|
return Vector<DNSAnswer> {};
|
||||||
}
|
}
|
||||||
|
|
||||||
// Verify the questions in our request and in their response match exactly, including case.
|
// Verify the questions in our request and in their response match exactly, including case.
|
||||||
|
@ -294,13 +280,13 @@ Vector<DNSAnswer> LookupServer::lookup(const DNSName& name, const String& namese
|
||||||
dbgln("Request and response questions do not match");
|
dbgln("Request and response questions do not match");
|
||||||
dbgln(" Request: name=_{}_, type={}, class={}", request_question.name().as_string(), response_question.record_type(), response_question.class_code());
|
dbgln(" Request: name=_{}_, type={}, class={}", request_question.name().as_string(), response_question.record_type(), response_question.class_code());
|
||||||
dbgln(" Response: name=_{}_, type={}, class={}", response_question.name().as_string(), response_question.record_type(), response_question.class_code());
|
dbgln(" Response: name=_{}_, type={}, class={}", response_question.name().as_string(), response_question.record_type(), response_question.class_code());
|
||||||
return {};
|
return Vector<DNSAnswer> {};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (response.answer_count() < 1) {
|
if (response.answer_count() < 1) {
|
||||||
dbgln("LookupServer: No answers :(");
|
dbgln("LookupServer: No answers :(");
|
||||||
return {};
|
return Vector<DNSAnswer> {};
|
||||||
}
|
}
|
||||||
|
|
||||||
Vector<DNSAnswer, 8> answers;
|
Vector<DNSAnswer, 8> answers;
|
||||||
|
|
|
@ -24,7 +24,7 @@ class LookupServer final : public Core::Object {
|
||||||
|
|
||||||
public:
|
public:
|
||||||
static LookupServer& the();
|
static LookupServer& the();
|
||||||
Vector<DNSAnswer> lookup(const DNSName& name, DNSRecordType record_type);
|
ErrorOr<Vector<DNSAnswer>> lookup(const DNSName& name, DNSRecordType record_type);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
LookupServer();
|
LookupServer();
|
||||||
|
@ -32,7 +32,7 @@ private:
|
||||||
void load_etc_hosts();
|
void load_etc_hosts();
|
||||||
void put_in_cache(const DNSAnswer&);
|
void put_in_cache(const DNSAnswer&);
|
||||||
|
|
||||||
Vector<DNSAnswer> lookup(const DNSName& hostname, const String& nameserver, bool& did_get_response, DNSRecordType record_type, ShouldRandomizeCase = ShouldRandomizeCase::Yes);
|
ErrorOr<Vector<DNSAnswer>> lookup(const DNSName& hostname, const String& nameserver, bool& did_get_response, DNSRecordType record_type, ShouldRandomizeCase = ShouldRandomizeCase::Yes);
|
||||||
|
|
||||||
OwnPtr<IPC::MultiServer<ClientConnection>> m_server;
|
OwnPtr<IPC::MultiServer<ClientConnection>> m_server;
|
||||||
RefPtr<DNSServer> m_dns_server;
|
RefPtr<DNSServer> m_dns_server;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue