diff options
author | Gerald Carter <jerry@samba.org> | 2007-05-06 21:06:55 +0000 |
---|---|---|
committer | Gerald (Jerry) Carter <jerry@samba.org> | 2007-10-10 12:21:49 -0500 |
commit | 89fd4444aff2c858605f71d46715e3b366e82749 (patch) | |
tree | c11d51e55e5196b2cf3dc07f61d96fd9a512ad65 /source3 | |
parent | c16059f1f0eab31835f577aa4985fd9d70a8982c (diff) | |
download | samba-89fd4444aff2c858605f71d46715e3b366e82749.tar.gz samba-89fd4444aff2c858605f71d46715e3b366e82749.tar.bz2 samba-89fd4444aff2c858605f71d46715e3b366e82749.zip |
r22714: Prevent DNS lookup storms when the DNS servers are unreachable.
Helps when transitioning from offline to online mode.
Note that this is a quick hack and a better solution
would be to start the DNS server's state between processes
(similar to the namecache entries).
(This used to be commit 4f05c6fe26f4abd7ca71eac339fee2ef5e254369)
Diffstat (limited to 'source3')
-rw-r--r-- | source3/libads/dns.c | 49 |
1 files changed, 40 insertions, 9 deletions
diff --git a/source3/libads/dns.c b/source3/libads/dns.c index f16cea2555..e0e696c99c 100644 --- a/source3/libads/dns.c +++ b/source3/libads/dns.c @@ -270,45 +270,76 @@ static int dnssrvcmp( struct dns_rr_srv *a, struct dns_rr_srv *b ) Simple wrapper for a DNS query *********************************************************************/ +#define DNS_FAILED_WAITTIME 30 + static NTSTATUS dns_send_req( TALLOC_CTX *ctx, const char *name, int q_type, uint8 **buf, int *resp_length ) { uint8 *buffer = NULL; size_t buf_len; int resp_len = NS_PACKETSZ; - + static time_t last_dns_check = 0; + static NTSTATUS last_dns_status = NT_STATUS_OK; + time_t now = time(NULL); + + /* Try to prevent bursts of DNS lookups if the server is down */ + + /* Protect against large clock changes */ + + if ( last_dns_check > now ) + last_dns_check = 0; + + /* IF we had a DNS timeout or a bad server and we are still + in the 30 second cache window, just return the previous + status and save the network timeout. */ + + if ( (NT_STATUS_EQUAL(last_dns_status,NT_STATUS_IO_TIMEOUT) || + NT_STATUS_EQUAL(last_dns_status,NT_STATUS_CONNECTION_REFUSED)) && + (last_dns_check+DNS_FAILED_WAITTIME) > now ) + { + DEBUG(10,("last_dns_check: Returning cached status (%s)\n", + nt_errstr(last_dns_status) )); + return last_dns_status; + } + + /* Send the Query */ do { if ( buffer ) TALLOC_FREE( buffer ); buf_len = resp_len * sizeof(uint8); - if (buf_len) { + if (buf_len) { if ( (buffer = TALLOC_ARRAY(ctx, uint8, buf_len)) == NULL ) { DEBUG(0,("ads_dns_lookup_srv: talloc() failed!\n")); - return NT_STATUS_NO_MEMORY; + last_dns_status = NT_STATUS_NO_MEMORY; + last_dns_check = time(NULL); + return last_dns_status; } - } else { - buffer = NULL; } if ( (resp_len = res_query(name, C_IN, q_type, buffer, buf_len)) < 0 ) { DEBUG(3,("ads_dns_lookup_srv: Failed to resolve %s (%s)\n", name, strerror(errno))); TALLOC_FREE( buffer ); + last_dns_status = NT_STATUS_UNSUCCESSFUL; + if (errno == ETIMEDOUT) { - return NT_STATUS_IO_TIMEOUT; + last_dns_status = NT_STATUS_IO_TIMEOUT; } if (errno == ECONNREFUSED) { - return NT_STATUS_CONNECTION_REFUSED; + last_dns_status = NT_STATUS_CONNECTION_REFUSED; } - return NT_STATUS_UNSUCCESSFUL; + last_dns_check = time(NULL); + return last_dns_status; } } while ( buf_len < resp_len && resp_len < MAX_DNS_PACKET_SIZE ); *buf = buffer; *resp_length = resp_len; - return NT_STATUS_OK; + last_dns_check = time(NULL); + last_dns_status = NT_STATUS_OK; + return last_dns_status; } /********************************************************************* |