diff options
author | Vsevolod Stakhov <vsevolod@rambler-co.ru> | 2011-03-23 17:08:58 +0300 |
---|---|---|
committer | Vsevolod Stakhov <vsevolod@rambler-co.ru> | 2011-03-23 17:08:58 +0300 |
commit | 689855e2bb5a1c911bab1f71da9aa963a8da62a2 (patch) | |
tree | be0ceaeb9aa5107ef8807625a06a1ff400e54994 | |
parent | 4aa3b704739e3398402fbbef1216ae588ff9be2e (diff) | |
download | rspamd-689855e2bb5a1c911bab1f71da9aa963a8da62a2.tar.gz rspamd-689855e2bb5a1c911bab1f71da9aa963a8da62a2.zip |
* Add throttling detection mechanic for dns resolver
* Improve phishing module adding ability to define 'strict' phishing domains
-rw-r--r-- | src/cfg_file.h | 2 | ||||
-rw-r--r-- | src/cfg_utils.c | 4 | ||||
-rw-r--r-- | src/cfg_xml.c | 12 | ||||
-rw-r--r-- | src/dns.c | 43 | ||||
-rw-r--r-- | src/dns.h | 5 | ||||
-rw-r--r-- | src/plugins/lua/phishing.lua | 55 |
6 files changed, 111 insertions, 10 deletions
diff --git a/src/cfg_file.h b/src/cfg_file.h index 3a9a7d12e..6fb4065e5 100644 --- a/src/cfg_file.h +++ b/src/cfg_file.h @@ -320,6 +320,8 @@ struct config_file { guint32 dns_timeout; /**< timeout in milliseconds for waiting for dns reply */ guint32 dns_retransmits; /**< maximum retransmits count */ + guint32 dns_throttling_errors; /**< maximum errors for starting resolver throttling */ + guint32 dns_throttling_time; /**< time in seconds for DNS throttling */ GList *nameservers; /**< list of nameservers or NULL to parse resolv.conf */ }; diff --git a/src/cfg_utils.c b/src/cfg_utils.c index f4cf48529..9acb442b5 100644 --- a/src/cfg_utils.c +++ b/src/cfg_utils.c @@ -170,7 +170,9 @@ init_defaults (struct config_file *cfg) cfg->dns_timeout = 1000; cfg->dns_retransmits = 5; - + /* After 20 errors do throttling for 10 seconds */ + cfg->dns_throttling_errors = 20; + cfg->dns_throttling_time = 10000; cfg->max_statfile_size = DEFAULT_STATFILE_SIZE; cfg->modules_opts = g_hash_table_new (g_str_hash, g_str_equal); diff --git a/src/cfg_xml.c b/src/cfg_xml.c index 2bc83e109..43b2e0441 100644 --- a/src/cfg_xml.c +++ b/src/cfg_xml.c @@ -184,6 +184,18 @@ static struct xml_parser_rule grammar[] = { G_STRUCT_OFFSET (struct config_file, dns_retransmits), NULL }, + { + "dns_throttling_errors", + xml_handle_uint32, + G_STRUCT_OFFSET (struct config_file, dns_throttling_errors), + NULL + }, + { + "dns_throttling_time", + xml_handle_seconds, + G_STRUCT_OFFSET (struct config_file, dns_throttling_time), + NULL + }, NULL_ATTR }, NULL_DEF_ATTR @@ -546,7 +546,7 @@ send_dns_request (struct rspamd_dns_request *req) } else { msg_err ("send failed: %s for server %s", strerror (errno), req->server->name); - upstream_fail (&req->server->up, time (NULL)); + upstream_fail (&req->server->up, req->time); return -1; } } @@ -964,6 +964,28 @@ dns_parse_reply (guint8 *in, gint r, struct rspamd_dns_resolver *resolver, struc } static void +dns_throttling_cb (gint fd, short what, void *arg) +{ + struct rspamd_dns_resolver *resolver = arg; + + resolver->throttling = FALSE; + resolver->errors = 0; + msg_info ("stop DNS throttling after %d seconds", (int)resolver->throttling_time.tv_sec); +} + +static void +dns_check_throttling (struct rspamd_dns_resolver *resolver) +{ + if (resolver->errors > resolver->max_errors) { + msg_info ("starting DNS throttling after %ud errors", resolver->errors); + /* Init throttling timeout */ + resolver->throttling = TRUE; + evtimer_set (&resolver->throttling_event, dns_throttling_cb, resolver); + event_add (&resolver->throttling_event, &resolver->throttling_time); + } +} + +static void dns_read_cb (gint fd, short what, void *arg) { struct rspamd_dns_resolver *resolver = arg; @@ -978,6 +1000,10 @@ dns_read_cb (gint fd, short what, void *arg) r = read (fd, in, sizeof (in)); if (r > sizeof (struct dns_header) + sizeof (struct dns_query)) { if ((rep = dns_parse_reply (in, r, resolver, &req)) != NULL) { + /* Decrease errors count */ + if (rep->request->resolver->errors > 0) { + rep->request->resolver->errors --; + } upstream_ok (&rep->request->server->up, rep->request->time); rep->request->func (rep, rep->request->arg); } @@ -1001,12 +1027,14 @@ dns_timer_cb (gint fd, short what, void *arg) upstream_fail (&rep->request->server->up, rep->request->time); remove_normal_event (req->session, dns_fin_cb, req); req->func (rep, req->arg); + dns_check_throttling (req->resolver); + req->resolver->errors ++; return; } /* Select other server */ req->server = (struct rspamd_dns_server *)get_upstream_round_robin (req->resolver->servers, req->resolver->servers_num, sizeof (struct rspamd_dns_server), - time (NULL), DEFAULT_UPSTREAM_ERROR_TIME, DEFAULT_UPSTREAM_DEAD_TIME, DEFAULT_UPSTREAM_MAXERRORS); + req->time, DEFAULT_UPSTREAM_ERROR_TIME, DEFAULT_UPSTREAM_DEAD_TIME, DEFAULT_UPSTREAM_MAXERRORS); if (req->server == NULL) { rep = memory_pool_alloc0 (req->pool, sizeof (struct rspamd_dns_reply)); rep->request = req; @@ -1065,7 +1093,8 @@ dns_retransmit_handler (gint fd, short what, void *arg) rep->code = DNS_RC_SERVFAIL; upstream_fail (&rep->request->server->up, rep->request->time); req->func (rep, req->arg); - + req->resolver->errors ++; + dns_check_throttling (req->resolver); return; } r = send_dns_request (req); @@ -1101,6 +1130,11 @@ make_dns_request (struct rspamd_dns_resolver *resolver, const gchar *name, *service, *proto; gint r; + /* Check throttling */ + if (resolver->throttling) { + return FALSE; + } + req = memory_pool_alloc (pool, sizeof (struct rspamd_dns_request)); req->pool = pool; req->session = session; @@ -1248,6 +1282,9 @@ dns_resolver_init (struct config_file *cfg) new->static_pool = cfg->cfg_pool; new->request_timeout = cfg->dns_timeout; new->max_retransmits = cfg->dns_retransmits; + new->max_errors = cfg->dns_throttling_errors; + new->throttling_time.tv_sec = cfg->dns_throttling_time / 1000; + new->throttling_time.tv_usec = (cfg->dns_throttling_time - new->throttling_time.tv_sec * 1000) * 1000; if (cfg->nameservers == NULL) { /* Parse resolv.conf */ @@ -49,7 +49,12 @@ struct rspamd_dns_resolver { struct dns_k_permutor *permutor; /**< permutor for randomizing request id */ guint request_timeout; guint max_retransmits; + guint max_errors; memory_pool_t *static_pool; /**< permament pool (cfg_pool) */ + gboolean throttling; /**< dns servers are busy */ + guint errors; /**< resolver errors */ + struct timeval throttling_time; /**< throttling time */ + struct event throttling_event; /**< throttling event */ }; struct dns_header; diff --git a/src/plugins/lua/phishing.lua b/src/plugins/lua/phishing.lua index 115688750..1e648768a 100644 --- a/src/plugins/lua/phishing.lua +++ b/src/plugins/lua/phishing.lua @@ -3,6 +3,7 @@ -- local symbol = 'PHISHED_URL' local domains = nil +local strict_domains = {} function phishing_cb (task) local urls = task:get_urls(); @@ -10,17 +11,31 @@ function phishing_cb (task) if urls then for _,url in ipairs(urls) do if url:is_phished() then + local purl = url:get_phished() + if table.maxn(strict_domains) > 0 then + local _,_,tld = string.find(purl:get_host(), '([a-zA-Z0-9%-]+\.[a-zA-Z0-9%-]+)$') + local found = false + if tld then + for _,rule in ipairs(strict_domains) do + if rule['map']:get_key(tld) then + task:insert_result(rule['symbol'], 1, purl:get_host()) + found = true + end + end + if found then + return + end + end + end if domains then - local _,_,tld = string.find(url:get_phished():get_host(), '([a-zA-Z0-9%-]+\.[a-zA-Z0-9%-]+)$') + local _,_,tld = string.find(purl:get_host(), '([a-zA-Z0-9%-]+\.[a-zA-Z0-9%-]+)$') if tld then if domains:get_key(tld) then - if url:is_phished() then - task:insert_result(symbol, 1, url:get_host()) - end + task:insert_result(symbol, 1, purl:get_host()) end end else - task:insert_result(symbol, 1, url:get_phished():get_host()) + task:insert_result(symbol, 1, purl:get_host()) end end end @@ -32,6 +47,7 @@ if type(rspamd_config.get_api_version) ~= 'nil' then if rspamd_config:get_api_version() >= 1 then rspamd_config:register_module_option('phishing', 'symbol', 'string') rspamd_config:register_module_option('phishing', 'domains', 'map') + rspamd_config:register_module_option('phishing', 'strict_domains', 'string') end end @@ -43,8 +59,35 @@ if opts then -- Register symbol's callback rspamd_config:register_symbol(symbol, 1.0, 'phishing_cb') end - if opts['domains'] then + if opts['domains'] and type(opt['domains']) == 'string' then domains = rspamd_config:add_hash_map (opts['domains']) end + if opts['strict_domains'] then + local sd = {} + if type(opts['strict_domains']) == 'table' then + sd = opts['strict_domains'] + else + sd[1] = opts['strict_domains'] + end + for _,d in ipairs(sd) do + local s, _ = string.find(d, ':') + if s then + local sym = string.sub(d, s + 1, -1) + local map = string.sub(d, 1, s - 1) + if type(rspamd_config.get_api_version) ~= 'nil' then + rspamd_config:register_virtual_symbol(sym, 1) + end + local rmap = rspamd_config:add_hash_map (map) + if rmap then + local rule = {symbol = sym, map = rmap} + table.insert(strict_domains, rule) + else + rspamd_logger.info('cannot add map: ' .. map .. ' for symbol: ' .. sym) + end + else + rspamd_logger.info('strict_domains option must be in format <map>:<symbol>') + end + end + end -- If no symbol defined, do not register this module end |