--[[ Copyright (c) 2016, Vsevolod Stakhov Copyright (c) 2016, Alexey Savelyev Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. ]]-- --[[ Example domains whitelist config: greylist { # Search "example.com" and "mail.example.com" for "mx.out.mail.example.com": whitelist_domains_url = [ "$LOCAL_CONFDIR/local.d/greylist-whitelist-domains.inc", "${CONFDIR}/maillist.inc", "${CONFDIR}/redirectors.inc", "${CONFDIR}/dmarc_whitelist.inc", "${CONFDIR}/spf_dkim_whitelist.inc", "${CONFDIR}/surbl-whitelist.inc", "https://maps.rspamd.com/freemail/free.txt.zst" ]; } Example config for exim users: greylist { action = "greylist"; } --]] if confighelp then return end -- A plugin that implements greylisting using redis local redis_params local whitelisted_ip local whitelist_domains_map = nil local toint =math.ifloor or math.floor local settings = { expire = 86400, -- 1 day by default timeout = 300, -- 5 minutes by default key_prefix = 'rg', -- default hash name max_data_len = 10240, -- default data limit to hash message = 'Try again later', -- default greylisted message symbol = 'GREYLIST', action = 'soft reject', -- default greylisted action ipv4_mask = 19, -- Mask bits for ipv4 ipv6_mask = 64, -- Mask bits for ipv6 report_time = false, -- Tell when greylisting is epired (appended to `message`) } local rspamd_logger = require "rspamd_logger" local rspamd_util = require "rspamd_util" local fun = require "fun" local hash = require "rspamd_cryptobox_hash" local rspamd_lua_utils = require "lua_util" local N = "greylist" local function data_key(task) local cached = task:get_mempool():get_variable("grey_bodyhash") if cached then return cached end local body = task:get_rawbody() if not body then return nil end local len = body:len() if len > settings['max_data_len'] then len = settings['max_data_len'] end local h = hash.create() h:update(body, len) local b32 = settings['key_prefix'] .. 'b' .. h:base32():sub(1, 20) task:get_mempool():set_variable("grey_bodyhash", b32) return b32 end local function envelope_key(task) local cached = task:get_mempool():get_variable("grey_metahash") if cached then return cached end local from = task:get_from('smtp') local h = hash.create() local addr = '<>' if from and from[1] then addr = from[1]['addr'] end h:update(addr) local rcpt = task:get_recipients('smtp') if rcpt then table.sort(rcpt, function(r1, r2) return r1['addr'] < r2['addr'] end) fun.each(function(r) h:update(r['addr']) end, rcpt) end local ip = task:get_ip() if ip and ip:is_valid() then local s if ip:get_version() == 4 then s = tostring(ip:apply_mask(settings['ipv4_mask'])) else s = tostring(ip:apply_mask(settings['ipv6_mask'])) end h:update(s) end local b32 = settings['key_prefix'] .. 'm' .. h:base32():sub(1, 20) task:get_mempool():set_variable("grey_metahash", b32) return b32 end -- Returns pair of booleans: found,greylisted local function check_time(task, tm, type) local t = tonumber(tm) if not t then rspamd_logger.errx(task, 'not a valid number: %s', tm) return false,false end local now = rspamd_util.get_time() if now - t < settings['timeout'] then return true,true else -- We just set variable to pass when in post-filter stage task:get_mempool():set_variable("grey_whitelisted", type) return true,false end end local function greylist_message(task, end_time, why) task:insert_result(settings['symbol'], 0.0, 'greylisted', end_time, why) if rspamd_lua_utils.is_rspamc_or_controller(task) then return end if settings.message_func then task:set_pre_result(settings['action'], settings.message_func(task, end_time)) else local message = settings['message'] if settings.report_time then message = string.format("%s: %s", message, end_time) end task:set_pre_result(settings['action'], message) end task:set_flag('greylisted') end local function greylist_check(task) local ip = task:get_ip() if task:get_user() or (ip and ip:is_local()) then return end if ip and ip:is_valid() and whitelisted_ip then if whitelisted_ip:get_key(ip) then -- Do not check whitelisted ip rspamd_logger.infox(task, 'skip greylisting for whitelisted IP') return end end local body_key = data_key(task) local meta_key = envelope_key(task) local hash_key = body_key .. meta_key local function redis_get_cb(err, data) local ret_body = false local greylisted_body = false local ret_meta = false local greylisted_meta = false if data then if data[1] and type(data[1]) ~= 'userdata' then ret_body,greylisted_body = check_time(task, data[1], 'body') if greylisted_body then local end_time = rspamd_util.time_to_string(rspamd_util.get_time() + settings['timeout']) task:get_mempool():set_variable("grey_greylisted_body", end_time) end end if data[2] and type(data[2]) ~= 'userdata' then if not ret_body or greylisted_body then ret_meta,greylisted_meta = check_time(task, data[2], 'meta') if greylisted_meta then local end_time = rspamd_util.time_to_string(rspamd_util.get_time() + settings['timeout']) task:get_mempool():set_variable("grey_greylisted_meta", end_time) end end end if not ret_body and not ret_meta then local end_time = rspamd_util.time_to_string(rspamd_util.get_time() + settings['timeout']) task:get_mempool():set_variable("grey_greylisted", end_time) elseif greylisted_body and greylisted_meta then local end_time = rspamd_util.time_to_string(rspamd_util.get_time() + settings['timeout']) rspamd_logger.infox(task, 'greylisted until "%s"', end_time) greylist_message(task, end_time, 'too early') end elseif err then rspamd_logger.errx(task, 'got error while getting greylisting keys: %1', err) return end end local ret = rspamd_redis_make_request(task, redis_params, -- connect params hash_key, -- hash key false, -- is write redis_get_cb, --callback 'MGET', -- command {body_key, meta_key} -- arguments ) if not ret then rspamd_logger.errx(task, 'cannot make redis request to check results') end end local function greylist_set(task) local action = task:get_metric_action('default') local ip = task:get_ip() if settings.greylist_min_score then local score = task:get_metric_score('default')[1] if score < settings.greylist_min_score then rspamd_logger.infox(task, 'Score too low - skip greylisting') if action == 'greylist' then -- We are going to accept message rspamd_logger.infox(task, 'Downgrading metric action from "greylist" to "no action"') task:disable_action('greylist') end return end end if task:get_user() or (ip and ip:is_local()) then if action == 'greylist' then -- We are going to accept message rspamd_logger.infox(task, 'Downgrading metric action from "greylist" to "no action"') task:disable_action('greylist') end return end if ip and ip:is_valid() and whitelisted_ip then if whitelisted_ip:get_key(ip) then if action == 'greylist' then -- We are going to accept message rspamd_logger.infox(task, 'Downgrading metric action from "greylist" to "no action"') task:disable_action('greylist') end return end end local is_whitelisted = task:get_mempool():get_variable("grey_whitelisted") local do_greylisting = task:get_mempool():get_variable("grey_greylisted") local do_greylisting_required = task:get_mempool():get_variable("grey_greylisted_required") -- Third and second level domains whitelist if not is_whitelisted and whitelist_domains_map then local hostname = task:get_hostname() if hostname then local domain = rspamd_util.get_tld(hostname) if whitelist_domains_map:get_key(hostname) or (domain and whitelist_domains_map:get_key(domain)) then is_whitelisted = 'meta' rspamd_logger.infox(task, 'skip greylisting for whitelisted domain') end end end if action == 'reject' or not do_greylisting_required and action == 'no action' then return end local body_key = data_key(task) local meta_key = envelope_key(task) local upstream, ret, conn local hash_key = body_key .. meta_key local function redis_set_cb(err) if err then rspamd_logger.errx(task, 'got error %s when setting greylisting record on server %s', err, upstream:get_addr()) end end local is_rspamc = rspamd_lua_utils.is_rspamc_or_controller(task) if is_whitelisted then if action == 'greylist' then -- We are going to accept message rspamd_logger.infox(task, 'Downgrading metric action from "greylist" to "no action"') task:disable_action('greylist') end task:insert_result(settings['symbol'], 0.0, 'pass', is_whitelisted) rspamd_logger.infox(task, 'greylisting pass (%s) until %s', is_whitelisted, rspamd_util.time_to_string(rspamd_util.get_time() + settings['expire'])) if is_rspamc then return end ret,conn,upstream = rspamd_redis_make_request(task, redis_params, -- connect params hash_key, -- hash key true, -- is write redis_set_cb, --callback 'EXPIRE', -- command {body_key, tostring(toint(settings['expire']))} -- arguments ) -- Update greylisting record expire if ret then conn:add_cmd('EXPIRE', { meta_key, tostring(toint(settings['expire'])) }) else rspamd_logger.errx(task, 'got error while connecting to redis') end elseif do_greylisting or do_greylisting_required then if is_rspamc then return end local t = tostring(toint(rspamd_util.get_time())) local end_time = rspamd_util.time_to_string(t + settings['timeout']) rspamd_logger.infox(task, 'greylisted until "%s", new record', end_time) greylist_message(task, end_time, 'new record') -- Create new record ret,conn,upstream = rspamd_redis_make_request(task, redis_params, -- connect params hash_key, -- hash key true, -- is write redis_set_cb, --callback 'SETEX', -- command {body_key, tostring(toint(settings['expire'])), t} -- arguments ) if ret then conn:add_cmd('SETEX', { meta_key, tostring(toint(settings['expire'])), t }) else rspamd_logger.errx(task, 'got error while connecting to redis') end else if action ~= 'no action' and action ~= 'reject' then local grey_res = task:get_mempool():get_variable("grey_greylisted_body") if grey_res then -- We need to delay message, hence set a temporary result rspamd_logger.infox(task, 'greylisting delayed until "%s": body', grey_res) greylist_message(task, grey_res, 'body') else grey_res = task:get_mempool():get_variable("grey_greylisted_meta") if grey_res then greylist_message(task, grey_res, 'meta') end end else task:insert_result(settings['symbol'], 0.0, 'greylisted', 'passed') end end end local opts = rspamd_config:get_all_opt('greylist') if opts then if opts['message_func'] then settings.message_func = assert(load(opts['message_func']))() end for k,v in pairs(opts) do if k ~= 'message_func' then settings[k] = v end end if settings['greylist_min_score'] then settings['greylist_min_score'] = tonumber(settings['greylist_min_score']) else local greylist_threshold = rspamd_config:get_metric_action('greylist') if greylist_threshold then settings['greylist_min_score'] = greylist_threshold end end whitelisted_ip = rspamd_map_add('greylist', 'whitelisted_ip', 'radix', 'Greylist whitelist ip map') whitelist_domains_map = rspamd_map_add('greylist', 'whitelist_domains_url', 'map', 'Greylist whitelist domains map') redis_params = rspamd_parse_redis_server('greylist') if not redis_params then rspamd_logger.infox(rspamd_config, 'no servers are specified, disabling module') rspamd_lua_utils.disable_module(N, "redis") else rspamd_config:register_symbol({ name = 'GREYLIST_SAVE', type = 'postfilter', callback = greylist_set, priority = 6, flags = 'empty', }) rspamd_config:register_symbol({ name = 'GREYLIST_CHECK', type = 'prefilter', callback = greylist_check, priority = 6, flags = 'empty', }) end end