mirror of
https://github.com/matrix-construct/construct
synced 2025-01-13 08:23:56 +01:00
modules: Split net_dns_cache from net_dns unit; same shared object.
This commit is contained in:
parent
7934756858
commit
fb3c6b47a4
4 changed files with 721 additions and 700 deletions
|
@ -59,7 +59,7 @@ moduledir = @moduledir@
|
||||||
|
|
||||||
conf_la_SOURCES = conf.cc
|
conf_la_SOURCES = conf.cc
|
||||||
stats_la_SOURCES = stats.cc
|
stats_la_SOURCES = stats.cc
|
||||||
net_dns_la_SOURCES = net_dns.cc net_dns_resolver.cc
|
net_dns_la_SOURCES = net_dns.cc net_dns_cache.cc net_dns_resolver.cc
|
||||||
net_dns_la_CPPFLAGS = $(AM_CPPFLAGS) @BOOST_CPPFLAGS@
|
net_dns_la_CPPFLAGS = $(AM_CPPFLAGS) @BOOST_CPPFLAGS@
|
||||||
net_dns_la_LDFLAGS = $(AM_LDFLAGS) @BOOST_LDFLAGS@
|
net_dns_la_LDFLAGS = $(AM_LDFLAGS) @BOOST_LDFLAGS@
|
||||||
net_dns_la_LIBADD = $(AM_LIBS) @BOOST_LIBS@
|
net_dns_la_LIBADD = $(AM_LIBS) @BOOST_LIBS@
|
||||||
|
|
|
@ -21,24 +21,14 @@ IRCD_MODULE
|
||||||
void
|
void
|
||||||
ircd::net::dns::init()
|
ircd::net::dns::init()
|
||||||
{
|
{
|
||||||
|
cache::init();
|
||||||
resolver_init(handle_resolved);
|
resolver_init(handle_resolved);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
ircd::net::dns::fini()
|
ircd::net::dns::fini()
|
||||||
{
|
{
|
||||||
if(!cache::waiting.empty())
|
cache::fini();
|
||||||
log::warning
|
|
||||||
{
|
|
||||||
log, "Waiting for %zu unfinished cache operations.",
|
|
||||||
cache::waiting.size(),
|
|
||||||
};
|
|
||||||
|
|
||||||
cache::dock.wait([]
|
|
||||||
{
|
|
||||||
return cache::waiting.empty();
|
|
||||||
});
|
|
||||||
|
|
||||||
resolver_fini();
|
resolver_fini();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -373,690 +363,3 @@ ircd::net::dns::new_record(mutable_buffer &buf,
|
||||||
consume(buf, sizeof(type));
|
consume(buf, sizeof(type));
|
||||||
return new (pos) type(answer);
|
return new (pos) type(answer);
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
|
||||||
// cache
|
|
||||||
//
|
|
||||||
|
|
||||||
decltype(ircd::net::dns::cache::error_ttl)
|
|
||||||
ircd::net::dns::cache::error_ttl
|
|
||||||
{
|
|
||||||
{ "name", "ircd.net.dns.cache.error_ttl" },
|
|
||||||
{ "default", 1200L },
|
|
||||||
};
|
|
||||||
|
|
||||||
decltype(ircd::net::dns::cache::nxdomain_ttl)
|
|
||||||
ircd::net::dns::cache::nxdomain_ttl
|
|
||||||
{
|
|
||||||
{ "name", "ircd.net.dns.cache.nxdomain_ttl" },
|
|
||||||
{ "default", 86400L },
|
|
||||||
};
|
|
||||||
|
|
||||||
decltype(ircd::net::dns::cache::min_ttl)
|
|
||||||
ircd::net::dns::cache::min_ttl
|
|
||||||
{
|
|
||||||
{ "name", "ircd.net.dns.cache.min_ttl" },
|
|
||||||
{ "default", 28800L },
|
|
||||||
};
|
|
||||||
|
|
||||||
decltype(ircd::net::dns::cache::room_id)
|
|
||||||
ircd::net::dns::cache::room_id
|
|
||||||
{
|
|
||||||
"dns", my_host()
|
|
||||||
};
|
|
||||||
|
|
||||||
decltype(ircd::net::dns::cache::hook)
|
|
||||||
ircd::net::dns::cache::hook
|
|
||||||
{
|
|
||||||
handle,
|
|
||||||
{
|
|
||||||
{ "_site", "vm.effect" },
|
|
||||||
{ "room_id", string_view{room_id} },
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
decltype(ircd::net::dns::cache::waiting)
|
|
||||||
ircd::net::dns::cache::waiting;
|
|
||||||
|
|
||||||
decltype(ircd::net::dns::cache::mutex)
|
|
||||||
ircd::net::dns::cache::mutex;
|
|
||||||
|
|
||||||
decltype(ircd::net::dns::cache::dock)
|
|
||||||
ircd::net::dns::cache::dock;
|
|
||||||
|
|
||||||
bool
|
|
||||||
IRCD_MODULE_EXPORT
|
|
||||||
ircd::net::dns::cache::put(const hostport &hp,
|
|
||||||
const opts &opts,
|
|
||||||
const uint &code,
|
|
||||||
const string_view &msg)
|
|
||||||
{
|
|
||||||
char type_buf[64];
|
|
||||||
const string_view type
|
|
||||||
{
|
|
||||||
make_type(type_buf, opts.qtype)
|
|
||||||
};
|
|
||||||
|
|
||||||
char state_key_buf[rfc1035::NAME_BUFSIZE * 2];
|
|
||||||
const string_view &state_key
|
|
||||||
{
|
|
||||||
opts.qtype == 33?
|
|
||||||
make_SRV_key(state_key_buf, host(hp), opts):
|
|
||||||
host(hp)
|
|
||||||
};
|
|
||||||
|
|
||||||
return put(type, state_key, code, msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
|
||||||
IRCD_MODULE_EXPORT
|
|
||||||
ircd::net::dns::cache::put(const hostport &hp,
|
|
||||||
const opts &opts,
|
|
||||||
const records &rrs)
|
|
||||||
{
|
|
||||||
const auto &type_code
|
|
||||||
{
|
|
||||||
!rrs.empty()? rrs.at(0)->type : opts.qtype
|
|
||||||
};
|
|
||||||
|
|
||||||
char type_buf[48];
|
|
||||||
const string_view type
|
|
||||||
{
|
|
||||||
make_type(type_buf, type_code)
|
|
||||||
};
|
|
||||||
|
|
||||||
char state_key_buf[rfc1035::NAME_BUFSIZE * 2];
|
|
||||||
const string_view &state_key
|
|
||||||
{
|
|
||||||
opts.qtype == 33?
|
|
||||||
make_SRV_key(state_key_buf, host(hp), opts):
|
|
||||||
host(hp)
|
|
||||||
};
|
|
||||||
|
|
||||||
return put(type, state_key, rrs);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
|
||||||
ircd::net::dns::cache::put(const string_view &type,
|
|
||||||
const string_view &state_key,
|
|
||||||
const uint &code,
|
|
||||||
const string_view &msg)
|
|
||||||
try
|
|
||||||
{
|
|
||||||
char content_buf[1024];
|
|
||||||
json::stack out{content_buf};
|
|
||||||
json::stack::object content{out};
|
|
||||||
json::stack::array array
|
|
||||||
{
|
|
||||||
content, ""
|
|
||||||
};
|
|
||||||
|
|
||||||
json::stack::object rr0
|
|
||||||
{
|
|
||||||
array
|
|
||||||
};
|
|
||||||
|
|
||||||
json::stack::member
|
|
||||||
{
|
|
||||||
rr0, "errcode", lex_cast(code)
|
|
||||||
};
|
|
||||||
|
|
||||||
json::stack::member
|
|
||||||
{
|
|
||||||
rr0, "error", msg
|
|
||||||
};
|
|
||||||
|
|
||||||
json::stack::member
|
|
||||||
{
|
|
||||||
rr0, "ttl", json::value
|
|
||||||
{
|
|
||||||
code == 3?
|
|
||||||
long(seconds(nxdomain_ttl).count()):
|
|
||||||
long(seconds(error_ttl).count())
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
rr0.~object();
|
|
||||||
array.~array();
|
|
||||||
content.~object();
|
|
||||||
send(room_id, m::me, type, state_key, json::object(out.completed()));
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
catch(const http::error &e)
|
|
||||||
{
|
|
||||||
const ctx::exception_handler eh;
|
|
||||||
log::error
|
|
||||||
{
|
|
||||||
log, "cache put (%s, %s) code:%u (%s) :%s %s",
|
|
||||||
type,
|
|
||||||
state_key,
|
|
||||||
code,
|
|
||||||
msg,
|
|
||||||
e.what(),
|
|
||||||
e.content,
|
|
||||||
};
|
|
||||||
|
|
||||||
const json::value error_value
|
|
||||||
{
|
|
||||||
json::object{e.content}
|
|
||||||
};
|
|
||||||
|
|
||||||
const json::value error_records{&error_value, 1};
|
|
||||||
const json::strung error{error_records};
|
|
||||||
call_waiters(type, state_key, error);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
catch(const std::exception &e)
|
|
||||||
{
|
|
||||||
const ctx::exception_handler eh;
|
|
||||||
log::error
|
|
||||||
{
|
|
||||||
log, "cache put (%s, %s) code:%u (%s) :%s",
|
|
||||||
type,
|
|
||||||
state_key,
|
|
||||||
code,
|
|
||||||
msg,
|
|
||||||
e.what()
|
|
||||||
};
|
|
||||||
|
|
||||||
const json::members error_object
|
|
||||||
{
|
|
||||||
{ "error", e.what() },
|
|
||||||
};
|
|
||||||
|
|
||||||
const json::value error_value{error_object};
|
|
||||||
const json::value error_records{&error_value, 1};
|
|
||||||
const json::strung error{error_records};
|
|
||||||
call_waiters(type, state_key, error);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
|
||||||
ircd::net::dns::cache::put(const string_view &type,
|
|
||||||
const string_view &state_key,
|
|
||||||
const records &rrs)
|
|
||||||
try
|
|
||||||
{
|
|
||||||
const unique_buffer<mutable_buffer> buf
|
|
||||||
{
|
|
||||||
8_KiB
|
|
||||||
};
|
|
||||||
|
|
||||||
json::stack out{buf};
|
|
||||||
json::stack::object content{out};
|
|
||||||
json::stack::array array
|
|
||||||
{
|
|
||||||
content, ""
|
|
||||||
};
|
|
||||||
|
|
||||||
if(rrs.empty())
|
|
||||||
{
|
|
||||||
// Add one object to the array with nothing except a ttl indicating no
|
|
||||||
// records (and no error) so we can cache that for the ttl. We use the
|
|
||||||
// nxdomain ttl for this value.
|
|
||||||
json::stack::object rr0{array};
|
|
||||||
json::stack::member
|
|
||||||
{
|
|
||||||
rr0, "ttl", json::value
|
|
||||||
{
|
|
||||||
long(seconds(nxdomain_ttl).count())
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
else for(const auto &record : rrs)
|
|
||||||
{
|
|
||||||
switch(record->type)
|
|
||||||
{
|
|
||||||
case 1: // A
|
|
||||||
{
|
|
||||||
json::stack::object object{array};
|
|
||||||
dynamic_cast<const rfc1035::record::A *>(record)->append(object);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
case 5: // CNAME
|
|
||||||
{
|
|
||||||
json::stack::object object{array};
|
|
||||||
dynamic_cast<const rfc1035::record::CNAME *>(record)->append(object);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
case 28: // AAAA
|
|
||||||
{
|
|
||||||
json::stack::object object{array};
|
|
||||||
dynamic_cast<const rfc1035::record::AAAA *>(record)->append(object);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
case 33: // SRV
|
|
||||||
{
|
|
||||||
json::stack::object object{array};
|
|
||||||
dynamic_cast<const rfc1035::record::SRV *>(record)->append(object);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// When the array has a zero value count we didn't know how to cache
|
|
||||||
// any of these records; don't send anything to the cache room.
|
|
||||||
if(!array.vc)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
array.~array();
|
|
||||||
content.~object();
|
|
||||||
send(room_id, m::me, type, state_key, json::object{out.completed()});
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
catch(const http::error &e)
|
|
||||||
{
|
|
||||||
const ctx::exception_handler eh;
|
|
||||||
log::error
|
|
||||||
{
|
|
||||||
log, "cache put (%s, %s) rrs:%zu :%s %s",
|
|
||||||
type,
|
|
||||||
state_key,
|
|
||||||
rrs.size(),
|
|
||||||
e.what(),
|
|
||||||
e.content,
|
|
||||||
};
|
|
||||||
|
|
||||||
const json::value error_value
|
|
||||||
{
|
|
||||||
json::object{e.content}
|
|
||||||
};
|
|
||||||
|
|
||||||
const json::value error_records{&error_value, 1};
|
|
||||||
const json::strung error{error_records};
|
|
||||||
call_waiters(type, state_key, error);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
catch(const std::exception &e)
|
|
||||||
{
|
|
||||||
const ctx::exception_handler eh;
|
|
||||||
log::error
|
|
||||||
{
|
|
||||||
log, "cache put (%s, %s) rrs:%zu :%s",
|
|
||||||
type,
|
|
||||||
state_key,
|
|
||||||
rrs.size(),
|
|
||||||
e.what(),
|
|
||||||
};
|
|
||||||
|
|
||||||
const json::members error_object
|
|
||||||
{
|
|
||||||
{ "error", e.what() },
|
|
||||||
};
|
|
||||||
|
|
||||||
const json::value error_value{error_object};
|
|
||||||
const json::value error_records{&error_value, 1};
|
|
||||||
const json::strung error{error_records};
|
|
||||||
call_waiters(type, state_key, error);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
|
||||||
IRCD_MODULE_EXPORT
|
|
||||||
ircd::net::dns::cache::get(const hostport &hp,
|
|
||||||
const opts &opts,
|
|
||||||
const callback &closure)
|
|
||||||
{
|
|
||||||
char type_buf[48];
|
|
||||||
const string_view type
|
|
||||||
{
|
|
||||||
make_type(type_buf, opts.qtype)
|
|
||||||
};
|
|
||||||
|
|
||||||
char state_key_buf[rfc1035::NAME_BUFSIZE * 2];
|
|
||||||
const string_view &state_key
|
|
||||||
{
|
|
||||||
opts.qtype == 33?
|
|
||||||
make_SRV_key(state_key_buf, host(hp), opts):
|
|
||||||
host(hp)
|
|
||||||
};
|
|
||||||
|
|
||||||
const m::room::state state
|
|
||||||
{
|
|
||||||
room_id
|
|
||||||
};
|
|
||||||
|
|
||||||
const m::event::idx &event_idx
|
|
||||||
{
|
|
||||||
state.get(std::nothrow, type, state_key)
|
|
||||||
};
|
|
||||||
|
|
||||||
if(!event_idx)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
time_t origin_server_ts;
|
|
||||||
if(!m::get<time_t>(event_idx, "origin_server_ts", origin_server_ts))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
bool ret{false};
|
|
||||||
const time_t ts{origin_server_ts / 1000L};
|
|
||||||
m::get(std::nothrow, event_idx, "content", [&hp, &closure, &ret, &ts]
|
|
||||||
(const json::object &content)
|
|
||||||
{
|
|
||||||
const json::array &rrs
|
|
||||||
{
|
|
||||||
content.get("")
|
|
||||||
};
|
|
||||||
|
|
||||||
// If all records are expired then skip; otherwise since this closure
|
|
||||||
// expects a single array we reveal both expired and valid records.
|
|
||||||
ret = !std::all_of(begin(rrs), end(rrs), [&ts]
|
|
||||||
(const json::object &rr)
|
|
||||||
{
|
|
||||||
return expired(rr, ts);
|
|
||||||
});
|
|
||||||
|
|
||||||
if(ret && closure)
|
|
||||||
closure(hp, rrs);
|
|
||||||
});
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
|
||||||
IRCD_MODULE_EXPORT
|
|
||||||
ircd::net::dns::cache::for_each(const hostport &hp,
|
|
||||||
const opts &opts,
|
|
||||||
const closure &closure)
|
|
||||||
{
|
|
||||||
char type_buf[48];
|
|
||||||
const string_view type
|
|
||||||
{
|
|
||||||
make_type(type_buf, opts.qtype)
|
|
||||||
};
|
|
||||||
|
|
||||||
char state_key_buf[rfc1035::NAME_BUFSIZE * 2];
|
|
||||||
const string_view &state_key
|
|
||||||
{
|
|
||||||
opts.qtype == 33?
|
|
||||||
make_SRV_key(state_key_buf, host(hp), opts):
|
|
||||||
host(hp)
|
|
||||||
};
|
|
||||||
|
|
||||||
const m::room::state state
|
|
||||||
{
|
|
||||||
room_id
|
|
||||||
};
|
|
||||||
|
|
||||||
const m::event::idx &event_idx
|
|
||||||
{
|
|
||||||
state.get(std::nothrow, type, state_key)
|
|
||||||
};
|
|
||||||
|
|
||||||
if(!event_idx)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
time_t origin_server_ts;
|
|
||||||
if(!m::get<time_t>(event_idx, "origin_server_ts", origin_server_ts))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
bool ret{true};
|
|
||||||
const time_t ts{origin_server_ts / 1000L};
|
|
||||||
m::get(std::nothrow, event_idx, "content", [&state_key, &closure, &ret, &ts]
|
|
||||||
(const json::object &content)
|
|
||||||
{
|
|
||||||
for(const json::object &rr : json::array(content.get("")))
|
|
||||||
{
|
|
||||||
if(expired(rr, ts))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if(!(ret = closure(state_key, rr)))
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
|
||||||
IRCD_MODULE_EXPORT
|
|
||||||
ircd::net::dns::cache::for_each(const string_view &type,
|
|
||||||
const closure &closure)
|
|
||||||
{
|
|
||||||
char type_buf[48];
|
|
||||||
const string_view full_type
|
|
||||||
{
|
|
||||||
make_type(type_buf, type)
|
|
||||||
};
|
|
||||||
|
|
||||||
const m::room::state state
|
|
||||||
{
|
|
||||||
room_id
|
|
||||||
};
|
|
||||||
|
|
||||||
return state.for_each(full_type, [&closure]
|
|
||||||
(const string_view &, const string_view &state_key, const m::event::idx &event_idx)
|
|
||||||
{
|
|
||||||
time_t origin_server_ts;
|
|
||||||
if(!m::get<time_t>(event_idx, "origin_server_ts", origin_server_ts))
|
|
||||||
return true;
|
|
||||||
|
|
||||||
bool ret{true};
|
|
||||||
const time_t ts{origin_server_ts / 1000L};
|
|
||||||
m::get(std::nothrow, event_idx, "content", [&state_key, &closure, &ret, &ts]
|
|
||||||
(const json::object &content)
|
|
||||||
{
|
|
||||||
for(const json::object &rr : json::array(content.get("")))
|
|
||||||
{
|
|
||||||
if(expired(rr, ts))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if(!(ret = closure(state_key, rr)))
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
ircd::net::dns::cache::handle(const m::event &event,
|
|
||||||
m::vm::eval &eval)
|
|
||||||
try
|
|
||||||
{
|
|
||||||
const string_view &type
|
|
||||||
{
|
|
||||||
json::get<"type"_>(event)
|
|
||||||
};
|
|
||||||
|
|
||||||
if(!startswith(type, "ircd.dns.rrs."))
|
|
||||||
return;
|
|
||||||
|
|
||||||
const string_view &state_key
|
|
||||||
{
|
|
||||||
json::get<"state_key"_>(event)
|
|
||||||
};
|
|
||||||
|
|
||||||
const json::array &rrs
|
|
||||||
{
|
|
||||||
json::get<"content"_>(event).get("")
|
|
||||||
};
|
|
||||||
|
|
||||||
call_waiters(type, state_key, rrs);
|
|
||||||
}
|
|
||||||
catch(const std::exception &e)
|
|
||||||
{
|
|
||||||
log::critical
|
|
||||||
{
|
|
||||||
log, "handle_cached() :%s", e.what()
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Note complications due to reentrance and other factors:
|
|
||||||
/// - This function is invoked from several different places on both the
|
|
||||||
/// timeout and receive contexts, in addition to any evaluator context.
|
|
||||||
/// - This function calls back to users making DNS queries, and they may
|
|
||||||
/// conduct another query in their callback frame -- mid-loop in this
|
|
||||||
/// function.
|
|
||||||
size_t
|
|
||||||
ircd::net::dns::cache::call_waiters(const string_view &type,
|
|
||||||
const string_view &state_key,
|
|
||||||
const json::array &rrs)
|
|
||||||
{
|
|
||||||
const ctx::uninterruptible::nothrow ui;
|
|
||||||
size_t ret(0), last; do
|
|
||||||
{
|
|
||||||
const std::lock_guard lock
|
|
||||||
{
|
|
||||||
mutex
|
|
||||||
};
|
|
||||||
|
|
||||||
auto it(begin(waiting));
|
|
||||||
for(last = ret; it != end(waiting); ++it)
|
|
||||||
if(call_waiter(type, state_key, rrs, *it))
|
|
||||||
{
|
|
||||||
it = waiting.erase(it);
|
|
||||||
++ret;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
while(last < ret);
|
|
||||||
|
|
||||||
if(ret)
|
|
||||||
dock.notify_all();
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
|
||||||
ircd::net::dns::cache::call_waiter(const string_view &type,
|
|
||||||
const string_view &state_key,
|
|
||||||
const json::array &rrs,
|
|
||||||
waiter &waiter)
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if(state_key != waiter.key)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if(lstrip(type, "ircd.dns.rrs.") != rfc1035::rqtype.at(waiter.opts.qtype))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
const hostport &target
|
|
||||||
{
|
|
||||||
waiter.opts.qtype == 33?
|
|
||||||
unmake_SRV_key(waiter.key):
|
|
||||||
waiter.key,
|
|
||||||
|
|
||||||
waiter.port
|
|
||||||
};
|
|
||||||
|
|
||||||
assert(waiter.callback);
|
|
||||||
waiter.callback(target, rrs);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
catch(const std::exception &e)
|
|
||||||
{
|
|
||||||
log::critical
|
|
||||||
{
|
|
||||||
log, "callback:%p %s,%s :%s",
|
|
||||||
(const void *)&waiter,
|
|
||||||
type,
|
|
||||||
state_key,
|
|
||||||
e.what(),
|
|
||||||
};
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// cache::waiter
|
|
||||||
//
|
|
||||||
|
|
||||||
bool
|
|
||||||
ircd::net::dns::cache::operator==(const waiter &a, const waiter &b)
|
|
||||||
{
|
|
||||||
return a.opts.qtype == b.opts.qtype &&
|
|
||||||
a.key && b.key &&
|
|
||||||
a.key == b.key;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
|
||||||
ircd::net::dns::cache::operator!=(const waiter &a, const waiter &b)
|
|
||||||
{
|
|
||||||
return !operator==(a, b);
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// cache::waiter::waiter
|
|
||||||
//
|
|
||||||
|
|
||||||
ircd::net::dns::cache::waiter::waiter(const hostport &hp,
|
|
||||||
const dns::opts &opts,
|
|
||||||
dns::callback &&callback)
|
|
||||||
:callback
|
|
||||||
{
|
|
||||||
std::move(callback)
|
|
||||||
}
|
|
||||||
,opts
|
|
||||||
{
|
|
||||||
opts
|
|
||||||
}
|
|
||||||
,port
|
|
||||||
{
|
|
||||||
net::port(hp)
|
|
||||||
}
|
|
||||||
,key
|
|
||||||
{
|
|
||||||
opts.qtype == 33?
|
|
||||||
make_SRV_key(keybuf, hp, opts):
|
|
||||||
strlcpy(keybuf, host(hp))
|
|
||||||
}
|
|
||||||
{
|
|
||||||
this->opts.srv = {};
|
|
||||||
this->opts.proto = {};
|
|
||||||
assert(this->opts.qtype);
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// cache room creation
|
|
||||||
//
|
|
||||||
|
|
||||||
namespace ircd::net::dns {
|
|
||||||
namespace [[gnu::visibility("hidden")]] cache
|
|
||||||
{
|
|
||||||
static void create_room();
|
|
||||||
extern m::hookfn<m::vm::eval &> create_room_hook;
|
|
||||||
}}
|
|
||||||
|
|
||||||
decltype(ircd::net::dns::cache::create_room_hook)
|
|
||||||
ircd::net::dns::cache::create_room_hook
|
|
||||||
{
|
|
||||||
{
|
|
||||||
{ "_site", "vm.effect" },
|
|
||||||
{ "room_id", "!ircd" },
|
|
||||||
{ "type", "m.room.create" },
|
|
||||||
},
|
|
||||||
[](const m::event &, m::vm::eval &)
|
|
||||||
{
|
|
||||||
create_room();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
void
|
|
||||||
ircd::net::dns::cache::create_room()
|
|
||||||
try
|
|
||||||
{
|
|
||||||
const m::room room
|
|
||||||
{
|
|
||||||
m::create(room_id, m::me, "internal")
|
|
||||||
};
|
|
||||||
|
|
||||||
log::debug
|
|
||||||
{
|
|
||||||
m::log, "Created '%s' for the DNS cache module.",
|
|
||||||
string_view{room.room_id}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
catch(const std::exception &e)
|
|
||||||
{
|
|
||||||
log::critical
|
|
||||||
{
|
|
||||||
m::log, "Creating the '%s' room failed :%s",
|
|
||||||
string_view{room_id},
|
|
||||||
e.what()
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
|
@ -47,6 +47,9 @@ namespace ircd::net::dns::cache
|
||||||
static bool put(const string_view &type, const string_view &state_key, const records &rrs);
|
static bool put(const string_view &type, const string_view &state_key, const records &rrs);
|
||||||
static bool put(const string_view &type, const string_view &state_key, const uint &code, const string_view &msg);
|
static bool put(const string_view &type, const string_view &state_key, const uint &code, const string_view &msg);
|
||||||
|
|
||||||
|
static void fini();
|
||||||
|
static void init();
|
||||||
|
|
||||||
extern conf::item<seconds> min_ttl IRCD_MODULE_EXPORT_DATA;
|
extern conf::item<seconds> min_ttl IRCD_MODULE_EXPORT_DATA;
|
||||||
extern conf::item<seconds> error_ttl IRCD_MODULE_EXPORT_DATA;
|
extern conf::item<seconds> error_ttl IRCD_MODULE_EXPORT_DATA;
|
||||||
extern conf::item<seconds> nxdomain_ttl IRCD_MODULE_EXPORT_DATA;
|
extern conf::item<seconds> nxdomain_ttl IRCD_MODULE_EXPORT_DATA;
|
||||||
|
|
715
modules/net_dns_cache.cc
Normal file
715
modules/net_dns_cache.cc
Normal file
|
@ -0,0 +1,715 @@
|
||||||
|
// Matrix Construct
|
||||||
|
//
|
||||||
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
||||||
|
// Copyright (C) 2016-2019 Jason Volk <jason@zemos.net>
|
||||||
|
//
|
||||||
|
// Permission to use, copy, modify, and/or distribute this software for any
|
||||||
|
// purpose with or without fee is hereby granted, provided that the above
|
||||||
|
// copyright notice and this permission notice is present in all copies. The
|
||||||
|
// full license for this software is available in the LICENSE file.
|
||||||
|
|
||||||
|
#include "net_dns.h"
|
||||||
|
|
||||||
|
decltype(ircd::net::dns::cache::error_ttl)
|
||||||
|
ircd::net::dns::cache::error_ttl
|
||||||
|
{
|
||||||
|
{ "name", "ircd.net.dns.cache.error_ttl" },
|
||||||
|
{ "default", 1200L },
|
||||||
|
};
|
||||||
|
|
||||||
|
decltype(ircd::net::dns::cache::nxdomain_ttl)
|
||||||
|
ircd::net::dns::cache::nxdomain_ttl
|
||||||
|
{
|
||||||
|
{ "name", "ircd.net.dns.cache.nxdomain_ttl" },
|
||||||
|
{ "default", 86400L },
|
||||||
|
};
|
||||||
|
|
||||||
|
decltype(ircd::net::dns::cache::min_ttl)
|
||||||
|
ircd::net::dns::cache::min_ttl
|
||||||
|
{
|
||||||
|
{ "name", "ircd.net.dns.cache.min_ttl" },
|
||||||
|
{ "default", 28800L },
|
||||||
|
};
|
||||||
|
|
||||||
|
decltype(ircd::net::dns::cache::room_id)
|
||||||
|
ircd::net::dns::cache::room_id
|
||||||
|
{
|
||||||
|
"dns", my_host()
|
||||||
|
};
|
||||||
|
|
||||||
|
decltype(ircd::net::dns::cache::hook)
|
||||||
|
ircd::net::dns::cache::hook
|
||||||
|
{
|
||||||
|
handle,
|
||||||
|
{
|
||||||
|
{ "_site", "vm.effect" },
|
||||||
|
{ "room_id", string_view{room_id} },
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
decltype(ircd::net::dns::cache::waiting)
|
||||||
|
ircd::net::dns::cache::waiting;
|
||||||
|
|
||||||
|
decltype(ircd::net::dns::cache::mutex)
|
||||||
|
ircd::net::dns::cache::mutex;
|
||||||
|
|
||||||
|
decltype(ircd::net::dns::cache::dock)
|
||||||
|
ircd::net::dns::cache::dock;
|
||||||
|
|
||||||
|
void
|
||||||
|
ircd::net::dns::cache::init()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ircd::net::dns::cache::fini()
|
||||||
|
{
|
||||||
|
if(!waiting.empty())
|
||||||
|
log::warning
|
||||||
|
{
|
||||||
|
log, "Waiting for %zu unfinished cache operations.",
|
||||||
|
waiting.size(),
|
||||||
|
};
|
||||||
|
|
||||||
|
dock.wait([]
|
||||||
|
{
|
||||||
|
return waiting.empty();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
IRCD_MODULE_EXPORT
|
||||||
|
ircd::net::dns::cache::put(const hostport &hp,
|
||||||
|
const opts &opts,
|
||||||
|
const uint &code,
|
||||||
|
const string_view &msg)
|
||||||
|
{
|
||||||
|
char type_buf[64];
|
||||||
|
const string_view type
|
||||||
|
{
|
||||||
|
make_type(type_buf, opts.qtype)
|
||||||
|
};
|
||||||
|
|
||||||
|
char state_key_buf[rfc1035::NAME_BUFSIZE * 2];
|
||||||
|
const string_view &state_key
|
||||||
|
{
|
||||||
|
opts.qtype == 33?
|
||||||
|
make_SRV_key(state_key_buf, host(hp), opts):
|
||||||
|
host(hp)
|
||||||
|
};
|
||||||
|
|
||||||
|
return put(type, state_key, code, msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
IRCD_MODULE_EXPORT
|
||||||
|
ircd::net::dns::cache::put(const hostport &hp,
|
||||||
|
const opts &opts,
|
||||||
|
const records &rrs)
|
||||||
|
{
|
||||||
|
const auto &type_code
|
||||||
|
{
|
||||||
|
!rrs.empty()? rrs.at(0)->type : opts.qtype
|
||||||
|
};
|
||||||
|
|
||||||
|
char type_buf[48];
|
||||||
|
const string_view type
|
||||||
|
{
|
||||||
|
make_type(type_buf, type_code)
|
||||||
|
};
|
||||||
|
|
||||||
|
char state_key_buf[rfc1035::NAME_BUFSIZE * 2];
|
||||||
|
const string_view &state_key
|
||||||
|
{
|
||||||
|
opts.qtype == 33?
|
||||||
|
make_SRV_key(state_key_buf, host(hp), opts):
|
||||||
|
host(hp)
|
||||||
|
};
|
||||||
|
|
||||||
|
return put(type, state_key, rrs);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
ircd::net::dns::cache::put(const string_view &type,
|
||||||
|
const string_view &state_key,
|
||||||
|
const uint &code,
|
||||||
|
const string_view &msg)
|
||||||
|
try
|
||||||
|
{
|
||||||
|
char content_buf[1024];
|
||||||
|
json::stack out{content_buf};
|
||||||
|
json::stack::object content{out};
|
||||||
|
json::stack::array array
|
||||||
|
{
|
||||||
|
content, ""
|
||||||
|
};
|
||||||
|
|
||||||
|
json::stack::object rr0
|
||||||
|
{
|
||||||
|
array
|
||||||
|
};
|
||||||
|
|
||||||
|
json::stack::member
|
||||||
|
{
|
||||||
|
rr0, "errcode", lex_cast(code)
|
||||||
|
};
|
||||||
|
|
||||||
|
json::stack::member
|
||||||
|
{
|
||||||
|
rr0, "error", msg
|
||||||
|
};
|
||||||
|
|
||||||
|
json::stack::member
|
||||||
|
{
|
||||||
|
rr0, "ttl", json::value
|
||||||
|
{
|
||||||
|
code == 3?
|
||||||
|
long(seconds(nxdomain_ttl).count()):
|
||||||
|
long(seconds(error_ttl).count())
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
rr0.~object();
|
||||||
|
array.~array();
|
||||||
|
content.~object();
|
||||||
|
send(room_id, m::me, type, state_key, json::object(out.completed()));
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
catch(const http::error &e)
|
||||||
|
{
|
||||||
|
const ctx::exception_handler eh;
|
||||||
|
log::error
|
||||||
|
{
|
||||||
|
log, "cache put (%s, %s) code:%u (%s) :%s %s",
|
||||||
|
type,
|
||||||
|
state_key,
|
||||||
|
code,
|
||||||
|
msg,
|
||||||
|
e.what(),
|
||||||
|
e.content,
|
||||||
|
};
|
||||||
|
|
||||||
|
const json::value error_value
|
||||||
|
{
|
||||||
|
json::object{e.content}
|
||||||
|
};
|
||||||
|
|
||||||
|
const json::value error_records{&error_value, 1};
|
||||||
|
const json::strung error{error_records};
|
||||||
|
call_waiters(type, state_key, error);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
catch(const std::exception &e)
|
||||||
|
{
|
||||||
|
const ctx::exception_handler eh;
|
||||||
|
log::error
|
||||||
|
{
|
||||||
|
log, "cache put (%s, %s) code:%u (%s) :%s",
|
||||||
|
type,
|
||||||
|
state_key,
|
||||||
|
code,
|
||||||
|
msg,
|
||||||
|
e.what()
|
||||||
|
};
|
||||||
|
|
||||||
|
const json::members error_object
|
||||||
|
{
|
||||||
|
{ "error", e.what() },
|
||||||
|
};
|
||||||
|
|
||||||
|
const json::value error_value{error_object};
|
||||||
|
const json::value error_records{&error_value, 1};
|
||||||
|
const json::strung error{error_records};
|
||||||
|
call_waiters(type, state_key, error);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
ircd::net::dns::cache::put(const string_view &type,
|
||||||
|
const string_view &state_key,
|
||||||
|
const records &rrs)
|
||||||
|
try
|
||||||
|
{
|
||||||
|
const unique_buffer<mutable_buffer> buf
|
||||||
|
{
|
||||||
|
8_KiB
|
||||||
|
};
|
||||||
|
|
||||||
|
json::stack out{buf};
|
||||||
|
json::stack::object content{out};
|
||||||
|
json::stack::array array
|
||||||
|
{
|
||||||
|
content, ""
|
||||||
|
};
|
||||||
|
|
||||||
|
if(rrs.empty())
|
||||||
|
{
|
||||||
|
// Add one object to the array with nothing except a ttl indicating no
|
||||||
|
// records (and no error) so we can cache that for the ttl. We use the
|
||||||
|
// nxdomain ttl for this value.
|
||||||
|
json::stack::object rr0{array};
|
||||||
|
json::stack::member
|
||||||
|
{
|
||||||
|
rr0, "ttl", json::value
|
||||||
|
{
|
||||||
|
long(seconds(nxdomain_ttl).count())
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
else for(const auto &record : rrs)
|
||||||
|
{
|
||||||
|
switch(record->type)
|
||||||
|
{
|
||||||
|
case 1: // A
|
||||||
|
{
|
||||||
|
json::stack::object object{array};
|
||||||
|
dynamic_cast<const rfc1035::record::A *>(record)->append(object);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 5: // CNAME
|
||||||
|
{
|
||||||
|
json::stack::object object{array};
|
||||||
|
dynamic_cast<const rfc1035::record::CNAME *>(record)->append(object);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 28: // AAAA
|
||||||
|
{
|
||||||
|
json::stack::object object{array};
|
||||||
|
dynamic_cast<const rfc1035::record::AAAA *>(record)->append(object);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 33: // SRV
|
||||||
|
{
|
||||||
|
json::stack::object object{array};
|
||||||
|
dynamic_cast<const rfc1035::record::SRV *>(record)->append(object);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// When the array has a zero value count we didn't know how to cache
|
||||||
|
// any of these records; don't send anything to the cache room.
|
||||||
|
if(!array.vc)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
array.~array();
|
||||||
|
content.~object();
|
||||||
|
send(room_id, m::me, type, state_key, json::object{out.completed()});
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
catch(const http::error &e)
|
||||||
|
{
|
||||||
|
const ctx::exception_handler eh;
|
||||||
|
log::error
|
||||||
|
{
|
||||||
|
log, "cache put (%s, %s) rrs:%zu :%s %s",
|
||||||
|
type,
|
||||||
|
state_key,
|
||||||
|
rrs.size(),
|
||||||
|
e.what(),
|
||||||
|
e.content,
|
||||||
|
};
|
||||||
|
|
||||||
|
const json::value error_value
|
||||||
|
{
|
||||||
|
json::object{e.content}
|
||||||
|
};
|
||||||
|
|
||||||
|
const json::value error_records{&error_value, 1};
|
||||||
|
const json::strung error{error_records};
|
||||||
|
call_waiters(type, state_key, error);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
catch(const std::exception &e)
|
||||||
|
{
|
||||||
|
const ctx::exception_handler eh;
|
||||||
|
log::error
|
||||||
|
{
|
||||||
|
log, "cache put (%s, %s) rrs:%zu :%s",
|
||||||
|
type,
|
||||||
|
state_key,
|
||||||
|
rrs.size(),
|
||||||
|
e.what(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const json::members error_object
|
||||||
|
{
|
||||||
|
{ "error", e.what() },
|
||||||
|
};
|
||||||
|
|
||||||
|
const json::value error_value{error_object};
|
||||||
|
const json::value error_records{&error_value, 1};
|
||||||
|
const json::strung error{error_records};
|
||||||
|
call_waiters(type, state_key, error);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
IRCD_MODULE_EXPORT
|
||||||
|
ircd::net::dns::cache::get(const hostport &hp,
|
||||||
|
const opts &opts,
|
||||||
|
const callback &closure)
|
||||||
|
{
|
||||||
|
char type_buf[48];
|
||||||
|
const string_view type
|
||||||
|
{
|
||||||
|
make_type(type_buf, opts.qtype)
|
||||||
|
};
|
||||||
|
|
||||||
|
char state_key_buf[rfc1035::NAME_BUFSIZE * 2];
|
||||||
|
const string_view &state_key
|
||||||
|
{
|
||||||
|
opts.qtype == 33?
|
||||||
|
make_SRV_key(state_key_buf, host(hp), opts):
|
||||||
|
host(hp)
|
||||||
|
};
|
||||||
|
|
||||||
|
const m::room::state state
|
||||||
|
{
|
||||||
|
room_id
|
||||||
|
};
|
||||||
|
|
||||||
|
const m::event::idx &event_idx
|
||||||
|
{
|
||||||
|
state.get(std::nothrow, type, state_key)
|
||||||
|
};
|
||||||
|
|
||||||
|
if(!event_idx)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
time_t origin_server_ts;
|
||||||
|
if(!m::get<time_t>(event_idx, "origin_server_ts", origin_server_ts))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
bool ret{false};
|
||||||
|
const time_t ts{origin_server_ts / 1000L};
|
||||||
|
m::get(std::nothrow, event_idx, "content", [&hp, &closure, &ret, &ts]
|
||||||
|
(const json::object &content)
|
||||||
|
{
|
||||||
|
const json::array &rrs
|
||||||
|
{
|
||||||
|
content.get("")
|
||||||
|
};
|
||||||
|
|
||||||
|
// If all records are expired then skip; otherwise since this closure
|
||||||
|
// expects a single array we reveal both expired and valid records.
|
||||||
|
ret = !std::all_of(begin(rrs), end(rrs), [&ts]
|
||||||
|
(const json::object &rr)
|
||||||
|
{
|
||||||
|
return expired(rr, ts);
|
||||||
|
});
|
||||||
|
|
||||||
|
if(ret && closure)
|
||||||
|
closure(hp, rrs);
|
||||||
|
});
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
IRCD_MODULE_EXPORT
|
||||||
|
ircd::net::dns::cache::for_each(const hostport &hp,
|
||||||
|
const opts &opts,
|
||||||
|
const closure &closure)
|
||||||
|
{
|
||||||
|
char type_buf[48];
|
||||||
|
const string_view type
|
||||||
|
{
|
||||||
|
make_type(type_buf, opts.qtype)
|
||||||
|
};
|
||||||
|
|
||||||
|
char state_key_buf[rfc1035::NAME_BUFSIZE * 2];
|
||||||
|
const string_view &state_key
|
||||||
|
{
|
||||||
|
opts.qtype == 33?
|
||||||
|
make_SRV_key(state_key_buf, host(hp), opts):
|
||||||
|
host(hp)
|
||||||
|
};
|
||||||
|
|
||||||
|
const m::room::state state
|
||||||
|
{
|
||||||
|
room_id
|
||||||
|
};
|
||||||
|
|
||||||
|
const m::event::idx &event_idx
|
||||||
|
{
|
||||||
|
state.get(std::nothrow, type, state_key)
|
||||||
|
};
|
||||||
|
|
||||||
|
if(!event_idx)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
time_t origin_server_ts;
|
||||||
|
if(!m::get<time_t>(event_idx, "origin_server_ts", origin_server_ts))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
bool ret{true};
|
||||||
|
const time_t ts{origin_server_ts / 1000L};
|
||||||
|
m::get(std::nothrow, event_idx, "content", [&state_key, &closure, &ret, &ts]
|
||||||
|
(const json::object &content)
|
||||||
|
{
|
||||||
|
for(const json::object &rr : json::array(content.get("")))
|
||||||
|
{
|
||||||
|
if(expired(rr, ts))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if(!(ret = closure(state_key, rr)))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
IRCD_MODULE_EXPORT
|
||||||
|
ircd::net::dns::cache::for_each(const string_view &type,
|
||||||
|
const closure &closure)
|
||||||
|
{
|
||||||
|
char type_buf[48];
|
||||||
|
const string_view full_type
|
||||||
|
{
|
||||||
|
make_type(type_buf, type)
|
||||||
|
};
|
||||||
|
|
||||||
|
const m::room::state state
|
||||||
|
{
|
||||||
|
room_id
|
||||||
|
};
|
||||||
|
|
||||||
|
return state.for_each(full_type, [&closure]
|
||||||
|
(const string_view &, const string_view &state_key, const m::event::idx &event_idx)
|
||||||
|
{
|
||||||
|
time_t origin_server_ts;
|
||||||
|
if(!m::get<time_t>(event_idx, "origin_server_ts", origin_server_ts))
|
||||||
|
return true;
|
||||||
|
|
||||||
|
bool ret{true};
|
||||||
|
const time_t ts{origin_server_ts / 1000L};
|
||||||
|
m::get(std::nothrow, event_idx, "content", [&state_key, &closure, &ret, &ts]
|
||||||
|
(const json::object &content)
|
||||||
|
{
|
||||||
|
for(const json::object &rr : json::array(content.get("")))
|
||||||
|
{
|
||||||
|
if(expired(rr, ts))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if(!(ret = closure(state_key, rr)))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
ircd::net::dns::cache::handle(const m::event &event,
|
||||||
|
m::vm::eval &eval)
|
||||||
|
try
|
||||||
|
{
|
||||||
|
const string_view &type
|
||||||
|
{
|
||||||
|
json::get<"type"_>(event)
|
||||||
|
};
|
||||||
|
|
||||||
|
if(!startswith(type, "ircd.dns.rrs."))
|
||||||
|
return;
|
||||||
|
|
||||||
|
const string_view &state_key
|
||||||
|
{
|
||||||
|
json::get<"state_key"_>(event)
|
||||||
|
};
|
||||||
|
|
||||||
|
const json::array &rrs
|
||||||
|
{
|
||||||
|
json::get<"content"_>(event).get("")
|
||||||
|
};
|
||||||
|
|
||||||
|
call_waiters(type, state_key, rrs);
|
||||||
|
}
|
||||||
|
catch(const std::exception &e)
|
||||||
|
{
|
||||||
|
log::critical
|
||||||
|
{
|
||||||
|
log, "handle_cached() :%s", e.what()
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Note complications due to reentrance and other factors:
|
||||||
|
/// - This function is invoked from several different places on both the
|
||||||
|
/// timeout and receive contexts, in addition to any evaluator context.
|
||||||
|
/// - This function calls back to users making DNS queries, and they may
|
||||||
|
/// conduct another query in their callback frame -- mid-loop in this
|
||||||
|
/// function.
|
||||||
|
size_t
|
||||||
|
ircd::net::dns::cache::call_waiters(const string_view &type,
|
||||||
|
const string_view &state_key,
|
||||||
|
const json::array &rrs)
|
||||||
|
{
|
||||||
|
const ctx::uninterruptible::nothrow ui;
|
||||||
|
size_t ret(0), last; do
|
||||||
|
{
|
||||||
|
const std::lock_guard lock
|
||||||
|
{
|
||||||
|
mutex
|
||||||
|
};
|
||||||
|
|
||||||
|
auto it(begin(waiting));
|
||||||
|
for(last = ret; it != end(waiting); ++it)
|
||||||
|
if(call_waiter(type, state_key, rrs, *it))
|
||||||
|
{
|
||||||
|
it = waiting.erase(it);
|
||||||
|
++ret;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
while(last < ret);
|
||||||
|
|
||||||
|
if(ret)
|
||||||
|
dock.notify_all();
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
ircd::net::dns::cache::call_waiter(const string_view &type,
|
||||||
|
const string_view &state_key,
|
||||||
|
const json::array &rrs,
|
||||||
|
waiter &waiter)
|
||||||
|
try
|
||||||
|
{
|
||||||
|
if(state_key != waiter.key)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if(lstrip(type, "ircd.dns.rrs.") != rfc1035::rqtype.at(waiter.opts.qtype))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
const hostport &target
|
||||||
|
{
|
||||||
|
waiter.opts.qtype == 33?
|
||||||
|
unmake_SRV_key(waiter.key):
|
||||||
|
waiter.key,
|
||||||
|
|
||||||
|
waiter.port
|
||||||
|
};
|
||||||
|
|
||||||
|
assert(waiter.callback);
|
||||||
|
waiter.callback(target, rrs);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
catch(const std::exception &e)
|
||||||
|
{
|
||||||
|
log::critical
|
||||||
|
{
|
||||||
|
log, "callback:%p %s,%s :%s",
|
||||||
|
(const void *)&waiter,
|
||||||
|
type,
|
||||||
|
state_key,
|
||||||
|
e.what(),
|
||||||
|
};
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// cache::waiter
|
||||||
|
//
|
||||||
|
|
||||||
|
bool
|
||||||
|
ircd::net::dns::cache::operator==(const waiter &a, const waiter &b)
|
||||||
|
{
|
||||||
|
return a.opts.qtype == b.opts.qtype &&
|
||||||
|
a.key && b.key &&
|
||||||
|
a.key == b.key;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
ircd::net::dns::cache::operator!=(const waiter &a, const waiter &b)
|
||||||
|
{
|
||||||
|
return !operator==(a, b);
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// cache::waiter::waiter
|
||||||
|
//
|
||||||
|
|
||||||
|
ircd::net::dns::cache::waiter::waiter(const hostport &hp,
|
||||||
|
const dns::opts &opts,
|
||||||
|
dns::callback &&callback)
|
||||||
|
:callback
|
||||||
|
{
|
||||||
|
std::move(callback)
|
||||||
|
}
|
||||||
|
,opts
|
||||||
|
{
|
||||||
|
opts
|
||||||
|
}
|
||||||
|
,port
|
||||||
|
{
|
||||||
|
net::port(hp)
|
||||||
|
}
|
||||||
|
,key
|
||||||
|
{
|
||||||
|
opts.qtype == 33?
|
||||||
|
make_SRV_key(keybuf, hp, opts):
|
||||||
|
strlcpy(keybuf, host(hp))
|
||||||
|
}
|
||||||
|
{
|
||||||
|
this->opts.srv = {};
|
||||||
|
this->opts.proto = {};
|
||||||
|
assert(this->opts.qtype);
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// cache room creation
|
||||||
|
//
|
||||||
|
|
||||||
|
namespace ircd::net::dns {
|
||||||
|
namespace [[gnu::visibility("hidden")]] cache
|
||||||
|
{
|
||||||
|
static void create_room();
|
||||||
|
extern m::hookfn<m::vm::eval &> create_room_hook;
|
||||||
|
}}
|
||||||
|
|
||||||
|
decltype(ircd::net::dns::cache::create_room_hook)
|
||||||
|
ircd::net::dns::cache::create_room_hook
|
||||||
|
{
|
||||||
|
{
|
||||||
|
{ "_site", "vm.effect" },
|
||||||
|
{ "room_id", "!ircd" },
|
||||||
|
{ "type", "m.room.create" },
|
||||||
|
},
|
||||||
|
[](const m::event &, m::vm::eval &)
|
||||||
|
{
|
||||||
|
create_room();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
void
|
||||||
|
ircd::net::dns::cache::create_room()
|
||||||
|
try
|
||||||
|
{
|
||||||
|
const m::room room
|
||||||
|
{
|
||||||
|
m::create(room_id, m::me, "internal")
|
||||||
|
};
|
||||||
|
|
||||||
|
log::debug
|
||||||
|
{
|
||||||
|
m::log, "Created '%s' for the DNS cache module.",
|
||||||
|
string_view{room.room_id}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
catch(const std::exception &e)
|
||||||
|
{
|
||||||
|
log::critical
|
||||||
|
{
|
||||||
|
m::log, "Creating the '%s' room failed :%s",
|
||||||
|
string_view{room_id},
|
||||||
|
e.what()
|
||||||
|
};
|
||||||
|
}
|
Loading…
Reference in a new issue