mirror of
https://github.com/chatmail/relay.git
synced 2026-05-12 17:14:36 +00:00
Compare commits
5 Commits
withlmtp
...
link2xt/au
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7e15094dd1 | ||
|
|
e19cce7c69 | ||
|
|
1d312f7cfe | ||
|
|
8bed8578ad | ||
|
|
0bfeb2ae5e |
76
README.md
76
README.md
@@ -1,40 +1,60 @@
|
|||||||
# Chat Mail server configuration
|
# Chat Mail server configuration
|
||||||
|
|
||||||
This repository setups a ready-to-go chatmail instance
|
This package deploys Postfix and Dovecot servers, including OpenDKIM for DKIM signing.
|
||||||
comprised of a minimal setup of the battle-tested
|
|
||||||
[postfix smtp server](https://www.postfix.org) and [dovecot imap server](https://www.dovecot.org).
|
Postfix uses Dovecot for authentication as described in <https://www.postfix.org/SASL_README.html#server_dovecot>
|
||||||
|
|
||||||
## Getting started
|
## Getting started
|
||||||
|
|
||||||
1. prepare your local system:
|
prepare:
|
||||||
|
|
||||||
scripts/init.sh
|
pip install -e chatmail-infra
|
||||||
|
|
||||||
2. set environment variable to the chatmail domain you want to setup:
|
|
||||||
|
|
||||||
export CHATMAIL_DOMAIN=c1.testrun.org # replace with your host
|
|
||||||
|
|
||||||
3. run the deploy of the chat mail instance:
|
|
||||||
|
|
||||||
scripts/deploy.sh
|
|
||||||
|
|
||||||
|
|
||||||
## Running tests and benchmarks (offline and online)
|
then run with pyinfra command line tool:
|
||||||
|
|
||||||
1. Set `CHATMAIL_SSH` so that `ssh root@$CHATMAIL_SSH` allows
|
CHATMAIL_DOMAIN=c1.testrun.org pyinfra --ssh-user root c1.testrun.org deploy.py
|
||||||
to login to the chatmail instance server.
|
|
||||||
|
|
||||||
2. To run local and online tests:
|
|
||||||
|
|
||||||
scripts/test.sh
|
## Structure (wip)
|
||||||
|
```
|
||||||
|
|
||||||
3. To run benchmarks against your chatmail instance:
|
# package doveauth tool and deploy chatmail server to a envvar-specified ssh-reachable host
|
||||||
|
deploy.py
|
||||||
|
|
||||||
scripts/bench.sh
|
# chatmail pyinfra deploy package
|
||||||
|
chatmail-pyinfra
|
||||||
|
pyproject.toml
|
||||||
|
chatmail/__init__ ...
|
||||||
|
|
||||||
## Running tests (offline and online)
|
# doveauth tool used by dovecot's auth mechanism on the host system
|
||||||
|
doveauth
|
||||||
|
README.md
|
||||||
|
pyproject.toml
|
||||||
|
doveauth.py
|
||||||
|
test_doveauth.py
|
||||||
|
|
||||||
|
# lmtp server to block (outgoing) unencrypted messages
|
||||||
|
filtermail
|
||||||
|
README.md
|
||||||
|
pyproject.toml
|
||||||
|
....
|
||||||
|
|
||||||
|
# online tests (after deploy)
|
||||||
|
|
||||||
|
online-tests # runnable via pytest
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# scripts for setup/development/deployment
|
||||||
|
|
||||||
|
scripts/
|
||||||
|
init.sh # create venv/other perequires
|
||||||
|
deploy.sh # run pyinfra based deploy of everything
|
||||||
|
test.sh # run all local and online tests
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
## Dovecot/Postfix configuration
|
## Dovecot/Postfix configuration
|
||||||
|
|
||||||
### Ports
|
### Ports
|
||||||
@@ -44,16 +64,4 @@ Dovecot listens on ports 143(imap) and 993 (imaps).
|
|||||||
|
|
||||||
## DNS
|
## DNS
|
||||||
|
|
||||||
For DKIM you must add a DNS entry as found in /etc/opendkim/selector.txt on your chatmail instance.
|
For DKIM you must add a DNS entry as in /etc/opendkim/selector.txt (where selector is the opendkim_selector configured in the chatmail inventory).
|
||||||
The above `scripts/deploy.sh` prints out the DKIM selector and DNS entry you
|
|
||||||
need to setup with your DNS provider.
|
|
||||||
|
|
||||||
## Emergency Commands
|
|
||||||
|
|
||||||
If you need to stop account creation,
|
|
||||||
e.g. because some script is wildly creating accounts,
|
|
||||||
just run `touch /tmp/nocreate`.
|
|
||||||
You can remove the file
|
|
||||||
as soon as the attacker was banned
|
|
||||||
by different means.
|
|
||||||
|
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ dependencies = [
|
|||||||
]
|
]
|
||||||
|
|
||||||
[project.scripts]
|
[project.scripts]
|
||||||
|
doveauth = "chatmaild.doveauth:main"
|
||||||
doveauth-dictproxy = "chatmaild.dictproxy:main"
|
doveauth-dictproxy = "chatmaild.dictproxy:main"
|
||||||
filtermail = "chatmaild.filtermail:main"
|
filtermail = "chatmaild.filtermail:main"
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
import logging
|
|
||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
import json
|
import json
|
||||||
@@ -12,8 +11,6 @@ import subprocess
|
|||||||
|
|
||||||
from .database import Database
|
from .database import Database
|
||||||
|
|
||||||
NOCREATE_FILE = "/etc/chatmail-nocreate"
|
|
||||||
|
|
||||||
|
|
||||||
def encrypt_password(password: str):
|
def encrypt_password(password: str):
|
||||||
password = password.encode("ascii")
|
password = password.encode("ascii")
|
||||||
@@ -30,11 +27,6 @@ def encrypt_password(password: str):
|
|||||||
|
|
||||||
|
|
||||||
def create_user(db, user, password):
|
def create_user(db, user, password):
|
||||||
if os.path.exists(NOCREATE_FILE):
|
|
||||||
logging.warning(
|
|
||||||
f"Didn't create account: {NOCREATE_FILE} exists. Delete the file to enable account creation."
|
|
||||||
)
|
|
||||||
return
|
|
||||||
with db.write_transaction() as conn:
|
with db.write_transaction() as conn:
|
||||||
conn.create_user(user, password)
|
conn.create_user(user, password)
|
||||||
return dict(home=f"/home/vmail/{user}", uid="vmail", gid="vmail", password=password)
|
return dict(home=f"/home/vmail/{user}", uid="vmail", gid="vmail", password=password)
|
||||||
@@ -61,7 +53,7 @@ def lookup_passdb(db, user, password):
|
|||||||
return userdata
|
return userdata
|
||||||
|
|
||||||
|
|
||||||
def handle_dovecot_request(msg, db, mail_domain):
|
def handle_dovecot_request(msg, db):
|
||||||
print(f"received msg: {msg!r}", file=sys.stderr)
|
print(f"received msg: {msg!r}", file=sys.stderr)
|
||||||
short_command = msg[0]
|
short_command = msg[0]
|
||||||
if short_command == "L": # LOOKUP
|
if short_command == "L": # LOOKUP
|
||||||
@@ -72,15 +64,13 @@ def handle_dovecot_request(msg, db, mail_domain):
|
|||||||
res = ""
|
res = ""
|
||||||
if namespace == "shared":
|
if namespace == "shared":
|
||||||
if type == "userdb":
|
if type == "userdb":
|
||||||
if user.endswith(f"@{mail_domain}"):
|
res = lookup_userdb(db, user)
|
||||||
res = lookup_userdb(db, user)
|
|
||||||
if res:
|
if res:
|
||||||
reply_command = "O"
|
reply_command = "O"
|
||||||
else:
|
else:
|
||||||
reply_command = "N"
|
reply_command = "N"
|
||||||
elif type == "passdb":
|
elif type == "passdb":
|
||||||
if user.endswith(f"@{mail_domain}"):
|
res = lookup_passdb(db, user, password=args[0])
|
||||||
res = lookup_passdb(db, user, password=args[0])
|
|
||||||
if res:
|
if res:
|
||||||
reply_command = "O"
|
reply_command = "O"
|
||||||
else:
|
else:
|
||||||
@@ -99,16 +89,14 @@ def main():
|
|||||||
socket = sys.argv[1]
|
socket = sys.argv[1]
|
||||||
passwd_entry = pwd.getpwnam(sys.argv[2])
|
passwd_entry = pwd.getpwnam(sys.argv[2])
|
||||||
db = Database(sys.argv[3])
|
db = Database(sys.argv[3])
|
||||||
with open("/etc/mailname", "r") as fp:
|
|
||||||
mail_domain = fp.read().strip()
|
|
||||||
|
|
||||||
class Handler(StreamRequestHandler):
|
class Handler(StreamRequestHandler):
|
||||||
def handle(self):
|
def handle(self):
|
||||||
while True:
|
while True:
|
||||||
msg = self.rfile.readline().strip().decode()
|
msg = self.rfile.readline().strip().decode()
|
||||||
if not msg:
|
if not msg:
|
||||||
break
|
continue
|
||||||
res = handle_dovecot_request(msg, db, mail_domain)
|
res = handle_dovecot_request(msg, db)
|
||||||
if res:
|
if res:
|
||||||
print(f"sending result: {res!r}", file=sys.stderr)
|
print(f"sending result: {res!r}", file=sys.stderr)
|
||||||
self.wfile.write(res.encode("ascii"))
|
self.wfile.write(res.encode("ascii"))
|
||||||
|
|||||||
65
chatmaild/src/chatmaild/doveauth.py
Normal file
65
chatmaild/src/chatmaild/doveauth.py
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
import base64
|
||||||
|
import sys
|
||||||
|
|
||||||
|
from .database import Database
|
||||||
|
|
||||||
|
|
||||||
|
def get_user_data(db, user):
|
||||||
|
with db.read_connection() as conn:
|
||||||
|
result = conn.get_user(user)
|
||||||
|
if result:
|
||||||
|
result["uid"] = "vmail"
|
||||||
|
result["gid"] = "vmail"
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
def create_user(db, user, password):
|
||||||
|
with db.write_transaction() as conn:
|
||||||
|
conn.create_user(user, password)
|
||||||
|
return dict(home=f"/home/vmail/{user}", uid="vmail", gid="vmail", password=password)
|
||||||
|
|
||||||
|
|
||||||
|
def verify_user(db, user, password):
|
||||||
|
userdata = get_user_data(db, user)
|
||||||
|
if userdata:
|
||||||
|
if userdata.get("password") == password:
|
||||||
|
userdata["status"] = "ok"
|
||||||
|
else:
|
||||||
|
userdata["status"] = "fail"
|
||||||
|
else:
|
||||||
|
userdata = create_user(db, user, password)
|
||||||
|
userdata["status"] = "ok"
|
||||||
|
|
||||||
|
return userdata
|
||||||
|
|
||||||
|
|
||||||
|
def lookup_user(db, user):
|
||||||
|
userdata = get_user_data(db, user)
|
||||||
|
if userdata:
|
||||||
|
userdata["status"] = "ok"
|
||||||
|
else:
|
||||||
|
userdata["status"] = "fail"
|
||||||
|
return userdata
|
||||||
|
|
||||||
|
|
||||||
|
def dump_result(res):
|
||||||
|
for key, value in res.items():
|
||||||
|
print(f"{key}={value}")
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
db = Database("/home/vmail/passdb.sqlite")
|
||||||
|
if sys.argv[1] == "hexauth":
|
||||||
|
login = base64.b16decode(sys.argv[2]).decode()
|
||||||
|
password = base64.b16decode(sys.argv[3]).decode()
|
||||||
|
res = verify_user(db, login, password)
|
||||||
|
dump_result(res)
|
||||||
|
elif sys.argv[1] == "hexlookup":
|
||||||
|
login = base64.b16decode(sys.argv[2]).decode()
|
||||||
|
res = lookup_user(db, login)
|
||||||
|
dump_result(res)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
[Unit]
|
|
||||||
Description=Chatmail Postfix AfterQueue filter
|
|
||||||
|
|
||||||
[Service]
|
|
||||||
ExecStart=/usr/local/bin/filtermail afterqueue /var/spool/postfix/private/filtermail-afterqueue
|
|
||||||
Restart=always
|
|
||||||
RestartSec=30
|
|
||||||
|
|
||||||
[Install]
|
|
||||||
WantedBy=multi-user.target
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
[Unit]
|
|
||||||
Description=Chatmail Postfix BeforeQeue filter
|
|
||||||
|
|
||||||
[Service]
|
|
||||||
ExecStart=/usr/local/bin/filtermail beforequeue 10080
|
|
||||||
Restart=always
|
|
||||||
RestartSec=30
|
|
||||||
|
|
||||||
[Install]
|
|
||||||
WantedBy=multi-user.target
|
|
||||||
@@ -1,20 +1,17 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
import asyncio
|
import asyncio
|
||||||
import logging
|
import logging
|
||||||
import time
|
|
||||||
import sys
|
|
||||||
from email.parser import BytesParser
|
from email.parser import BytesParser
|
||||||
from email import policy
|
from email import policy
|
||||||
from email.utils import parseaddr
|
|
||||||
|
|
||||||
from aiosmtpd.lmtp import LMTP
|
from aiosmtpd.lmtp import LMTP
|
||||||
from aiosmtpd.smtp import SMTP
|
from aiosmtpd.controller import UnixSocketController
|
||||||
from aiosmtpd.controller import UnixSocketController, Controller
|
|
||||||
from smtplib import SMTP as SMTPClient
|
from smtplib import SMTP as SMTPClient
|
||||||
|
|
||||||
|
|
||||||
def check_encrypted(message):
|
def check_encrypted(content):
|
||||||
"""Check that the message is an OpenPGP-encrypted message."""
|
"""Check that the message is an OpenPGP-encrypted message."""
|
||||||
|
message = BytesParser(policy=policy.default).parsebytes(content)
|
||||||
if not message.is_multipart():
|
if not message.is_multipart():
|
||||||
return False
|
return False
|
||||||
if message.get("subject") != "...":
|
if message.get("subject") != "...":
|
||||||
@@ -35,138 +32,72 @@ def check_encrypted(message):
|
|||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
class BeforeQueueHandler:
|
class ExampleController(UnixSocketController):
|
||||||
def __init__(self):
|
def factory(self):
|
||||||
self.send_rate_limiter = SendRateLimiter()
|
return LMTP(self.handler, **self.SMTP_kwargs)
|
||||||
|
|
||||||
async def handle_MAIL(self, server, session, envelope, address, mail_options):
|
|
||||||
logging.info(f"handle_MAIL from {address}")
|
|
||||||
if self.send_rate_limiter.is_sending_allowed(address):
|
|
||||||
envelope.mail_from = address
|
|
||||||
return "250 OK"
|
|
||||||
return f"450 4.7.1: Too much mail from {address}"
|
|
||||||
|
|
||||||
async def handle_DATA(self, server, session, envelope):
|
|
||||||
logging.info("handle_DATA before-queue: re-injecting the mail")
|
|
||||||
client = SMTPClient("localhost", "10026")
|
|
||||||
client.sendmail(envelope.mail_from, envelope.rcpt_tos, envelope.content)
|
|
||||||
return "250 OK"
|
|
||||||
|
|
||||||
|
|
||||||
class SendRateLimiter:
|
class ExampleHandler:
|
||||||
MAX_USER_SEND_PER_MINUTE = 80
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
self.addr2timestamps = {}
|
|
||||||
|
|
||||||
def is_sending_allowed(self, mail_from):
|
|
||||||
last = self.addr2timestamps.setdefault(mail_from, [])
|
|
||||||
now = time.time()
|
|
||||||
last[:] = [ts for ts in last if ts >= (now - 60)]
|
|
||||||
if len(last) <= self.MAX_USER_SEND_PER_MINUTE:
|
|
||||||
last.append(now)
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
class AfterQueueHandler:
|
|
||||||
async def handle_RCPT(self, server, session, envelope, address, rcpt_options):
|
async def handle_RCPT(self, server, session, envelope, address, rcpt_options):
|
||||||
envelope.rcpt_tos.append(address)
|
envelope.rcpt_tos.append(address)
|
||||||
return "250 OK"
|
return "250 OK"
|
||||||
|
|
||||||
async def handle_DATA(self, server, session, envelope):
|
async def handle_DATA(self, server, session, envelope):
|
||||||
valid_recipients, res = lmtp_handle_DATA(envelope)
|
logging.info("Processing DATA message from %s", envelope.mail_from)
|
||||||
|
|
||||||
|
valid_recipients = []
|
||||||
|
|
||||||
|
mail_encrypted = check_encrypted(envelope.content)
|
||||||
|
|
||||||
|
res = []
|
||||||
|
for recipient in envelope.rcpt_tos:
|
||||||
|
my_local_domain = envelope.mail_from.split("@")
|
||||||
|
if len(my_local_domain) != 2:
|
||||||
|
res += [f"500 Invalid from address <{envelope.mail_from}>"]
|
||||||
|
continue
|
||||||
|
|
||||||
|
if envelope.mail_from == recipient:
|
||||||
|
# Always allow sending emails to self.
|
||||||
|
valid_recipients += [recipient]
|
||||||
|
res += ["250 OK"]
|
||||||
|
continue
|
||||||
|
|
||||||
|
recipient_local_domain = recipient.split("@")
|
||||||
|
if len(recipient_local_domain) != 2:
|
||||||
|
res += [f"500 Invalid address <{recipient}>"]
|
||||||
|
continue
|
||||||
|
|
||||||
|
is_outgoing = recipient_local_domain[1] != my_local_domain[1]
|
||||||
|
if is_outgoing and not mail_encrypted:
|
||||||
|
res += ["500 Outgoing mail must be encrypted"]
|
||||||
|
continue
|
||||||
|
|
||||||
|
valid_recipients += [recipient]
|
||||||
|
res += ["250 OK"]
|
||||||
|
|
||||||
# Reinject the mail back into Postfix.
|
# Reinject the mail back into Postfix.
|
||||||
if valid_recipients:
|
if valid_recipients:
|
||||||
logging.info("afterqueue: re-injecting the mail")
|
logging.info("Reinjecting the mail")
|
||||||
client = SMTPClient("localhost", "10027")
|
client = SMTPClient("localhost", "10026")
|
||||||
client.sendmail(envelope.mail_from, valid_recipients, envelope.content)
|
client.sendmail(envelope.mail_from, valid_recipients, envelope.content)
|
||||||
else:
|
|
||||||
logging.info("no valid recipients, ignoring mail")
|
|
||||||
|
|
||||||
return "\r\n".join(res)
|
return "\r\n".join(res)
|
||||||
|
|
||||||
|
|
||||||
def lmtp_handle_DATA(envelope):
|
async def asyncmain(loop):
|
||||||
"""the central filtering function for e-mails."""
|
controller = ExampleController(
|
||||||
logging.info(f"Processing DATA message from {envelope.mail_from}")
|
ExampleHandler(), unix_socket="/var/spool/postfix/private/filtermail"
|
||||||
|
)
|
||||||
message = BytesParser(policy=policy.default).parsebytes(envelope.content)
|
controller.start()
|
||||||
mail_encrypted = check_encrypted(message)
|
|
||||||
|
|
||||||
valid_recipients = []
|
|
||||||
res = []
|
|
||||||
for recipient in envelope.rcpt_tos:
|
|
||||||
my_local_domain = envelope.mail_from.split("@")
|
|
||||||
if len(my_local_domain) != 2:
|
|
||||||
res += [f"500 Invalid from address <{envelope.mail_from}>"]
|
|
||||||
continue
|
|
||||||
|
|
||||||
_, from_addr = parseaddr(message.get("from").strip())
|
|
||||||
logging.info(f"mime-from: {from_addr} envelope-from: {envelope.mail_from}")
|
|
||||||
if envelope.mail_from.lower() != from_addr.lower():
|
|
||||||
res += [f"500 Invalid FROM <{from_addr}> for <{envelope.mail_from}>"]
|
|
||||||
continue
|
|
||||||
|
|
||||||
if envelope.mail_from == recipient:
|
|
||||||
# Always allow sending emails to self.
|
|
||||||
valid_recipients += [recipient]
|
|
||||||
res += ["250 OK"]
|
|
||||||
continue
|
|
||||||
|
|
||||||
recipient_local_domain = recipient.split("@")
|
|
||||||
if len(recipient_local_domain) != 2:
|
|
||||||
res += [f"500 Invalid address <{recipient}>"]
|
|
||||||
continue
|
|
||||||
|
|
||||||
is_outgoing = recipient_local_domain[1] != my_local_domain[1]
|
|
||||||
|
|
||||||
if (
|
|
||||||
is_outgoing
|
|
||||||
and not mail_encrypted
|
|
||||||
and message.get("secure-join") != "vc-request"
|
|
||||||
and message.get("secure-join") != "vg-request"
|
|
||||||
):
|
|
||||||
res += ["500 Outgoing mail must be encrypted"]
|
|
||||||
continue
|
|
||||||
|
|
||||||
valid_recipients += [recipient]
|
|
||||||
res += ["250 OK"]
|
|
||||||
|
|
||||||
assert len(envelope.rcpt_tos) == len(res)
|
|
||||||
assert len(valid_recipients) <= len(res)
|
|
||||||
return valid_recipients, res
|
|
||||||
|
|
||||||
|
|
||||||
class UnixController(UnixSocketController):
|
|
||||||
def factory(self):
|
|
||||||
return LMTP(self.handler, **self.SMTP_kwargs)
|
|
||||||
|
|
||||||
|
|
||||||
class SMTPController(Controller):
|
|
||||||
def factory(self):
|
|
||||||
return SMTP(self.handler, **self.SMTP_kwargs)
|
|
||||||
|
|
||||||
|
|
||||||
async def asyncmain_afterqueue(loop, unix_socket_fn):
|
|
||||||
UnixController(AfterQueueHandler(), unix_socket=unix_socket_fn).start()
|
|
||||||
|
|
||||||
|
|
||||||
async def asyncmain_beforequeue(loop, port):
|
|
||||||
Controller(BeforeQueueHandler(), hostname="127.0.0.1", port=port).start()
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
args = sys.argv[1:]
|
|
||||||
assert len(args) == 2
|
|
||||||
logging.basicConfig(level=logging.INFO)
|
logging.basicConfig(level=logging.INFO)
|
||||||
loop = asyncio.new_event_loop()
|
loop = asyncio.new_event_loop()
|
||||||
asyncio.set_event_loop(loop)
|
asyncio.set_event_loop(loop)
|
||||||
if args[0] == "afterqueue":
|
loop.create_task(asyncmain(loop=loop))
|
||||||
task = asyncmain_afterqueue(loop, args[1])
|
|
||||||
elif args[0] == "beforequeue":
|
|
||||||
task = asyncmain_beforequeue(loop, port=int(args[1]))
|
|
||||||
else:
|
|
||||||
raise SystemExit(1)
|
|
||||||
loop.create_task(task)
|
|
||||||
loop.run_forever()
|
loop.run_forever()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
||||||
|
|||||||
10
chatmaild/src/chatmaild/filtermail.service
Normal file
10
chatmaild/src/chatmaild/filtermail.service
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
[Unit]
|
||||||
|
Description=Email filter for chatmail servers
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
ExecStart=/usr/local/bin/filtermail
|
||||||
|
Restart=always
|
||||||
|
RestartSec=30
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
@@ -1,9 +1,7 @@
|
|||||||
import os
|
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
import chatmaild.dictproxy
|
from .dictproxy import get_user_data
|
||||||
from .dictproxy import get_user_data, lookup_passdb
|
from .doveauth import verify_user
|
||||||
from .database import Database, DBError
|
from .database import Database, DBError
|
||||||
|
|
||||||
|
|
||||||
@@ -15,31 +13,16 @@ def db(tmpdir):
|
|||||||
|
|
||||||
|
|
||||||
def test_basic(db):
|
def test_basic(db):
|
||||||
chatmaild.dictproxy.NOCREATE_FILE = "/tmp/nocreate"
|
verify_user(db, "link2xt@c1.testrun.org", "asdf")
|
||||||
if os.path.exists(chatmaild.dictproxy.NOCREATE_FILE):
|
|
||||||
os.remove(chatmaild.dictproxy.NOCREATE_FILE)
|
|
||||||
lookup_passdb(db, "link2xt@c1.testrun.org", "asdf")
|
|
||||||
data = get_user_data(db, "link2xt@c1.testrun.org")
|
data = get_user_data(db, "link2xt@c1.testrun.org")
|
||||||
assert data
|
assert data
|
||||||
|
|
||||||
|
|
||||||
def test_dont_overwrite_password_on_wrong_login(db):
|
def test_verify_or_create(db):
|
||||||
"""Test that logging in with a different password doesn't create a new user"""
|
res = verify_user(db, "newuser1@something.org", "kajdlkajsldk12l3kj1983")
|
||||||
res = lookup_passdb(db, "newuser1@something.org", "kajdlkajsldk12l3kj1983")
|
assert res["status"] == "ok"
|
||||||
assert res["password"]
|
res = verify_user(db, "newuser1@something.org", "kajdlqweqwe")
|
||||||
res2 = lookup_passdb(db, "newuser1@something.org", "kajdlqweqwe")
|
assert res["status"] == "fail"
|
||||||
# this function always returns a password hash, which is actually compared by dovecot.
|
|
||||||
assert res["password"] == res2["password"]
|
|
||||||
|
|
||||||
|
|
||||||
def test_nocreate_file(db):
|
|
||||||
chatmaild.dictproxy.NOCREATE_FILE = "/tmp/nocreate"
|
|
||||||
with open(chatmaild.dictproxy.NOCREATE_FILE, "w+") as f:
|
|
||||||
f.write("")
|
|
||||||
assert os.path.exists(chatmaild.dictproxy.NOCREATE_FILE)
|
|
||||||
lookup_passdb(db, "newuser1@something.org", "kajdlqweqwe")
|
|
||||||
assert not get_user_data(db, "newuser1@something.org")
|
|
||||||
os.remove(chatmaild.dictproxy.NOCREATE_FILE)
|
|
||||||
|
|
||||||
|
|
||||||
def test_db_version(db):
|
def test_db_version(db):
|
||||||
|
|||||||
@@ -1,55 +1,12 @@
|
|||||||
from .filtermail import check_encrypted, lmtp_handle_DATA, SendRateLimiter
|
|
||||||
from email.parser import BytesParser
|
|
||||||
from email import policy
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
|
from .filtermail import check_encrypted
|
||||||
def test_reject_forged_from():
|
|
||||||
def makemail(from_addr):
|
|
||||||
return BytesParser(policy=policy.default).parsebytes(
|
|
||||||
"\r\n".join(
|
|
||||||
[
|
|
||||||
f"From: <{from_addr}",
|
|
||||||
"To: <barbaz@c3.testrun.org>",
|
|
||||||
"Date: Sun, 15 Oct 2023 16:41:44 +0000",
|
|
||||||
"Message-ID: <Mr.3gckbNy5bch.uK3Hd2Ws6-w@c2.testrun.org>",
|
|
||||||
"References: <Mr.3gckbNy5bch.uK3Hd2Ws6-w@c2.testrun.org>",
|
|
||||||
"Chat-Version: 1.0",
|
|
||||||
"MIME-Version: 1.0",
|
|
||||||
"Content-Type: text/plain; charset=utf-8; format=flowed; delsp=no",
|
|
||||||
"",
|
|
||||||
"Hi!",
|
|
||||||
"",
|
|
||||||
"",
|
|
||||||
]
|
|
||||||
).encode()
|
|
||||||
)
|
|
||||||
|
|
||||||
class envelope:
|
|
||||||
mail_from = "bob@c3.testrun.org"
|
|
||||||
rcpt_tos = ["somebody@c3.testrun.org"]
|
|
||||||
|
|
||||||
# test that the filter lets good mail through
|
|
||||||
envelope.content = makemail(envelope.mail_from).as_bytes()
|
|
||||||
valid_recipients, res = lmtp_handle_DATA(envelope=envelope)
|
|
||||||
assert valid_recipients == envelope.rcpt_tos
|
|
||||||
assert len(res) == 1 and "250" in res[0]
|
|
||||||
|
|
||||||
# test that the filter rejects forged mail
|
|
||||||
envelope.content = makemail("forged@c3.testrun.org").as_bytes()
|
|
||||||
valid_recipients, res = lmtp_handle_DATA(envelope=envelope)
|
|
||||||
assert not valid_recipients
|
|
||||||
assert len(res) == 1 and "500" in res[0]
|
|
||||||
|
|
||||||
|
|
||||||
def test_filtermail():
|
def test_filtermail():
|
||||||
def check_encrypted_bstr(content):
|
assert not check_encrypted(b"foo")
|
||||||
message = BytesParser(policy=policy.default).parsebytes(content)
|
|
||||||
return check_encrypted(message)
|
|
||||||
|
|
||||||
assert not check_encrypted_bstr(b"foo")
|
assert not check_encrypted(
|
||||||
|
|
||||||
assert not check_encrypted_bstr(
|
|
||||||
"\r\n".join(
|
"\r\n".join(
|
||||||
[
|
[
|
||||||
"Subject: =?utf-8?q?Message_from_foobar=40c2=2Etestrun=2Eorg?=",
|
"Subject: =?utf-8?q?Message_from_foobar=40c2=2Etestrun=2Eorg?=",
|
||||||
@@ -79,7 +36,7 @@ def test_filtermail():
|
|||||||
).encode()
|
).encode()
|
||||||
)
|
)
|
||||||
|
|
||||||
assert not check_encrypted_bstr(
|
assert not check_encrypted(
|
||||||
"\r\n".join(
|
"\r\n".join(
|
||||||
[
|
[
|
||||||
"Subject: =?utf-8?q?Message_from_foobar=40c2=2Etestrun=2Eorg?=",
|
"Subject: =?utf-8?q?Message_from_foobar=40c2=2Etestrun=2Eorg?=",
|
||||||
@@ -110,7 +67,7 @@ def test_filtermail():
|
|||||||
)
|
)
|
||||||
|
|
||||||
# https://xkcd.com/1181/
|
# https://xkcd.com/1181/
|
||||||
assert not check_encrypted_bstr(
|
assert not check_encrypted(
|
||||||
"\r\n".join(
|
"\r\n".join(
|
||||||
[
|
[
|
||||||
"Subject: =?utf-8?q?Message_from_foobar=40c2=2Etestrun=2Eorg?=",
|
"Subject: =?utf-8?q?Message_from_foobar=40c2=2Etestrun=2Eorg?=",
|
||||||
@@ -142,7 +99,7 @@ def test_filtermail():
|
|||||||
).encode()
|
).encode()
|
||||||
)
|
)
|
||||||
|
|
||||||
assert check_encrypted_bstr(
|
assert check_encrypted(
|
||||||
"\r\n".join(
|
"\r\n".join(
|
||||||
[
|
[
|
||||||
"Subject: ...",
|
"Subject: ...",
|
||||||
@@ -215,7 +172,7 @@ def test_filtermail():
|
|||||||
).encode()
|
).encode()
|
||||||
)
|
)
|
||||||
|
|
||||||
assert not check_encrypted_bstr(
|
assert not check_encrypted(
|
||||||
"\r\n".join(
|
"\r\n".join(
|
||||||
[
|
[
|
||||||
"Subject: Buy Penis Enlargement at www.malicious-domain.com",
|
"Subject: Buy Penis Enlargement at www.malicious-domain.com",
|
||||||
@@ -288,7 +245,7 @@ def test_filtermail():
|
|||||||
).encode()
|
).encode()
|
||||||
)
|
)
|
||||||
|
|
||||||
assert not check_encrypted_bstr(
|
assert not check_encrypted(
|
||||||
"\r\n".join(
|
"\r\n".join(
|
||||||
[
|
[
|
||||||
"Subject: Message opened",
|
"Subject: Message opened",
|
||||||
@@ -327,15 +284,3 @@ def test_filtermail():
|
|||||||
]
|
]
|
||||||
).encode()
|
).encode()
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
def test_send_rate_limiter():
|
|
||||||
limiter = SendRateLimiter()
|
|
||||||
for i in range(100):
|
|
||||||
if limiter.is_sending_allowed("some@example.org"):
|
|
||||||
if i <= SendRateLimiter.MAX_USER_SEND_PER_MINUTE:
|
|
||||||
continue
|
|
||||||
pytest.fail("limiter didn't work")
|
|
||||||
else:
|
|
||||||
assert i == SendRateLimiter.MAX_USER_SEND_PER_MINUTE + 1
|
|
||||||
break
|
|
||||||
|
|||||||
@@ -53,25 +53,24 @@ def _install_chatmaild() -> None:
|
|||||||
daemon_reload=True,
|
daemon_reload=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
for fn in ("filtermail-after", "filtermail-before"):
|
files.put(
|
||||||
files.put(
|
name="upload filtermail.service",
|
||||||
name=f"upload {fn}.service",
|
src=importlib.resources.files("chatmaild")
|
||||||
src=importlib.resources.files("chatmaild")
|
.joinpath("filtermail.service")
|
||||||
.joinpath(f"{fn}.service")
|
.open("rb"),
|
||||||
.open("rb"),
|
dest="/etc/systemd/system/filtermail.service",
|
||||||
dest=f"/etc/systemd/system/{fn}.service",
|
user="root",
|
||||||
user="root",
|
group="root",
|
||||||
group="root",
|
mode="644",
|
||||||
mode="644",
|
)
|
||||||
)
|
systemd.service(
|
||||||
systemd.service(
|
name="Setup filtermail service",
|
||||||
name=f"Setup {fn} service",
|
service="filtermail.service",
|
||||||
service=f"{fn}.service",
|
running=True,
|
||||||
running=True,
|
enabled=True,
|
||||||
enabled=True,
|
restarted=True,
|
||||||
restarted=True,
|
daemon_reload=True,
|
||||||
daemon_reload=True,
|
)
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def _configure_opendkim(domain: str, dkim_selector: str) -> bool:
|
def _configure_opendkim(domain: str, dkim_selector: str) -> bool:
|
||||||
@@ -111,7 +110,7 @@ def _configure_opendkim(domain: str, dkim_selector: str) -> bool:
|
|||||||
return need_restart
|
return need_restart
|
||||||
|
|
||||||
|
|
||||||
def _configure_postfix(domain: str, debug: bool = False) -> bool:
|
def _configure_postfix(domain: str) -> bool:
|
||||||
"""Configures Postfix SMTP server."""
|
"""Configures Postfix SMTP server."""
|
||||||
need_restart = False
|
need_restart = False
|
||||||
|
|
||||||
@@ -125,20 +124,21 @@ def _configure_postfix(domain: str, debug: bool = False) -> bool:
|
|||||||
)
|
)
|
||||||
need_restart |= main_config.changed
|
need_restart |= main_config.changed
|
||||||
|
|
||||||
master_config = files.template(
|
master_config = files.put(
|
||||||
src=importlib.resources.files(__package__).joinpath("postfix/master.cf.j2"),
|
src=importlib.resources.files(__package__)
|
||||||
|
.joinpath("postfix/master.cf")
|
||||||
|
.open("rb"),
|
||||||
dest="/etc/postfix/master.cf",
|
dest="/etc/postfix/master.cf",
|
||||||
user="root",
|
user="root",
|
||||||
group="root",
|
group="root",
|
||||||
mode="644",
|
mode="644",
|
||||||
debug=debug,
|
|
||||||
)
|
)
|
||||||
need_restart |= master_config.changed
|
need_restart |= master_config.changed
|
||||||
|
|
||||||
return need_restart
|
return need_restart
|
||||||
|
|
||||||
|
|
||||||
def _configure_dovecot(mail_server: str, debug: bool = False) -> bool:
|
def _configure_dovecot(mail_server: str) -> bool:
|
||||||
"""Configures Dovecot IMAP server."""
|
"""Configures Dovecot IMAP server."""
|
||||||
need_restart = False
|
need_restart = False
|
||||||
|
|
||||||
@@ -149,7 +149,6 @@ def _configure_dovecot(mail_server: str, debug: bool = False) -> bool:
|
|||||||
group="root",
|
group="root",
|
||||||
mode="644",
|
mode="644",
|
||||||
config={"hostname": mail_server},
|
config={"hostname": mail_server},
|
||||||
debug=debug,
|
|
||||||
)
|
)
|
||||||
need_restart |= main_config.changed
|
need_restart |= main_config.changed
|
||||||
auth_config = files.put(
|
auth_config = files.put(
|
||||||
@@ -161,43 +160,6 @@ def _configure_dovecot(mail_server: str, debug: bool = False) -> bool:
|
|||||||
)
|
)
|
||||||
need_restart |= auth_config.changed
|
need_restart |= auth_config.changed
|
||||||
|
|
||||||
files.put(
|
|
||||||
src=importlib.resources.files(__package__)
|
|
||||||
.joinpath("dovecot/expunge.cron")
|
|
||||||
.open("rb"),
|
|
||||||
dest="/etc/cron.d/expunge",
|
|
||||||
user="root",
|
|
||||||
group="root",
|
|
||||||
mode="644",
|
|
||||||
)
|
|
||||||
|
|
||||||
return need_restart
|
|
||||||
|
|
||||||
|
|
||||||
def _configure_nginx(domain: str, debug: bool = False) -> bool:
|
|
||||||
"""Configures nginx HTTP server."""
|
|
||||||
need_restart = False
|
|
||||||
|
|
||||||
main_config = files.template(
|
|
||||||
src=importlib.resources.files(__package__).joinpath("nginx.conf.j2"),
|
|
||||||
dest="/etc/nginx/nginx.conf",
|
|
||||||
user="root",
|
|
||||||
group="root",
|
|
||||||
mode="644",
|
|
||||||
config={"domain_name": domain},
|
|
||||||
)
|
|
||||||
need_restart |= main_config.changed
|
|
||||||
|
|
||||||
autoconfig = files.template(
|
|
||||||
src=importlib.resources.files(__package__).joinpath("autoconfig.xml.j2"),
|
|
||||||
dest="/var/www/html/.well-known/autoconfig/mail/config-v1.1.xml",
|
|
||||||
user="root",
|
|
||||||
group="root",
|
|
||||||
mode="644",
|
|
||||||
config={"domain_name": domain},
|
|
||||||
)
|
|
||||||
need_restart |= autoconfig.changed
|
|
||||||
|
|
||||||
return need_restart
|
return need_restart
|
||||||
|
|
||||||
|
|
||||||
@@ -222,7 +184,7 @@ def deploy_chatmail(mail_domain: str, mail_server: str, dkim_selector: str) -> N
|
|||||||
)
|
)
|
||||||
|
|
||||||
# Deploy acmetool to have TLS certificates.
|
# Deploy acmetool to have TLS certificates.
|
||||||
deploy_acmetool(nginx_hook=True, domains=[mail_server])
|
deploy_acmetool(domains=[mail_server])
|
||||||
|
|
||||||
apt.packages(
|
apt.packages(
|
||||||
name="Install Postfix",
|
name="Install Postfix",
|
||||||
@@ -242,17 +204,10 @@ def deploy_chatmail(mail_domain: str, mail_server: str, dkim_selector: str) -> N
|
|||||||
],
|
],
|
||||||
)
|
)
|
||||||
|
|
||||||
apt.packages(
|
|
||||||
name="Install nginx",
|
|
||||||
packages=["nginx"],
|
|
||||||
)
|
|
||||||
|
|
||||||
_install_chatmaild()
|
_install_chatmaild()
|
||||||
debug = False
|
dovecot_need_restart = _configure_dovecot(mail_server)
|
||||||
dovecot_need_restart = _configure_dovecot(mail_server, debug=debug)
|
postfix_need_restart = _configure_postfix(mail_domain)
|
||||||
postfix_need_restart = _configure_postfix(mail_domain, debug=debug)
|
|
||||||
opendkim_need_restart = _configure_opendkim(mail_domain, dkim_selector)
|
opendkim_need_restart = _configure_opendkim(mail_domain, dkim_selector)
|
||||||
nginx_need_restart = _configure_nginx(mail_domain)
|
|
||||||
|
|
||||||
systemd.service(
|
systemd.service(
|
||||||
name="Start and enable OpenDKIM",
|
name="Start and enable OpenDKIM",
|
||||||
@@ -278,21 +233,6 @@ def deploy_chatmail(mail_domain: str, mail_server: str, dkim_selector: str) -> N
|
|||||||
restarted=dovecot_need_restart,
|
restarted=dovecot_need_restart,
|
||||||
)
|
)
|
||||||
|
|
||||||
systemd.service(
|
|
||||||
name="Start and enable nginx",
|
|
||||||
service="nginx.service",
|
|
||||||
running=True,
|
|
||||||
enabled=True,
|
|
||||||
restarted=nginx_need_restart,
|
|
||||||
)
|
|
||||||
|
|
||||||
# This file is used by auth proxy.
|
|
||||||
# https://wiki.debian.org/EtcMailName
|
|
||||||
server.shell(
|
|
||||||
name="Setup /etc/mailname",
|
|
||||||
commands=[f"echo {mail_domain} >/etc/mailname; chmod 644 /etc/mailname"],
|
|
||||||
)
|
|
||||||
|
|
||||||
def callback():
|
def callback():
|
||||||
result = server.shell(
|
result = server.shell(
|
||||||
commands=[
|
commands=[
|
||||||
|
|||||||
@@ -38,13 +38,22 @@ def deploy_acmetool(nginx_hook=False, email="", domains=[]):
|
|||||||
email=email,
|
email=email,
|
||||||
)
|
)
|
||||||
|
|
||||||
files.template(
|
service_file = files.put(
|
||||||
src=importlib.resources.files(__package__).joinpath("target.yaml.j2"),
|
src=importlib.resources.files(__package__)
|
||||||
dest="/var/lib/acme/conf/target",
|
.joinpath("acmetool-redirector.service")
|
||||||
|
.open("rb"),
|
||||||
|
dest="/etc/systemd/system/acmetool-redirector.service",
|
||||||
user="root",
|
user="root",
|
||||||
group="root",
|
group="root",
|
||||||
mode="644",
|
mode="644",
|
||||||
)
|
)
|
||||||
|
systemd.service(
|
||||||
|
name="Setup acmetool-redirector service",
|
||||||
|
service="acmetool-redirector.service",
|
||||||
|
running=True,
|
||||||
|
enabled=True,
|
||||||
|
restarted=service_file.changed,
|
||||||
|
)
|
||||||
|
|
||||||
for domain in domains:
|
for domain in domains:
|
||||||
server.shell(
|
server.shell(
|
||||||
|
|||||||
@@ -0,0 +1,11 @@
|
|||||||
|
[Unit]
|
||||||
|
Description=acmetool HTTP redirector
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
Type=notify
|
||||||
|
ExecStart=/usr/bin/acmetool redirector --service.uid=daemon
|
||||||
|
Restart=always
|
||||||
|
RestartSec=30
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
request:
|
|
||||||
provider: https://acme-v02.api.letsencrypt.org/directory
|
|
||||||
key:
|
|
||||||
type: rsa
|
|
||||||
challenge:
|
|
||||||
webroot-paths:
|
|
||||||
- /var/www/html/.well-known/acme-challenge
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
|
|
||||||
<clientConfig version="1.1">
|
|
||||||
<emailProvider id="{{ config.domain_name }}">
|
|
||||||
<domain>{{ config.domain_name }}</domain>
|
|
||||||
<displayName>{{ config.domain_name }} chatmail</displayName>
|
|
||||||
<displayShortName>{{ config.domain_name }}</displayShortName>
|
|
||||||
<incomingServer type="imap">
|
|
||||||
<hostname>{{ config.domain_name }}</hostname>
|
|
||||||
<port>993</port>
|
|
||||||
<socketType>SSL</socketType>
|
|
||||||
<authentication>password-cleartext</authentication>
|
|
||||||
<username>%EMAILADDRESS%</username>
|
|
||||||
</incomingServer>
|
|
||||||
<incomingServer type="imap">
|
|
||||||
<hostname>{{ config.domain_name }}</hostname>
|
|
||||||
<port>143</port>
|
|
||||||
<socketType>STARTTLS</socketType>
|
|
||||||
<authentication>password-cleartext</authentication>
|
|
||||||
<username>%EMAILADDRESS%</username>
|
|
||||||
</incomingServer>
|
|
||||||
<outgoingServer type="smtp">
|
|
||||||
<hostname>{{ config.domain_name }}</hostname>
|
|
||||||
<port>465</port>
|
|
||||||
<socketType>SSL</socketType>
|
|
||||||
<authentication>password-cleartext</authentication>
|
|
||||||
<username>%EMAILADDRESS%</username>
|
|
||||||
</outgoingServer>
|
|
||||||
<outgoingServer type="smtp">
|
|
||||||
<hostname>{{ config.domain_name }}</hostname>
|
|
||||||
<port>587</port>
|
|
||||||
<socketType>STARTTLS</socketType>
|
|
||||||
<authentication>password-cleartext</authentication>
|
|
||||||
<username>%EMAILADDRESS%</username>
|
|
||||||
</outgoingServer>
|
|
||||||
</emailProvider>
|
|
||||||
</clientConfig>
|
|
||||||
@@ -4,23 +4,11 @@ protocols = imap lmtp
|
|||||||
|
|
||||||
auth_mechanisms = plain
|
auth_mechanisms = plain
|
||||||
|
|
||||||
{% if debug == true %}
|
|
||||||
auth_verbose = yes
|
auth_verbose = yes
|
||||||
auth_debug = yes
|
auth_debug = yes
|
||||||
auth_debug_passwords = yes
|
auth_debug_passwords = yes
|
||||||
auth_verbose_passwords = plain
|
auth_verbose_passwords = plain
|
||||||
auth_cache_size = 100M
|
auth_cache_size = 100M
|
||||||
mail_debug = yes
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
|
|
||||||
mail_plugins = quota
|
|
||||||
|
|
||||||
# these are the capabilities Delta Chat cares about actually
|
|
||||||
# so let's keep the network overhead per login small
|
|
||||||
# https://github.com/deltachat/deltachat-core-rust/blob/master/src/imap/capabilities.rs
|
|
||||||
imap_capability = IMAP4rev1 IDLE MOVE QUOTA CONDSTORE
|
|
||||||
|
|
||||||
|
|
||||||
# Authentication for system users.
|
# Authentication for system users.
|
||||||
passdb {
|
passdb {
|
||||||
@@ -72,28 +60,13 @@ mail_privileged_group = vmail
|
|||||||
# Enable IMAP COMPRESS (RFC 4978).
|
# Enable IMAP COMPRESS (RFC 4978).
|
||||||
# <https://datatracker.ietf.org/doc/html/rfc4978.html>
|
# <https://datatracker.ietf.org/doc/html/rfc4978.html>
|
||||||
protocol imap {
|
protocol imap {
|
||||||
mail_plugins = $mail_plugins imap_zlib imap_quota
|
mail_plugins = $mail_plugins imap_zlib
|
||||||
}
|
|
||||||
|
|
||||||
protocol lmtp {
|
|
||||||
mail_plugins = $mail_plugins quota
|
|
||||||
}
|
}
|
||||||
|
|
||||||
plugin {
|
plugin {
|
||||||
imap_compress_deflate_level = 6
|
imap_compress_deflate_level = 6
|
||||||
}
|
}
|
||||||
|
|
||||||
plugin {
|
|
||||||
# for now we define static quota-rules for all users
|
|
||||||
quota = maildir:User quota
|
|
||||||
quota_rule = *:storage=100M
|
|
||||||
quota_max_mail_size=30M
|
|
||||||
quota_grace = 0
|
|
||||||
# quota_over_flag_value = TRUE
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
service lmtp {
|
service lmtp {
|
||||||
user=vmail
|
user=vmail
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +0,0 @@
|
|||||||
2 0 * * * dovecot doveadm expunge -A SEEN BEFORE 40d INBOX
|
|
||||||
2 0 * * * dovecot doveadm expunge -A SEEN BEFORE 40d Deltachat
|
|
||||||
2 0 * * * dovecot doveadm expunge -A SEEN BEFORE 40d Trash
|
|
||||||
2 30 * * * dovecot doveadm purge -A
|
|
||||||
@@ -1,47 +0,0 @@
|
|||||||
user www-data;
|
|
||||||
worker_processes auto;
|
|
||||||
pid /run/nginx.pid;
|
|
||||||
error_log /var/log/nginx/error.log;
|
|
||||||
|
|
||||||
events {
|
|
||||||
worker_connections 768;
|
|
||||||
# multi_accept on;
|
|
||||||
}
|
|
||||||
|
|
||||||
http {
|
|
||||||
sendfile on;
|
|
||||||
tcp_nopush on;
|
|
||||||
|
|
||||||
# Do not emit nginx version on error pages.
|
|
||||||
server_tokens off;
|
|
||||||
|
|
||||||
include /etc/nginx/mime.types;
|
|
||||||
default_type application/octet-stream;
|
|
||||||
|
|
||||||
ssl_protocols TLSv1 TLSv1.1 TLSv1.2 TLSv1.3;
|
|
||||||
ssl_prefer_server_ciphers on;
|
|
||||||
ssl_certificate /var/lib/acme/live/{{ config.domain_name }}/fullchain;
|
|
||||||
ssl_certificate_key /var/lib/acme/live/{{ config.domain_name }}/privkey;
|
|
||||||
|
|
||||||
gzip on;
|
|
||||||
|
|
||||||
server {
|
|
||||||
listen 80 default_server;
|
|
||||||
listen [::]:80 default_server;
|
|
||||||
listen 443 ssl default_server;
|
|
||||||
listen [::]:443 ssl default_server;
|
|
||||||
|
|
||||||
root /var/www/html;
|
|
||||||
|
|
||||||
index index.html index.htm;
|
|
||||||
|
|
||||||
server_name _;
|
|
||||||
|
|
||||||
location / {
|
|
||||||
# First attempt to serve request as file, then
|
|
||||||
# as directory, then fall back to displaying a 404.
|
|
||||||
try_files $uri $uri/ =404;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -37,8 +37,6 @@ mydestination =
|
|||||||
relayhost =
|
relayhost =
|
||||||
mynetworks = 127.0.0.0/8 [::ffff:127.0.0.0]/104 [::1]/128
|
mynetworks = 127.0.0.0/8 [::ffff:127.0.0.0]/104 [::1]/128
|
||||||
mailbox_size_limit = 0
|
mailbox_size_limit = 0
|
||||||
# maximum 30MB sized messages
|
|
||||||
message_size_limit = 31457280
|
|
||||||
recipient_delimiter = +
|
recipient_delimiter = +
|
||||||
inet_interfaces = all
|
inet_interfaces = all
|
||||||
inet_protocols = all
|
inet_protocols = all
|
||||||
|
|||||||
@@ -9,11 +9,7 @@
|
|||||||
# service type private unpriv chroot wakeup maxproc command + args
|
# service type private unpriv chroot wakeup maxproc command + args
|
||||||
# (yes) (yes) (no) (never) (100)
|
# (yes) (yes) (no) (never) (100)
|
||||||
# ==========================================================================
|
# ==========================================================================
|
||||||
{% if debug == true %}
|
|
||||||
smtp inet n - y - - smtpd -v
|
|
||||||
{% else %}
|
|
||||||
smtp inet n - y - - smtpd
|
smtp inet n - y - - smtpd
|
||||||
{% endif %}
|
|
||||||
#smtp inet n - y - 1 postscreen
|
#smtp inet n - y - 1 postscreen
|
||||||
#smtpd pass - - y - - smtpd
|
#smtpd pass - - y - - smtpd
|
||||||
#dnsblog unix - - y - 0 dnsblog
|
#dnsblog unix - - y - 0 dnsblog
|
||||||
@@ -32,8 +28,7 @@ submission inet n - y - - smtpd
|
|||||||
-o smtpd_recipient_restrictions=
|
-o smtpd_recipient_restrictions=
|
||||||
-o smtpd_relay_restrictions=permit_sasl_authenticated,reject
|
-o smtpd_relay_restrictions=permit_sasl_authenticated,reject
|
||||||
-o milter_macro_daemon_name=ORIGINATING
|
-o milter_macro_daemon_name=ORIGINATING
|
||||||
-o smtpd_proxy_filter=127.0.0.1:10080
|
-o content_filter=filter:unix:private/filtemail
|
||||||
-o content_filter=filter:unix:private/filtermail-afterqueue
|
|
||||||
smtps inet n - y - - smtpd
|
smtps inet n - y - - smtpd
|
||||||
-o syslog_name=postfix/smtps
|
-o syslog_name=postfix/smtps
|
||||||
-o smtpd_tls_wrappermode=yes
|
-o smtpd_tls_wrappermode=yes
|
||||||
@@ -48,7 +43,7 @@ smtps inet n - y - - smtpd
|
|||||||
-o smtpd_recipient_restrictions=
|
-o smtpd_recipient_restrictions=
|
||||||
-o smtpd_relay_restrictions=permit_sasl_authenticated,reject
|
-o smtpd_relay_restrictions=permit_sasl_authenticated,reject
|
||||||
-o milter_macro_daemon_name=ORIGINATING
|
-o milter_macro_daemon_name=ORIGINATING
|
||||||
-o smtpd_proxy_filter=127.0.0.1:10080
|
-o content_filter=filter:unix:private/filtermail
|
||||||
#628 inet n - y - - qmqpd
|
#628 inet n - y - - qmqpd
|
||||||
pickup unix n - y 60 1 pickup
|
pickup unix n - y 60 1 pickup
|
||||||
cleanup unix n - y - 0 cleanup
|
cleanup unix n - y - 0 cleanup
|
||||||
@@ -71,6 +66,8 @@ showq unix n - y - - showq
|
|||||||
error unix - - y - - error
|
error unix - - y - - error
|
||||||
retry unix - - y - - error
|
retry unix - - y - - error
|
||||||
discard unix - - y - - discard
|
discard unix - - y - - discard
|
||||||
|
local unix - n n - - local
|
||||||
|
virtual unix - n n - - virtual
|
||||||
lmtp unix - - y - - lmtp
|
lmtp unix - - y - - lmtp
|
||||||
anvil unix - - y - 1 anvil
|
anvil unix - - y - 1 anvil
|
||||||
scache unix - - y - 1 scache
|
scache unix - - y - 1 scache
|
||||||
@@ -78,6 +75,4 @@ postlog unix-dgram n - n - 1 postlogd
|
|||||||
filter unix - n n - - lmtp
|
filter unix - n n - - lmtp
|
||||||
# Local SMTP server for reinjecting filered mail.
|
# Local SMTP server for reinjecting filered mail.
|
||||||
localhost:10026 inet n - n - 10 smtpd
|
localhost:10026 inet n - n - 10 smtpd
|
||||||
-o content_filter=filter:unix:private/filtermail-afterqueue
|
|
||||||
localhost:10027 inet n - n - 10 smtpd
|
|
||||||
-o content_filter=
|
-o content_filter=
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
def test_tls_serialized_connect(benchmark, imap_or_smtp):
|
|
||||||
def connect():
|
|
||||||
imap_or_smtp.connect()
|
|
||||||
|
|
||||||
benchmark(connect)
|
|
||||||
|
|
||||||
|
|
||||||
def test_login(benchmark, imap_or_smtp, gencreds):
|
|
||||||
cls = imap_or_smtp.__class__
|
|
||||||
conns = []
|
|
||||||
for i in range(20):
|
|
||||||
conn = cls(imap_or_smtp.host)
|
|
||||||
conn.connect()
|
|
||||||
conns.append(conn)
|
|
||||||
|
|
||||||
def login():
|
|
||||||
conn = conns.pop()
|
|
||||||
conn.login(*gencreds())
|
|
||||||
|
|
||||||
benchmark(login)
|
|
||||||
|
|
||||||
|
|
||||||
def test_send_and_receive_10(benchmark, cmfactory, lp):
|
|
||||||
"""send many messages between two accounts"""
|
|
||||||
ac1, ac2 = cmfactory.get_online_accounts(2)
|
|
||||||
chat = cmfactory.get_accepted_chat(ac1, ac2)
|
|
||||||
|
|
||||||
def send_10_receive_all():
|
|
||||||
for i in range(10):
|
|
||||||
chat.send_text(f"hello {i}")
|
|
||||||
for i in range(10):
|
|
||||||
ac2.wait_next_incoming_message()
|
|
||||||
|
|
||||||
benchmark(send_10_receive_all)
|
|
||||||
@@ -1,57 +1,15 @@
|
|||||||
import os
|
import os
|
||||||
import io
|
import io
|
||||||
import random
|
|
||||||
import subprocess
|
|
||||||
import imaplib
|
import imaplib
|
||||||
import smtplib
|
import smtplib
|
||||||
import itertools
|
import itertools
|
||||||
import pytest
|
import pytest
|
||||||
|
import time
|
||||||
|
|
||||||
def pytest_addoption(parser):
|
|
||||||
parser.addoption(
|
|
||||||
"--slow", action="store_true", default=False, help="also run slow tests"
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def pytest_runtest_setup(item):
|
|
||||||
markers = list(item.iter_markers(name="slow"))
|
|
||||||
if markers:
|
|
||||||
if not item.config.getoption("--slow"):
|
|
||||||
pytest.skip("skipping slow test, use --slow to run")
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def maildomain():
|
def maildomain():
|
||||||
domain = os.environ.get("CHATMAIL_DOMAIN")
|
return os.environ.get("CHATMAIL_DOMAIN", "c1.testrun.org")
|
||||||
if not domain:
|
|
||||||
pytest.skip("set CHATMAIL_DOMAIN to a ssh-reachable chatmail instance")
|
|
||||||
return domain
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def sshdomain(maildomain):
|
|
||||||
return os.environ.get("CHATMAIL_SSH", maildomain)
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def maildomain2():
|
|
||||||
domain = os.environ.get("CHATMAIL_DOMAIN2")
|
|
||||||
if not domain:
|
|
||||||
pytest.skip("set CHATMAIL_DOMAIN2 to a ssh-reachable chatmail instance")
|
|
||||||
return domain
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def sshdomain2(maildomain2):
|
|
||||||
return os.environ.get("CHATMAIL_SSH2", maildomain2)
|
|
||||||
|
|
||||||
|
|
||||||
def pytest_report_header():
|
|
||||||
domain = os.environ.get("CHATMAIL_DOMAIN")
|
|
||||||
if domain:
|
|
||||||
text = f"chatmail test instance: {domain}"
|
|
||||||
return ["-" * len(text), text, "-" * len(text)]
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
@@ -60,10 +18,6 @@ def imap(maildomain):
|
|||||||
|
|
||||||
|
|
||||||
class ImapConn:
|
class ImapConn:
|
||||||
AuthError = imaplib.IMAP4.error
|
|
||||||
logcmd = "journalctl -f -u dovecot"
|
|
||||||
name = "dovecot"
|
|
||||||
|
|
||||||
def __init__(self, host):
|
def __init__(self, host):
|
||||||
self.host = host
|
self.host = host
|
||||||
|
|
||||||
@@ -75,24 +29,6 @@ class ImapConn:
|
|||||||
print(f"imap-login {user!r} {password!r}")
|
print(f"imap-login {user!r} {password!r}")
|
||||||
self.conn.login(user, password)
|
self.conn.login(user, password)
|
||||||
|
|
||||||
def fetch_all(self):
|
|
||||||
print("imap-fetch all")
|
|
||||||
status, res = self.conn.select()
|
|
||||||
if int(res[0]) == 0:
|
|
||||||
raise ValueError("no messages in imap folder")
|
|
||||||
status, results = self.conn.fetch("1:*", "(RFC822)")
|
|
||||||
assert status == "OK"
|
|
||||||
return results
|
|
||||||
|
|
||||||
def fetch_all_messages(self):
|
|
||||||
print("imap-fetch all messages")
|
|
||||||
results = self.fetch_all()
|
|
||||||
messages = []
|
|
||||||
for item in results:
|
|
||||||
if len(item) == 2:
|
|
||||||
messages.append(item[1].decode())
|
|
||||||
return messages
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def smtp(maildomain):
|
def smtp(maildomain):
|
||||||
@@ -100,10 +36,6 @@ def smtp(maildomain):
|
|||||||
|
|
||||||
|
|
||||||
class SmtpConn:
|
class SmtpConn:
|
||||||
AuthError = smtplib.SMTPAuthenticationError
|
|
||||||
logcmd = "journalctl -f -t postfix/smtpd -t postfix/smtp -t postfix/lmtp"
|
|
||||||
name = "postfix"
|
|
||||||
|
|
||||||
def __init__(self, host):
|
def __init__(self, host):
|
||||||
self.host = host
|
self.host = host
|
||||||
|
|
||||||
@@ -115,33 +47,18 @@ class SmtpConn:
|
|||||||
print(f"smtp-login {user!r} {password!r}")
|
print(f"smtp-login {user!r} {password!r}")
|
||||||
self.conn.login(user, password)
|
self.conn.login(user, password)
|
||||||
|
|
||||||
def sendmail(self, from_addr, to_addrs, msg):
|
|
||||||
print(f"smtp-sendmail from={from_addr!r} to_addrs={to_addrs!r}")
|
|
||||||
print(f"smtp-sendmail message size: {len(msg)}")
|
|
||||||
return self.conn.sendmail(from_addr=from_addr, to_addrs=to_addrs, msg=msg)
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(params=["imap", "smtp"])
|
|
||||||
def imap_or_smtp(request):
|
|
||||||
return request.getfixturevalue(request.param)
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def gencreds(maildomain):
|
def gencreds(maildomain):
|
||||||
|
prefix = str(time.time())
|
||||||
count = itertools.count()
|
count = itertools.count()
|
||||||
next(count)
|
|
||||||
|
|
||||||
def gen(domain=None):
|
def gen():
|
||||||
domain = domain if domain else maildomain
|
|
||||||
while 1:
|
while 1:
|
||||||
num = next(count)
|
num = next(count)
|
||||||
alphanumeric = "abcdefghijklmnopqrstuvwxyz1234567890"
|
yield f"user{prefix}_{num}@{maildomain}", f"password{prefix}_{num}"
|
||||||
user = "".join(random.choices(alphanumeric, k=10))
|
|
||||||
user = f"ac{num}_{user}"
|
|
||||||
password = "".join(random.choices(alphanumeric, k=10))
|
|
||||||
yield f"{user}@{domain}", f"{password}"
|
|
||||||
|
|
||||||
return lambda domain=None: next(gen(domain))
|
return lambda: next(gen())
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
@@ -156,20 +73,13 @@ class ChatmailTestProcess:
|
|||||||
def __init__(self, pytestconfig, maildomain, gencreds):
|
def __init__(self, pytestconfig, maildomain, gencreds):
|
||||||
self.pytestconfig = pytestconfig
|
self.pytestconfig = pytestconfig
|
||||||
self.maildomain = maildomain
|
self.maildomain = maildomain
|
||||||
assert "." in self.maildomain, maildomain
|
|
||||||
self.gencreds = gencreds
|
self.gencreds = gencreds
|
||||||
self._addr2files = {}
|
self._addr2files = {}
|
||||||
|
|
||||||
def get_liveconfig_producer(self):
|
def get_liveconfig_producer(self):
|
||||||
while 1:
|
while 1:
|
||||||
user, password = self.gencreds(self.maildomain)
|
user, password = self.gencreds()
|
||||||
config = {
|
config = {"addr": user, "mail_pw": password}
|
||||||
"addr": user,
|
|
||||||
"mail_pw": password,
|
|
||||||
}
|
|
||||||
# speed up account configuration
|
|
||||||
config["mail_server"] = self.maildomain
|
|
||||||
config["send_server"] = self.maildomain
|
|
||||||
yield config
|
yield config
|
||||||
|
|
||||||
def cache_maybe_retrieve_configured_db_files(self, cache_addr, db_target_path):
|
def cache_maybe_retrieve_configured_db_files(self, cache_addr, db_target_path):
|
||||||
@@ -180,107 +90,17 @@ class ChatmailTestProcess:
|
|||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def cmfactory(request, gencreds, tmpdir, data, maildomain):
|
def cmfactory(request, maildomain, gencreds, tmpdir, data):
|
||||||
# cloned from deltachat.testplugin.amfactory
|
# cloned from deltachat.testplugin.amfactory
|
||||||
pytest.importorskip("deltachat")
|
pytest.importorskip("deltachat")
|
||||||
from deltachat.testplugin import ACFactory
|
from deltachat.testplugin import ACFactory
|
||||||
|
|
||||||
testproc = ChatmailTestProcess(request.config, maildomain, gencreds)
|
testproc = ChatmailTestProcess(request.config, maildomain, gencreds)
|
||||||
am = ACFactory(request=request, tmpdir=tmpdir, testprocess=testproc, data=data)
|
am = ACFactory(request=request, tmpdir=tmpdir, testprocess=testproc, data=data)
|
||||||
|
|
||||||
# nb. a bit hacky
|
|
||||||
# would probably be better if deltachat's test machinery grows native support
|
|
||||||
def switch_maildomain(maildomain2):
|
|
||||||
am.testprocess.maildomain = maildomain2
|
|
||||||
|
|
||||||
am.switch_maildomain = switch_maildomain
|
|
||||||
|
|
||||||
yield am
|
yield am
|
||||||
if hasattr(request.node, "rep_call") and request.node.rep_call.failed:
|
if hasattr(request.node, "rep_call") and request.node.rep_call.failed:
|
||||||
if testproc.pytestconfig.getoption("--extra-info"):
|
if testprocess.pytestconfig.getoption("--extra-info"):
|
||||||
logfile = io.StringIO()
|
logfile = io.StringIO()
|
||||||
am.dump_imap_summary(logfile=logfile)
|
am.dump_imap_summary(logfile=logfile)
|
||||||
print(logfile.getvalue())
|
print(logfile.getvalue())
|
||||||
# request.node.add_report_section("call", "imap-server-state", s)
|
# request.node.add_report_section("call", "imap-server-state", s)
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def remote(sshdomain):
|
|
||||||
return Remote(sshdomain)
|
|
||||||
|
|
||||||
|
|
||||||
class Remote:
|
|
||||||
def __init__(self, sshdomain):
|
|
||||||
self.sshdomain = sshdomain
|
|
||||||
|
|
||||||
def iter_output(self, logcmd=""):
|
|
||||||
getjournal = f"journalctl -f" if not logcmd else logcmd
|
|
||||||
self.popen = subprocess.Popen(
|
|
||||||
["ssh", f"root@{self.sshdomain}", getjournal],
|
|
||||||
stdout=subprocess.PIPE,
|
|
||||||
)
|
|
||||||
while 1:
|
|
||||||
line = self.popen.stdout.readline()
|
|
||||||
res = line.decode().strip().lower()
|
|
||||||
if res:
|
|
||||||
yield res
|
|
||||||
else:
|
|
||||||
break
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def mailgen(request):
|
|
||||||
class Mailgen:
|
|
||||||
def get_encrypted(self, from_addr, to_addr):
|
|
||||||
data = request.fspath.dirpath("mailgen/encrypted.eml").read()
|
|
||||||
return data.format(from_addr=from_addr, to_addr=to_addr)
|
|
||||||
|
|
||||||
return Mailgen()
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def cmsetup(maildomain, gencreds):
|
|
||||||
return CMSetup(maildomain, gencreds)
|
|
||||||
|
|
||||||
|
|
||||||
class CMSetup:
|
|
||||||
def __init__(self, maildomain, gencreds):
|
|
||||||
self.maildomain = maildomain
|
|
||||||
self.gencreds = gencreds
|
|
||||||
|
|
||||||
def gen_users(self, num):
|
|
||||||
print(f"Creating {num} online users")
|
|
||||||
users = []
|
|
||||||
for i in range(num):
|
|
||||||
addr, password = self.gencreds()
|
|
||||||
user = CMUser(self.maildomain, addr, password)
|
|
||||||
assert user.smtp
|
|
||||||
users.append(user)
|
|
||||||
return users
|
|
||||||
|
|
||||||
|
|
||||||
class CMUser:
|
|
||||||
def __init__(self, maildomain, addr, password):
|
|
||||||
self.maildomain = maildomain
|
|
||||||
self.addr = addr
|
|
||||||
self.password = password
|
|
||||||
self._smtp = None
|
|
||||||
self._imap = None
|
|
||||||
|
|
||||||
@property
|
|
||||||
def smtp(self):
|
|
||||||
if not self._smtp:
|
|
||||||
handle = SmtpConn(self.maildomain)
|
|
||||||
handle.connect()
|
|
||||||
handle.login(self.addr, self.password)
|
|
||||||
self._smtp = handle
|
|
||||||
return self._smtp
|
|
||||||
|
|
||||||
@property
|
|
||||||
def imap(self):
|
|
||||||
if not self._imap:
|
|
||||||
imap = ImapConn(self.maildomain)
|
|
||||||
imap.connect()
|
|
||||||
imap.login(self.addr, self.password)
|
|
||||||
self._imap = imap
|
|
||||||
return self._imap
|
|
||||||
|
|||||||
@@ -1,66 +0,0 @@
|
|||||||
From: {from_addr}
|
|
||||||
To: {to_addr}
|
|
||||||
Subject: ...
|
|
||||||
Date: Sun, 15 Oct 2023 16:43:21 +0000
|
|
||||||
Message-ID: <Mr.UVyJWZmkCKM.hGzNc6glBE_@c2.testrun.org>
|
|
||||||
In-Reply-To: <Mr.MvmCz-GQbi_.6FGRkhDf05c@c2.testrun.org>
|
|
||||||
References: <Mr.3gckbNy5bch.uK3Hd2Ws6-w@c2.testrun.org>
|
|
||||||
<Mr.MvmCz-GQbi_.6FGRkhDf05c@c2.testrun.org>
|
|
||||||
Chat-Version: 1.0
|
|
||||||
Autocrypt: addr={from_addr}; prefer-encrypt=mutual;
|
|
||||||
keydata=xjMEZSwWjhYJKwYBBAHaRw8BAQdAQBEhqeJh0GueHB6kF/DUQqYCxARNBVokg/AzT+7LqH
|
|
||||||
rNFzxiYXJiYXpAYzIudGVzdHJ1bi5vcmc+wosEEBYIADMCGQEFAmUsFo4CGwMECwkIBwYVCAkKCwID
|
|
||||||
FgIBFiEEFTfUNvVnY3b9F7yHnmme1PfUhX8ACgkQnmme1PfUhX9A4AEAnHWHp49eBCMHK5t66gYPiW
|
|
||||||
XQuB1mwUjzGfYWB+0RXUoA/0xcQ3FbUNlGKW7Blp6eMFfViv6Mv2d3kNSXACB6nmcMzjgEZSwWjhIK
|
|
||||||
KwYBBAGXVQEFAQEHQBpY5L2M1XHo0uxf8SX1wNLBp/OVvidoWHQF2Jz+kJsUAwEIB8J4BBgWCAAgBQ
|
|
||||||
JlLBaOAhsMFiEEFTfUNvVnY3b9F7yHnmme1PfUhX8ACgkQnmme1PfUhX/INgEA37AJaNvruYsJVanP
|
|
||||||
IXnYw4CKd55UAwl8Zcy+M2diAbkA/0fHHcGV4r78hpbbL1Os52DPOdqYQRauIeJUeG+G6bQO
|
|
||||||
MIME-Version: 1.0
|
|
||||||
Content-Type: multipart/encrypted; protocol="application/pgp-encrypted";
|
|
||||||
boundary="YFrteb74qSXmggbOxZL9dRnhymywAi"
|
|
||||||
|
|
||||||
|
|
||||||
--YFrteb74qSXmggbOxZL9dRnhymywAi
|
|
||||||
Content-Description: PGP/MIME version identification
|
|
||||||
Content-Type: application/pgp-encrypted
|
|
||||||
|
|
||||||
Version: 1
|
|
||||||
|
|
||||||
|
|
||||||
--YFrteb74qSXmggbOxZL9dRnhymywAi
|
|
||||||
Content-Description: OpenPGP encrypted message
|
|
||||||
Content-Disposition: inline; filename="encrypted.asc";
|
|
||||||
Content-Type: application/octet-stream; name="encrypted.asc"
|
|
||||||
|
|
||||||
-----BEGIN PGP MESSAGE-----
|
|
||||||
|
|
||||||
wU4DhW3gBZ/VvCYSAQdA8bMs2spwbKdGjVsL1ByPkNrqD7frpB73maeL6I6SzDYg
|
|
||||||
O5G53tv339RdKq3WRcCtEEvxjHlUx2XNwXzC04BpmfvBTgNfPUyLDzjXnxIBB0Ae
|
|
||||||
8ymwGvXMCCimHXN0Dg8Ui62KOi03h0UgheoHWovJSCDF4CKre/xtFr3nL7lq/PKI
|
|
||||||
JsjVNz7/RK9FSXF6WwfONtLCyQGEuVAsB/KXfCBEyfKhaMwGHvhujRidGW5uV1no
|
|
||||||
lMGl3ODmo29Lgeu2uSE7EpJRZoe6hU6ddmBkqxax61ZtkaFlGFFpdo2K8balNNdz
|
|
||||||
ZsJ/9mmI9x3oOJ4/l1nhQbUO9ADbs7gJhFdV5Qkp30b5fCI7bU+aoe1ccBbLe/WM
|
|
||||||
YUty1PqcuQT7XjA+XmYuL261tvW8pBetT+i33/E2d8PzzYt2IuK9qeevyS+yxdwA
|
|
||||||
kfwejFWzzsUlJaDxs1x4XOxkMgSj+jo+g12dFOb7fyClsAnq23iDb8AuaT/BScAI
|
|
||||||
+lO+gher69+6LmM7VGHLG5k762J1jTaQCaKt1s8TAWV99Eo4491vL6fyvk3l/Cfg
|
|
||||||
RXSwiWFgj19Pn0Rq7CD9v22UE2vdUMBTcV4aw79mClk1YQ23jbF0y5DCjPdJ62Zo
|
|
||||||
tskBgFt3NoWV80jZ76zIBLrrjLwCCll8JjJtFwSkt2GX5RFBsVa4A8IDht9RtEk7
|
|
||||||
rrHgbSZQfkauEi/mH3/6CDZoLqSHudUZ7d4MaJwun1TkFYGe2ORwGJd4OBj3oGJp
|
|
||||||
H8YBwCpk///L/fKjX0Gg3M8nrpM4wrRFhPKidAgO/kcm25X4+ZHlVkWBTCt5RWKI
|
|
||||||
fHh6oLDZCqCfcgMkE1KKmwfIHaUkhq5BPRigwy6i5dh1DM4+1UCLh3dxzVbqE9b9
|
|
||||||
61NB19nXdRtDA2sOUnj9ve6m/wEPyCb6/zBQZqvCBYb1/AjdXpUrFT+DbpfyxaXN
|
|
||||||
XfhDVb5mNqNM/IVj0V5fvTc6vOfYbzQtPm10H+FdWWfb+rJRfyC3MA2w2IqstFe3
|
|
||||||
w3bu2iE6CQvSqRvge+ZqLKt/NqYwOURiUmpuklbl3kPJ97+mfKWoiqk8Iz1VY+bb
|
|
||||||
NMUC7aoGv+jcoj+WS6PYO8N6BeRVUUB3ZJSf8nzjgxm1/BcM+UD3BPrlhT11ODRs
|
|
||||||
baifGbprMWwt3dhb8cQgRT8GPdpO1OsDkzL6iikMjLHWWiA99GV6ruiHsIPw6boW
|
|
||||||
A6/uSOskbDHOROotKmddGTBd0iiHXAoQsJFt1ZjUkt6EHrgWs+GAvrvKpXs1mrz8
|
|
||||||
uj3GwEFrHS+Xuf2UDgpszYT3hI2cL/kUtGakVR7m7vVMZqXBUbZdGAEb1PZNPwsI
|
|
||||||
E4aMK02+EVB+tSN4Fzj99N2YD0inVYt+oPjr2tHhUS6aSGBNS/48Ki47DOg4Sxkn
|
|
||||||
lkOWnEbCD+XTnbDd
|
|
||||||
=agR5
|
|
||||||
-----END PGP MESSAGE-----
|
|
||||||
|
|
||||||
|
|
||||||
--YFrteb74qSXmggbOxZL9dRnhymywAi--
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,3 +1,2 @@
|
|||||||
[pytest]
|
[pytest]
|
||||||
addopts = -vrsx --strict-markers
|
addopts = -vrsx
|
||||||
markers = slow: mark test as slow (requires --slow option to run)
|
|
||||||
|
|||||||
@@ -1,52 +0,0 @@
|
|||||||
import smtplib
|
|
||||||
import pytest
|
|
||||||
|
|
||||||
|
|
||||||
def test_remote(remote, imap_or_smtp):
|
|
||||||
lineproducer = remote.iter_output(imap_or_smtp.logcmd)
|
|
||||||
imap_or_smtp.connect()
|
|
||||||
assert imap_or_smtp.name in next(lineproducer)
|
|
||||||
|
|
||||||
|
|
||||||
def test_use_two_chatmailservers(cmfactory, maildomain2):
|
|
||||||
ac1 = cmfactory.new_online_configuring_account(cache=False)
|
|
||||||
cmfactory.switch_maildomain(maildomain2)
|
|
||||||
ac2 = cmfactory.new_online_configuring_account(cache=False)
|
|
||||||
cmfactory.bring_accounts_online()
|
|
||||||
cmfactory.get_accepted_chat(ac1, ac2)
|
|
||||||
domain1 = ac1.get_config("addr").split("@")[1]
|
|
||||||
domain2 = ac2.get_config("addr").split("@")[1]
|
|
||||||
assert domain1 != domain2
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("forgeaddr", ["internal", "someone@example.org"])
|
|
||||||
def test_reject_forged_from(cmsetup, mailgen, lp, remote, forgeaddr):
|
|
||||||
user1, user3 = cmsetup.gen_users(2)
|
|
||||||
|
|
||||||
lp.sec("send encrypted message with forged from")
|
|
||||||
print("envelope_from", user1.addr)
|
|
||||||
if forgeaddr == "internal":
|
|
||||||
addr_to_forge = cmsetup.gen_users(1)[0].addr
|
|
||||||
else:
|
|
||||||
addr_to_forge = "someone@example.org"
|
|
||||||
|
|
||||||
print("message to inject:")
|
|
||||||
msg = mailgen.get_encrypted(from_addr=addr_to_forge, to_addr=user3.addr)
|
|
||||||
for line in msg.split("\n")[:4]:
|
|
||||||
print(f" {line}")
|
|
||||||
|
|
||||||
lp.sec("Send forged mail and check remote postfix lmtp processing result")
|
|
||||||
remote_log = remote.iter_output("journalctl -t postfix/lmtp")
|
|
||||||
user1.smtp.sendmail(from_addr=user1.addr, to_addrs=[user3.addr], msg=msg)
|
|
||||||
for line in remote_log:
|
|
||||||
# print(line)
|
|
||||||
if "500 invalid from" in line and user3.addr in line:
|
|
||||||
break
|
|
||||||
else:
|
|
||||||
pytest.fail("remote postfix/filtermail failed to reject message")
|
|
||||||
|
|
||||||
# check that the logged in user (who sent the forged msg) got a non-delivery notice
|
|
||||||
for message in user1.imap.fetch_all_messages():
|
|
||||||
if "Invalid FROM" in message and addr_to_forge in message:
|
|
||||||
return
|
|
||||||
pytest.fail(f"forged From={addr_to_forge} did not cause non-delivery notice")
|
|
||||||
@@ -1,56 +1,55 @@
|
|||||||
import pytest
|
import pytest
|
||||||
|
import imaplib
|
||||||
import smtplib
|
import smtplib
|
||||||
|
|
||||||
|
|
||||||
def test_login_basic_functioning(imap_or_smtp, gencreds, lp):
|
class TestDovecot:
|
||||||
"""Test a) that an initial login creates a user automatically
|
def test_login_ok(self, imap, gencreds):
|
||||||
and b) verify we can also login a second time with the same password
|
user, password = gencreds()
|
||||||
and c) that using a different password fails the login."""
|
imap.connect()
|
||||||
user, password = gencreds()
|
imap.login(user, password)
|
||||||
lp.sec(f"login first time with {user} {password}")
|
# verify it works on another connection
|
||||||
imap_or_smtp.connect()
|
imap.connect()
|
||||||
imap_or_smtp.login(user, password)
|
imap.login(user, password)
|
||||||
lp.indent("success")
|
|
||||||
|
|
||||||
lp.sec(f"reconnect and login second time {user} {password}")
|
def test_login_same_password(self, imap, gencreds):
|
||||||
imap_or_smtp.connect()
|
"""Test two different users logging in with the same password.
|
||||||
imap_or_smtp.login(user, password)
|
|
||||||
imap_or_smtp.connect()
|
|
||||||
lp.sec("success")
|
|
||||||
|
|
||||||
lp.sec(f"reconnect and verify wrong password fails {user} ")
|
This ensures that authentication process does not confuse the users
|
||||||
imap_or_smtp.connect()
|
by using only the password hash as a key.
|
||||||
with pytest.raises(imap_or_smtp.AuthError):
|
"""
|
||||||
imap_or_smtp.login(user, password + "wrong")
|
user1, password1 = gencreds()
|
||||||
|
user2, _password2 = gencreds()
|
||||||
|
imap.connect()
|
||||||
|
imap.login(user1, password1)
|
||||||
|
imap.connect()
|
||||||
|
imap.login(user2, password1)
|
||||||
|
|
||||||
|
def test_login_fail(self, imap, gencreds):
|
||||||
|
user, password = gencreds()
|
||||||
|
imap.connect()
|
||||||
|
imap.login(user, password)
|
||||||
|
imap.connect()
|
||||||
|
with pytest.raises(imaplib.IMAP4.error) as excinfo:
|
||||||
|
imap.login(user, password + "wrong")
|
||||||
|
assert "AUTHENTICATIONFAILED" in str(excinfo)
|
||||||
|
|
||||||
|
|
||||||
def test_login_same_password(imap_or_smtp, gencreds):
|
class TestPostfix:
|
||||||
"""Test two different users logging in with the same password
|
def test_login_ok(self, smtp, gencreds):
|
||||||
to ensure that authentication process does not confuse the users
|
user, password = gencreds()
|
||||||
by using only the password hash as a key.
|
smtp.connect()
|
||||||
"""
|
smtp.login(user, password)
|
||||||
user1, password1 = gencreds()
|
# verify it works on another connection
|
||||||
user2, _ = gencreds()
|
smtp.connect()
|
||||||
imap_or_smtp.connect()
|
smtp.login(user, password)
|
||||||
imap_or_smtp.login(user1, password1)
|
|
||||||
imap_or_smtp.connect()
|
|
||||||
imap_or_smtp.login(user2, password1)
|
|
||||||
|
|
||||||
|
def test_login_fail(self, smtp, gencreds):
|
||||||
@pytest.mark.slow
|
user, password = gencreds()
|
||||||
def test_exceed_rate_limit(cmsetup, gencreds, mailgen):
|
smtp.connect()
|
||||||
"""Test that the per-account send-mail limit is exceeded."""
|
smtp.login(user, password)
|
||||||
user1, user2 = cmsetup.gen_users(2)
|
smtp.connect()
|
||||||
mail = mailgen.get_encrypted(user1.addr, user2.addr)
|
with pytest.raises(smtplib.SMTPAuthenticationError) as excinfo:
|
||||||
for i in range(100):
|
smtp.login(user, password + "wrong")
|
||||||
print("Sending mail", str(i))
|
assert excinfo.value.smtp_code == 535
|
||||||
try:
|
assert "authentication failed" in str(excinfo)
|
||||||
user1.smtp.sendmail(user1.addr, [user2.addr], mail)
|
|
||||||
except smtplib.SMTPException as e:
|
|
||||||
if i < 80:
|
|
||||||
pytest.fail(f"rate limit was exceeded too early with msg {i}")
|
|
||||||
outcome = e.recipients[user2.addr]
|
|
||||||
assert outcome[0] == 450
|
|
||||||
assert b'4.7.1: Too much mail from' in outcome[1]
|
|
||||||
return
|
|
||||||
pytest.fail("Rate limit was not exceeded")
|
|
||||||
|
|||||||
@@ -1,83 +1,11 @@
|
|||||||
import random
|
class TestMailSending:
|
||||||
import pytest
|
|
||||||
|
|
||||||
|
|
||||||
class TestEndToEndDeltaChat:
|
|
||||||
"Tests that use Delta Chat accounts on the chat mail instance."
|
|
||||||
|
|
||||||
def test_one_on_one(self, cmfactory, lp):
|
def test_one_on_one(self, cmfactory, lp):
|
||||||
"""Test that a DC account can send a message to a second DC account
|
|
||||||
on the same chat-mail instance."""
|
|
||||||
ac1, ac2 = cmfactory.get_online_accounts(2)
|
ac1, ac2 = cmfactory.get_online_accounts(2)
|
||||||
chat = cmfactory.get_accepted_chat(ac1, ac2)
|
chat = cmfactory.get_accepted_chat(ac1, ac2)
|
||||||
|
|
||||||
lp.sec("ac1: prepare and send text message to ac2")
|
lp.sec("ac1: prepare and send text message to ac2")
|
||||||
chat.send_text("message0")
|
msg1 = chat.send_text("message0")
|
||||||
|
|
||||||
lp.sec("wait for ac2 to receive message")
|
lp.sec("wait for ac2 to receive message")
|
||||||
msg2 = ac2._evtracker.wait_next_incoming_message()
|
msg2 = ac2._evtracker.wait_next_incoming_message()
|
||||||
assert msg2.text == "message0"
|
assert msg2.text == "message0"
|
||||||
|
|
||||||
@pytest.mark.slow
|
|
||||||
def test_exceed_quota(self, cmfactory, lp, tmpdir, remote):
|
|
||||||
"""This is a very slow test as it needs to upload >100MB of mail data
|
|
||||||
before quota is exceeded, and thus depends on the speed of the upload.
|
|
||||||
"""
|
|
||||||
ac1, ac2 = cmfactory.get_online_accounts(2)
|
|
||||||
chat = cmfactory.get_accepted_chat(ac1, ac2)
|
|
||||||
|
|
||||||
quota = 1024 * 1024 * 100
|
|
||||||
attachsize = 1 * 1024 * 1024
|
|
||||||
num_to_send = quota // attachsize + 2
|
|
||||||
lp.sec(f"ac1: send {num_to_send} large files to ac2")
|
|
||||||
lp.indent(f"per-user quota is assumed to be: {quota/(1024*1024)}MB")
|
|
||||||
alphanumeric = "abcdefghijklmnopqrstuvwxyz1234567890"
|
|
||||||
msgs = []
|
|
||||||
for i in range(num_to_send):
|
|
||||||
attachment = tmpdir / f"attachment{i}"
|
|
||||||
data = "".join(random.choice(alphanumeric) for i in range(1024))
|
|
||||||
with open(attachment, "w+") as f:
|
|
||||||
for j in range(attachsize // len(data)):
|
|
||||||
f.write(data)
|
|
||||||
|
|
||||||
msg = chat.send_file(str(attachment))
|
|
||||||
msgs.append(msg)
|
|
||||||
lp.indent(f"Sent out msg {i}, size {attachsize/(1024*1024)}MB")
|
|
||||||
|
|
||||||
lp.sec("ac2: check messages are arriving until quota is reached")
|
|
||||||
|
|
||||||
addr = ac2.get_config("addr").lower()
|
|
||||||
saved_ok = 0
|
|
||||||
for line in remote.iter_output("journalctl -f -u dovecot"):
|
|
||||||
if addr not in line:
|
|
||||||
# print(line)
|
|
||||||
continue
|
|
||||||
if "quota" in line:
|
|
||||||
if "quota exceeded" in line:
|
|
||||||
if saved_ok < num_to_send // 2:
|
|
||||||
pytest.fail(
|
|
||||||
f"quota exceeded too early: after {saved_ok} messages already"
|
|
||||||
)
|
|
||||||
lp.indent("good, message sending failed because quota was exceeded")
|
|
||||||
return
|
|
||||||
if "saved mail to inbox" in line:
|
|
||||||
saved_ok += 1
|
|
||||||
print(f"{saved_ok}: {line}")
|
|
||||||
if saved_ok >= num_to_send:
|
|
||||||
break
|
|
||||||
|
|
||||||
pytest.fail("sending succeeded although messages should exceed quota")
|
|
||||||
|
|
||||||
def test_securejoin(self, cmfactory, lp, maildomain2):
|
|
||||||
ac1 = cmfactory.new_online_configuring_account(cache=False)
|
|
||||||
cmfactory.switch_maildomain(maildomain2)
|
|
||||||
ac2 = cmfactory.new_online_configuring_account(cache=False)
|
|
||||||
cmfactory.bring_accounts_online()
|
|
||||||
|
|
||||||
lp.sec("ac1: create QR code and let ac2 scan it, starting the securejoin")
|
|
||||||
qr = ac1.get_setup_contact_qr()
|
|
||||||
|
|
||||||
lp.sec("ac2: start QR-code based setup contact protocol")
|
|
||||||
ch = ac2.qr_setup_contact(qr)
|
|
||||||
assert ch.id >= 10
|
|
||||||
ac1._evtracker.wait_securejoin_inviter_progress(1000)
|
|
||||||
|
|||||||
15
plan.txt
15
plan.txt
@@ -2,10 +2,19 @@
|
|||||||
|
|
||||||
## Dovecot goals/steps
|
## Dovecot goals/steps
|
||||||
|
|
||||||
- automatic expiry of messages older than M days
|
2. (holger) per-user storage quota (adaptive)
|
||||||
- also expunge unread messages
|
a) define a static 100MB per-user quota
|
||||||
|
|
||||||
- limit: configure max-connections per account
|
3. automatic expiry of messages older than M days
|
||||||
|
- delete unconditionally messages older than 40 days
|
||||||
|
|
||||||
|
4. limit: max-connections per account
|
||||||
|
|
||||||
|
|
||||||
|
## Filtermail
|
||||||
|
|
||||||
|
- (alex, Only allow (outgoing) mails if secure-join or autocrypt-pgp-encrypted format.
|
||||||
|
TODO: mime-parse mails and check/add tests
|
||||||
|
|
||||||
|
|
||||||
## nami: send out rate limit / rspamd
|
## nami: send out rate limit / rspamd
|
||||||
|
|||||||
@@ -1,4 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -e
|
|
||||||
|
|
||||||
online-tests/venv/bin/pytest online-tests/benchmark.py -vrx
|
|
||||||
@@ -2,9 +2,8 @@
|
|||||||
: ${CHATMAIL_DOMAIN:=c1.testrun.org}
|
: ${CHATMAIL_DOMAIN:=c1.testrun.org}
|
||||||
export CHATMAIL_DOMAIN
|
export CHATMAIL_DOMAIN
|
||||||
|
|
||||||
chatmaild/venv/bin/python3 -m build -n --sdist chatmaild --outdir dist
|
venv/bin/python3 -m build -n --sdist chatmaild --outdir dist
|
||||||
|
|
||||||
deploy-chatmail/venv/bin/pyinfra --ssh-user root "$CHATMAIL_DOMAIN" \
|
deploy-chatmail/venv/bin/pyinfra --ssh-user root "$CHATMAIL_DOMAIN" deploy.py
|
||||||
deploy-chatmail/src/deploy_chatmail/deploy.py
|
|
||||||
|
|
||||||
rm -r dist/
|
rm -r dist/
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ domain = os.environ.get("CHATMAIL_DOMAIN", "c3.testrun.org")
|
|||||||
print("connecting")
|
print("connecting")
|
||||||
conn = imaplib.IMAP4_SSL(domain)
|
conn = imaplib.IMAP4_SSL(domain)
|
||||||
print("logging in")
|
print("logging in")
|
||||||
conn.login(f"imapcapa", "pass")
|
conn.login(f"measure{time.time()}", "pass")
|
||||||
status, res = conn.capability()
|
status, res = conn.capability()
|
||||||
for capa in sorted(res[0].decode().split()):
|
for capa in sorted(res[0].decode().split()):
|
||||||
print(capa)
|
print(capa)
|
||||||
|
|||||||
@@ -6,9 +6,12 @@ deploy-chatmail/venv/bin/pip install -e deploy-chatmail
|
|||||||
deploy-chatmail/venv/bin/pip install -e chatmaild
|
deploy-chatmail/venv/bin/pip install -e chatmaild
|
||||||
|
|
||||||
python3 -m venv chatmaild/venv
|
python3 -m venv chatmaild/venv
|
||||||
sudo apt install -y dovecot-core && sudo systemctl disable --now dovecot
|
chatmaild/venv/bin/pip install pytest
|
||||||
chatmaild/venv/bin/pip install --upgrade pytest build 'setuptools>=68'
|
|
||||||
chatmaild/venv/bin/pip install -e chatmaild
|
chatmaild/venv/bin/pip install -e chatmaild
|
||||||
|
|
||||||
python3 -m venv online-tests/venv
|
python3 -m venv online-tests/venv
|
||||||
online-tests/venv/bin/pip install pytest pytest-timeout pdbpp deltachat pytest-benchmark
|
online-tests/venv/bin/pip install pytest pytest-timeout pdbpp deltachat
|
||||||
|
|
||||||
|
python3 -m venv venv
|
||||||
|
venv/bin/pip install build
|
||||||
|
venv/bin/pip install 'setuptools>=68'
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
set -e
|
set -e
|
||||||
|
|
||||||
: ${CHATMAIL_DOMAIN:=c1.testrun.org}
|
: ${CHATMAIL_DOMAIN:=c1.testrun.org}
|
||||||
: ${CHATMAIL_SSH:=$CHATMAIL_DOMAIN}
|
: ${CHATMAIL_SSH_HOST:=$CHATMAIL_DOMAIN}
|
||||||
|
|
||||||
rsync -avz . "root@$CHATMAIL_SSH:/root/chatmail" --exclude='/.git' --filter="dir-merge,- .gitignore"
|
rsync -avz . "root@$CHATMAIL_SSH_HOST:/root/chatmail" --exclude='/.git' --filter="dir-merge,- .gitignore"
|
||||||
ssh "root@$CHATMAIL_SSH" "cd /root/chatmail; apt install -y python3-venv; python3 -m venv venv; venv/bin/pip install pyinfra build; venv/bin/python3 -m build -n --sdist chatmaild --outdir dist; venv/bin/pip install -e ./deploy-chatmail -e ./chatmaild; export CHATMAIL_DOMAIN=$CHATMAIL_DOMAIN; venv/bin/pyinfra @local deploy.py"
|
ssh "root@$CHATMAIL_SSH_HOST" "cd /root/chatmail; apt install -y python3-venv; python3 -m venv venv; venv/bin/pip install pyinfra build; venv/bin/python3 -m build -n --sdist chatmaild --outdir dist; venv/bin/pip install -e ./deploy-chatmail -e ./chatmaild; export CHATMAIL_DOMAIN=$CHATMAIL_DOMAIN; venv/bin/pyinfra @local deploy.py"
|
||||||
|
|||||||
@@ -1,3 +1,7 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
chatmaild/venv/bin/pytest chatmaild/ $@
|
set -e
|
||||||
online-tests/venv/bin/pytest online-tests/ -vrx --durations=5 $@
|
pushd chatmaild/src/chatmaild
|
||||||
|
../../venv/bin/pytest
|
||||||
|
popd
|
||||||
|
|
||||||
|
online-tests/venv/bin/pytest online-tests/ -vrx --durations=5
|
||||||
|
|||||||
Reference in New Issue
Block a user