created arg recursion_count to async_resolve and many other to avoid infinite loops
[pingcheck] / src / dns / dnsmaster.h
1 /*
2  The software in this package is distributed under the GNU General
3  Public License version 2 (with a special exception described below).
4
5  A copy of GNU General Public License (GPL) is included in this distribution,
6  in the file COPYING.GPL.
7
8  As a special exception, if other files instantiate templates or use macros
9  or inline functions from this file, or you compile this file and link it
10  with other works to produce a work based on this file, this file
11  does not by itself cause the resulting work to be covered
12  by the GNU General Public License.
13
14  However the source code for this file must still be made available
15  in accordance with section (3) of the GNU General Public License.
16
17  This exception does not invalidate any other reasons why a work based
18  on this file might be covered by the GNU General Public License.
19
20  Christian Herdtweck, Intra2net AG 2015
21  */
22
23 /**
24  * Two in one: a DNS resolver factory and a DNS cache
25  *
26  * Put these two things into one class because it is easier this way to avoid
27  *   sync problems if e.g. there are 2 resolvers for the same host name
28  *
29  * This class is a Singleton. In case there are problems with this approach,
30  * there is an alternative:
31  * give every ResolverBase object a DnsMasterItem as variable, that is set
32  *   during construction and call update / get_cached_results on that instance
33  */
34
35 #ifndef DNS_MASTER_H
36 #define DNS_MASTER_H
37
38 #include <map>
39 #include <utility>   // pair
40
41 #include <boost/shared_ptr.hpp>
42 #include <boost/noncopyable.hpp>
43 #include <boost/net/dns.hpp>
44
45 #include "host/pinger.h"    // for IoserviceItem
46 #include "host/pingprotocol.h"
47 #include "dns/dnscache.h"
48 #include "dns/resolverbase.h"
49
50 class DnsMaster;
51
52 typedef boost::shared_ptr<DnsMaster> DnsMasterItem;
53
54 typedef boost::net::dns::type_t DnsIpProtocol;
55 const DnsIpProtocol DNS_IPv4 = boost::net::dns::type_a;
56 const DnsIpProtocol DNS_IPv6 = boost::net::dns::type_a6;
57 const DnsIpProtocol DNS_IPALL = boost::net::dns::type_all;
58
59 typedef std::pair<std::string, DnsIpProtocol> resolver_key_type;
60 typedef std::map<resolver_key_type, ResolverItem> resolver_map_type;
61
62 std::string to_string(const DnsIpProtocol &protocol);
63
64 /**
65  * Factory and Cache of DNS resolvers
66  * 
67  * to avoid having several resolvers resolving the same hostname which might
68  * result in conflicts with caching, this class is a singleton factory and the
69  * only place where Resolvers are constructed. They are remembered in an
70  * internal cache by hostname and IP version requested (v4, v6 or both).
71  * 
72  * During resolving, several different name servers will have to be queried for
73  * the same hostname. These recursive resolvers are created using
74  * get_recursor_for and are NOT cached, so they should only be used from another
75  * "regular" resolver (created using get_resolver_for)
76  *
77  * The DnsMaster also remembers a few global variables that can be queried
78  * using public getter functions and it creates the DnsCache used by all its
79  * resolvers
80  */
81 class DnsMaster : boost::noncopyable
82 {
83 // Resolver factory
84 public:
85     ResolverItem& get_resolver_for(const std::string &hostname,
86                                    const PingProtocol &ping_protocol);
87     ResolverItem& get_resolver_for(const std::string &hostname,
88                                    const DnsIpProtocol &protocol);
89
90 // implementation of singleton
91 private:
92     static DnsMasterItem TheOnlyInstance;
93
94     DnsMaster(const IoServiceItem &io_serv,
95               const boost::asio::ip::address &default_name_server,
96               const int resolved_ip_ttl_threshold,
97               const int max_address_resolution_attempts,
98               const int max_recursion_count,
99               const DnsCacheItem &cache);
100 public:
101     static void create_master(const IoServiceItem &io_serv,
102                             const boost::asio::ip::address &default_name_server,
103                             const int resolved_ip_ttl_threshold,
104                             const int min_time_between_resolves,
105                             const int max_address_resolution_attempts,
106                             const int max_recursion_count,
107                             const std::string &cache_file);
108     static void create_master(const IoServiceItem &io_serv,
109                             const boost::asio::ip::address &default_name_server,
110                             const int resolved_ip_ttl_threshold,
111                             const int max_address_resolution_attempts,
112                             const int max_recursion_count,
113                             const DnsCacheItem &cache);  // needed for unit test
114     static DnsMasterItem& get_instance();
115     ~DnsMaster();
116
117 // storage of global variables
118 public:
119     //boost::asio::ip::address &get_default_name_server() const;  // unused
120     int get_resolved_ip_ttl_threshold() const;
121     int get_max_address_resolution_attempts() const;
122     int get_max_recursion_count() const;
123
124 // variables
125 private:
126     IoServiceItem IoService;
127     const boost::asio::ip::address DefaultNameServer;
128     const int ResolvedIpTtlThreshold;
129     const int MaxAddressResolutionAttempts;
130     const int MaxRecursionCount;
131     DnsCacheItem Cache;
132     resolver_map_type ResolverMap;
133
134 // internal helper functions
135 private:
136     bool is_ip(const std::string &hostname) const;
137     static DnsIpProtocol ping2dns_protocol(const PingProtocol& pprot);
138 };
139
140 #endif
141
142 // (created using vim -- the world's best text editor)
143