475d29e0fc31cab8281ed052af3ebec9251825d3
[bpdyndnsd] / src / service_dyndns.cpp
1 /** @file
2  * @brief DYNDNS Service class implementation. This class represents the DYNDNS service.
3  *
4  *
5  *
6  * @copyright Intra2net AG
7  * @license GPLv2
8 */
9
10 #include "service_dyndns.hpp"
11
12 #include <boost/foreach.hpp>
13
14 using namespace std;
15
16
17 /**
18  * Default Constructor, needed for object serialization.
19  */
20 ServiceDyndns::ServiceDyndns()
21 {
22 }
23
24
25 /**
26  * Constructor.
27  * @param _hostname The hostname to update
28  * @param _login The login name.
29  * @param _password The corresponding password.
30  */
31 ServiceDyndns::ServiceDyndns(const string& _protocol, const string& _hostname, const string& _login, const string& _password, const Logger::Ptr& _logger, const int _update_interval, const int _max_updates_within_interval, const int _max_equal_updates_in_succession, const int _dns_cache_ttl, const string& _proxy, const int _proxy_port, const string& _alternative_server)
32     : AlternativeServer(_alternative_server)
33 {
34     if ( _update_interval == -1 )        // If _update_interval is default po::option_desc (not specified via config)
35         set_update_interval(15);              // use default protocol value
36     else
37         set_update_interval(_update_interval);
38
39     if ( _max_updates_within_interval == -1 )
40         set_max_updates_within_interval(3);
41     else
42         set_max_updates_within_interval(_max_updates_within_interval);
43
44     if ( _max_equal_updates_in_succession == -1 )
45         set_max_equal_updates_in_succession(2);
46     else
47         set_max_equal_updates_in_succession(_max_equal_updates_in_succession);
48
49     if ( _dns_cache_ttl == -1 )
50         set_dns_cache_ttl(60);
51     else
52         set_dns_cache_ttl(_dns_cache_ttl);
53
54     set_protocol(_protocol);
55     set_hostname(_hostname);
56     set_login(_login);
57     set_password(_password);
58     set_logger(_logger);
59
60     // create http helper class
61     HTTPHelp = HTTPHelper::Ptr(new HTTPHelper(_logger,_proxy,_proxy_port,_login,_password));
62
63     BaseUrl = assemble_base_url(get_hostname());
64 }
65
66
67 /**
68  * Default destructor
69  */
70 ServiceDyndns::~ServiceDyndns()
71 {
72 }
73
74
75 /**
76  * Assemble the dyndns update url from the given fqhn
77  * @param hostname The fqhn hostname to update IP for.
78  * @return The assembled update url without IP.
79  */
80 string ServiceDyndns::assemble_base_url(const string& fqhn) const
81 {
82     string base_url = "https://";
83
84     // Test if a AlternativeServer name is given, needed for e.g. NO-IP
85     if ( AlternativeServer.empty() )
86         base_url.append("members.dyndns.org");
87     else
88         base_url.append(AlternativeServer);
89
90     base_url.append("/nic/update?hostname=");
91     base_url.append(fqhn);
92     base_url.append("&wildcard=NOCHG&mx=NOCHG&backmx=NOCHG&myip=");
93
94     return base_url;
95 }
96
97
98 /**
99  * Performs the Service update.
100  * @param ip IP Address to set.
101  * @return 0 if all is fine, -1 otherwise.
102  */
103 Service::UpdateErrorCode ServiceDyndns::perform_update(const std::string& ip)
104 {
105     string url = BaseUrl;
106     url.append(ip);
107
108     if ( !HTTPHelp->is_initialized() )
109     {
110         get_logger()->print_httphelper_not_initialized();
111         HTTPHelp->re_initialize();
112         return GenericError;
113     }
114
115     // Perform curl operation on given url
116     long http_status_code = HTTPHelp->http_get(url);
117
118     get_logger()->print_http_status_code(url,http_status_code);
119
120     // HTTP operation completed successful.
121     // Now we have to parse the data received by curl,
122     // cause http status code is not significant for dyndns update errors
123     if ( http_status_code == 200 )
124     {
125         // Get the received http data.
126         string curl_data = HTTPHelp->get_curl_data();
127
128         if ( curl_data.compare(0,4,"good") == 0 )
129         {
130             return UpdateOk;
131         }
132         else if (curl_data.compare(0,5,"nochg") == 0 )
133         {
134             // IP didn't change, this might indicate a problem at the
135             // dyndns backend or another client is running elsewhere.
136             get_logger()->print_update_failure(url, curl_data);
137             return NoChange;
138         }
139         else if (curl_data.compare(0,5,"abuse") == 0 )
140         {
141             // Account is blocked. Log it as a successful update
142             // so the IP address will be "burnt" until it changes.
143             get_logger()->print_update_failure(url, curl_data);
144             return Blocked;
145         }
146         else if ( curl_data == "badauth" )
147         {
148             get_logger()->print_service_not_authorized(url,get_login(),get_password());
149             return NotAuth;
150         }
151         else
152         {
153             get_logger()->print_update_failure(url, curl_data);
154             return UpdateError;
155         }
156     }
157     else
158     {
159         get_logger()->print_update_failure(url,http_status_code);
160     }
161
162     return GenericError;
163 }