Fixed undefined var ;-).
[bpdyndnsd] / src / service.cpp
CommitLineData
b1be615b 1/** @file
2bc1878a 2 * @brief The abstract service class. This class represents all services.
b1be615b
BS
3 *
4 *
5 *
6 * @copyright Intra2net AG
7 * @license GPLv2
8*/
9
4de6a9b8 10#include "service.hpp"
3c0cd271 11#include <boost/foreach.hpp>
ca5d6889
BS
12
13using namespace std;
14
0f0908e1
TJ
15/// Number of update errors until a service will be blocked
16const int MaxErrorCount = 3;
17/// Number of seconds a service will be blocked if MaxErrorCount is reached
18const int ErrorBlockServiceSeconds = 15 * 60;
2bc1878a 19
85a0abf9
BS
20/**
21 * Default Constructor
22 */
4545a371 23Service::Service()
9a0aff44 24 : Login("NOT SERIALIZED")
27baf279 25 , Password("NOT SERIALIZED")
025abebb 26 , ActualIP("0.0.0.0")
b67d84ea
BS
27 , UpdateInterval(15)
28 , MaxUpdatesWithinInterval(3)
4553e833 29 , MaxEqualUpdatesInSuccession(2)
d5a516ba 30 , DNSCacheTTL(0)
0f0908e1
TJ
31 , ErrorCount(0)
32 , ErrorServiceBlockedUntil(0)
2e956a36 33 , Log(new Logger())
4545a371
BS
34{
35}
36
2bc1878a 37
85a0abf9 38/**
27baf279 39 * Default Destructor needed for deserialization.
85a0abf9 40 */
4545a371
BS
41Service::~Service()
42{
43}
44
45
2bc1878a 46/**
3a89ac31
BS
47 * Setter for member Protocol.
48 * @param _protocol Value to set Protocol to.
49 */
50void Service::set_protocol(const string& _protocol)
51{
52 Protocol = _protocol;
53}
54
55
56/**
57 * Getter for memeber Protocol.
58 * @return Value of member Protocol.
59 */
b38684ce 60string Service::get_protocol() const
3a89ac31
BS
61{
62 return Protocol;
63}
64
65
66/**
67 * Setter for member Hostname.
68 * @param _hostname Value to set Hostname to.
69 */
70void Service::set_hostname(const string& _hostname)
71{
72 Hostname = _hostname;
73}
74
75
76/**
77 * Getter for member Hostname.
78 * @return Value of member Hostname.
79 */
b38684ce 80string Service::get_hostname() const
3a89ac31
BS
81{
82 return Hostname;
83}
84
85
86/**
87 * Setter for member Login.
88 * @param _login Value to set Login to.
89 */
90void Service::set_login(const string& _login)
91{
92 Login = _login;
93}
94
95
96/**
97 * Getter for member Login.
98 * @return Value of member Login.
99 */
b38684ce 100string Service::get_login() const
3a89ac31
BS
101{
102 return Login;
103}
104
105
106/**
107 * Setter for member Password.
108 * @param _password Value to set Password to.
109 */
110void Service::set_password(const string& _password)
111{
112 Password = _password;
113}
114
115
116/**
117 * Getter for member Password.
118 * @return Value of member Password.
119 */
b38684ce 120string Service::get_password() const
3a89ac31
BS
121{
122 return Password;
123}
124
125
d55fbd67 126
88a594e8 127void Service::set_logger(const Logger::Ptr& _log)
3a89ac31
BS
128{
129 Log = _log;
130}
131
132
133/**
134 * Getter for member Log.
135 * @return Shared pointer to Logger object.
136 */
b38684ce 137Logger::Ptr Service::get_logger() const
3a89ac31
BS
138{
139 return Log;
2bc1878a
BS
140}
141
142
4553e833
BS
143/**
144 * Setter for member LastUpdates.
145 * @param _last_updates Value to set LastUpdates to.
146 */
147void Service::set_last_updates(std::map<time_t,std::string> _last_updates)
2bc1878a 148{
3c0cd271 149 LastUpdates.clear();
4553e833 150 for ( std::map<time_t,std::string>::iterator iter = _last_updates.begin(); iter != _last_updates.end(); iter++ )
3c0cd271 151 {
4553e833 152 LastUpdates.insert(make_pair(iter->first,iter->second));
3c0cd271 153 }
2bc1878a
BS
154}
155
156
157/**
4553e833
BS
158 * Getter for member LastUpdates.
159 * @return Value of member LastUpdates.
2bc1878a 160 */
4553e833 161const std::map<time_t,std::string> Service::get_last_updates() const
2bc1878a 162{
c1b8cb79 163 return LastUpdates;
2bc1878a
BS
164}
165
166
167/**
025abebb
BS
168 * Setter for member ActualIP.
169 * @param _actual_ip Value to set ActualIP to.
27baf279
BS
170 */
171void Service::set_actual_ip(const std::string& _actual_ip)
172{
025abebb 173 ActualIP = _actual_ip;
27baf279
BS
174}
175
176
177/**
025abebb
BS
178 * Getter for member ActualIP.
179 * @return Value of member ActualIP.
27baf279 180 */
b38684ce 181std::string Service::get_actual_ip() const
27baf279 182{
025abebb 183 return ActualIP;
27baf279
BS
184}
185
186
27baf279 187/**
3a89ac31
BS
188 * Overloading of comparison operator.
189 * @param other Reference to other Service object.
190 * @return True if they equal, false if not.
2bc1878a 191 */
3a89ac31 192bool Service::operator== (const Service& other) const
2bc1878a 193{
3a89ac31
BS
194 if ( ( this->Protocol == other.Protocol ) && ( this->Hostname == other.Hostname ) )
195 return true;
196 return false;
2bc1878a
BS
197}
198
199
200/**
3a89ac31
BS
201 * Overloading of disparate operator.
202 * @param other Reference to other Service object.
203 * @return True if they differ, false if they are equal.
2bc1878a 204 */
3a89ac31 205bool Service::operator!= (const Service& other) const
2bc1878a 206{
3a89ac31 207 return !(*this == other);
2bc1878a 208}
3c0cd271
BS
209
210
211/**
212 * Checks if update will exceed max update interval.
213 * @param current_time Current time.
0ebcd4ef 214 * @param changed_to_online True if we just changed to online, false if we were already online
3c0cd271
BS
215 * @return True if update is allowed, false if update would exceed max update interval.
216 */
0ebcd4ef 217bool Service::update_allowed(const time_t current_time, bool changed_to_online)
3c0cd271 218{
4553e833
BS
219 int i = 0;
220 for ( std::map<time_t,std::string>::reverse_iterator r_iter = LastUpdates.rbegin(); (r_iter != LastUpdates.rend()) && ( i < MaxUpdatesWithinInterval ); r_iter++)
3c0cd271 221 {
4553e833 222 if ( (i == (MaxUpdatesWithinInterval-1)) && ( (r_iter->first + ((time_t)UpdateInterval*60)) >= current_time ) )
3c0cd271 223 {
4553e833 224 Log->print_update_not_allowed(changed_to_online,current_time,r_iter->first,MaxUpdatesWithinInterval,get_service_name());
3c0cd271
BS
225 return false;
226 }
227 i++;
228 }
229 return true;
230}
231
232
233/**
c3dea5dc
BS
234 * Service update method, common to each service.
235 * @param ip The new ip to set for the hostname.
0ebcd4ef
TJ
236 * @param current_time Current time
237 * @param changed_to_online True if we just changed to online, false if we were already online
c3dea5dc 238 */
0ebcd4ef 239void Service::update(const string& ip, const time_t current_time, bool changed_to_online)
c3dea5dc 240{
0f0908e1
TJ
241 const std::string service_name = get_service_name();
242
243 // Check if service is blocked for a short period of time (because of update errors)
244 if (ErrorServiceBlockedUntil && current_time < ErrorServiceBlockedUntil)
245 {
246 Log->print_update_service_is_blocked(service_name, ErrorServiceBlockedUntil - current_time);
247 return;
248 }
249
c3dea5dc 250 // test if update is permitted by UpdateInterval Status
0ebcd4ef 251 if ( update_allowed(current_time, changed_to_online) )
c3dea5dc 252 {
0ebcd4ef
TJ
253 Log->print_update_service(service_name);
254
c3dea5dc
BS
255 if ( perform_update(ip) == 0 )
256 {
257 // if update was successful, we need to set the Lastupdated and ActualIP base member.
4553e833 258 set_last_update(current_time,ip);
c3dea5dc 259 ActualIP = ip;
0f0908e1
TJ
260 Log->print_update_service_successful(service_name);
261
262 ErrorCount = 0;
263 ErrorServiceBlockedUntil = 0;
c3dea5dc
BS
264 }
265 else
266 {
267 // problem while trying to update service
0f0908e1
TJ
268 Log->print_update_service_failure(service_name);
269
270 ++ErrorCount;
271 if (ErrorCount >= MaxErrorCount)
272 {
273 Log->print_block_service(service_name, ErrorBlockServiceSeconds);
274
275 ErrorServiceBlockedUntil = time(NULL) + ErrorBlockServiceSeconds;
276 ErrorCount = 0;
277 }
c3dea5dc
BS
278 }
279 }
280}
281
282
283/**
fc613942
BS
284* Sets the given time into the LastUpdates member and deletes expired entries.
285* @param _timeout Value to set into LastUpdates.
286*/
4553e833 287void Service::set_last_update(const time_t current_time, const string& ip)
fc613942
BS
288{
289 // Insert value into the list.
4553e833
BS
290 LastUpdates.insert(make_pair(current_time,ip));
291
292 // Get the maximum of MaxUpdatesWithinInterval and MaxEqualUpdatesInSuccession
293 int maximum = max(MaxUpdatesWithinInterval,MaxEqualUpdatesInSuccession);
fc613942
BS
294
295 // Check for expired entries:
296
297 // MaxUpdatesWithinInterval given in service config, then use this to check for expired entries.
4553e833 298 if ( maximum > 0 )
fc613942 299 {
4553e833
BS
300 // Delete the oldest entry if there are more than max(MaxUpdatesWithinInterval,MaxEqualUpdatesInSuccession)+1 entries in the list.
301 if ( LastUpdates.size() > (size_t)(maximum+1) )
302 LastUpdates.erase(LastUpdates.upper_bound(0));
fc613942
BS
303 return;
304 }
305 // UpdateInterval given in service config, then use this to check for expired entries.
306 else if ( UpdateInterval > 0 )
307 {
308 // Delete the oldest entry if it's older than current_time - UpdateInterval(minutes) + 1.
162bffff 309 if ( (current_time - ((time_t)UpdateInterval*60) + 1) > LastUpdates.upper_bound(0)->first )
4553e833 310 LastUpdates.erase(LastUpdates.upper_bound(0));
fc613942
BS
311 return;
312 }
313 // Neither MaxUpdatesWithinInterval nor UpdateInterval are given, so keep fix number of 10 entries.
314 else
315 {
316 if ( LastUpdates.size() > 10 )
4553e833 317 LastUpdates.erase(LastUpdates.upper_bound(0));
fc613942
BS
318 return;
319 }
320}
321
322
323/**
4553e833
BS
324 * Getter the last updated time.
325 * @return Value of the last update as time_t.
326 */
327time_t Service::get_last_update_time( )
328{
329 std::map<time_t,std::string>::reverse_iterator r_iter = LastUpdates.rbegin();
330 return r_iter->first;
331}
332
333
334/**
3c0cd271
BS
335 * Setter for member Timeout.
336 * @param _timeout Value to set Timeout to.
337 */
338void Service::set_update_interval(const int _update_interval)
339{
340 UpdateInterval = _update_interval;
341}
342
343
344/**
345 * Getter for member Timeout.
346 * @return Value of Timeout.
347 */
348int Service::get_update_interval() const
349{
350 return UpdateInterval;
351}
352
353
354/**
355 * Setter for member Max_updates_per_timeout.
356 * @param _max_updates_per_timeout Value to set Max_updates_per_timeout to.
357 */
358void Service::set_max_updates_within_interval(const int _max_updates_within_interval)
359{
360 MaxUpdatesWithinInterval = _max_updates_within_interval;
361}
362
363
364/**
365 * Getter for member Max_updates_per_timeout.
366 * @return Value of Max_updates_per_timeout.
367 */
368int Service::get_max_updates_within_interval() const
369{
370 return MaxUpdatesWithinInterval;
371}
c3dea5dc
BS
372
373
374/**
4553e833
BS
375 * Setter for member MaxEqualUpdatesInSuccession.
376 * @param _max_equal_updates_in_succession Value to set MaxEqualUpdatesInSuccession to.
377 */
378void Service::set_max_equal_updates_in_succession(const int _max_equal_updates_in_succession)
379{
380 MaxEqualUpdatesInSuccession = _max_equal_updates_in_succession;
381}
382
383
384/**
385 * Getter for member MaxEqualUpdatesInSuccession.
386 * @return Value of MaxEqualUpdatesInSuccession.
387 */
388int Service::get_max_equal_updates_in_succession() const
389{
390 return MaxEqualUpdatesInSuccession;
391}
392
393
394/**
c3dea5dc
BS
395 * Get a unique service identify string
396 * @return A unique service identify string
397 */
398string Service::get_service_name() const
399{
400 string service_name;
401
402 service_name.append(Protocol);
403 service_name.append(" ");
404 service_name.append(Hostname);
405
406 return service_name;
407}
408
409
410/**
411 * Get member DNSCacheTTL
412 * @return DNSCacheTTL
413 */
414int Service::get_dns_cache_ttl() const
415{
416 return DNSCacheTTL;
417}
418
419
420/**
421 * Set member DNSCacheTTL
422 * @param _dns_cache_ttl DNSCacheTTL
423 */
424void Service::set_dns_cache_ttl(const int _dns_cache_ttl)
425{
426 DNSCacheTTL = _dns_cache_ttl;
427}