load "${REPOSITORY_ROOT}/test/helper/setup" load "${REPOSITORY_ROOT}/test/helper/common" BATS_TEST_NAME_PREFIX='[LDAP] ' CONTAINER1_NAME='dms-test_ldap' CONTAINER2_NAME='dms-test_ldap_provider' # Single test-case specific containers: CONTAINER3_NAME='dms-test_ldap_custom-config' function setup_file() { export DMS_TEST_NETWORK='test-network-ldap' export DMS_DOMAIN='example.test' export FQDN_MAIL="mail.${DMS_DOMAIN}" export FQDN_LDAP="ldap.${DMS_DOMAIN}" # LDAP is provisioned with two domains (via `.ldif` files) unrelated to the FQDN of DMS: export FQDN_LOCALHOST_A='localhost.localdomain' export FQDN_LOCALHOST_B='localhost.otherdomain' # Link the test containers to separate network: # NOTE: If the network already exists, test will fail to start. docker network create "${DMS_TEST_NETWORK}" # Setup local openldap service: docker run --rm -d --name "${CONTAINER2_NAME}" \ --env LDAP_ADMIN_PASSWORD=admin \ --env LDAP_ROOT='dc=example,dc=test' \ --env LDAP_PORT_NUMBER=389 \ --env LDAP_SKIP_DEFAULT_TREE=yes \ --volume "${REPOSITORY_ROOT}/test/config/ldap/openldap/ldifs/:/ldifs/:ro" \ --volume "${REPOSITORY_ROOT}/test/config/ldap/openldap/schemas/:/schemas/:ro" \ --hostname "${FQDN_LDAP}" \ --network "${DMS_TEST_NETWORK}" \ bitnami/openldap:latest _run_until_success_or_timeout 20 sh -c "docker logs ${CONTAINER2_NAME} 2>&1 | grep 'LDAP setup finished'" # # Setup DMS container # # LDAP filter queries explained. # NOTE: All LDAP configs for Postfix (with the exception of `ldap-senders.cf`), return the `mail` attribute value of matched results. # This is through the config key `result_attribute`, which the ENV substitution feature can only replace across all configs, not selectively like `query_filter`. # NOTE: The queries below rely specifically upon attributes and classes defined by the schema `postfix-book.ldif`. These are not compatible with all LDAP setups. # `mailAlias`` is supported by both classes provided from the schema `postfix-book.ldif`, but `mailEnabled` is only available to `PostfixBookMailAccount` class: local QUERY_ALIAS='(&(mailAlias=%s) (| (objectClass=PostfixBookMailForward) (&(objectClass=PostfixBookMailAccount)(mailEnabled=TRUE)) ))' # Postfix does domain lookups with the domain of the recipient to check if DMS manages the mail domain. # For this lookup `%s` only represents the domain, not a full email address. Hence the match pattern using a wildcard prefix `*@`. # For a breakdown, see QUERY_SENDERS comment. # NOTE: Although `result_attribute = mail` will return each accounts full email address, Postfix will only compare to domain-part. local QUERY_DOMAIN='(| (& (|(mail=*@%s) (mailAlias=*@%s) (mailGroupMember=*@%s)) (&(objectClass=PostfixBookMailAccount)(mailEnabled=TRUE)) ) (&(mailAlias=*@%s)(objectClass=PostfixBookMailForward)) )' # Simple queries for a single attribute that additionally requires `mailEnabled=TRUE` from the `PostfixBookMailAccount` class: # NOTE: `mail` attribute is not unique to `PostfixBookMailAccount`. The `mailEnabled` attribute is to further control valid mail accounts. # TODO: For tests, since `mailEnabled` is not relevant (always configured as TRUE currently), # a simpler query like `mail=%s` or `mailGroupMember=%s` would be sufficient. The additional constraints could be covered in our docs instead. local QUERY_GROUP='(&(mailGroupMember=%s) (&(objectClass=PostfixBookMailAccount)(mailEnabled=TRUE)) )' local QUERY_USER='(&(mail=%s) (&(objectClass=PostfixBookMailAccount)(mailEnabled=TRUE)) )' # Given the sender address `%s` from Postfix, query LDAP for accounts that meet the search filter, # the `result_attribute` is `mail` + `uid` (`userID`) attributes for login names that are authorized to use that sender address. # One of the result values returned must match the login name of the user trying to send mail with that sender address. # # Filter: Search for accounts that meet any of the following conditions: # - Has any attribute (`mail`, `mailAlias`, `mailGroupMember`) with a value of `%s`, AND is of class `PostfixBookMailAccount` with `mailEnabled=TRUE` attribute set. # - Has attribute `mailAlias` with value of `%s` AND is of class PostfixBookMailForward # - Has the attribute `userID` with value `some.user.id` local QUERY_SENDERS='(| (& (|(mail=%s) (mailAlias=%s) (mailGroupMember=%s)) (&(objectClass=PostfixBookMailAccount)(mailEnabled=TRUE)) ) (&(mailAlias=%s)(objectClass=PostfixBookMailForward)) (userID=some.user.id))' # NOTE: The remaining queries below have been left as they were instead of rewritten, `mailEnabled` has no associated class requirement, nor does the class requirement ensure `mailEnabled=TRUE`. # When using SASLAuthd for login auth (only valid to Postfix in DMS), # Postfix will pass on the login username and SASLAuthd will use it's backend to do a lookup. # The current default is `uniqueIdentifier=%u`, but `%u` is inaccurate currently with LDAP backend # as SASLAuthd will ignore the domain-part received (if any) and forward only the local-part as the query. # TODO: Fix this by having supervisor start the service with `-r` option like it does `rimap` backend. # NOTE: `%u` (full login with realm/domain-part) is presently equivalent to `%U` (local-part) only, # As the `userID` is not a mail address, we ensure any domain-part is ignored, as a login name is not # required to match the mail accounts actual `mail` attribute (nor the local-part), they are distinct. # TODO: Docs should better cover this difference, as it does confuse some users of DMS (and past contributors to our tests..). local SASLAUTHD_QUERY='(&(userID=%U)(mailEnabled=TRUE))' # Dovecot is configured to lookup a user account by their login name (`userID` in this case, but it could be any attribute like `mail`). # Dovecot syntax token `%n` is the local-part of the full email address supplied as the login name. There must be a unique match on `userID` (which there will be as each account is configured via LDIF to use it in their DN) # NOTE: We already have a constraint on the LDAP tree to search (`LDAP_SEARCH_BASE`), if all objects in that subtree use `PostfixBookMailAccount` class then there is no benefit in the extra constraint. # TODO: For tests, that additional constraint is meaningless. We can detail it in our docs instead and just use `userID=%n`. local DOVECOT_QUERY_PASS='(&(userID=%n)(objectClass=PostfixBookMailAccount))' local DOVECOT_QUERY_USER='(&(userID=%n)(objectClass=PostfixBookMailAccount))' local ENV_LDAP_CONFIG=( --env ACCOUNT_PROVISIONER=LDAP # Common LDAP ENV: # NOTE: `scripts/startup/setup.d/ldap.sh:_setup_ldap()` uses `_replace_by_env_in_file()` to configure settings (stripping `DOVECOT_` / `LDAP_` prefixes): --env LDAP_SERVER_HOST="ldap://${FQDN_LDAP}" --env LDAP_SEARCH_BASE='ou=users,dc=example,dc=test' --env LDAP_START_TLS=no # Credentials needed for read access to LDAP_SEARCH_BASE: --env LDAP_BIND_DN='cn=admin,dc=example,dc=test' --env LDAP_BIND_PW='admin' # Postfix SASL auth provider (SASLAuthd instead of default Dovecot provider): --env ENABLE_SASLAUTHD=1 --env SASLAUTHD_MECHANISMS=ldap --env SASLAUTHD_LDAP_FILTER="${SASLAUTHD_QUERY}" # ENV to configure LDAP configs for Dovecot + Postfix: # Dovecot: --env DOVECOT_PASS_FILTER="${DOVECOT_QUERY_PASS}" --env DOVECOT_USER_FILTER="${DOVECOT_QUERY_USER}" --env DOVECOT_TLS=no # Postfix: --env LDAP_QUERY_FILTER_ALIAS="${QUERY_ALIAS}" --env LDAP_QUERY_FILTER_DOMAIN="${QUERY_DOMAIN}" --env LDAP_QUERY_FILTER_GROUP="${QUERY_GROUP}" --env LDAP_QUERY_FILTER_SENDERS="${QUERY_SENDERS}" --env LDAP_QUERY_FILTER_USER="${QUERY_USER}" ) # Extra ENV needed to support specific test-cases: local ENV_SUPPORT=( # Required for openssl commands to be successul: # NOTE: snakeoil cert is created (for `docker-mailserver.invalid`) via Debian post-install script for Postfix package. # TODO: Use proper TLS cert --env SSL_TYPE='snakeoil' # TODO; All below are questionable value to LDAP tests? --env POSTMASTER_ADDRESS="postmaster@${FQDN_LOCALHOST_A}" # TODO: Only required because LDAP accounts use unrelated domain part. FQDN_LOCALHOST_A / ldif files can be adjusted to FQDN_MAIL --env PFLOGSUMM_TRIGGER=logrotate --env REPORT_RECIPIENT=1 # TODO: Invalid value, should be a recipient address (if not default postmaster), remove? --env SPOOF_PROTECTION=1 ) export CONTAINER_NAME=${CONTAINER1_NAME} local CUSTOM_SETUP_ARGUMENTS=( "${ENV_LDAP_CONFIG[@]}" "${ENV_SUPPORT[@]}" --hostname "${FQDN_MAIL}" --network "${DMS_TEST_NETWORK}" ) _init_with_defaults _common_container_setup 'CUSTOM_SETUP_ARGUMENTS' _wait_for_smtp_port_in_container # Single test-case containers below cannot be defined in a test-case when expanding arrays # defined in `setup()`. Those arrays would need to be hoisted up to the top of the file vars. # Alternatively for ENV overrides, separate `.env` files could be used. Better options # are available once switching to `compose.yaml` in tests. export CONTAINER_NAME=${CONTAINER3_NAME} local CUSTOM_SETUP_ARGUMENTS=( "${ENV_LDAP_CONFIG[@]}" # `hostname` should be unique when connecting containers via network: --hostname "custom-config.${DMS_DOMAIN}" --network "${DMS_TEST_NETWORK}" ) _init_with_defaults # NOTE: `test/config/` has now been duplicated, can move test specific files to host-side `/tmp/docker-mailserver`: mv "${TEST_TMP_CONFIG}/ldap/overrides/"*.cf "${TEST_TMP_CONFIG}/" _common_container_setup 'CUSTOM_SETUP_ARGUMENTS' # Set default implicit container fallback for helpers: export CONTAINER_NAME=${CONTAINER1_NAME} } function teardown_file() { docker rm -f "${CONTAINER1_NAME}" "${CONTAINER2_NAME}" docker network rm "${DMS_TEST_NETWORK}" } # Could optionally call `_default_teardown` in test-cases that have specific containers. # This will otherwise handle it implicitly which is helpful when the test-case hits a failure, # As failure will bail early missing teardown, which then prevents network cleanup. This way is safer: function teardown() { if [[ ${CONTAINER_NAME} != "${CONTAINER1_NAME}" ]] \ && [[ ${CONTAINER_NAME} != "${CONTAINER2_NAME}" ]] then _default_teardown fi } # postfix # NOTE: Each of the 3 user accounts tested below are defined in separate LDIF config files, # Those are bundled into the locally built OpenLDAP Dockerfile. @test "postfix: ldap lookup works correctly" { _should_exist_in_ldap_tables "some.user@${FQDN_LOCALHOST_A}" # Test email receiving from a other domain then the primary domain of the mailserver _should_exist_in_ldap_tables "some.other.user@${FQDN_LOCALHOST_B}" # Should not require `userID` / `uid` to match the local part of `mail` (`.ldif` defined settings): # REF: https://github.com/docker-mailserver/docker-mailserver/pull/642#issuecomment-313916384 # NOTE: This account has no `mailAlias` or `mailGroupMember` defined in it's `.ldif`. local MAIL_ACCOUNT="some.user.email@${FQDN_LOCALHOST_A}" _run_in_container postmap -q "${MAIL_ACCOUNT}" ldap:/etc/postfix/ldap-users.cf assert_success assert_output "${MAIL_ACCOUNT}" } # Custom LDAP config files support: @test "postfix: ldap custom config files copied" { # Use the test-case specific container from `setup()` (change only applies to test-case): export CONTAINER_NAME=${CONTAINER3_NAME} local LDAP_CONFIGS_POSTFIX=( /etc/postfix/ldap-users.cf /etc/postfix/ldap-groups.cf /etc/postfix/ldap-aliases.cf ) for LDAP_CONFIG in "${LDAP_CONFIGS_POSTFIX[@]}"; do _run_in_container grep '# Testconfig for ldap integration' "${LDAP_CONFIG}" assert_success done } @test "postfix: ldap config overwrites success" { local LDAP_SETTINGS_POSTFIX=( "server_host = ldap://${FQDN_LDAP}" 'start_tls = no' 'search_base = ou=users,dc=example,dc=test' 'bind_dn = cn=admin,dc=example,dc=test' ) for LDAP_SETTING in "${LDAP_SETTINGS_POSTFIX[@]}"; do # "${LDAP_SETTING%=*}" is to match only the key portion of the var (helpful for assert_output error messages) # NOTE: `start_tls = no` is a default setting, but the white-space differs when ENV `LDAP_START_TLS` is not set explicitly. _run_in_container grep "${LDAP_SETTING%=*}" /etc/postfix/ldap-users.cf assert_output "${LDAP_SETTING}" assert_success _run_in_container grep "${LDAP_SETTING%=*}" /etc/postfix/ldap-groups.cf assert_output "${LDAP_SETTING}" assert_success _run_in_container grep "${LDAP_SETTING%=*}" /etc/postfix/ldap-aliases.cf assert_output "${LDAP_SETTING}" assert_success done } # dovecot @test "dovecot: ldap imap connection and authentication works" { _nc_wrapper 'auth/imap-ldap-auth.txt' '-w 1 0.0.0.0 143' assert_success } @test "dovecot: ldap mail delivery works" { _should_successfully_deliver_mail_to "some.user@${FQDN_LOCALHOST_A}" "/var/mail/${FQDN_LOCALHOST_A}/some.user/new/" # Should support delivering to a local recipient with a different domain (and disjoint mail location): # NOTE: Mail is delivered to location defined in `.ldif` (an account config setting, either `mailHomeDirectory` or `mailStorageDirectory`). # `some.other.user` has been configured to use a mailbox domain different from it's address domain part, hence the difference here: _should_successfully_deliver_mail_to "some.other.user@${FQDN_LOCALHOST_B}" "/var/mail/${FQDN_LOCALHOST_A}/some.other.user/new/" } @test "dovecot: ldap config overwrites success" { local LDAP_SETTINGS_DOVECOT=( "uris = ldap://${FQDN_LDAP}" 'tls = no' 'base = ou=users,dc=example,dc=test' 'dn = cn=admin,dc=example,dc=test' ) for LDAP_SETTING in "${LDAP_SETTINGS_DOVECOT[@]}"; do _run_in_container grep "${LDAP_SETTING%=*}" /etc/dovecot/dovecot-ldap.conf.ext assert_output "${LDAP_SETTING}" assert_success done } # Requires ENV `POSTMASTER_ADDRESS` # NOTE: Not important to LDAP feature tests? @test "dovecot: postmaster address" { _run_in_container grep "postmaster_address = postmaster@${FQDN_LOCALHOST_A}" /etc/dovecot/conf.d/15-lda.conf assert_success } # NOTE: `target/scripts/startup/setup.d/dovecot.sh` should prevent enabling the quotas feature when using LDAP: @test "dovecot: quota plugin is disabled" { # Dovecot configs have not enabled the quota plugins: _run_in_container grep "\$mail_plugins quota" /etc/dovecot/conf.d/10-mail.conf assert_failure _run_in_container grep "\$mail_plugins imap_quota" /etc/dovecot/conf.d/20-imap.conf assert_failure # Dovecot Quota config only present with disabled extension: _run_in_container_bash '[[ -f /etc/dovecot/conf.d/90-quota.conf ]]' assert_failure _run_in_container_bash '[[ -f /etc/dovecot/conf.d/90-quota.conf.disab ]]' assert_success # Postfix quotas policy service not configured in `main.cf`: _run_in_container postconf smtpd_recipient_restrictions refute_output --partial 'check_policy_service inet:localhost:65265' } @test "saslauthd: sasl ldap authentication works" { _run_in_container testsaslauthd -u some.user -p secret assert_success } # Requires ENV `PFLOGSUMM_TRIGGER=logrotate` @test "pflogsumm delivery" { # Verify default sender is `mailserver-report` when ENV `PFLOGSUMM_SENDER` + `REPORT_SENDER` are unset: # NOTE: Mail is sent from Postfix (configured hostname used as domain part) _run_in_container grep "mailserver-report@${FQDN_MAIL}" /etc/logrotate.d/maillog assert_success # When `LOGROTATE_INTERVAL` is unset, the default should be configured as `weekly`: _run_in_container grep 'weekly' /etc/logrotate.d/maillog assert_success } # ATTENTION: Remaining tests must come after "dovecot: ldap mail delivery works" since the below tests would affect the expected count (by delivering extra mail), # Thus not friendly for running testcases in this file in parallel # Requires ENV `SPOOF_PROTECTION=1` for the expected assert_output @test "spoofing (with LDAP): rejects sender forging" { _wait_for_smtp_port_in_container_to_respond dms-test_ldap _send_email --expect-rejection \ --port 465 -tlsc --auth PLAIN \ --auth-user some.user@localhost.localdomain \ --auth-password secret \ --ehlo mail \ --from ldap@localhost.localdomain \ --data 'auth/ldap-smtp-auth-spoofed.txt' assert_failure assert_output --partial 'Sender address rejected: not owned by user' } @test "spoofing (with LDAP): accepts sending as alias" { _send_email \ --port 465 -tlsc --auth PLAIN \ --auth-user some.user@localhost.localdomain \ --auth-password secret \ --ehlo mail \ --from postmaster@localhost.localdomain \ --to some.user@localhost.localdomain \ --data 'auth/ldap-smtp-auth-spoofed-alias.txt' assert_output --partial 'End data with' } @test "spoofing (with LDAP): uses senders filter" { # skip introduced with #3006, changing port 25 to 465 # Template used has invalid AUTH: https://github.com/docker-mailserver/docker-mailserver/pull/3006#discussion_r1073321432 skip 'TODO: This test seems to have been broken from the start (?)' _send_email --expect-rejection \ --port 465 -tlsc --auth PLAIN \ --auth-user some.user.email@localhost.localdomain \ --auth-password secret \ --ehlo mail \ --from randomspoofedaddress@localhost.localdomain \ --to some.user@localhost.localdomain \ --data 'auth/ldap-smtp-auth-spoofed-sender-with-filter-exception.txt' assert_failure assert_output --partial 'Sender address rejected: not owned by user' } @test "saslauthd: ldap smtp authentication" { _send_email --expect-rejection \ --auth PLAIN \ --auth-user some.user@localhost.localdomain \ --auth-password wrongpassword \ --quit-after AUTH assert_failure assert_output --partial 'Host did not advertise authentication' _send_email \ --port 465 -tlsc \ --auth LOGIN \ --auth-user some.user@localhost.localdomain \ --auth-password secret \ --quit-after AUTH assert_output --partial 'Authentication successful' _send_email \ --port 587 -tls \ --auth PLAIN \ --auth-user some.user@localhost.localdomain \ --auth-password secret \ --quit-after AUTH assert_success assert_output --partial 'Authentication successful' } # # Test helper methods: # function _should_exist_in_ldap_tables() { local MAIL_ACCOUNT=${1:?Mail account is required} local DOMAIN_PART="${MAIL_ACCOUNT#*@}" # Each LDAP config file sets `query_filter` to lookup a key in LDAP (values defined in `.ldif` test files) # `mail` (ldap-users), `mailAlias` (ldap-aliases), `mailGroupMember` (ldap-groups) # `postmap` is queried with the mail account address, and the LDAP service should respond with # `result_attribute` which is the LDAP `mail` value (should match what we'r'e quering `postmap` with) _run_in_container postmap -q "${MAIL_ACCOUNT}" ldap:/etc/postfix/ldap-users.cf assert_success assert_output "${MAIL_ACCOUNT}" # Check which account has the `postmaster` virtual alias: _run_in_container postmap -q "postmaster@${DOMAIN_PART}" ldap:/etc/postfix/ldap-aliases.cf assert_success assert_output "${MAIL_ACCOUNT}" _run_in_container postmap -q "employees@${DOMAIN_PART}" ldap:/etc/postfix/ldap-groups.cf assert_success assert_output "${MAIL_ACCOUNT}" } # NOTE: `test-email.txt` is only used for these two LDAP tests with `sendmail` command. # The file excludes sender/recipient addresses, thus not usable with `_send_email()` helper (`nc` command)? # TODO: Could probably adapt? function _should_successfully_deliver_mail_to() { local SENDER_ADDRESS='user@external.tld' local RECIPIENT_ADDRESS=${1:?Recipient address is required} local MAIL_STORAGE_RECIPIENT=${2:?Recipient storage location is required} local MAIL_TEMPLATE='/tmp/docker-mailserver-test/emails/test-email.txt' _run_in_container_bash "sendmail -f ${SENDER_ADDRESS} ${RECIPIENT_ADDRESS} < ${MAIL_TEMPLATE}" _wait_for_empty_mail_queue_in_container _run_in_container grep -R 'This is a test mail.' "${MAIL_STORAGE_RECIPIENT}" assert_success _should_output_number_of_lines 1 # NOTE: Prevents compatibility for running testcases in parallel (for same container) when the count could become racey: _count_files_in_directory_in_container "${MAIL_STORAGE_RECIPIENT}" 1 }