mirror of
https://github.com/mail-in-a-box/mailinabox.git
synced 2025-04-03 00:07:05 +00:00
add a quota test
This commit is contained in:
parent
8aa76dc5de
commit
2040d4b193
@ -400,3 +400,34 @@ mgmt_assert_admin_login() {
|
|||||||
fi
|
fi
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
mgmt_get_user_quota() {
|
||||||
|
local user="$1"
|
||||||
|
record "[get user $user quota]"
|
||||||
|
mgmt_rest GET "/admin/mail/users/quota?email=$user"
|
||||||
|
local rc=$?
|
||||||
|
# REST_OUTPUT contains json, eg:
|
||||||
|
# { "email": "alice@somedomain.com", "quota": "5000" }
|
||||||
|
if [ $rc -eq 0 ]; then
|
||||||
|
# output to stdout the quota value
|
||||||
|
QUOTA="$(/usr/bin/jq -r ".quota" <<<"$REST_OUTPUT" 2>>$TEST_OF)"
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
record "could not obtain quota member from json using jq"
|
||||||
|
rc=1
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
QUOTA="error"
|
||||||
|
fi
|
||||||
|
return $rc
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
mgmt_set_user_quota() {
|
||||||
|
local user="$1"
|
||||||
|
local quota="$2"
|
||||||
|
record "[set user $user quota to $quota]"
|
||||||
|
mgmt_rest POST "/admin/mail/users/quota" "email=$user" "quota=$quota"
|
||||||
|
local rc=$?
|
||||||
|
return $rc
|
||||||
|
}
|
||||||
|
@ -361,12 +361,123 @@ test_totp() {
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
test_mailbox_quotas() {
|
||||||
|
test_start "mailbox-quotas"
|
||||||
|
|
||||||
|
# create standard user alice
|
||||||
|
local alice="alice@somedomain.com"
|
||||||
|
create_user "$alice" "alice"
|
||||||
|
|
||||||
|
# quota should be unlimited for newly added users
|
||||||
|
if ! mgmt_get_user_quota "$alice"; then
|
||||||
|
test_failure "Unable to get $alice's quota: $REST_ERROR"
|
||||||
|
elif [ "$QUOTA" != "0" -a "$QUOTA" != "unlimited" ]; then
|
||||||
|
test_failure "A newly created user should have unlimited quota"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# get alice's current total number of messages. should be 0 unless
|
||||||
|
# the account was "archived"
|
||||||
|
local count_messages="$(doveadm -f json quota get -u "$alice" | jq -r '.[] | select(.type=="MESSAGE") | .value')"
|
||||||
|
record "$alice currently has $count_messages messages"
|
||||||
|
|
||||||
|
# set alice's quota to a small number
|
||||||
|
local quota_value="5K"
|
||||||
|
if ! mgmt_set_user_quota "$alice" "$quota_value"
|
||||||
|
then
|
||||||
|
test_failure "Unable to set $alice's quota: $REST_ERROR"
|
||||||
|
else
|
||||||
|
# read back the quota - make sure it's what we set
|
||||||
|
if ! mgmt_get_user_quota "$alice" || [ "$QUOTA" != "$quota_value" ]
|
||||||
|
then
|
||||||
|
test_failure "Setting quota failed - expected quota does not match current quota: $REST_OUTPUT $REST_ERROR QUOTA=$QUOTA"
|
||||||
|
|
||||||
|
else
|
||||||
|
record_search "(mail=$alice)"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! have_test_failures; then
|
||||||
|
# send messages large enough to exceed the quota
|
||||||
|
local output
|
||||||
|
local subjects=()
|
||||||
|
local msgidx=0
|
||||||
|
local body="$(python3 -c 'for i in range(0,int(512/4)): print("abc\n", end="")')"
|
||||||
|
local quota_exceeded="no"
|
||||||
|
|
||||||
|
while ! have_test_failures && [ $msgidx -lt 10 ]; do
|
||||||
|
record ""
|
||||||
|
record "[send msg $msgidx]"
|
||||||
|
local subj="msg$msgidx - $(generate_password)"
|
||||||
|
output="$($PYMAIL -smtp-debug -body-from-stdin -no-delete -subj "$subj" $PRIVATE_IP $alice alice <<<"$body" 2>&1)"
|
||||||
|
if ! assert_python_success $? "$output"; then
|
||||||
|
break
|
||||||
|
fi
|
||||||
|
|
||||||
|
# You'd expect that the send would fail when the quota is
|
||||||
|
# exceeded, but it doesn't. Postfix accepts it into it's
|
||||||
|
# queue, then bounces the message back to sender with
|
||||||
|
# delivery status notification (DSN) of 5.2.2 when it
|
||||||
|
# processes the queue.
|
||||||
|
#
|
||||||
|
# The debugging messages (turned on by the -smtp-debug
|
||||||
|
# argument) hold the internal postfix message id, so
|
||||||
|
# extract that, then grep the logs to see if the message
|
||||||
|
# was bounced due to 5.2.2.
|
||||||
|
|
||||||
|
local postid="$(awk '/^data: .* queued as/ { match($0," as "); print substr($0,RSTART+4,10); exit }' <<<"$output" 2>>$TEST_OF)"
|
||||||
|
record "Extracted POSTID=$postid"
|
||||||
|
if [ ! -z "$postid" ]; then
|
||||||
|
/usr/sbin/postqueue -f >>"$TEST_OF" 2>&1
|
||||||
|
flush_logs
|
||||||
|
record "[dovecot and postfix logs for msg $msgidx]"
|
||||||
|
record "logs: $(grep "$postid" /var/log/mail.log)"
|
||||||
|
|
||||||
|
if grep "$postid" /var/log/mail.log | grep "status=bounced" | grep -Fq "5.2.2"; then
|
||||||
|
# success - message was rejected
|
||||||
|
quota_exceeded="yes"
|
||||||
|
break
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
subjects+=( "$subj" )
|
||||||
|
let msgidx+=1
|
||||||
|
# doveadm quota get -u "$alice" >>"$TEST_OF" 2>&1
|
||||||
|
done
|
||||||
|
|
||||||
|
if ! have_test_failures && [ "$quota_exceeded" = "no" ]; then
|
||||||
|
test_failure "Quota restriction was not enforced by dovecot after sending $msgidx messages"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# cleanup: delete the messages
|
||||||
|
msgidx=0
|
||||||
|
for subj in "${subjects[@]}"; do
|
||||||
|
record "[delete msg $msgidx]"
|
||||||
|
record "subj=$subj"
|
||||||
|
$PYMAIL -no-send -timeout 2 -subj "$subj" $PRIVATE_IP $alice alice >>$TEST_OF 2>&1
|
||||||
|
let msgidx+=1
|
||||||
|
done
|
||||||
|
|
||||||
|
# verify cleanup worked
|
||||||
|
local cur_count_messages="$(doveadm -f json quota get -u "$alice" | jq -r '.[] | select(.type=="MESSAGE") | .value')"
|
||||||
|
if [ $count_messages -ne $cur_count_messages ]; then
|
||||||
|
test_failure "Cleanup failed: test account $alice started with $count_messages but ended up with $cur_count_messages"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# cleanup: delete the test user
|
||||||
|
delete_user "$alice"
|
||||||
|
test_end
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
suite_start "management-users" mgmt_start
|
suite_start "management-users" mgmt_start
|
||||||
|
|
||||||
test_totp
|
test_totp
|
||||||
test_mixed_case_domains
|
test_mixed_case_domains
|
||||||
test_mixed_case_users
|
test_mixed_case_users
|
||||||
test_intl_domains
|
test_intl_domains
|
||||||
|
test_mailbox_quotas
|
||||||
|
|
||||||
suite_end mgmt_end
|
suite_end mgmt_end
|
||||||
|
|
||||||
|
@ -28,6 +28,8 @@ def usage():
|
|||||||
print(" -no-send: don't send, just delete")
|
print(" -no-send: don't send, just delete")
|
||||||
print(" -no-delete: don't delete, just send")
|
print(" -no-delete: don't delete, just send")
|
||||||
print(" -timeout <seconds>: how long to wait for message")
|
print(" -timeout <seconds>: how long to wait for message")
|
||||||
|
print(" -body-from-stdin: read the message body from stdin")
|
||||||
|
print(" -smtp-debug: output debugging messages")
|
||||||
print("");
|
print("");
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
@ -48,6 +50,8 @@ delete_msg=True # login to imap and delete message
|
|||||||
wait_timeout=30 # abandon timeout wiating for message delivery
|
wait_timeout=30 # abandon timeout wiating for message delivery
|
||||||
wait_cycle_sleep=5 # delay between delivery checks
|
wait_cycle_sleep=5 # delay between delivery checks
|
||||||
subject="Mail-in-a-Box Automated Test Message " + uuid.uuid4().hex # message subject
|
subject="Mail-in-a-Box Automated Test Message " + uuid.uuid4().hex # message subject
|
||||||
|
body_from_stdin=False
|
||||||
|
smtp_debug=False
|
||||||
|
|
||||||
# process command line
|
# process command line
|
||||||
argi=1
|
argi=1
|
||||||
@ -81,6 +85,12 @@ while argi<len(sys.argv):
|
|||||||
elif arg=="-timeout" and arg_remaining>1:
|
elif arg=="-timeout" and arg_remaining>1:
|
||||||
wait_timeout=int(sys.argv[argi+1])
|
wait_timeout=int(sys.argv[argi+1])
|
||||||
argi+=2
|
argi+=2
|
||||||
|
elif arg=="-body-from-stdin":
|
||||||
|
body_from_stdin = True
|
||||||
|
argi+=1
|
||||||
|
elif arg=="-smtp-debug":
|
||||||
|
smtp_debug = True
|
||||||
|
argi+=1
|
||||||
else:
|
else:
|
||||||
usage()
|
usage()
|
||||||
|
|
||||||
@ -100,14 +110,20 @@ headerfrom = if_unset(headerfrom, emailfrom)
|
|||||||
emailto = if_unset(emailto, login)
|
emailto = if_unset(emailto, login)
|
||||||
emailto_pw = if_unset(emailto_pw, pw)
|
emailto_pw = if_unset(emailto_pw, pw)
|
||||||
|
|
||||||
|
if body_from_stdin:
|
||||||
|
body=sys.stdin.readlines()
|
||||||
|
else:
|
||||||
|
body=['This is a test message. It should be automatically deleted by the test script.']
|
||||||
|
|
||||||
msg = """From: {headerfrom}
|
msg = """From: {headerfrom}
|
||||||
To: {emailto}
|
To: {emailto}
|
||||||
Subject: {subject}
|
Subject: {subject}
|
||||||
|
|
||||||
This is a test message. It should be automatically deleted by the test script.""".format(
|
{body}""".format(
|
||||||
headerfrom=headerfrom,
|
headerfrom=headerfrom,
|
||||||
emailto=emailto,
|
emailto=emailto,
|
||||||
subject=subject,
|
subject=subject,
|
||||||
|
body=''.join(body)
|
||||||
)
|
)
|
||||||
|
|
||||||
def imap_login(host, login, pw):
|
def imap_login(host, login, pw):
|
||||||
@ -162,7 +178,8 @@ def smtp_login(host, login, pw, port):
|
|||||||
server.starttls()
|
server.starttls()
|
||||||
else:
|
else:
|
||||||
server = smtplib.SMTP_SSL(host)
|
server = smtplib.SMTP_SSL(host)
|
||||||
#server.set_debuglevel(1)
|
if smtp_debug:
|
||||||
|
server.set_debuglevel(1)
|
||||||
|
|
||||||
# Verify that the EHLO name matches the server's reverse DNS.
|
# Verify that the EHLO name matches the server's reverse DNS.
|
||||||
ipaddr = socket.gethostbyname(host) # IPv4 only!
|
ipaddr = socket.gethostbyname(host) # IPv4 only!
|
||||||
@ -191,7 +208,10 @@ def smtp_login(host, login, pw, port):
|
|||||||
if send_msg:
|
if send_msg:
|
||||||
# Attempt to send a mail.
|
# Attempt to send a mail.
|
||||||
server = smtp_login(host, login, pw, port)
|
server = smtp_login(host, login, pw, port)
|
||||||
server.sendmail(emailfrom, [emailto], msg)
|
# sendmail: "If this method does not raise an exception, it returns a dictionary, with one entry for each recipient that was refused. Each entry contains a tuple of the SMTP error code and the accompanying error message sent by the server."
|
||||||
|
errors = server.sendmail(emailfrom, [emailto], msg)
|
||||||
|
#print(errors)
|
||||||
|
#if errors: raise ValueError(errors)
|
||||||
server.quit()
|
server.quit()
|
||||||
print("SMTP submission is OK.")
|
print("SMTP submission is OK.")
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user