mirror of https://github.com/sipwise/sems.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
313 lines
6.4 KiB
313 lines
6.4 KiB
/*
|
|
* $Id: resolver.h 1460 2009-07-08 12:50:39Z rco $
|
|
*
|
|
* Copyright (C) 2007 Raphael Coeffic
|
|
*
|
|
* This file is part of SEMS, a free SIP media server.
|
|
*
|
|
* SEMS is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version. This program is released under
|
|
* the GPL with the additional exemption that compiling, linking,
|
|
* and/or using OpenSSL is allowed.
|
|
*
|
|
* For a license to use the SEMS software under conditions
|
|
* other than those described here, or to purchase support for this
|
|
* software, please contact iptel.org by e-mail at the following addresses:
|
|
* info@iptel.org
|
|
*
|
|
* SEMS is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
#ifndef _resolver_h_
|
|
#define _resolver_h_
|
|
|
|
#include "singleton.h"
|
|
#include "hash_table.h"
|
|
#include "atomic_types.h"
|
|
#include "parse_dns.h"
|
|
#include "parse_next_hop.h"
|
|
|
|
#include <string>
|
|
#include <vector>
|
|
#include <map>
|
|
using std::string;
|
|
using std::vector;
|
|
using std::map;
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#define DNS_CACHE_SIZE 128
|
|
|
|
enum address_type {
|
|
|
|
IPnone=0,
|
|
IPv4=1,
|
|
IPv6=2
|
|
};
|
|
|
|
enum proto_type {
|
|
|
|
TCP=1,
|
|
UDP=2
|
|
};
|
|
|
|
struct dns_handle;
|
|
|
|
struct dns_base_entry
|
|
{
|
|
u_int64_t expire;
|
|
|
|
dns_base_entry()
|
|
:expire(0)
|
|
{}
|
|
|
|
virtual ~dns_base_entry() {}
|
|
virtual string to_str() = 0;
|
|
};
|
|
|
|
class dns_entry
|
|
: public atomic_ref_cnt,
|
|
public dns_base_entry
|
|
{
|
|
virtual dns_base_entry* get_rr(dns_record* rr, u_char* begin, u_char* end)=0;
|
|
|
|
public:
|
|
vector<dns_base_entry*> ip_vec;
|
|
|
|
static dns_entry* make_entry(dns_rr_type t);
|
|
|
|
dns_entry();
|
|
virtual ~dns_entry();
|
|
virtual void init()=0;
|
|
virtual void add_rr(dns_record* rr, u_char* begin, u_char* end, long now);
|
|
virtual int next_ip(dns_handle* h, sockaddr_storage* sa)=0;
|
|
|
|
virtual string to_str();
|
|
};
|
|
|
|
typedef ht_map_bucket<string,dns_entry> dns_bucket_base;
|
|
|
|
class dns_bucket
|
|
: protected dns_bucket_base
|
|
{
|
|
friend class _resolver;
|
|
public:
|
|
dns_bucket(unsigned long id);
|
|
bool insert(const string& name, dns_entry* e);
|
|
bool remove(const string& name);
|
|
dns_entry* find(const string& name);
|
|
};
|
|
|
|
typedef hash_table<dns_bucket> dns_cache;
|
|
|
|
struct ip_entry
|
|
: public dns_base_entry
|
|
{
|
|
address_type type;
|
|
|
|
union {
|
|
in_addr addr;
|
|
in6_addr addr6;
|
|
};
|
|
|
|
virtual void to_sa(sockaddr_storage* sa);
|
|
virtual string to_str();
|
|
};
|
|
|
|
struct ip_port_entry
|
|
: public ip_entry
|
|
{
|
|
unsigned short port;
|
|
|
|
virtual void to_sa(sockaddr_storage* sa);
|
|
virtual string to_str();
|
|
};
|
|
|
|
class dns_ip_entry
|
|
: public dns_entry
|
|
{
|
|
public:
|
|
dns_ip_entry()
|
|
: dns_entry()
|
|
{}
|
|
|
|
void init(){};
|
|
dns_base_entry* get_rr(dns_record* rr, u_char* begin, u_char* end);
|
|
int next_ip(dns_handle* h, sockaddr_storage* sa);
|
|
|
|
int fill_ip_list(const list<sip_destination>& ip_list);
|
|
};
|
|
|
|
class dns_srv_entry;
|
|
|
|
struct dns_handle
|
|
{
|
|
dns_handle();
|
|
dns_handle(const dns_handle& h);
|
|
~dns_handle();
|
|
|
|
bool valid();
|
|
bool eoip();
|
|
|
|
int next_ip(sockaddr_storage* sa);
|
|
const dns_handle& operator = (const dns_handle& rh);
|
|
|
|
private:
|
|
friend class _resolver;
|
|
friend class dns_entry;
|
|
friend class dns_srv_entry;
|
|
friend class dns_ip_entry;
|
|
|
|
dns_srv_entry* srv_e;
|
|
int srv_n;
|
|
unsigned int srv_used;
|
|
unsigned short port;
|
|
|
|
dns_ip_entry* ip_e;
|
|
int ip_n;
|
|
};
|
|
|
|
struct naptr_record
|
|
: public dns_base_entry
|
|
{
|
|
unsigned short order;
|
|
unsigned short pref;
|
|
|
|
string flags;
|
|
string services;
|
|
string regexp;
|
|
string replace;
|
|
|
|
virtual string to_str()
|
|
{ return string(); }
|
|
};
|
|
|
|
class dns_naptr_entry
|
|
: public dns_entry
|
|
{
|
|
public:
|
|
dns_naptr_entry()
|
|
: dns_entry()
|
|
{}
|
|
|
|
void init();
|
|
dns_base_entry* get_rr(dns_record* rr, u_char* begin, u_char* end);
|
|
|
|
// not needed
|
|
int next_ip(dns_handle* h, sockaddr_storage* sa) { return -1; }
|
|
};
|
|
|
|
#define SIP_TRSP_SIZE_MAX 4
|
|
|
|
struct sip_target
|
|
{
|
|
sockaddr_storage ss;
|
|
char trsp[SIP_TRSP_SIZE_MAX+1];
|
|
|
|
sip_target() : ss{}, trsp{} {};
|
|
sip_target(const sip_target& target);
|
|
|
|
void clear();
|
|
const sip_target& operator = (const sip_target& target);
|
|
};
|
|
|
|
struct sip_target_set
|
|
{
|
|
list<sip_target> dest_list;
|
|
list<sip_target>::iterator dest_list_it;
|
|
|
|
sip_target_set();
|
|
|
|
void reset_iterator();
|
|
bool has_next();
|
|
int get_next(sockaddr_storage* ss, cstring& next_trsp,
|
|
unsigned int flags);
|
|
bool next();
|
|
|
|
void debug();
|
|
|
|
private:
|
|
sip_target_set(const sip_target_set&) {}
|
|
};
|
|
|
|
typedef map<string,dns_entry*> dns_entry_map_base;
|
|
|
|
class dns_entry_map
|
|
: public dns_entry_map_base
|
|
{
|
|
public:
|
|
dns_entry_map();
|
|
~dns_entry_map();
|
|
|
|
bool insert(const key_type& key, mapped_type e);
|
|
dns_entry* fetch(const key_type& key);
|
|
|
|
private:
|
|
// forbid some inherited methods
|
|
mapped_type& operator[](const key_type& k);
|
|
std::pair<iterator, bool> insert(const value_type& x);
|
|
};
|
|
|
|
class _resolver
|
|
: AmThread
|
|
{
|
|
public:
|
|
// disable SRV lookups
|
|
static bool disable_srv;
|
|
|
|
int resolve_name(const char* name,
|
|
dns_handle* h,
|
|
sockaddr_storage* sa,
|
|
const address_type types,
|
|
dns_rr_type t = dns_r_a);
|
|
|
|
int str2ip(const char* name,
|
|
sockaddr_storage* sa,
|
|
const address_type types);
|
|
|
|
int query_dns(const char* name, dns_entry_map& entry_map, dns_rr_type t);
|
|
|
|
/**
|
|
* Transforms all elements of a destination list into
|
|
* a target set, thus resolving all DNS names and
|
|
* converting IPs into a sockaddr_storage.
|
|
*/
|
|
int resolve_targets(const list<sip_destination>& dest_list,
|
|
sip_target_set* targets);
|
|
|
|
protected:
|
|
_resolver();
|
|
~_resolver();
|
|
|
|
int set_destination_ip(const cstring& next_hop,
|
|
unsigned short next_port,
|
|
const cstring& next_trsp,
|
|
sockaddr_storage* remote_ip,
|
|
dns_handle* h_dns);
|
|
|
|
void run();
|
|
void on_stop() {}
|
|
|
|
private:
|
|
dns_cache cache;
|
|
};
|
|
|
|
typedef singleton<_resolver> resolver;
|
|
|
|
#endif
|
|
|
|
/** EMACS **
|
|
* Local variables:
|
|
* mode: c++
|
|
* c-basic-offset: 4
|
|
* End:
|
|
*/
|