2023-01-25 09:28:59 +00:00
|
|
|
load "${REPOSITORY_ROOT}/test/helper/setup"
|
|
|
|
load "${REPOSITORY_ROOT}/test/helper/common"
|
|
|
|
|
2023-01-29 13:52:38 +00:00
|
|
|
BATS_TEST_NAME_PREFIX='[Rspamd] '
|
2023-01-25 09:28:59 +00:00
|
|
|
CONTAINER_NAME='dms-test_rspamd'
|
|
|
|
|
|
|
|
function setup_file() {
|
|
|
|
_init_with_defaults
|
|
|
|
|
|
|
|
# Comment for maintainers about `PERMIT_DOCKER=host`:
|
|
|
|
# https://github.com/docker-mailserver/docker-mailserver/pull/2815/files#r991087509
|
|
|
|
local CUSTOM_SETUP_ARGUMENTS=(
|
|
|
|
--env ENABLE_CLAMAV=1
|
|
|
|
--env ENABLE_RSPAMD=1
|
|
|
|
--env ENABLE_OPENDKIM=0
|
|
|
|
--env ENABLE_OPENDMARC=0
|
|
|
|
--env PERMIT_DOCKER=host
|
|
|
|
--env LOG_LEVEL=trace
|
2023-03-18 15:32:48 +00:00
|
|
|
--env MOVE_SPAM_TO_JUNK=1
|
|
|
|
--env RSPAMD_LEARN=1
|
2023-04-11 07:16:57 +00:00
|
|
|
--env RSPAMD_GREYLISTING=1
|
2023-01-25 09:28:59 +00:00
|
|
|
)
|
|
|
|
|
2023-03-18 15:32:48 +00:00
|
|
|
mv "${TEST_TMP_CONFIG}"/rspamd/* "${TEST_TMP_CONFIG}/"
|
2023-01-25 09:28:59 +00:00
|
|
|
_common_container_setup 'CUSTOM_SETUP_ARGUMENTS'
|
|
|
|
|
|
|
|
# wait for ClamAV to be fully setup or we will get errors on the log
|
|
|
|
_repeat_in_container_until_success_or_timeout 60 "${CONTAINER_NAME}" test -e /var/run/clamav/clamd.ctl
|
|
|
|
|
2023-03-04 09:45:43 +00:00
|
|
|
_wait_for_service rspamd-redis
|
2023-01-25 09:28:59 +00:00
|
|
|
_wait_for_service rspamd
|
2023-02-19 11:36:43 +00:00
|
|
|
_wait_for_service clamav
|
2023-01-25 09:28:59 +00:00
|
|
|
_wait_for_service postfix
|
|
|
|
_wait_for_smtp_port_in_container
|
|
|
|
|
|
|
|
# We will send 3 emails: the first one should pass just fine; the second one should
|
|
|
|
# be rejected due to spam; the third one should be rejected due to a virus.
|
2023-03-18 15:32:48 +00:00
|
|
|
export MAIL_ID1=$(_send_email_and_get_id 'email-templates/rspamd-pass')
|
2023-02-24 09:44:18 +00:00
|
|
|
export MAIL_ID2=$(_send_email_and_get_id 'email-templates/rspamd-spam')
|
|
|
|
export MAIL_ID3=$(_send_email_and_get_id 'email-templates/rspamd-virus')
|
2023-03-18 15:32:48 +00:00
|
|
|
export MAIL_ID4=$(_send_email_and_get_id 'email-templates/rspamd-spam-header')
|
2023-02-19 11:36:43 +00:00
|
|
|
|
2023-03-18 15:32:48 +00:00
|
|
|
for ID in MAIL_ID{1,2,3,4}
|
|
|
|
do
|
|
|
|
[[ -n ${!ID} ]] || { echo "${ID} is empty - aborting!" ; return 1 ; }
|
|
|
|
done
|
2023-01-25 09:28:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function teardown_file() { _default_teardown ; }
|
|
|
|
|
|
|
|
@test "Postfix's main.cf was adjusted" {
|
2023-02-26 10:42:14 +00:00
|
|
|
_run_in_container grep -F 'smtpd_milters = $rspamd_milter' /etc/postfix/main.cf
|
2023-01-25 09:28:59 +00:00
|
|
|
assert_success
|
2023-03-18 15:32:48 +00:00
|
|
|
_run_in_container postconf rspamd_milter
|
|
|
|
assert_success
|
|
|
|
assert_output 'rspamd_milter = inet:localhost:11332'
|
2023-01-25 09:28:59 +00:00
|
|
|
}
|
|
|
|
|
2023-02-19 11:36:43 +00:00
|
|
|
@test 'logs exist and contains proper content' {
|
2023-01-29 13:52:38 +00:00
|
|
|
_service_log_should_contain_string 'rspamd' 'rspamd .* is loading configuration'
|
|
|
|
_service_log_should_contain_string 'rspamd' 'lua module clickhouse is disabled in the configuration'
|
|
|
|
_service_log_should_contain_string 'rspamd' 'lua module elastic is disabled in the configuration'
|
2023-01-30 07:23:58 +00:00
|
|
|
_service_log_should_contain_string 'rspamd' 'lua module neural is disabled in the configuration'
|
2023-01-29 13:52:38 +00:00
|
|
|
_service_log_should_contain_string 'rspamd' 'lua module reputation is disabled in the configuration'
|
|
|
|
_service_log_should_contain_string 'rspamd' 'lua module spamassassin is disabled in the configuration'
|
|
|
|
_service_log_should_contain_string 'rspamd' 'lua module url_redirector is disabled in the configuration'
|
|
|
|
_service_log_should_contain_string 'rspamd' 'lua module metric_exporter is disabled in the configuration'
|
2023-01-25 09:28:59 +00:00
|
|
|
}
|
|
|
|
|
2023-02-19 11:36:43 +00:00
|
|
|
@test 'normal mail passes fine' {
|
2023-01-29 13:52:38 +00:00
|
|
|
_service_log_should_contain_string 'rspamd' 'F \(no action\)'
|
2023-01-25 09:28:59 +00:00
|
|
|
|
2023-01-29 13:52:38 +00:00
|
|
|
_print_mail_log_for_id "${MAIL_ID1}"
|
2023-01-25 09:28:59 +00:00
|
|
|
assert_output --partial "stored mail into mailbox 'INBOX'"
|
2023-03-18 15:32:48 +00:00
|
|
|
|
|
|
|
_count_files_in_directory_in_container /var/mail/localhost.localdomain/user1/new/ 1
|
2023-01-25 09:28:59 +00:00
|
|
|
}
|
|
|
|
|
2023-02-19 11:36:43 +00:00
|
|
|
@test 'detects and rejects spam' {
|
2023-01-29 13:52:38 +00:00
|
|
|
_service_log_should_contain_string 'rspamd' 'S \(reject\)'
|
|
|
|
_service_log_should_contain_string 'rspamd' 'reject "Gtube pattern"'
|
2023-01-25 09:28:59 +00:00
|
|
|
|
2023-01-29 13:52:38 +00:00
|
|
|
_print_mail_log_for_id "${MAIL_ID2}"
|
2023-01-25 09:28:59 +00:00
|
|
|
assert_output --partial 'milter-reject'
|
|
|
|
assert_output --partial '5.7.1 Gtube pattern'
|
2023-03-18 15:32:48 +00:00
|
|
|
|
|
|
|
_count_files_in_directory_in_container /var/mail/localhost.localdomain/user1/new/ 1
|
2023-01-25 09:28:59 +00:00
|
|
|
}
|
|
|
|
|
2023-02-19 11:36:43 +00:00
|
|
|
@test 'detects and rejects virus' {
|
2023-01-29 13:52:38 +00:00
|
|
|
_service_log_should_contain_string 'rspamd' 'T \(reject\)'
|
|
|
|
_service_log_should_contain_string 'rspamd' 'reject "ClamAV FOUND VIRUS "Eicar-Signature"'
|
2023-01-25 09:28:59 +00:00
|
|
|
|
2023-01-29 13:52:38 +00:00
|
|
|
_print_mail_log_for_id "${MAIL_ID3}"
|
2023-01-25 09:28:59 +00:00
|
|
|
assert_output --partial 'milter-reject'
|
|
|
|
assert_output --partial '5.7.1 ClamAV FOUND VIRUS "Eicar-Signature"'
|
|
|
|
refute_output --partial "stored mail into mailbox 'INBOX'"
|
2023-03-18 15:32:48 +00:00
|
|
|
|
|
|
|
_count_files_in_directory_in_container /var/mail/localhost.localdomain/user1/new/ 1
|
2023-01-25 09:28:59 +00:00
|
|
|
}
|
2023-02-19 11:36:43 +00:00
|
|
|
|
|
|
|
@test 'custom commands work correctly' {
|
|
|
|
# check `testmodule1` which should be disabled
|
|
|
|
local MODULE_PATH='/etc/rspamd/override.d/testmodule1.conf'
|
|
|
|
_run_in_container_bash "[[ -f ${MODULE_PATH} ]]"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F '# documentation: https://rspamd.com/doc/modules/testmodule1.html' "${MODULE_PATH}"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F 'enabled = false;' "${MODULE_PATH}"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F 'someoption = somevalue;' "${MODULE_PATH}"
|
|
|
|
assert_failure
|
|
|
|
|
|
|
|
# check `testmodule2` which should be enabled and it should have extra options set
|
|
|
|
MODULE_PATH='/etc/rspamd/override.d/testmodule2.conf'
|
|
|
|
_run_in_container_bash "[[ -f ${MODULE_PATH} ]]"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F '# documentation: https://rspamd.com/doc/modules/testmodule2.html' "${MODULE_PATH}"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F 'enabled = true;' "${MODULE_PATH}"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F 'someoption = somevalue;' "${MODULE_PATH}"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F 'anotheroption = whatAvaLue;' "${MODULE_PATH}"
|
|
|
|
assert_success
|
|
|
|
|
|
|
|
# check whether writing the same option twice overwrites the first value in `testmodule3`
|
|
|
|
MODULE_PATH='/etc/rspamd/override.d/testmodule3.conf'
|
|
|
|
_run_in_container grep -F 'someoption = somevalue;' "${MODULE_PATH}"
|
|
|
|
assert_failure
|
|
|
|
_run_in_container grep -F 'someoption = somevalue2;' "${MODULE_PATH}"
|
|
|
|
assert_success
|
|
|
|
|
|
|
|
# check whether adding a single line writes the line properly in `testmodule4.something`
|
|
|
|
MODULE_PATH='/etc/rspamd/override.d/testmodule4.something'
|
|
|
|
_run_in_container_bash "[[ -f ${MODULE_PATH} ]]"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F 'some very long line with "weird $charact"ers' "${MODULE_PATH}"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F 'and! ano. ther &line' "${MODULE_PATH}"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F '# some comment' "${MODULE_PATH}"
|
|
|
|
assert_success
|
|
|
|
|
|
|
|
# check whether spaces in front of options are handles properly in `testmodule_complicated`
|
|
|
|
MODULE_PATH='/etc/rspamd/override.d/testmodule_complicated.conf'
|
|
|
|
_run_in_container_bash "[[ -f ${MODULE_PATH} ]]"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F ' anOption = anotherValue;' "${MODULE_PATH}"
|
|
|
|
|
|
|
|
# check whether controller option was written properly
|
|
|
|
MODULE_PATH='/etc/rspamd/override.d/worker-controller.inc'
|
|
|
|
_run_in_container_bash "[[ -f ${MODULE_PATH} ]]"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F 'someOption = someValue42;' "${MODULE_PATH}"
|
|
|
|
assert_success
|
|
|
|
|
|
|
|
# check whether controller option was written properly
|
|
|
|
MODULE_PATH='/etc/rspamd/override.d/worker-proxy.inc'
|
|
|
|
_run_in_container_bash "[[ -f ${MODULE_PATH} ]]"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F 'abcdefg71 = RAAAANdooM;' "${MODULE_PATH}"
|
|
|
|
assert_success
|
|
|
|
|
|
|
|
# check whether basic options are written properly
|
|
|
|
MODULE_PATH='/etc/rspamd/override.d/options.inc'
|
|
|
|
_run_in_container_bash "[[ -f ${MODULE_PATH} ]]"
|
|
|
|
assert_success
|
|
|
|
_run_in_container grep -F 'OhMy = "PraiseBeLinters !";' "${MODULE_PATH}"
|
|
|
|
assert_success
|
|
|
|
}
|
2023-03-18 15:32:48 +00:00
|
|
|
|
|
|
|
@test 'Check MOVE_SPAM_TO_JUNK works for Rspamd' {
|
|
|
|
_run_in_container_bash '[[ -f /usr/lib/dovecot/sieve-global/after/spam_to_junk.sieve ]]'
|
|
|
|
assert_success
|
|
|
|
_run_in_container_bash '[[ -f /usr/lib/dovecot/sieve-global/after/spam_to_junk.svbin ]]'
|
|
|
|
assert_success
|
|
|
|
|
|
|
|
_service_log_should_contain_string 'rspamd' 'S \(add header\)'
|
|
|
|
_service_log_should_contain_string 'rspamd' 'add header "Gtube pattern"'
|
|
|
|
|
|
|
|
_print_mail_log_for_id "${MAIL_ID4}"
|
|
|
|
assert_output --partial "fileinto action: stored mail into mailbox 'Junk'"
|
|
|
|
|
|
|
|
_count_files_in_directory_in_container /var/mail/localhost.localdomain/user1/new/ 1
|
|
|
|
_count_files_in_directory_in_container /var/mail/localhost.localdomain/user1/.Junk/new/ 1
|
|
|
|
}
|
|
|
|
|
|
|
|
@test 'Check RSPAMD_LEARN works' {
|
|
|
|
for FILE in learn-{ham,spam}.{sieve,svbin}
|
|
|
|
do
|
|
|
|
_run_in_container_bash "[[ -f /usr/lib/dovecot/sieve-pipe/${FILE} ]]"
|
|
|
|
assert_success
|
|
|
|
done
|
|
|
|
|
|
|
|
_run_in_container grep 'mail_plugins.*imap_sieve' /etc/dovecot/conf.d/20-imap.conf
|
|
|
|
local SIEVE_CONFIG_FILE='/etc/dovecot/conf.d/90-sieve.conf'
|
|
|
|
_run_in_container grep 'sieve_plugins.*sieve_imapsieve' "${SIEVE_CONFIG_FILE}"
|
|
|
|
_run_in_container grep 'sieve_global_extensions.*\+vnd\.dovecot\.pipe' "${SIEVE_CONFIG_FILE}"
|
|
|
|
_run_in_container grep -F 'sieve_pipe_bin_dir = /usr/lib/dovecot/sieve-pipe' "${SIEVE_CONFIG_FILE}"
|
|
|
|
|
|
|
|
# Move an email to the "Junk" folder from "INBOX"; the first email we
|
|
|
|
# sent should pass fine, hence we can now move it
|
|
|
|
_send_email 'nc_templates/rspamd_imap_move_to_junk' '0.0.0.0 143'
|
|
|
|
sleep 1 # wait for the transaction to finish
|
|
|
|
|
|
|
|
local MOVE_TO_JUNK_LINES=(
|
|
|
|
'imapsieve: mailbox Junk: MOVE event'
|
|
|
|
'imapsieve: Matched static mailbox rule [1]'
|
|
|
|
"sieve: file storage: script: Opened script \`learn-spam'"
|
|
|
|
'sieve: file storage: Using Sieve script path: /usr/lib/dovecot/sieve-pipe/learn-spam.sieve'
|
|
|
|
"sieve: Executing script from \`/usr/lib/dovecot/sieve-pipe/learn-spam.svbin'"
|
|
|
|
"Finished running script \`/usr/lib/dovecot/sieve-pipe/learn-spam.svbin'"
|
|
|
|
'sieve: action pipe: running program: rspamc'
|
|
|
|
"pipe action: piped message to program \`rspamc'"
|
|
|
|
"left message in mailbox 'Junk'"
|
|
|
|
)
|
|
|
|
|
|
|
|
_run_in_container cat /var/log/mail/mail.log
|
|
|
|
assert_success
|
|
|
|
for LINE in "${MOVE_TO_JUNK_LINES[@]}"
|
|
|
|
do
|
|
|
|
assert_output --partial "${LINE}"
|
|
|
|
done
|
|
|
|
|
|
|
|
# Move an email to the "INBOX" folder from "Junk"; there should be two mails
|
|
|
|
# in the "Junk" folder
|
|
|
|
_send_email 'nc_templates/rspamd_imap_move_to_inbox' '0.0.0.0 143'
|
|
|
|
sleep 1 # wait for the transaction to finish
|
|
|
|
|
|
|
|
local MOVE_TO_JUNK_LINES=(
|
|
|
|
'imapsieve: Matched static mailbox rule [2]'
|
|
|
|
"sieve: file storage: script: Opened script \`learn-ham'"
|
|
|
|
'sieve: file storage: Using Sieve script path: /usr/lib/dovecot/sieve-pipe/learn-ham.sieve'
|
|
|
|
"sieve: Executing script from \`/usr/lib/dovecot/sieve-pipe/learn-ham.svbin'"
|
|
|
|
"Finished running script \`/usr/lib/dovecot/sieve-pipe/learn-ham.svbin'"
|
|
|
|
"left message in mailbox 'INBOX'"
|
|
|
|
)
|
|
|
|
|
|
|
|
_run_in_container cat /var/log/mail/mail.log
|
|
|
|
assert_success
|
|
|
|
for LINE in "${MOVE_TO_JUNK_LINES[@]}"
|
|
|
|
do
|
|
|
|
assert_output --partial "${LINE}"
|
|
|
|
done
|
|
|
|
}
|
2023-04-11 07:16:57 +00:00
|
|
|
|
|
|
|
@test 'Check greylisting is enabled' {
|
|
|
|
_run_in_container grep 'enabled = true;' /etc/rspamd/local.d/greylist.conf
|
|
|
|
assert_success
|
|
|
|
_run_in_container rspamadm configdump greylist
|
|
|
|
assert_success
|
|
|
|
assert_output --partial 'enabled = true;'
|
|
|
|
}
|