2018-01-28 18:37:13 +01:00
|
|
|
// Matrix Construct
|
|
|
|
//
|
|
|
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
|
|
|
// Copyright (C) 2016-2018 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.
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
#define HAVE_IRCD_NET_DNS_H
|
|
|
|
|
|
|
|
/// DNS resolution suite.
|
|
|
|
///
|
2019-03-23 08:59:25 +01:00
|
|
|
/// Note that at this time these functions must be called on an ircd::ctx to
|
|
|
|
/// conduct any queries of the cache.
|
2018-03-02 08:08:22 +01:00
|
|
|
///
|
2018-10-01 20:52:59 +02:00
|
|
|
namespace ircd::net::dns
|
2018-01-28 18:37:13 +01:00
|
|
|
{
|
2019-10-06 02:02:50 +02:00
|
|
|
struct init;
|
2019-05-04 01:09:57 +02:00
|
|
|
struct opts;
|
2018-10-03 23:17:27 +02:00
|
|
|
using records = vector_view<const rfc1035::record *>;
|
2019-03-22 02:24:36 +01:00
|
|
|
using callback = std::function<void (const hostport &, const json::array &)>;
|
|
|
|
using callback_one = std::function<void (const hostport &, const json::object &)>;
|
|
|
|
using callback_ipport = std::function<void (std::exception_ptr, const hostport &, const ipport &)>;
|
2018-04-15 01:20:21 +02:00
|
|
|
|
2019-05-04 01:09:57 +02:00
|
|
|
extern const struct opts opts_default;
|
|
|
|
extern log::log log;
|
2018-10-01 20:52:59 +02:00
|
|
|
|
2019-03-24 22:50:26 +01:00
|
|
|
// Utilities
|
|
|
|
bool is_error(const json::object &rr);
|
|
|
|
bool is_error(const json::array &rr);
|
2019-03-25 04:12:15 +01:00
|
|
|
bool is_empty(const json::object &rr);
|
|
|
|
bool is_empty(const json::array &rr);
|
2019-03-24 22:50:26 +01:00
|
|
|
time_t get_ttl(const json::object &rr);
|
|
|
|
bool expired(const json::object &rr, const time_t &rr_ts, const time_t &min_ttl);
|
|
|
|
bool expired(const json::object &rr, const time_t &rr_ts);
|
|
|
|
json::object random_choice(const json::array &);
|
2020-03-06 20:41:09 +01:00
|
|
|
|
2018-10-01 20:52:59 +02:00
|
|
|
string_view make_SRV_key(const mutable_buffer &out, const hostport &, const opts &);
|
|
|
|
string_view unmake_SRV_key(const string_view &);
|
2019-03-12 02:08:46 +01:00
|
|
|
|
2020-03-06 20:41:09 +01:00
|
|
|
uint16_t service_port(std::nothrow_t, const string_view &name, const string_view &prot = {});
|
|
|
|
uint16_t service_port(const string_view &name, const string_view &prot = {});
|
|
|
|
|
2019-05-04 01:09:57 +02:00
|
|
|
// Callback-based interface
|
|
|
|
void resolve(const hostport &, const opts &, callback);
|
|
|
|
void resolve(const hostport &, const opts &, callback_one); // convenience
|
|
|
|
void resolve(const hostport &, const opts &, callback_ipport); // convenience
|
2019-03-23 08:59:25 +01:00
|
|
|
}
|
2018-01-28 23:19:35 +01:00
|
|
|
|
2018-02-03 22:15:55 +01:00
|
|
|
/// DNS resolution options
|
|
|
|
struct ircd::net::dns::opts
|
2018-01-28 23:19:35 +01:00
|
|
|
{
|
2018-10-03 07:38:08 +02:00
|
|
|
/// Specifies the rfc1035 query type. If this is non-zero a query of this
|
|
|
|
/// type will be made or an exception will be thrown if it is not possible
|
|
|
|
/// to make the query with this type. A zero value means automatic and the
|
|
|
|
/// type will deduced and set internally. To translate a string type like
|
|
|
|
/// "SRV" to the type integer for this value see ircd/rfc1035.h.
|
|
|
|
uint16_t qtype {0};
|
|
|
|
|
2018-02-03 22:15:55 +01:00
|
|
|
/// Overrides the SRV query to make for this resolution. If empty an
|
|
|
|
/// SRV query may still be made from other deductions. This string is
|
|
|
|
/// copied at the start of the resolution. It must be a fully qualified
|
|
|
|
/// SRV query string: Example: "_matrix._tcp."
|
|
|
|
string_view srv;
|
2018-01-28 23:19:35 +01:00
|
|
|
|
2018-02-03 22:15:55 +01:00
|
|
|
/// Specifies the SRV protocol part when deducing the SRV query to be
|
|
|
|
/// made. This is used when the net::hostport.service string is just
|
|
|
|
/// the name of the service like "matrix" so we then use this value
|
|
|
|
/// to build the protocol part of the SRV query string. It is ignored
|
|
|
|
/// if `srv` is set or if no service is specified (thus no SRV query is
|
|
|
|
/// made in the first place).
|
|
|
|
string_view proto{"tcp"};
|
2018-03-02 08:08:22 +01:00
|
|
|
|
|
|
|
/// Whether the dns::cache is checked and may respond to the query.
|
|
|
|
bool cache_check {true};
|
|
|
|
|
|
|
|
/// Whether the result of this lookup from the nameserver should be
|
2019-03-22 02:24:36 +01:00
|
|
|
/// added to the cache.
|
2018-03-02 08:08:22 +01:00
|
|
|
bool cache_result {true};
|
2018-04-29 01:42:18 +02:00
|
|
|
|
|
|
|
/// When false, nxdomain errors are not treated as exceptions and the
|
|
|
|
/// eptr of a callback will not be set. Instead, the returned record
|
|
|
|
/// will contain some nulled / empty data and the user is obliged to
|
|
|
|
/// know that this is not a real DNS answer but the error's result.
|
|
|
|
/// Note: Requires that cache_result is set to true. If not, this value
|
|
|
|
/// is ignored and always considered to be set to true; this is because
|
|
|
|
/// the returned record is a reference to the cached error.
|
|
|
|
bool nxdomain_exceptions {true};
|
2019-08-18 11:06:45 +02:00
|
|
|
|
2020-03-06 20:41:09 +01:00
|
|
|
/// When false, queries to translate service strings to port numbers using
|
|
|
|
/// netdb (i.e. /etc/services) will be disabled if they were to occur.
|
|
|
|
bool service_port {true};
|
|
|
|
|
2019-08-18 11:06:45 +02:00
|
|
|
opts() = default;
|
2018-03-02 08:08:22 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/// (internal) DNS cache
|
2018-10-01 20:52:59 +02:00
|
|
|
namespace ircd::net::dns::cache
|
2018-03-02 08:08:22 +01:00
|
|
|
{
|
2019-03-22 02:24:36 +01:00
|
|
|
using closure = std::function<bool (const string_view &, const json::object &)>;
|
|
|
|
|
2019-10-03 19:47:33 +02:00
|
|
|
extern conf::item<seconds> min_ttl;
|
|
|
|
extern conf::item<seconds> error_ttl;
|
|
|
|
extern conf::item<seconds> nxdomain_ttl;
|
|
|
|
|
2019-03-22 02:24:36 +01:00
|
|
|
string_view make_type(const mutable_buffer &out, const string_view &);
|
|
|
|
string_view make_type(const mutable_buffer &out, const uint16_t &);
|
2018-10-01 04:01:06 +02:00
|
|
|
|
2019-03-22 02:24:36 +01:00
|
|
|
bool for_each(const string_view &type, const closure &); // do not make_type() here
|
|
|
|
bool for_each(const hostport &, const opts &, const closure &);
|
2018-04-15 01:39:13 +02:00
|
|
|
bool get(const hostport &, const opts &, const callback &);
|
2019-03-22 02:24:36 +01:00
|
|
|
bool put(const hostport &, const opts &, const records &);
|
|
|
|
bool put(const hostport &, const opts &, const uint &code, const string_view &msg = {});
|
2019-03-23 08:59:25 +01:00
|
|
|
}
|
2019-10-06 02:02:50 +02:00
|
|
|
|
|
|
|
/// (internal)
|
|
|
|
struct ircd::net::dns::init
|
|
|
|
{
|
|
|
|
init(), ~init() noexcept;
|
|
|
|
};
|