mirror of
https://github.com/chatmail/relay.git
synced 2026-05-10 16:04:37 +00:00
Compare commits
238 Commits
socks-setu
...
hpk/cidebu
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
aa1891fc54 | ||
|
|
37e02445ce | ||
|
|
2e5a1a3a67 | ||
|
|
be5b25b0ab | ||
|
|
254fe95394 | ||
|
|
ac61ac082e | ||
|
|
02df395dab | ||
|
|
39584c7b7d | ||
|
|
4ebc4f3069 | ||
|
|
1eca8aa143 | ||
|
|
9c09d50e8f | ||
|
|
d73e896e66 | ||
|
|
283045dc4a | ||
|
|
180cfb3951 | ||
|
|
610637da80 | ||
|
|
73e6f5e6da | ||
|
|
b7e6926880 | ||
|
|
a7ef6ee35b | ||
|
|
920e062293 | ||
|
|
794a0608a1 | ||
|
|
fc09653de3 | ||
|
|
c8661fd135 | ||
|
|
4b0600a453 | ||
|
|
f1c10cac2b | ||
|
|
af83ca0235 | ||
|
|
8f6870ebb7 | ||
|
|
0e8bdbd3e3 | ||
|
|
0d593c22d1 | ||
|
|
a1f0a3e23b | ||
|
|
9b15d8de24 | ||
|
|
aaa51cf234 | ||
|
|
66c7115cfc | ||
|
|
823386d824 | ||
|
|
433cb71211 | ||
|
|
62c60d3070 | ||
|
|
698d328620 | ||
|
|
4292355310 | ||
|
|
85bb301255 | ||
|
|
0d61c13c58 | ||
|
|
15f79e0826 | ||
|
|
3d96f0fdfa | ||
|
|
733b9604ba | ||
|
|
969fdd7995 | ||
|
|
b1d11d7747 | ||
|
|
e948bdaea8 | ||
|
|
17389b8667 | ||
|
|
635b5de304 | ||
|
|
67be981176 | ||
|
|
0b8402c187 | ||
|
|
7c98c1f8c9 | ||
|
|
0483603d4a | ||
|
|
6b59b8be44 | ||
|
|
07ffc003e4 | ||
|
|
4cb62df33f | ||
|
|
ef58f011fb | ||
|
|
f7ef236ac8 | ||
|
|
dbe906a331 | ||
|
|
3899f41c61 | ||
|
|
57c29c14a4 | ||
|
|
2b5d903cc5 | ||
|
|
c8d270a853 | ||
|
|
72f4e9edbf | ||
|
|
1ce0a2b0ba | ||
|
|
044ebfb9a2 | ||
|
|
a41b034aa2 | ||
|
|
e00f0b852d | ||
|
|
501b12564c | ||
|
|
229ad15a28 | ||
|
|
e4f35d8dae | ||
|
|
4271573e15 | ||
|
|
b651a9046b | ||
|
|
6b84eaf8af | ||
|
|
1b076bcd22 | ||
|
|
30437f6c46 | ||
|
|
3171e40a26 | ||
|
|
61c915995b | ||
|
|
073bd86344 | ||
|
|
777a7addd2 | ||
|
|
4f28476c47 | ||
|
|
b05aec72c2 | ||
|
|
610675452e | ||
|
|
83387f5d08 | ||
|
|
142206529c | ||
|
|
c0f200b1a9 | ||
|
|
6d55f75bee | ||
|
|
c68cbf1806 | ||
|
|
9677617c7f | ||
|
|
d8cf282953 | ||
|
|
b959f57058 | ||
|
|
8768e6fd0b | ||
|
|
acbf370383 | ||
|
|
80dfdaee06 | ||
|
|
4d15ae9452 | ||
|
|
9a68d42ee8 | ||
|
|
d732d099ac | ||
|
|
582a2af799 | ||
|
|
fba3963d47 | ||
|
|
e80d33e2e0 | ||
|
|
6a3001bf22 | ||
|
|
368c41ba27 | ||
|
|
fa0d8432bc | ||
|
|
2811e08563 | ||
|
|
846a4066d8 | ||
|
|
6e1477666e | ||
|
|
013def94f9 | ||
|
|
468bb04149 | ||
|
|
30a23dad17 | ||
|
|
17af249f90 | ||
|
|
4e65291304 | ||
|
|
505ad36b36 | ||
|
|
dcb614911a | ||
|
|
e06c3631b2 | ||
|
|
da236e6e1b | ||
|
|
2796730a87 | ||
|
|
f32e18c32a | ||
|
|
1a5fd331b6 | ||
|
|
772b86a4b5 | ||
|
|
e0013b9bee | ||
|
|
127d9d6460 | ||
|
|
cb7de8019b | ||
|
|
2b5b06316d | ||
|
|
76b56d7b78 | ||
|
|
c1163228f6 | ||
|
|
8af825d7ea | ||
|
|
0a968aae93 | ||
|
|
879cffc056 | ||
|
|
462e92cca0 | ||
|
|
e1b1a945b1 | ||
|
|
0493e27312 | ||
|
|
e4f8c78efe | ||
|
|
e2cbf4e3e4 | ||
|
|
f35d98bb40 | ||
|
|
7ce1a5e841 | ||
|
|
0a72c2fba7 | ||
|
|
824f70f463 | ||
|
|
39f5f64998 | ||
|
|
1752803199 | ||
|
|
e372599ce7 | ||
|
|
ce9fb02a75 | ||
|
|
4526f5e772 | ||
|
|
616a42c8f3 | ||
|
|
ecb5ef8a10 | ||
|
|
824c3dc1d7 | ||
|
|
9b76d46558 | ||
|
|
cc4920ddc7 | ||
|
|
2af10175fa | ||
|
|
ae455fa9e1 | ||
|
|
60d7e516dd | ||
|
|
bf18905e02 | ||
|
|
4d6f520f18 | ||
|
|
9da626dfc8 | ||
|
|
1cca9aa441 | ||
|
|
3d054847a0 | ||
|
|
a31d998e67 | ||
|
|
d313bea97f | ||
|
|
da04226594 | ||
|
|
eb2de26638 | ||
|
|
f5652cdbc4 | ||
|
|
13172c92f3 | ||
|
|
09df636183 | ||
|
|
2b45ace3ba | ||
|
|
9e05a7d1eb | ||
|
|
21e7c09c43 | ||
|
|
14d96e0a9b | ||
|
|
459ffcabd6 | ||
|
|
75cc3fdab0 | ||
|
|
2d26a40c2b | ||
|
|
a78d4e6198 | ||
|
|
2a1e004962 | ||
|
|
5e55cc205d | ||
|
|
476c732373 | ||
|
|
71c50b7936 | ||
|
|
79cb390f16 | ||
|
|
c1452c9c6f | ||
|
|
6e903d7498 | ||
|
|
221f4a2b0c | ||
|
|
080ae058d8 | ||
|
|
edb84c0b3b | ||
|
|
04ef477d51 | ||
|
|
5696788d3a | ||
|
|
1c2bf919ed | ||
|
|
d15c22c1e8 | ||
|
|
9c6e90ae27 | ||
|
|
481791c277 | ||
|
|
a25c7981f9 | ||
|
|
53519f2865 | ||
|
|
3a50d82657 | ||
|
|
c640087498 | ||
|
|
2089f3ab58 | ||
|
|
cbaa6924c1 | ||
|
|
6ab3e9657d | ||
|
|
16f237dc60 | ||
|
|
554c33423f | ||
|
|
5d5e2b199c | ||
|
|
989ce70f97 | ||
|
|
f5dc4cb71e | ||
|
|
76512dfa2d | ||
|
|
850112502f | ||
|
|
888fa88aa3 | ||
|
|
15e7458666 | ||
|
|
0a93c76e66 | ||
|
|
312f86223c | ||
|
|
27a60418ad | ||
|
|
46d31a91da | ||
|
|
a8765d8847 | ||
|
|
8ee6ca1b80 | ||
|
|
1a2b73a862 | ||
|
|
c44f4efced | ||
|
|
9fdf4fd2af | ||
|
|
33353ccaf6 | ||
|
|
5fe3a269be | ||
|
|
0b4770018d | ||
|
|
75fcbd03ce | ||
|
|
377121bdee | ||
|
|
e5e58f4e38 | ||
|
|
04517f284c | ||
|
|
e32fb37b5d | ||
|
|
8d9019b1c5 | ||
|
|
63d3e05674 | ||
|
|
e466a03055 | ||
|
|
1819a276cb | ||
|
|
9ec6430b71 | ||
|
|
2097233fd6 | ||
|
|
4bca7891a2 | ||
|
|
2e23e743fd | ||
|
|
edc593586b | ||
|
|
1e229ad2de | ||
|
|
8baee557ee | ||
|
|
42e50b089f | ||
|
|
e6a3fab6aa | ||
|
|
ccd6e3e99c | ||
|
|
21778fa4f3 | ||
|
|
14342383cf | ||
|
|
926de76010 | ||
|
|
ee25d35db1 | ||
|
|
ee2115584b | ||
|
|
1c9c088657 | ||
|
|
b5afac2f1a |
4
.github/workflows/ci.yaml
vendored
4
.github/workflows/ci.yaml
vendored
@@ -9,7 +9,7 @@ jobs:
|
|||||||
name: isolated chatmaild tests
|
name: isolated chatmaild tests
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: run chatmaild tests
|
- name: run chatmaild tests
|
||||||
working-directory: chatmaild
|
working-directory: chatmaild
|
||||||
@@ -19,7 +19,7 @@ jobs:
|
|||||||
name: deploy-chatmail tests
|
name: deploy-chatmail tests
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: initenv
|
- name: initenv
|
||||||
run: scripts/initenv.sh
|
run: scripts/initenv.sh
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
;; Zone file for staging.testrun.org
|
;; Zone file for staging2.testrun.org
|
||||||
|
|
||||||
$ORIGIN staging.testrun.org.
|
$ORIGIN staging2.testrun.org.
|
||||||
$TTL 300
|
$TTL 300
|
||||||
|
|
||||||
@ IN SOA ns.testrun.org. root.nine.testrun.org (
|
@ IN SOA ns.testrun.org. root.nine.testrun.org (
|
||||||
@@ -15,6 +15,7 @@ $TTL 300
|
|||||||
@ IN NS ns.testrun.org.
|
@ IN NS ns.testrun.org.
|
||||||
|
|
||||||
;; DNS records.
|
;; DNS records.
|
||||||
@ IN A 37.27.37.98
|
@ IN A 37.27.24.139
|
||||||
mta-sts.staging.testrun.org. CNAME staging.testrun.org.
|
mta-sts.staging2.testrun.org. CNAME staging2.testrun.org.
|
||||||
www.staging.testrun.org. CNAME staging.testrun.org.
|
www.staging2.testrun.org. CNAME staging2.testrun.org.
|
||||||
|
|
||||||
|
|||||||
84
.github/workflows/test-and-deploy.yaml
vendored
84
.github/workflows/test-and-deploy.yaml
vendored
@@ -1,72 +1,96 @@
|
|||||||
name: deploy on staging.testrun.org, and run tests
|
name: deploy on staging2.testrun.org, and run tests
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- main
|
- main
|
||||||
- staging-ci
|
pull_request:
|
||||||
|
paths-ignore:
|
||||||
|
- 'scripts/**'
|
||||||
|
- '**/README.md'
|
||||||
|
- 'CHANGELOG.md'
|
||||||
|
- 'LICENSE'
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
deploy:
|
deploy:
|
||||||
name: deploy on staging.testrun.org, and run tests
|
name: deploy on staging2.testrun.org, and run tests
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
timeout-minutes: 30
|
||||||
|
concurrency:
|
||||||
|
group: ci-${{ github.workflow }}-${{ github.ref }}
|
||||||
|
cancel-in-progress: ${{ !contains(github.ref, '$GITHUB_REF') }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: jsok/serialize-workflow-action@v1
|
||||||
|
with:
|
||||||
|
github-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: prepare SSH
|
- name: prepare SSH
|
||||||
run: |
|
run: |
|
||||||
mkdir ~/.ssh
|
mkdir ~/.ssh
|
||||||
echo "${{ secrets.STAGING_SSH_KEY }}" >> ~/.ssh/id_ed25519
|
echo "${{ secrets.STAGING_SSH_KEY }}" >> ~/.ssh/id_ed25519
|
||||||
chmod 600 ~/.ssh/id_ed25519
|
chmod 600 ~/.ssh/id_ed25519
|
||||||
ssh-keyscan staging.testrun.org > ~/.ssh/known_hosts
|
ssh-keyscan staging2.testrun.org > ~/.ssh/known_hosts
|
||||||
# rsync -avz root@staging.testrun.org:/var/lib/acme . || true
|
# save previous acme & dkim state
|
||||||
# rsync -avz root@staging.testrun.org:/var/lib/rspamd/dkim . || true
|
rsync -avz root@staging2.testrun.org:/var/lib/acme . || true
|
||||||
|
rsync -avz root@staging2.testrun.org:/etc/dkimkeys . || true
|
||||||
|
# store previous acme & dkim state on ns.testrun.org, if it contains useful certs
|
||||||
|
if [ -f dkimkeys/opendkim.private ]; then rsync -avz -e "ssh -o StrictHostKeyChecking=accept-new" dkimkeys root@ns.testrun.org:/tmp/ || true; fi
|
||||||
|
if [ "$(ls -A acme/certs)" ]; then rsync -avz -e "ssh -o StrictHostKeyChecking=accept-new" acme root@ns.testrun.org:/tmp/ || true; fi
|
||||||
|
# make sure CAA record isn't set
|
||||||
|
ssh -o StrictHostKeyChecking=accept-new root@ns.testrun.org sed -i '/CAA/d' /etc/nsd/staging2.testrun.org.zone
|
||||||
|
ssh root@ns.testrun.org systemctl reload nsd
|
||||||
|
|
||||||
#- name: rebuild staging.testrun.org to have a clean VPS
|
- name: rebuild staging2.testrun.org to have a clean VPS
|
||||||
# run: |
|
run: |
|
||||||
# curl -X POST \
|
curl -X POST \
|
||||||
# -H "Authorization: Bearer ${{ secrets.HETZNER_API_TOKEN }}" \
|
-H "Authorization: Bearer ${{ secrets.HETZNER_API_TOKEN }}" \
|
||||||
# -H "Content-Type: application/json" \
|
-H "Content-Type: application/json" \
|
||||||
# -d '{"image":"debian-12"}' \
|
-d '{"image":"debian-12"}' \
|
||||||
# "https://api.hetzner.cloud/v1/servers/${{ secrets.STAGING_SERVER_ID }}/actions/rebuild"
|
"https://api.hetzner.cloud/v1/servers/${{ secrets.STAGING_SERVER_ID }}/actions/rebuild"
|
||||||
|
|
||||||
- run: scripts/initenv.sh
|
- run: scripts/initenv.sh
|
||||||
|
|
||||||
- name: append venv/bin to PATH
|
- name: append venv/bin to PATH
|
||||||
run: echo venv/bin >>$GITHUB_PATH
|
run: echo venv/bin >>$GITHUB_PATH
|
||||||
|
|
||||||
|
- name: upload TLS cert after rebuilding
|
||||||
|
run: |
|
||||||
|
echo " --- wait until staging2.testrun.org VPS is rebuilt --- "
|
||||||
|
rm ~/.ssh/known_hosts
|
||||||
|
while ! ssh -o ConnectTimeout=180 -o StrictHostKeyChecking=accept-new -v root@staging2.testrun.org id -u ; do sleep 1 ; done
|
||||||
|
ssh -o StrictHostKeyChecking=accept-new -v root@staging2.testrun.org id -u
|
||||||
|
# download acme & dkim state from ns.testrun.org
|
||||||
|
rsync -e "ssh -o StrictHostKeyChecking=accept-new" -avz root@ns.testrun.org:/tmp/acme acme-restore || true
|
||||||
|
rsync -avz root@ns.testrun.org:/tmp/dkimkeys dkimkeys-restore || true
|
||||||
|
# restore acme & dkim state to staging2.testrun.org
|
||||||
|
rsync -avz acme-restore/acme/ root@staging2.testrun.org:/var/lib/acme || true
|
||||||
|
rsync -avz dkimkeys-restore/dkimkeys/ root@staging2.testrun.org:/etc/dkimkeys || true
|
||||||
|
ssh -o StrictHostKeyChecking=accept-new -v root@staging2.testrun.org chown root:root -R /var/lib/acme || true
|
||||||
|
|
||||||
- name: run formatting checks
|
- name: run formatting checks
|
||||||
run: cmdeploy fmt -v
|
run: cmdeploy fmt -v
|
||||||
|
|
||||||
- name: run deploy-chatmail offline tests
|
- name: run deploy-chatmail offline tests
|
||||||
run: pytest --pyargs cmdeploy
|
run: pytest --pyargs cmdeploy
|
||||||
|
|
||||||
#- name: upload TLS cert after rebuilding
|
- run: cmdeploy init staging2.testrun.org
|
||||||
# run: |
|
|
||||||
# echo " --- wait until staging.testrun.org VPS is rebuilt --- "
|
|
||||||
# rm ~/.ssh/known_hosts
|
|
||||||
# while ! ssh -o ConnectTimeout=180 -o StrictHostKeyChecking=accept-new -v root@staging.testrun.org id -u ; do sleep 1 ; done
|
|
||||||
# ssh -o StrictHostKeyChecking=accept-new -v root@staging.testrun.org id -u
|
|
||||||
# rsync -avz acme root@staging.testrun.org:/var/lib/ || true
|
|
||||||
# rsync -avz dkim root@staging.testrun.org:/var/lib/rspamd/ || true
|
|
||||||
|
|
||||||
- run: cmdeploy init staging.testrun.org
|
- run: cmdeploy run --verbose
|
||||||
|
|
||||||
- run: cmdeploy run
|
|
||||||
|
|
||||||
- name: set DNS entries
|
- name: set DNS entries
|
||||||
run: |
|
run: |
|
||||||
#ssh -o StrictHostKeyChecking=accept-new -v root@staging.testrun.org chown _rspamd:_rspamd -R /var/lib/rspamd/dkim
|
ssh -o StrictHostKeyChecking=accept-new root@staging2.testrun.org chown opendkim:opendkim -R /etc/dkimkeys
|
||||||
cmdeploy dns --zonefile staging-generated.zone
|
cmdeploy dns --zonefile staging-generated.zone --verbose
|
||||||
cat staging-generated.zone >> .github/workflows/staging.testrun.org-default.zone
|
cat staging-generated.zone >> .github/workflows/staging.testrun.org-default.zone
|
||||||
cat .github/workflows/staging.testrun.org-default.zone
|
cat .github/workflows/staging.testrun.org-default.zone
|
||||||
scp -o StrictHostKeyChecking=accept-new .github/workflows/staging.testrun.org-default.zone root@ns.testrun.org:/etc/nsd/staging.testrun.org.zone
|
scp .github/workflows/staging.testrun.org-default.zone root@ns.testrun.org:/etc/nsd/staging2.testrun.org.zone
|
||||||
ssh root@ns.testrun.org nsd-checkzone staging.testrun.org /etc/nsd/staging.testrun.org.zone
|
ssh root@ns.testrun.org nsd-checkzone staging2.testrun.org /etc/nsd/staging2.testrun.org.zone
|
||||||
ssh root@ns.testrun.org systemctl reload nsd
|
ssh root@ns.testrun.org systemctl reload nsd
|
||||||
|
|
||||||
- name: cmdeploy test
|
- name: cmdeploy test
|
||||||
run: CHATMAIL_DOMAIN2=nine.testrun.org cmdeploy test --slow
|
run: CHATMAIL_DOMAIN2=nine.testrun.org cmdeploy test --slow
|
||||||
|
|
||||||
- name: cmdeploy dns (try 3 times)
|
- name: cmdeploy dns (try 3 times)
|
||||||
run: cmdeploy dns || cmdeploy dns || cmdeploy dns
|
run: cmdeploy dns -v || cmdeploy dns -v || cmdeploy dns -v
|
||||||
|
|
||||||
|
|||||||
155
CHANGELOG.md
Normal file
155
CHANGELOG.md
Normal file
@@ -0,0 +1,155 @@
|
|||||||
|
# Changelog for chatmail deployment
|
||||||
|
|
||||||
|
## untagged
|
||||||
|
|
||||||
|
- BREAKING: new required chatmail.ini values:
|
||||||
|
|
||||||
|
mailboxes_dir = /home/vmail/mail/{mail_domain}
|
||||||
|
passdb = /home/vmail/passdb.sqlite
|
||||||
|
|
||||||
|
reducing hardcoding these two paths all over the files, also improving testability.
|
||||||
|
([#351](https://github.com/deltachat/chatmail/pull/351))
|
||||||
|
|
||||||
|
- BREAKING: new required chatmail.ini value 'delete_inactive_users_after = 100'
|
||||||
|
which removes users from database and mails after 100 days without any login.
|
||||||
|
([#350](https://github.com/deltachat/chatmail/pull/350))
|
||||||
|
|
||||||
|
- reload nginx in the acmetool cronjob
|
||||||
|
([#360](https://github.com/deltachat/chatmail/pull/360))
|
||||||
|
|
||||||
|
- remove checking of reverse-DNS PTR records. Chatmail-servers don't
|
||||||
|
depend on it and even in the wider e-mail system it's not common anymore.
|
||||||
|
If it's an issue, a chatmail operator can still care to properly set reverse DNS.
|
||||||
|
([#348](https://github.com/deltachat/chatmail/pull/348))
|
||||||
|
|
||||||
|
- Make DNS-checking faster and more interactive, run it fully during "cmdeploy run",
|
||||||
|
also introducing a generic mechanism for rapid remote ssh-based python function execution.
|
||||||
|
([#346](https://github.com/deltachat/chatmail/pull/346))
|
||||||
|
|
||||||
|
- Don't fix file owner ship of /home/vmail
|
||||||
|
([#345](https://github.com/deltachat/chatmail/pull/345))
|
||||||
|
|
||||||
|
- Support iterating over all users with doveadm commands
|
||||||
|
([#344](https://github.com/deltachat/chatmail/pull/344))
|
||||||
|
|
||||||
|
- Test and fix for attempts to create inadmissible accounts
|
||||||
|
([#333](https://github.com/deltachat/chatmail/pull/321))
|
||||||
|
|
||||||
|
- check that OpenPGP has only PKESK, SKESK and SEIPD packets
|
||||||
|
([#323](https://github.com/deltachat/chatmail/pull/323),
|
||||||
|
[#324](https://github.com/deltachat/chatmail/pull/324))
|
||||||
|
|
||||||
|
- improve filtermail checks for encrypted messages and drop support for unencrypted MDNs
|
||||||
|
([#320](https://github.com/deltachat/chatmail/pull/320))
|
||||||
|
|
||||||
|
- replace `bash` with `/bin/sh`
|
||||||
|
([#334](https://github.com/deltachat/chatmail/pull/334))
|
||||||
|
|
||||||
|
- Increase number of logged in IMAP sessions to 50000
|
||||||
|
([#335](https://github.com/deltachat/chatmail/pull/335))
|
||||||
|
|
||||||
|
- filtermail: do not allow ASCII armor without actual payload
|
||||||
|
([#325](https://github.com/deltachat/chatmail/pull/325))
|
||||||
|
|
||||||
|
- Remove sieve to enable hardlink deduplication in LMTP
|
||||||
|
([#343](https://github.com/deltachat/chatmail/pull/343))
|
||||||
|
|
||||||
|
- dovecot: enable gzip compression on disk
|
||||||
|
([#341](https://github.com/deltachat/chatmail/pull/341))
|
||||||
|
|
||||||
|
- DKIM-sign Content-Type and oversign all signed headers
|
||||||
|
([#296](https://github.com/deltachat/chatmail/pull/296))
|
||||||
|
|
||||||
|
- Add nonci_accounts metric
|
||||||
|
([#347](https://github.com/deltachat/chatmail/pull/347))
|
||||||
|
|
||||||
|
- doveauth: log when a new account is created
|
||||||
|
([#349](https://github.com/deltachat/chatmail/pull/349))
|
||||||
|
|
||||||
|
- Multiplex HTTPS, IMAP and SMTP on port 443
|
||||||
|
([#357](https://github.com/deltachat/chatmail/pull/357))
|
||||||
|
|
||||||
|
## 1.3.0 - 2024-06-06
|
||||||
|
|
||||||
|
- don't check necessary DNS records on cmdeploy init anymore
|
||||||
|
([#316](https://github.com/deltachat/chatmail/pull/316))
|
||||||
|
|
||||||
|
- ensure cron and acl are installed
|
||||||
|
([#293](https://github.com/deltachat/chatmail/pull/293),
|
||||||
|
[#310](https://github.com/deltachat/chatmail/pull/310))
|
||||||
|
|
||||||
|
- change default for delete_mails_after from 40 to 20 days
|
||||||
|
([#300](https://github.com/deltachat/chatmail/pull/300))
|
||||||
|
|
||||||
|
- save journald logs only to memory and save nginx logs to journald instead of file
|
||||||
|
([#299](https://github.com/deltachat/chatmail/pull/299))
|
||||||
|
|
||||||
|
- fix writing of multiple obs repositories in `/etc/apt/sources.list`
|
||||||
|
([#290](https://github.com/deltachat/chatmail/pull/290))
|
||||||
|
|
||||||
|
- metadata: add support for `/shared/vendor/deltachat/irohrelay`
|
||||||
|
([#284](https://github.com/deltachat/chatmail/pull/284))
|
||||||
|
|
||||||
|
- Emit "XCHATMAIL" capability from IMAP server
|
||||||
|
([#278](https://github.com/deltachat/chatmail/pull/278))
|
||||||
|
|
||||||
|
- Move echobot `into /var/lib/echobot`
|
||||||
|
([#281](https://github.com/deltachat/chatmail/pull/281))
|
||||||
|
|
||||||
|
- Accept Let's Encrypt's new Terms of Services
|
||||||
|
([#275](https://github.com/deltachat/chatmail/pull/276))
|
||||||
|
|
||||||
|
- Reload Dovecot and Postfix when TLS certificate updates
|
||||||
|
([#271](https://github.com/deltachat/chatmail/pull/271))
|
||||||
|
|
||||||
|
- Use forked version of dovecot without hardcoded delays
|
||||||
|
([#270](https://github.com/deltachat/chatmail/pull/270))
|
||||||
|
|
||||||
|
## 1.2.0 - 2024-04-04
|
||||||
|
|
||||||
|
- Install dig on the server to resolve DNS records
|
||||||
|
([#267](https://github.com/deltachat/chatmail/pull/267))
|
||||||
|
|
||||||
|
- preserve notification order and exponentially backoff with
|
||||||
|
retries for tokens where we didn't get a successful return
|
||||||
|
([#265](https://github.com/deltachat/chatmail/pull/263))
|
||||||
|
|
||||||
|
- Run chatmail-metadata and doveauth as vmail
|
||||||
|
([#261](https://github.com/deltachat/chatmail/pull/261))
|
||||||
|
|
||||||
|
- Apply systemd restrictions to echobot
|
||||||
|
([#259](https://github.com/deltachat/chatmail/pull/259))
|
||||||
|
|
||||||
|
- re-enable running the CI in pull requests, but not concurrently
|
||||||
|
([#258](https://github.com/deltachat/chatmail/pull/258))
|
||||||
|
|
||||||
|
|
||||||
|
## 1.1.0 - 2024-03-28
|
||||||
|
|
||||||
|
### The changelog starts to record changes from March 15th, 2024
|
||||||
|
|
||||||
|
- Move systemd unit templates to cmdeploy package
|
||||||
|
([#255](https://github.com/deltachat/chatmail/pull/255))
|
||||||
|
|
||||||
|
- Persist push tokens and support multiple device per address
|
||||||
|
([#254](https://github.com/deltachat/chatmail/pull/254))
|
||||||
|
|
||||||
|
- Avoid warning for regular doveauth protocol's hello message.
|
||||||
|
([#250](https://github.com/deltachat/chatmail/pull/250))
|
||||||
|
|
||||||
|
- Fix various tests to pass again with "cmdeploy test".
|
||||||
|
([#245](https://github.com/deltachat/chatmail/pull/245),
|
||||||
|
[#242](https://github.com/deltachat/chatmail/pull/242)
|
||||||
|
|
||||||
|
- Ensure lets-encrypt certificates are reloaded after renewal
|
||||||
|
([#244]) https://github.com/deltachat/chatmail/pull/244
|
||||||
|
|
||||||
|
- Persist tokens to avoid iOS users loosing push-notifications when the
|
||||||
|
chatmail metadata service is restarted (happens regularly during deploys)
|
||||||
|
([#238](https://github.com/deltachat/chatmail/pull/239)
|
||||||
|
|
||||||
|
- Fix failing sieve-script compile errors on incoming messages
|
||||||
|
([#237](https://github.com/deltachat/chatmail/pull/239)
|
||||||
|
|
||||||
|
- Fix quota reporting after expunging of old mails
|
||||||
|
([#233](https://github.com/deltachat/chatmail/pull/239)
|
||||||
28
README.md
28
README.md
@@ -15,6 +15,8 @@ after which the initially specified password is required for using them.
|
|||||||
|
|
||||||
## Deploying your own chatmail server
|
## Deploying your own chatmail server
|
||||||
|
|
||||||
|
To deploy chatmail on your own server, you must have set-up ssh authentication and need to use an ed25519 key, due to an [upstream bug in paramiko](https://github.com/paramiko/paramiko/issues/2191). You also need to add your private key to the local ssh-agent, because you can't type in your password during deployment.
|
||||||
|
|
||||||
We use `chat.example.org` as the chatmail domain in the following steps.
|
We use `chat.example.org` as the chatmail domain in the following steps.
|
||||||
Please substitute it with your own domain.
|
Please substitute it with your own domain.
|
||||||
|
|
||||||
@@ -153,10 +155,34 @@ While this file is present, account creation will be blocked.
|
|||||||
|
|
||||||
[Postfix](http://www.postfix.org/) listens on ports 25 (smtp) and 587 (submission) and 465 (submissions).
|
[Postfix](http://www.postfix.org/) listens on ports 25 (smtp) and 587 (submission) and 465 (submissions).
|
||||||
[Dovecot](https://www.dovecot.org/) listens on ports 143 (imap) and 993 (imaps).
|
[Dovecot](https://www.dovecot.org/) listens on ports 143 (imap) and 993 (imaps).
|
||||||
[nginx](https://www.nginx.com/) listens on port 443 (https).
|
[nginx](https://www.nginx.com/) listens on port 8443 (https-alt) and 443 (https).
|
||||||
|
Port 443 multiplexes HTTPS, IMAP and SMTP using ALPN to redirect connections to ports 8443, 465 or 993.
|
||||||
[acmetool](https://hlandau.github.io/acmetool/) listens on port 80 (http).
|
[acmetool](https://hlandau.github.io/acmetool/) listens on port 80 (http).
|
||||||
|
|
||||||
Delta Chat apps will, however, discover all ports and configurations
|
Delta Chat apps will, however, discover all ports and configurations
|
||||||
automatically by reading the [autoconfig XML file](https://www.ietf.org/archive/id/draft-bucksch-autoconfig-00.html) from the chatmail service.
|
automatically by reading the [autoconfig XML file](https://www.ietf.org/archive/id/draft-bucksch-autoconfig-00.html) from the chatmail service.
|
||||||
|
|
||||||
|
## Email authentication
|
||||||
|
|
||||||
|
chatmail servers rely on [DKIM](https://www.rfc-editor.org/rfc/rfc6376)
|
||||||
|
to authenticate incoming emails.
|
||||||
|
Incoming emails must have a valid DKIM signature with
|
||||||
|
Signing Domain Identifier (SDID, `d=` parameter in the DKIM-Signature header)
|
||||||
|
equal to the `From:` header domain.
|
||||||
|
This property is checked by OpenDKIM screen policy script
|
||||||
|
before validating the signatures.
|
||||||
|
This correpsonds to strict [DMARC](https://www.rfc-editor.org/rfc/rfc7489) alignment (`adkim=s`),
|
||||||
|
but chatmail does not rely on DMARC and does not consult the sender policy published in DMARC records.
|
||||||
|
Other legacy authentication mechanisms such as [iprev](https://www.rfc-editor.org/rfc/rfc8601#section-2.7.3)
|
||||||
|
and [SPF](https://www.rfc-editor.org/rfc/rfc7208) are also not taken into account.
|
||||||
|
If there is no valid DKIM signature on the incoming email,
|
||||||
|
the sender receives a "5.7.1 No valid DKIM signature found" error.
|
||||||
|
|
||||||
|
Outgoing emails must be sent over authenticated connection
|
||||||
|
with envelope MAIL FROM (return path) corresponding to the login.
|
||||||
|
This is ensured by Postfix which maps login username
|
||||||
|
to MAIL FROM with
|
||||||
|
[`smtpd_sender_login_maps`](https://www.postfix.org/postconf.5.html#smtpd_sender_login_maps)
|
||||||
|
and rejects incorrectly authenticated emails with [`reject_sender_login_mismatch`](reject_sender_login_mismatch) policy.
|
||||||
|
`From:` header must correspond to envelope MAIL FROM,
|
||||||
|
this is ensured by `filtermail` proxy.
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
include src/chatmaild/*.f
|
|
||||||
include src/chatmaild/ini/*.ini.f
|
include src/chatmaild/ini/*.ini.f
|
||||||
include src/chatmaild/ini/*.ini
|
include src/chatmaild/ini/*.ini
|
||||||
include src/chatmaild/tests/mail-data/*
|
include src/chatmaild/tests/mail-data/*
|
||||||
|
|||||||
@@ -10,6 +10,8 @@ dependencies = [
|
|||||||
"iniconfig",
|
"iniconfig",
|
||||||
"deltachat-rpc-server",
|
"deltachat-rpc-server",
|
||||||
"deltachat-rpc-client",
|
"deltachat-rpc-client",
|
||||||
|
"filelock",
|
||||||
|
"requests",
|
||||||
]
|
]
|
||||||
|
|
||||||
[tool.setuptools]
|
[tool.setuptools]
|
||||||
@@ -20,9 +22,11 @@ where = ['src']
|
|||||||
|
|
||||||
[project.scripts]
|
[project.scripts]
|
||||||
doveauth = "chatmaild.doveauth:main"
|
doveauth = "chatmaild.doveauth:main"
|
||||||
|
chatmail-metadata = "chatmaild.metadata:main"
|
||||||
filtermail = "chatmaild.filtermail:main"
|
filtermail = "chatmaild.filtermail:main"
|
||||||
echobot = "chatmaild.echo:main"
|
echobot = "chatmaild.echo:main"
|
||||||
chatmail-metrics = "chatmaild.metrics:main"
|
chatmail-metrics = "chatmaild.metrics:main"
|
||||||
|
delete_inactive_users = "chatmaild.delete_inactive_users:main"
|
||||||
|
|
||||||
[project.entry-points.pytest11]
|
[project.entry-points.pytest11]
|
||||||
"chatmaild.testplugin" = "chatmaild.tests.plugin"
|
"chatmaild.testplugin" = "chatmaild.tests.plugin"
|
||||||
@@ -33,6 +37,16 @@ log_format = "%(asctime)s %(levelname)s %(message)s"
|
|||||||
log_date_format = "%Y-%m-%d %H:%M:%S"
|
log_date_format = "%Y-%m-%d %H:%M:%S"
|
||||||
log_level = "INFO"
|
log_level = "INFO"
|
||||||
|
|
||||||
|
[tool.ruff]
|
||||||
|
lint.select = [
|
||||||
|
"F", # Pyflakes
|
||||||
|
"I", # isort
|
||||||
|
|
||||||
|
"PLC", # Pylint Convention
|
||||||
|
"PLE", # Pylint Error
|
||||||
|
"PLW", # Pylint Warning
|
||||||
|
]
|
||||||
|
|
||||||
[tool.tox]
|
[tool.tox]
|
||||||
legacy_tox_ini = """
|
legacy_tox_ini = """
|
||||||
[tox]
|
[tox]
|
||||||
@@ -44,10 +58,9 @@ skipdist = True
|
|||||||
skip_install = True
|
skip_install = True
|
||||||
deps =
|
deps =
|
||||||
ruff
|
ruff
|
||||||
black
|
|
||||||
commands =
|
commands =
|
||||||
black --quiet --check --diff src/
|
ruff format --quiet --diff src/
|
||||||
ruff src/
|
ruff check src/
|
||||||
|
|
||||||
[testenv]
|
[testenv]
|
||||||
deps = pytest
|
deps = pytest
|
||||||
|
|||||||
@@ -1,9 +1,13 @@
|
|||||||
|
from pathlib import Path
|
||||||
|
|
||||||
import iniconfig
|
import iniconfig
|
||||||
|
|
||||||
|
|
||||||
def read_config(inipath):
|
def read_config(inipath):
|
||||||
|
assert Path(inipath).exists(), inipath
|
||||||
cfg = iniconfig.IniConfig(inipath)
|
cfg = iniconfig.IniConfig(inipath)
|
||||||
return Config(inipath, params=cfg.sections["params"])
|
params = cfg.sections["params"]
|
||||||
|
return Config(inipath, params=params)
|
||||||
|
|
||||||
|
|
||||||
class Config:
|
class Config:
|
||||||
@@ -13,13 +17,17 @@ class Config:
|
|||||||
self.max_user_send_per_minute = int(params["max_user_send_per_minute"])
|
self.max_user_send_per_minute = int(params["max_user_send_per_minute"])
|
||||||
self.max_mailbox_size = params["max_mailbox_size"]
|
self.max_mailbox_size = params["max_mailbox_size"]
|
||||||
self.delete_mails_after = params["delete_mails_after"]
|
self.delete_mails_after = params["delete_mails_after"]
|
||||||
|
self.delete_inactive_users_after = int(params["delete_inactive_users_after"])
|
||||||
self.username_min_length = int(params["username_min_length"])
|
self.username_min_length = int(params["username_min_length"])
|
||||||
self.username_max_length = int(params["username_max_length"])
|
self.username_max_length = int(params["username_max_length"])
|
||||||
self.password_min_length = int(params["password_min_length"])
|
self.password_min_length = int(params["password_min_length"])
|
||||||
self.passthrough_senders = params["passthrough_senders"].split()
|
self.passthrough_senders = params["passthrough_senders"].split()
|
||||||
self.passthrough_recipients = params["passthrough_recipients"].split()
|
self.passthrough_recipients = params["passthrough_recipients"].split()
|
||||||
|
self.mailboxes_dir = Path(params["mailboxes_dir"].strip())
|
||||||
|
self.passdb_path = Path(params["passdb_path"].strip())
|
||||||
self.filtermail_smtp_port = int(params["filtermail_smtp_port"])
|
self.filtermail_smtp_port = int(params["filtermail_smtp_port"])
|
||||||
self.postfix_reinject_port = int(params["postfix_reinject_port"])
|
self.postfix_reinject_port = int(params["postfix_reinject_port"])
|
||||||
|
self.iroh_relay = params.get("iroh_relay")
|
||||||
self.privacy_postal = params.get("privacy_postal")
|
self.privacy_postal = params.get("privacy_postal")
|
||||||
self.privacy_mail = params.get("privacy_mail")
|
self.privacy_mail = params.get("privacy_mail")
|
||||||
self.privacy_pdo = params.get("privacy_pdo")
|
self.privacy_pdo = params.get("privacy_pdo")
|
||||||
@@ -28,14 +36,36 @@ class Config:
|
|||||||
def _getbytefile(self):
|
def _getbytefile(self):
|
||||||
return open(self._inipath, "rb")
|
return open(self._inipath, "rb")
|
||||||
|
|
||||||
|
def get_user_maildir(self, addr):
|
||||||
|
if addr and addr != "." and "/" not in addr:
|
||||||
|
res = self.mailboxes_dir.joinpath(addr).resolve()
|
||||||
|
if res.is_relative_to(self.mailboxes_dir):
|
||||||
|
return res
|
||||||
|
raise ValueError(f"invalid address {addr!r}")
|
||||||
|
|
||||||
def write_initial_config(inipath, mail_domain):
|
|
||||||
|
def write_initial_config(inipath, mail_domain, overrides):
|
||||||
|
"""Write out default config file, using the specified config value overrides."""
|
||||||
from importlib.resources import files
|
from importlib.resources import files
|
||||||
|
|
||||||
inidir = files(__package__).joinpath("ini")
|
inidir = files(__package__).joinpath("ini")
|
||||||
content = (
|
source_inipath = inidir.joinpath("chatmail.ini.f")
|
||||||
inidir.joinpath("chatmail.ini.f").read_text().format(mail_domain=mail_domain)
|
content = source_inipath.read_text().format(mail_domain=mail_domain)
|
||||||
)
|
|
||||||
|
# apply config overrides
|
||||||
|
new_lines = []
|
||||||
|
for line in content.split("\n"):
|
||||||
|
new_line = line.strip()
|
||||||
|
if new_line and new_line[0] not in "#[":
|
||||||
|
name, value = map(str.strip, new_line.split("=", maxsplit=1))
|
||||||
|
value = overrides.get(name, value)
|
||||||
|
new_line = f"{name} = {value}"
|
||||||
|
new_lines.append(new_line)
|
||||||
|
|
||||||
|
content = "\n".join(new_lines)
|
||||||
|
|
||||||
|
# apply testrun privacy overrides
|
||||||
|
|
||||||
if mail_domain.endswith(".testrun.org"):
|
if mail_domain.endswith(".testrun.org"):
|
||||||
override_inipath = inidir.joinpath("override-testrun.ini")
|
override_inipath = inidir.joinpath("override-testrun.ini")
|
||||||
privacy = iniconfig.IniConfig(override_inipath)["privacy"]
|
privacy = iniconfig.IniConfig(override_inipath)["privacy"]
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import sqlite3
|
|
||||||
import contextlib
|
import contextlib
|
||||||
|
import sqlite3
|
||||||
import time
|
import time
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
|
|||||||
33
chatmaild/src/chatmaild/delete_inactive_users.py
Normal file
33
chatmaild/src/chatmaild/delete_inactive_users.py
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
"""
|
||||||
|
Remove inactive users
|
||||||
|
"""
|
||||||
|
|
||||||
|
import shutil
|
||||||
|
import sys
|
||||||
|
import time
|
||||||
|
|
||||||
|
from .config import read_config
|
||||||
|
from .database import Database
|
||||||
|
from .doveauth import iter_userdb_lastlogin_before
|
||||||
|
|
||||||
|
|
||||||
|
def delete_inactive_users(db, config, CHUNK=100):
|
||||||
|
cutoff_date = time.time() - config.delete_inactive_users_after * 86400
|
||||||
|
|
||||||
|
old_users = iter_userdb_lastlogin_before(db, cutoff_date)
|
||||||
|
chunks = (old_users[i : i + CHUNK] for i in range(0, len(old_users), CHUNK))
|
||||||
|
for sublist in chunks:
|
||||||
|
for user in sublist:
|
||||||
|
user_mail_dir = config.get_user_maildir(user)
|
||||||
|
shutil.rmtree(user_mail_dir, ignore_errors=True)
|
||||||
|
|
||||||
|
with db.write_transaction() as conn:
|
||||||
|
for user in sublist:
|
||||||
|
conn.execute("DELETE FROM users WHERE addr = ?", (user,))
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
(cfgpath,) = sys.argv[1:]
|
||||||
|
config = read_config(cfgpath)
|
||||||
|
db = Database(config.passdb_path)
|
||||||
|
delete_inactive_users(db, config)
|
||||||
@@ -1,22 +1,26 @@
|
|||||||
|
import crypt
|
||||||
|
import json
|
||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
import time
|
|
||||||
import sys
|
import sys
|
||||||
import json
|
import time
|
||||||
import crypt
|
from pathlib import Path
|
||||||
from socketserver import (
|
from socketserver import (
|
||||||
UnixStreamServer,
|
|
||||||
StreamRequestHandler,
|
StreamRequestHandler,
|
||||||
ThreadingMixIn,
|
ThreadingMixIn,
|
||||||
|
UnixStreamServer,
|
||||||
)
|
)
|
||||||
import pwd
|
|
||||||
|
|
||||||
|
from .config import Config, read_config
|
||||||
from .database import Database
|
from .database import Database
|
||||||
from .config import read_config, Config
|
|
||||||
|
|
||||||
NOCREATE_FILE = "/etc/chatmail-nocreate"
|
NOCREATE_FILE = "/etc/chatmail-nocreate"
|
||||||
|
|
||||||
|
|
||||||
|
class UnknownCommand(ValueError):
|
||||||
|
"""dictproxy handler received an unkown command"""
|
||||||
|
|
||||||
|
|
||||||
def encrypt_password(password: str):
|
def encrypt_password(password: str):
|
||||||
# https://doc.dovecot.org/configuration_manual/authentication/password_schemes/
|
# https://doc.dovecot.org/configuration_manual/authentication/password_schemes/
|
||||||
passhash = crypt.crypt(password, crypt.METHOD_SHA512)
|
passhash = crypt.crypt(password, crypt.METHOD_SHA512)
|
||||||
@@ -42,44 +46,75 @@ def is_allowed_to_create(config: Config, user, cleartext_password) -> bool:
|
|||||||
return False
|
return False
|
||||||
localpart, domain = parts
|
localpart, domain = parts
|
||||||
|
|
||||||
|
if localpart == "echo":
|
||||||
|
# echobot account should not be created in the database
|
||||||
|
return False
|
||||||
|
|
||||||
if (
|
if (
|
||||||
len(localpart) > config.username_max_length
|
len(localpart) > config.username_max_length
|
||||||
or len(localpart) < config.username_min_length
|
or len(localpart) < config.username_min_length
|
||||||
):
|
):
|
||||||
if localpart != "echo":
|
logging.warning(
|
||||||
logging.warning(
|
"localpart %s has to be between %s and %s chars long",
|
||||||
"localpart %s has to be between %s and %s chars long",
|
localpart,
|
||||||
localpart,
|
config.username_min_length,
|
||||||
config.username_min_length,
|
config.username_max_length,
|
||||||
config.username_max_length,
|
)
|
||||||
)
|
return False
|
||||||
return False
|
|
||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
def get_user_data(db, user):
|
def get_user_data(db, config: Config, user):
|
||||||
|
if user == f"echo@{config.mail_domain}":
|
||||||
|
return dict(
|
||||||
|
home=str(config.get_user_maildir(user)),
|
||||||
|
uid="vmail",
|
||||||
|
gid="vmail",
|
||||||
|
)
|
||||||
|
|
||||||
with db.read_connection() as conn:
|
with db.read_connection() as conn:
|
||||||
result = conn.get_user(user)
|
result = conn.get_user(user)
|
||||||
if result:
|
if result:
|
||||||
|
result["home"] = str(config.get_user_maildir(user))
|
||||||
result["uid"] = "vmail"
|
result["uid"] = "vmail"
|
||||||
result["gid"] = "vmail"
|
result["gid"] = "vmail"
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
|
||||||
def lookup_userdb(db, user):
|
def lookup_userdb(db, config: Config, user):
|
||||||
return get_user_data(db, user)
|
return get_user_data(db, config, user)
|
||||||
|
|
||||||
|
|
||||||
def lookup_passdb(db, config: Config, user, cleartext_password):
|
def lookup_passdb(db, config: Config, user, cleartext_password, last_login=None):
|
||||||
|
if user == f"echo@{config.mail_domain}":
|
||||||
|
# Echobot writes password it wants to log in with into /run/echobot/password
|
||||||
|
try:
|
||||||
|
password = Path("/run/echobot/password").read_text()
|
||||||
|
except Exception:
|
||||||
|
logging.exception("Exception when trying to read /run/echobot/password")
|
||||||
|
return None
|
||||||
|
|
||||||
|
return dict(
|
||||||
|
home=str(config.get_user_maildir(user)),
|
||||||
|
uid="vmail",
|
||||||
|
gid="vmail",
|
||||||
|
password=encrypt_password(password),
|
||||||
|
)
|
||||||
|
|
||||||
|
if last_login is None:
|
||||||
|
last_login = time.time()
|
||||||
|
last_login = int(last_login)
|
||||||
|
|
||||||
with db.write_transaction() as conn:
|
with db.write_transaction() as conn:
|
||||||
userdata = conn.get_user(user)
|
userdata = conn.get_user(user)
|
||||||
if userdata:
|
if userdata:
|
||||||
# Update last login time.
|
# Update last login time.
|
||||||
conn.execute(
|
conn.execute(
|
||||||
"UPDATE users SET last_login=? WHERE addr=?", (int(time.time()), user)
|
"UPDATE users SET last_login=? WHERE addr=?", (last_login, user)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
userdata["home"] = str(config.get_user_maildir(user))
|
||||||
userdata["uid"] = "vmail"
|
userdata["uid"] = "vmail"
|
||||||
userdata["gid"] = "vmail"
|
userdata["gid"] = "vmail"
|
||||||
return userdata
|
return userdata
|
||||||
@@ -89,15 +124,34 @@ def lookup_passdb(db, config: Config, user, cleartext_password):
|
|||||||
encrypted_password = encrypt_password(cleartext_password)
|
encrypted_password = encrypt_password(cleartext_password)
|
||||||
q = """INSERT INTO users (addr, password, last_login)
|
q = """INSERT INTO users (addr, password, last_login)
|
||||||
VALUES (?, ?, ?)"""
|
VALUES (?, ?, ?)"""
|
||||||
conn.execute(q, (user, encrypted_password, int(time.time())))
|
conn.execute(q, (user, encrypted_password, last_login))
|
||||||
|
print(f"Created address: {user}", file=sys.stderr)
|
||||||
return dict(
|
return dict(
|
||||||
home=f"/home/vmail/mail/{config.mail_domain}/{user}",
|
home=str(config.get_user_maildir(user)),
|
||||||
uid="vmail",
|
uid="vmail",
|
||||||
gid="vmail",
|
gid="vmail",
|
||||||
password=encrypted_password,
|
password=encrypted_password,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def iter_userdb(db) -> list:
|
||||||
|
"""Get a list of all user addresses."""
|
||||||
|
with db.read_connection() as conn:
|
||||||
|
rows = conn.execute(
|
||||||
|
"SELECT addr from users",
|
||||||
|
).fetchall()
|
||||||
|
return [x[0] for x in rows]
|
||||||
|
|
||||||
|
|
||||||
|
def iter_userdb_lastlogin_before(db, cutoff_date):
|
||||||
|
"""Get a list of users where last login was before cutoff_date."""
|
||||||
|
with db.read_connection() as conn:
|
||||||
|
rows = conn.execute(
|
||||||
|
"SELECT addr FROM users WHERE last_login < ?", (cutoff_date,)
|
||||||
|
).fetchall()
|
||||||
|
return [x[0] for x in rows]
|
||||||
|
|
||||||
|
|
||||||
def split_and_unescape(s):
|
def split_and_unescape(s):
|
||||||
"""Split strings using double quote as a separator and backslash as escape character
|
"""Split strings using double quote as a separator and backslash as escape character
|
||||||
into parts."""
|
into parts."""
|
||||||
@@ -125,8 +179,12 @@ def split_and_unescape(s):
|
|||||||
|
|
||||||
|
|
||||||
def handle_dovecot_request(msg, db, config: Config):
|
def handle_dovecot_request(msg, db, config: Config):
|
||||||
|
# see https://doc.dovecot.org/3.0/developer_manual/design/dict_protocol/
|
||||||
short_command = msg[0]
|
short_command = msg[0]
|
||||||
if short_command == "L": # LOOKUP
|
if short_command == "H": # HELLO
|
||||||
|
# we don't do any checking on versions and just return
|
||||||
|
return
|
||||||
|
elif short_command == "L": # LOOKUP
|
||||||
parts = msg[1:].split("\t")
|
parts = msg[1:].split("\t")
|
||||||
|
|
||||||
# Dovecot <2.3.17 has only one part,
|
# Dovecot <2.3.17 has only one part,
|
||||||
@@ -142,7 +200,7 @@ def handle_dovecot_request(msg, db, config: Config):
|
|||||||
if type == "userdb":
|
if type == "userdb":
|
||||||
user = args[0]
|
user = args[0]
|
||||||
if user.endswith(f"@{config.mail_domain}"):
|
if user.endswith(f"@{config.mail_domain}"):
|
||||||
res = lookup_userdb(db, user)
|
res = lookup_userdb(db, config, user)
|
||||||
if res:
|
if res:
|
||||||
reply_command = "O"
|
reply_command = "O"
|
||||||
else:
|
else:
|
||||||
@@ -157,7 +215,14 @@ def handle_dovecot_request(msg, db, config: Config):
|
|||||||
reply_command = "N"
|
reply_command = "N"
|
||||||
json_res = json.dumps(res) if res else ""
|
json_res = json.dumps(res) if res else ""
|
||||||
return f"{reply_command}{json_res}\n"
|
return f"{reply_command}{json_res}\n"
|
||||||
return None
|
elif short_command == "I": # ITERATE
|
||||||
|
# example: I0\t0\tshared/userdb/
|
||||||
|
parts = msg[1:].split("\t")
|
||||||
|
if parts[2] == "shared/userdb/":
|
||||||
|
result = "".join(f"Oshared/userdb/{user}\t\n" for user in iter_userdb(db))
|
||||||
|
return f"{result}\n"
|
||||||
|
|
||||||
|
raise UnknownCommand(msg)
|
||||||
|
|
||||||
|
|
||||||
def handle_dovecot_protocol(rfile, wfile, db: Database, config: Config):
|
def handle_dovecot_protocol(rfile, wfile, db: Database, config: Config):
|
||||||
@@ -165,12 +230,14 @@ def handle_dovecot_protocol(rfile, wfile, db: Database, config: Config):
|
|||||||
msg = rfile.readline().strip().decode()
|
msg = rfile.readline().strip().decode()
|
||||||
if not msg:
|
if not msg:
|
||||||
break
|
break
|
||||||
res = handle_dovecot_request(msg, db, config)
|
try:
|
||||||
if res:
|
res = handle_dovecot_request(msg, db, config)
|
||||||
wfile.write(res.encode("ascii"))
|
except UnknownCommand:
|
||||||
wfile.flush()
|
logging.warning("unknown command: %r", msg)
|
||||||
else:
|
else:
|
||||||
logging.warning("request had no answer: %r", msg)
|
if res:
|
||||||
|
wfile.write(res.encode("ascii"))
|
||||||
|
wfile.flush()
|
||||||
|
|
||||||
|
|
||||||
class ThreadedUnixStreamServer(ThreadingMixIn, UnixStreamServer):
|
class ThreadedUnixStreamServer(ThreadingMixIn, UnixStreamServer):
|
||||||
@@ -178,10 +245,9 @@ class ThreadedUnixStreamServer(ThreadingMixIn, UnixStreamServer):
|
|||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
socket = sys.argv[1]
|
socket, cfgpath = sys.argv[1:]
|
||||||
passwd_entry = pwd.getpwnam(sys.argv[2])
|
config = read_config(cfgpath)
|
||||||
db = Database(sys.argv[3])
|
db = Database(config.passdb_path)
|
||||||
config = read_config(sys.argv[4])
|
|
||||||
|
|
||||||
class Handler(StreamRequestHandler):
|
class Handler(StreamRequestHandler):
|
||||||
def handle(self):
|
def handle(self):
|
||||||
@@ -197,7 +263,6 @@ def main():
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
with ThreadedUnixStreamServer(socket, Handler) as server:
|
with ThreadedUnixStreamServer(socket, Handler) as server:
|
||||||
os.chown(socket, uid=passwd_entry.pw_uid, gid=passwd_entry.pw_gid)
|
|
||||||
try:
|
try:
|
||||||
server.serve_forever()
|
server.serve_forever()
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
|
|||||||
@@ -3,14 +3,17 @@
|
|||||||
|
|
||||||
it will echo back any message that has non-empty text and also supports the /help command.
|
it will echo back any message that has non-empty text and also supports the /help command.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
|
import subprocess
|
||||||
import sys
|
import sys
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
from deltachat_rpc_client import Bot, DeltaChat, EventType, Rpc, events
|
from deltachat_rpc_client import Bot, DeltaChat, EventType, Rpc, events
|
||||||
|
|
||||||
from chatmaild.newemail import create_newemail_dict
|
|
||||||
from chatmaild.config import read_config
|
from chatmaild.config import read_config
|
||||||
|
from chatmaild.newemail import create_newemail_dict
|
||||||
|
|
||||||
hooks = events.HookCollection()
|
hooks = events.HookCollection()
|
||||||
|
|
||||||
@@ -18,14 +21,14 @@ hooks = events.HookCollection()
|
|||||||
@hooks.on(events.RawEvent)
|
@hooks.on(events.RawEvent)
|
||||||
def log_event(event):
|
def log_event(event):
|
||||||
if event.kind == EventType.INFO:
|
if event.kind == EventType.INFO:
|
||||||
logging.info(event.msg)
|
logging.info("%s", event.msg)
|
||||||
elif event.kind == EventType.WARNING:
|
elif event.kind == EventType.WARNING:
|
||||||
logging.warning(event.msg)
|
logging.warning("%s", event.msg)
|
||||||
|
|
||||||
|
|
||||||
@hooks.on(events.RawEvent(EventType.ERROR))
|
@hooks.on(events.RawEvent(EventType.ERROR))
|
||||||
def log_error(event):
|
def log_error(event):
|
||||||
logging.error(event.msg)
|
logging.error("%s", event.msg)
|
||||||
|
|
||||||
|
|
||||||
@hooks.on(events.MemberListChanged)
|
@hooks.on(events.MemberListChanged)
|
||||||
@@ -48,6 +51,9 @@ def on_group_name_changed(event):
|
|||||||
@hooks.on(events.NewMessage(func=lambda e: not e.command))
|
@hooks.on(events.NewMessage(func=lambda e: not e.command))
|
||||||
def echo(event):
|
def echo(event):
|
||||||
snapshot = event.message_snapshot
|
snapshot = event.message_snapshot
|
||||||
|
if snapshot.is_info:
|
||||||
|
# Ignore info messages
|
||||||
|
return
|
||||||
if snapshot.text or snapshot.file:
|
if snapshot.text or snapshot.file:
|
||||||
snapshot.chat.send_message(text=snapshot.text, file=snapshot.file)
|
snapshot.chat.send_message(text=snapshot.text, file=snapshot.file)
|
||||||
|
|
||||||
@@ -59,6 +65,7 @@ def help_command(event):
|
|||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
|
logging.basicConfig(level=logging.INFO)
|
||||||
path = os.environ.get("PATH")
|
path = os.environ.get("PATH")
|
||||||
venv_path = sys.argv[0].strip("echobot")
|
venv_path = sys.argv[0].strip("echobot")
|
||||||
os.environ["PATH"] = path + ":" + venv_path
|
os.environ["PATH"] = path + ":" + venv_path
|
||||||
@@ -71,14 +78,27 @@ def main():
|
|||||||
account = accounts[0] if accounts else deltachat.add_account()
|
account = accounts[0] if accounts else deltachat.add_account()
|
||||||
|
|
||||||
bot = Bot(account, hooks)
|
bot = Bot(account, hooks)
|
||||||
|
|
||||||
|
config = read_config(sys.argv[1])
|
||||||
|
|
||||||
|
# Create password file
|
||||||
|
if bot.is_configured():
|
||||||
|
password = bot.account.get_config("mail_pw")
|
||||||
|
else:
|
||||||
|
password = create_newemail_dict(config)["password"]
|
||||||
|
Path("/run/echobot/password").write_text(password)
|
||||||
|
|
||||||
|
# Give the user which doveauth runs as access to the password file.
|
||||||
|
subprocess.run(
|
||||||
|
["/usr/bin/setfacl", "-m", "user:vmail:r", "/run/echobot/password"],
|
||||||
|
check=True,
|
||||||
|
)
|
||||||
|
|
||||||
if not bot.is_configured():
|
if not bot.is_configured():
|
||||||
config = read_config(sys.argv[1])
|
|
||||||
password = create_newemail_dict(config).get("password")
|
|
||||||
email = "echo@" + config.mail_domain
|
email = "echo@" + config.mail_domain
|
||||||
bot.configure(email, password)
|
bot.configure(email, password)
|
||||||
bot.run_forever()
|
bot.run_forever()
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
logging.basicConfig(level=logging.INFO)
|
|
||||||
main()
|
main()
|
||||||
|
|||||||
@@ -1,11 +0,0 @@
|
|||||||
[Unit]
|
|
||||||
Description=Chatmail echo bot for testing it works
|
|
||||||
|
|
||||||
[Service]
|
|
||||||
ExecStart={execpath} {config_path}
|
|
||||||
Environment="PATH={remote_venv_dir}:$PATH"
|
|
||||||
Restart=always
|
|
||||||
RestartSec=30
|
|
||||||
|
|
||||||
[Install]
|
|
||||||
WantedBy=multi-user.target
|
|
||||||
36
chatmaild/src/chatmaild/filedict.py
Normal file
36
chatmaild/src/chatmaild/filedict.py
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
import json
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
from contextlib import contextmanager
|
||||||
|
|
||||||
|
import filelock
|
||||||
|
|
||||||
|
|
||||||
|
class FileDict:
|
||||||
|
"""Concurrency-safe multi-reader/single-writer persistent dict."""
|
||||||
|
|
||||||
|
def __init__(self, path):
|
||||||
|
self.path = path
|
||||||
|
self.lock_path = path.with_name(path.name + ".lock")
|
||||||
|
|
||||||
|
@contextmanager
|
||||||
|
def modify(self):
|
||||||
|
# the OS will release the lock if the process dies,
|
||||||
|
# and the contextmanager will otherwise guarantee release
|
||||||
|
with filelock.FileLock(self.lock_path):
|
||||||
|
data = self.read()
|
||||||
|
yield data
|
||||||
|
write_path = self.path.with_name(self.path.name + ".tmp")
|
||||||
|
with write_path.open("w") as f:
|
||||||
|
json.dump(data, f)
|
||||||
|
os.rename(write_path, self.path)
|
||||||
|
|
||||||
|
def read(self):
|
||||||
|
try:
|
||||||
|
with self.path.open("r") as f:
|
||||||
|
return json.load(f)
|
||||||
|
except FileNotFoundError:
|
||||||
|
return {}
|
||||||
|
except Exception:
|
||||||
|
logging.warning("corrupt serialization state at: %r", self.path)
|
||||||
|
return {}
|
||||||
@@ -1,20 +1,114 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
import asyncio
|
import asyncio
|
||||||
|
import base64
|
||||||
|
import binascii
|
||||||
import logging
|
import logging
|
||||||
import time
|
|
||||||
import sys
|
import sys
|
||||||
from email.parser import BytesParser
|
import time
|
||||||
from email import policy
|
from email import policy
|
||||||
|
from email.parser import BytesParser
|
||||||
from email.utils import parseaddr
|
from email.utils import parseaddr
|
||||||
|
from smtplib import SMTP as SMTPClient
|
||||||
|
|
||||||
from aiosmtpd.controller import Controller
|
from aiosmtpd.controller import Controller
|
||||||
from smtplib import SMTP as SMTPClient
|
|
||||||
|
|
||||||
from .config import read_config
|
from .config import read_config
|
||||||
|
|
||||||
|
|
||||||
|
def check_openpgp_payload(payload: bytes):
|
||||||
|
"""Checks the OpenPGP payload.
|
||||||
|
|
||||||
|
OpenPGP payload must consist only of PKESK and SKESK packets
|
||||||
|
terminated by a single SEIPD packet.
|
||||||
|
|
||||||
|
Returns True if OpenPGP payload is correct,
|
||||||
|
False otherwise.
|
||||||
|
|
||||||
|
May raise IndexError while trying to read OpenPGP packet header
|
||||||
|
if it is truncated.
|
||||||
|
"""
|
||||||
|
i = 0
|
||||||
|
while i < len(payload):
|
||||||
|
# Only OpenPGP format is allowed.
|
||||||
|
if payload[i] & 0xC0 != 0xC0:
|
||||||
|
return False
|
||||||
|
|
||||||
|
packet_type_id = payload[i] & 0x3F
|
||||||
|
i += 1
|
||||||
|
if payload[i] < 192:
|
||||||
|
# One-octet length.
|
||||||
|
body_len = payload[i]
|
||||||
|
i += 1
|
||||||
|
elif payload[i] < 224:
|
||||||
|
# Two-octet length.
|
||||||
|
body_len = ((payload[i] - 192) << 8) + payload[i + 1] + 192
|
||||||
|
i += 2
|
||||||
|
elif payload[i] == 255:
|
||||||
|
# Five-octet length.
|
||||||
|
body_len = (
|
||||||
|
(payload[i + 1] << 24)
|
||||||
|
| (payload[i + 2] << 16)
|
||||||
|
| (payload[i + 3] << 8)
|
||||||
|
| payload[i + 4]
|
||||||
|
)
|
||||||
|
i += 5
|
||||||
|
else:
|
||||||
|
# Partial body length is not allowed.
|
||||||
|
return False
|
||||||
|
|
||||||
|
i += body_len
|
||||||
|
|
||||||
|
if i == len(payload):
|
||||||
|
if packet_type_id == 18:
|
||||||
|
# Last packet should be
|
||||||
|
# Symmetrically Encrypted and Integrity Protected Data Packet (SEIPD)
|
||||||
|
return True
|
||||||
|
elif packet_type_id not in [1, 3]:
|
||||||
|
# All packets except the last one must be either
|
||||||
|
# Public-Key Encrypted Session Key Packet (PKESK)
|
||||||
|
# or
|
||||||
|
# Symmetric-Key Encrypted Session Key Packet (SKESK)
|
||||||
|
return False
|
||||||
|
|
||||||
|
if i == 0:
|
||||||
|
return False
|
||||||
|
|
||||||
|
if i > len(payload):
|
||||||
|
# Payload is truncated.
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
def check_armored_payload(payload: str):
|
||||||
|
prefix = "-----BEGIN PGP MESSAGE-----\r\n\r\n"
|
||||||
|
if not payload.startswith(prefix):
|
||||||
|
return False
|
||||||
|
payload = payload.removeprefix(prefix)
|
||||||
|
|
||||||
|
suffix = "-----END PGP MESSAGE-----\r\n\r\n"
|
||||||
|
if not payload.endswith(suffix):
|
||||||
|
return False
|
||||||
|
payload = payload.removesuffix(suffix)
|
||||||
|
|
||||||
|
# Remove CRC24.
|
||||||
|
payload = payload.rpartition("=")[0]
|
||||||
|
|
||||||
|
try:
|
||||||
|
payload = base64.b64decode(payload)
|
||||||
|
except binascii.Error:
|
||||||
|
return False
|
||||||
|
|
||||||
|
try:
|
||||||
|
return check_openpgp_payload(payload)
|
||||||
|
except IndexError:
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
def check_encrypted(message):
|
def check_encrypted(message):
|
||||||
"""Check that the message is an OpenPGP-encrypted message."""
|
"""Check that the message is an OpenPGP-encrypted message.
|
||||||
|
|
||||||
|
MIME structure of the message must correspond to <https://www.rfc-editor.org/rfc/rfc3156>.
|
||||||
|
"""
|
||||||
if not message.is_multipart():
|
if not message.is_multipart():
|
||||||
return False
|
return False
|
||||||
if message.get("subject") != "...":
|
if message.get("subject") != "...":
|
||||||
@@ -23,46 +117,30 @@ def check_encrypted(message):
|
|||||||
return False
|
return False
|
||||||
parts_count = 0
|
parts_count = 0
|
||||||
for part in message.iter_parts():
|
for part in message.iter_parts():
|
||||||
|
# We explicitly check Content-Type of each part later,
|
||||||
|
# but this is to be absolutely sure `get_payload()` returns string and not list.
|
||||||
|
if part.is_multipart():
|
||||||
|
return False
|
||||||
|
|
||||||
if parts_count == 0:
|
if parts_count == 0:
|
||||||
if part.get_content_type() != "application/pgp-encrypted":
|
if part.get_content_type() != "application/pgp-encrypted":
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
payload = part.get_payload()
|
||||||
|
if payload.strip() != "Version: 1":
|
||||||
|
return False
|
||||||
elif parts_count == 1:
|
elif parts_count == 1:
|
||||||
if part.get_content_type() != "application/octet-stream":
|
if part.get_content_type() != "application/octet-stream":
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
if not check_armored_payload(part.get_payload()):
|
||||||
|
return False
|
||||||
else:
|
else:
|
||||||
return False
|
return False
|
||||||
parts_count += 1
|
parts_count += 1
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
def check_mdn(message, envelope):
|
|
||||||
if len(envelope.rcpt_tos) != 1:
|
|
||||||
return False
|
|
||||||
|
|
||||||
for name in ["auto-submitted", "chat-version"]:
|
|
||||||
if not message.get(name):
|
|
||||||
return False
|
|
||||||
|
|
||||||
if message.get_content_type() != "multipart/report":
|
|
||||||
return False
|
|
||||||
|
|
||||||
body = message.get_body()
|
|
||||||
if body.get_content_type() != "text/plain":
|
|
||||||
return False
|
|
||||||
|
|
||||||
if list(body.iter_attachments()) or list(body.iter_parts()):
|
|
||||||
return False
|
|
||||||
|
|
||||||
# even with all mime-structural checks an attacker
|
|
||||||
# could try to abuse the subject or body to contain links or other
|
|
||||||
# annoyance -- we skip on checking subject/body for now as Delta Chat
|
|
||||||
# should evolve to create E2E-encrypted read receipts anyway.
|
|
||||||
# and then MDNs are just encrypted mail and can pass the border
|
|
||||||
# to other instances.
|
|
||||||
|
|
||||||
return True
|
|
||||||
|
|
||||||
|
|
||||||
async def asyncmain_beforequeue(config):
|
async def asyncmain_beforequeue(config):
|
||||||
port = config.filtermail_smtp_port
|
port = config.filtermail_smtp_port
|
||||||
Controller(BeforeQueueHandler(config), hostname="127.0.0.1", port=port).start()
|
Controller(BeforeQueueHandler(config), hostname="127.0.0.1", port=port).start()
|
||||||
@@ -74,7 +152,7 @@ class BeforeQueueHandler:
|
|||||||
self.send_rate_limiter = SendRateLimiter()
|
self.send_rate_limiter = SendRateLimiter()
|
||||||
|
|
||||||
async def handle_MAIL(self, server, session, envelope, address, mail_options):
|
async def handle_MAIL(self, server, session, envelope, address, mail_options):
|
||||||
logging.info(f"handle_MAIL from {address}")
|
logging.info("handle_MAIL from %s", address)
|
||||||
envelope.mail_from = address
|
envelope.mail_from = address
|
||||||
max_sent = self.config.max_user_send_per_minute
|
max_sent = self.config.max_user_send_per_minute
|
||||||
if not self.send_rate_limiter.is_sending_allowed(address, max_sent):
|
if not self.send_rate_limiter.is_sending_allowed(address, max_sent):
|
||||||
@@ -98,19 +176,16 @@ class BeforeQueueHandler:
|
|||||||
|
|
||||||
def check_DATA(self, envelope):
|
def check_DATA(self, envelope):
|
||||||
"""the central filtering function for e-mails."""
|
"""the central filtering function for e-mails."""
|
||||||
logging.info(f"Processing DATA message from {envelope.mail_from}")
|
logging.info("Processing DATA message from %s", envelope.mail_from)
|
||||||
|
|
||||||
message = BytesParser(policy=policy.default).parsebytes(envelope.content)
|
message = BytesParser(policy=policy.default).parsebytes(envelope.content)
|
||||||
mail_encrypted = check_encrypted(message)
|
mail_encrypted = check_encrypted(message)
|
||||||
|
|
||||||
_, from_addr = parseaddr(message.get("from").strip())
|
_, from_addr = parseaddr(message.get("from").strip())
|
||||||
logging.info(f"mime-from: {from_addr} envelope-from: {envelope.mail_from!r}")
|
logging.info("mime-from: %s envelope-from: %r", from_addr, envelope.mail_from)
|
||||||
if envelope.mail_from.lower() != from_addr.lower():
|
if envelope.mail_from.lower() != from_addr.lower():
|
||||||
return f"500 Invalid FROM <{from_addr!r}> for <{envelope.mail_from!r}>"
|
return f"500 Invalid FROM <{from_addr!r}> for <{envelope.mail_from!r}>"
|
||||||
|
|
||||||
if not mail_encrypted and check_mdn(message, envelope):
|
|
||||||
return
|
|
||||||
|
|
||||||
if envelope.mail_from in self.config.passthrough_senders:
|
if envelope.mail_from in self.config.passthrough_senders:
|
||||||
return
|
return
|
||||||
|
|
||||||
|
|||||||
@@ -8,17 +8,20 @@ mail_domain = {mail_domain}
|
|||||||
#
|
#
|
||||||
|
|
||||||
#
|
#
|
||||||
# Account Restrictions
|
# Restrictions on user addresses
|
||||||
#
|
#
|
||||||
|
|
||||||
# how many mails a user can send out per minute
|
# how many mails a user can send out per minute
|
||||||
max_user_send_per_minute = 60
|
max_user_send_per_minute = 60
|
||||||
|
|
||||||
# maximum mailbox size of a chatmail account
|
# maximum mailbox size of a chatmail address
|
||||||
max_mailbox_size = 100M
|
max_mailbox_size = 100M
|
||||||
|
|
||||||
# days after which mails are unconditionally deleted
|
# days after which mails are unconditionally deleted
|
||||||
delete_mails_after = 40
|
delete_mails_after = 20
|
||||||
|
|
||||||
|
# days after which users without a login are deleted (database and mails)
|
||||||
|
delete_inactive_users_after = 100
|
||||||
|
|
||||||
# minimum length a username must have
|
# minimum length a username must have
|
||||||
username_min_length = 9
|
username_min_length = 9
|
||||||
@@ -29,7 +32,7 @@ username_max_length = 9
|
|||||||
# minimum length a password must have
|
# minimum length a password must have
|
||||||
password_min_length = 9
|
password_min_length = 9
|
||||||
|
|
||||||
# list of chatmail accounts which can send outbound un-encrypted mail
|
# list of chatmail addresses which can send outbound un-encrypted mail
|
||||||
passthrough_senders =
|
passthrough_senders =
|
||||||
|
|
||||||
# list of e-mail recipients for which to accept outbound un-encrypted mails
|
# list of e-mail recipients for which to accept outbound un-encrypted mails
|
||||||
@@ -39,6 +42,12 @@ passthrough_recipients = xstore@testrun.org groupsbot@hispanilandia.net
|
|||||||
# Deployment Details
|
# Deployment Details
|
||||||
#
|
#
|
||||||
|
|
||||||
|
# Directory where user mailboxes are stored
|
||||||
|
mailboxes_dir = /home/vmail/mail/{mail_domain}
|
||||||
|
|
||||||
|
# user address sqlite database path
|
||||||
|
passdb_path = /home/vmail/passdb.sqlite
|
||||||
|
|
||||||
# where the filtermail SMTP service listens
|
# where the filtermail SMTP service listens
|
||||||
filtermail_smtp_port = 10080
|
filtermail_smtp_port = 10080
|
||||||
|
|
||||||
@@ -60,4 +69,3 @@ privacy_pdo =
|
|||||||
|
|
||||||
# postal address of the privacy supervisor
|
# postal address of the privacy supervisor
|
||||||
privacy_supervisor =
|
privacy_supervisor =
|
||||||
|
|
||||||
|
|||||||
165
chatmaild/src/chatmaild/metadata.py
Normal file
165
chatmaild/src/chatmaild/metadata.py
Normal file
@@ -0,0 +1,165 @@
|
|||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
from socketserver import (
|
||||||
|
StreamRequestHandler,
|
||||||
|
ThreadingMixIn,
|
||||||
|
UnixStreamServer,
|
||||||
|
)
|
||||||
|
|
||||||
|
from .config import read_config
|
||||||
|
from .filedict import FileDict
|
||||||
|
from .notifier import Notifier
|
||||||
|
|
||||||
|
DICTPROXY_HELLO_CHAR = "H"
|
||||||
|
DICTPROXY_LOOKUP_CHAR = "L"
|
||||||
|
DICTPROXY_ITERATE_CHAR = "I"
|
||||||
|
DICTPROXY_BEGIN_TRANSACTION_CHAR = "B"
|
||||||
|
DICTPROXY_SET_CHAR = "S"
|
||||||
|
DICTPROXY_COMMIT_TRANSACTION_CHAR = "C"
|
||||||
|
DICTPROXY_TRANSACTION_CHARS = "BSC"
|
||||||
|
|
||||||
|
|
||||||
|
class Metadata:
|
||||||
|
# each SETMETADATA on this key appends to a list of unique device tokens
|
||||||
|
# which only ever get removed if the upstream indicates the token is invalid
|
||||||
|
DEVICETOKEN_KEY = "devicetoken"
|
||||||
|
|
||||||
|
def __init__(self, vmail_dir):
|
||||||
|
self.vmail_dir = vmail_dir
|
||||||
|
|
||||||
|
def get_metadata_dict(self, addr):
|
||||||
|
return FileDict(self.vmail_dir / addr / "metadata.json")
|
||||||
|
|
||||||
|
def add_token_to_addr(self, addr, token):
|
||||||
|
with self.get_metadata_dict(addr).modify() as data:
|
||||||
|
tokens = data.setdefault(self.DEVICETOKEN_KEY, [])
|
||||||
|
if token not in tokens:
|
||||||
|
tokens.append(token)
|
||||||
|
|
||||||
|
def remove_token_from_addr(self, addr, token):
|
||||||
|
with self.get_metadata_dict(addr).modify() as data:
|
||||||
|
tokens = data.get(self.DEVICETOKEN_KEY, [])
|
||||||
|
if token in tokens:
|
||||||
|
tokens.remove(token)
|
||||||
|
|
||||||
|
def get_tokens_for_addr(self, addr):
|
||||||
|
mdict = self.get_metadata_dict(addr).read()
|
||||||
|
return mdict.get(self.DEVICETOKEN_KEY, [])
|
||||||
|
|
||||||
|
|
||||||
|
def handle_dovecot_protocol(rfile, wfile, notifier, metadata, iroh_relay=None):
|
||||||
|
transactions = {}
|
||||||
|
while True:
|
||||||
|
msg = rfile.readline().strip().decode()
|
||||||
|
if not msg:
|
||||||
|
break
|
||||||
|
|
||||||
|
res = handle_dovecot_request(msg, transactions, notifier, metadata, iroh_relay)
|
||||||
|
if res:
|
||||||
|
wfile.write(res.encode("ascii"))
|
||||||
|
wfile.flush()
|
||||||
|
|
||||||
|
|
||||||
|
def handle_dovecot_request(msg, transactions, notifier, metadata, iroh_relay=None):
|
||||||
|
# see https://doc.dovecot.org/3.0/developer_manual/design/dict_protocol/
|
||||||
|
short_command = msg[0]
|
||||||
|
parts = msg[1:].split("\t")
|
||||||
|
if short_command == DICTPROXY_LOOKUP_CHAR:
|
||||||
|
# Lpriv/43f5f508a7ea0366dff30200c15250e3/devicetoken\tlkj123poi@c2.testrun.org
|
||||||
|
keyparts = parts[0].split("/", 2)
|
||||||
|
if keyparts[0] == "priv":
|
||||||
|
keyname = keyparts[2]
|
||||||
|
addr = parts[1]
|
||||||
|
if keyname == metadata.DEVICETOKEN_KEY:
|
||||||
|
res = " ".join(metadata.get_tokens_for_addr(addr))
|
||||||
|
return f"O{res}\n"
|
||||||
|
elif keyparts[0] == "shared":
|
||||||
|
keyname = keyparts[2]
|
||||||
|
if (
|
||||||
|
keyname == "vendor/vendor.dovecot/pvt/server/vendor/deltachat/irohrelay"
|
||||||
|
and iroh_relay
|
||||||
|
):
|
||||||
|
# Handle `GETMETADATA "" /shared/vendor/deltachat/irohrelay`
|
||||||
|
return f"O{iroh_relay}\n"
|
||||||
|
logging.warning("lookup ignored: %r", msg)
|
||||||
|
return "N\n"
|
||||||
|
elif short_command == DICTPROXY_ITERATE_CHAR:
|
||||||
|
# Empty line means ITER_FINISHED.
|
||||||
|
# If we don't return empty line Dovecot will timeout.
|
||||||
|
return "\n"
|
||||||
|
elif short_command == DICTPROXY_HELLO_CHAR:
|
||||||
|
return # no version checking
|
||||||
|
|
||||||
|
if short_command not in (DICTPROXY_TRANSACTION_CHARS):
|
||||||
|
logging.warning("unknown dictproxy request: %r", msg)
|
||||||
|
return
|
||||||
|
|
||||||
|
transaction_id = parts[0]
|
||||||
|
|
||||||
|
if short_command == DICTPROXY_BEGIN_TRANSACTION_CHAR:
|
||||||
|
addr = parts[1]
|
||||||
|
transactions[transaction_id] = dict(addr=addr, res="O\n")
|
||||||
|
elif short_command == DICTPROXY_COMMIT_TRANSACTION_CHAR:
|
||||||
|
# each set devicetoken operation persists directly
|
||||||
|
# and does not wait until a "commit" comes
|
||||||
|
# because our dovecot config does not involve
|
||||||
|
# multiple set-operations in a single commit
|
||||||
|
return transactions.pop(transaction_id)["res"]
|
||||||
|
elif short_command == DICTPROXY_SET_CHAR:
|
||||||
|
# For documentation on key structure see
|
||||||
|
# https://github.com/dovecot/core/blob/main/src/lib-storage/mailbox-attribute.h
|
||||||
|
|
||||||
|
keyname = parts[1].split("/")
|
||||||
|
value = parts[2] if len(parts) > 2 else ""
|
||||||
|
addr = transactions[transaction_id]["addr"]
|
||||||
|
if keyname[0] == "priv" and keyname[2] == metadata.DEVICETOKEN_KEY:
|
||||||
|
metadata.add_token_to_addr(addr, value)
|
||||||
|
elif keyname[0] == "priv" and keyname[2] == "messagenew":
|
||||||
|
notifier.new_message_for_addr(addr, metadata)
|
||||||
|
else:
|
||||||
|
# Transaction failed.
|
||||||
|
transactions[transaction_id]["res"] = "F\n"
|
||||||
|
|
||||||
|
|
||||||
|
class ThreadedUnixStreamServer(ThreadingMixIn, UnixStreamServer):
|
||||||
|
request_queue_size = 100
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
socket, config_path = sys.argv[1:]
|
||||||
|
|
||||||
|
config = read_config(config_path)
|
||||||
|
iroh_relay = config.iroh_relay
|
||||||
|
|
||||||
|
vmail_dir = config.mailboxes_dir
|
||||||
|
if not vmail_dir.exists():
|
||||||
|
logging.error("vmail dir does not exist: %r", vmail_dir)
|
||||||
|
return 1
|
||||||
|
|
||||||
|
queue_dir = vmail_dir / "pending_notifications"
|
||||||
|
queue_dir.mkdir(exist_ok=True)
|
||||||
|
metadata = Metadata(vmail_dir)
|
||||||
|
notifier = Notifier(queue_dir)
|
||||||
|
notifier.start_notification_threads(metadata.remove_token_from_addr)
|
||||||
|
|
||||||
|
class Handler(StreamRequestHandler):
|
||||||
|
def handle(self):
|
||||||
|
try:
|
||||||
|
handle_dovecot_protocol(
|
||||||
|
self.rfile, self.wfile, notifier, metadata, iroh_relay
|
||||||
|
)
|
||||||
|
except Exception:
|
||||||
|
logging.exception("Exception in the dovecot dictproxy handler")
|
||||||
|
raise
|
||||||
|
|
||||||
|
try:
|
||||||
|
os.unlink(socket)
|
||||||
|
except FileNotFoundError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
with ThreadedUnixStreamServer(socket, Handler) as server:
|
||||||
|
try:
|
||||||
|
server.serve_forever()
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
pass
|
||||||
@@ -1,7 +1,6 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
from pathlib import Path
|
|
||||||
import time
|
|
||||||
import sys
|
import sys
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
|
||||||
def main(vmail_dir=None):
|
def main(vmail_dir=None):
|
||||||
@@ -16,9 +15,15 @@ def main(vmail_dir=None):
|
|||||||
if path.name[:3] in ("ci-", "ac_"):
|
if path.name[:3] in ("ci-", "ac_"):
|
||||||
ci_accounts += 1
|
ci_accounts += 1
|
||||||
|
|
||||||
timestamp = int(time.time() * 1000)
|
print("# HELP total number of accounts")
|
||||||
print(f"accounts {accounts} {timestamp}")
|
print("# TYPE accounts gauge")
|
||||||
print(f"ci_accounts {ci_accounts} {timestamp}")
|
print(f"accounts {accounts}")
|
||||||
|
print("# HELP number of CI accounts")
|
||||||
|
print("# TYPE ci_accounts gauge")
|
||||||
|
print(f"ci_accounts {ci_accounts}")
|
||||||
|
print("# HELP number of non-CI accounts")
|
||||||
|
print("# TYPE nonci_accounts gauge")
|
||||||
|
print(f"nonci_accounts {accounts - ci_accounts}")
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|||||||
@@ -1,13 +1,13 @@
|
|||||||
#!/usr/local/lib/chatmaild/venv/bin/python3
|
#!/usr/local/lib/chatmaild/venv/bin/python3
|
||||||
|
|
||||||
""" CGI script for creating new accounts. """
|
"""CGI script for creating new accounts."""
|
||||||
|
|
||||||
import json
|
import json
|
||||||
import random
|
import random
|
||||||
import secrets
|
import secrets
|
||||||
import string
|
import string
|
||||||
|
|
||||||
from chatmaild.config import read_config, Config
|
from chatmaild.config import Config, read_config
|
||||||
|
|
||||||
CONFIG_PATH = "/usr/local/lib/chatmaild/chatmail.ini"
|
CONFIG_PATH = "/usr/local/lib/chatmaild/chatmail.ini"
|
||||||
ALPHANUMERIC = string.ascii_lowercase + string.digits
|
ALPHANUMERIC = string.ascii_lowercase + string.digits
|
||||||
|
|||||||
166
chatmaild/src/chatmaild/notifier.py
Normal file
166
chatmaild/src/chatmaild/notifier.py
Normal file
@@ -0,0 +1,166 @@
|
|||||||
|
"""
|
||||||
|
This modules provides notification machinery for transmitting device tokens to
|
||||||
|
a central notification server which in turn contacts a phone provider's notification server
|
||||||
|
to trigger Delta Chat apps to retrieve messages and provide instant notifications to users.
|
||||||
|
|
||||||
|
The Notifier class arranges the queuing of tokens in separate PriorityQueues
|
||||||
|
from which NotifyThreads take and transmit them via HTTPS
|
||||||
|
to the `notifications.delta.chat` service.
|
||||||
|
The current lack of proper HTTP/2-support in Python leads us
|
||||||
|
to use multiple threads and connections to the Rust-implemented `notifications.delta.chat`
|
||||||
|
which itself uses HTTP/2 and thus only a single connection to phone-notification providers.
|
||||||
|
|
||||||
|
If a token fails to cause a successful notification
|
||||||
|
it is moved to a retry-number specific PriorityQueue
|
||||||
|
which handles all tokens that failed a particular number of times
|
||||||
|
and which are scheduled for retry using exponential back-off timing.
|
||||||
|
If a token notification would be scheduled more than DROP_DEADLINE seconds
|
||||||
|
after its first attempt, it is dropped with a log error.
|
||||||
|
|
||||||
|
Note that tokens are completely opaque to the notification machinery here
|
||||||
|
and will in the future be encrypted foreclosing all ability to distinguish
|
||||||
|
which device token ultimately goes to which phone-provider notification service,
|
||||||
|
or to understand the relation of "device tokens" and chatmail addresses.
|
||||||
|
The meaning and format of tokens is basically a matter of Delta-Chat Core and
|
||||||
|
the `notification.delta.chat` service.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import logging
|
||||||
|
import math
|
||||||
|
import os
|
||||||
|
import time
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from pathlib import Path
|
||||||
|
from queue import PriorityQueue
|
||||||
|
from threading import Thread
|
||||||
|
from uuid import uuid4
|
||||||
|
|
||||||
|
import requests
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class PersistentQueueItem:
|
||||||
|
path: Path
|
||||||
|
addr: str
|
||||||
|
start_ts: int
|
||||||
|
token: str
|
||||||
|
|
||||||
|
def delete(self):
|
||||||
|
self.path.unlink(missing_ok=True)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def create(cls, queue_dir, addr, start_ts, token):
|
||||||
|
queue_id = uuid4().hex
|
||||||
|
start_ts = int(start_ts)
|
||||||
|
path = queue_dir.joinpath(queue_id)
|
||||||
|
tmp_path = path.with_name(path.name + ".tmp")
|
||||||
|
tmp_path.write_text(f"{addr}\n{start_ts}\n{token}")
|
||||||
|
os.rename(tmp_path, path)
|
||||||
|
return cls(path, addr, start_ts, token)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def read_from_path(cls, path):
|
||||||
|
addr, start_ts, token = path.read_text().split("\n", maxsplit=2)
|
||||||
|
return cls(path, addr, int(start_ts), token)
|
||||||
|
|
||||||
|
def __lt__(self, other):
|
||||||
|
return self.start_ts < other.start_ts
|
||||||
|
|
||||||
|
|
||||||
|
class Notifier:
|
||||||
|
URL = "https://notifications.delta.chat/notify"
|
||||||
|
CONNECTION_TIMEOUT = 60.0 # seconds until http-request is given up
|
||||||
|
BASE_DELAY = 8.0 # base seconds for exponential back-off delay
|
||||||
|
DROP_DEADLINE = 5 * 60 * 60 # drop notifications after 5 hours
|
||||||
|
|
||||||
|
def __init__(self, queue_dir):
|
||||||
|
self.queue_dir = queue_dir
|
||||||
|
max_tries = int(math.log(self.DROP_DEADLINE, self.BASE_DELAY)) + 1
|
||||||
|
self.retry_queues = [PriorityQueue() for _ in range(max_tries)]
|
||||||
|
|
||||||
|
def compute_delay(self, retry_num):
|
||||||
|
return 0 if retry_num == 0 else pow(self.BASE_DELAY, retry_num)
|
||||||
|
|
||||||
|
def new_message_for_addr(self, addr, metadata):
|
||||||
|
start_ts = int(time.time())
|
||||||
|
for token in metadata.get_tokens_for_addr(addr):
|
||||||
|
queue_item = PersistentQueueItem.create(
|
||||||
|
self.queue_dir, addr, start_ts, token
|
||||||
|
)
|
||||||
|
self.queue_for_retry(queue_item)
|
||||||
|
|
||||||
|
def requeue_persistent_queue_items(self):
|
||||||
|
for queue_path in self.queue_dir.iterdir():
|
||||||
|
if queue_path.name.endswith(".tmp"):
|
||||||
|
logging.warning("removing spurious queue item: %r", queue_path)
|
||||||
|
queue_path.unlink()
|
||||||
|
continue
|
||||||
|
queue_item = PersistentQueueItem.read_from_path(queue_path)
|
||||||
|
self.queue_for_retry(queue_item)
|
||||||
|
|
||||||
|
def queue_for_retry(self, queue_item, retry_num=0):
|
||||||
|
delay = self.compute_delay(retry_num)
|
||||||
|
when = int(time.time()) + delay
|
||||||
|
deadline = queue_item.start_ts + self.DROP_DEADLINE
|
||||||
|
if retry_num >= len(self.retry_queues) or when > deadline:
|
||||||
|
queue_item.delete()
|
||||||
|
logging.error("notification exceeded deadline: %r", queue_item.token)
|
||||||
|
return
|
||||||
|
|
||||||
|
self.retry_queues[retry_num].put((when, queue_item))
|
||||||
|
|
||||||
|
def start_notification_threads(self, remove_token_from_addr):
|
||||||
|
self.requeue_persistent_queue_items()
|
||||||
|
threads = {}
|
||||||
|
for retry_num in range(len(self.retry_queues)):
|
||||||
|
# use 4 threads for first-try tokens and less for subsequent tries
|
||||||
|
num_threads = 4 if retry_num == 0 else 2
|
||||||
|
threads[retry_num] = []
|
||||||
|
for _ in range(num_threads):
|
||||||
|
thread = NotifyThread(self, retry_num, remove_token_from_addr)
|
||||||
|
threads[retry_num].append(thread)
|
||||||
|
thread.start()
|
||||||
|
return threads
|
||||||
|
|
||||||
|
|
||||||
|
class NotifyThread(Thread):
|
||||||
|
def __init__(self, notifier, retry_num, remove_token_from_addr):
|
||||||
|
super().__init__(daemon=True)
|
||||||
|
self.notifier = notifier
|
||||||
|
self.retry_num = retry_num
|
||||||
|
self.remove_token_from_addr = remove_token_from_addr
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
self.notifier.retry_queues[self.retry_num].put((None, None))
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
requests_session = requests.Session()
|
||||||
|
while self.retry_one(requests_session):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def retry_one(self, requests_session, sleep=time.sleep):
|
||||||
|
when, queue_item = self.notifier.retry_queues[self.retry_num].get()
|
||||||
|
if when is None:
|
||||||
|
return False
|
||||||
|
wait_time = when - int(time.time())
|
||||||
|
if wait_time > 0:
|
||||||
|
sleep(wait_time)
|
||||||
|
self.perform_request_to_notification_server(requests_session, queue_item)
|
||||||
|
return True
|
||||||
|
|
||||||
|
def perform_request_to_notification_server(self, requests_session, queue_item):
|
||||||
|
timeout = self.notifier.CONNECTION_TIMEOUT
|
||||||
|
token = queue_item.token
|
||||||
|
try:
|
||||||
|
res = requests_session.post(self.notifier.URL, data=token, timeout=timeout)
|
||||||
|
except requests.exceptions.RequestException as e:
|
||||||
|
res = e
|
||||||
|
else:
|
||||||
|
if res.status_code in (200, 410):
|
||||||
|
if res.status_code == 410:
|
||||||
|
self.remove_token_from_addr(queue_item.addr, token)
|
||||||
|
queue_item.delete()
|
||||||
|
return
|
||||||
|
|
||||||
|
logging.warning("Notification request failed: %r", res)
|
||||||
|
self.notifier.queue_for_retry(queue_item, retry_num=self.retry_num + 1)
|
||||||
44
chatmaild/src/chatmaild/tests/mail-data/literal.eml
Normal file
44
chatmaild/src/chatmaild/tests/mail-data/literal.eml
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
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"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,12 +1,14 @@
|
|||||||
import random
|
|
||||||
import importlib.resources
|
import importlib.resources
|
||||||
import itertools
|
import itertools
|
||||||
from email.parser import BytesParser
|
import os
|
||||||
|
import random
|
||||||
from email import policy
|
from email import policy
|
||||||
import pytest
|
from email.parser import BytesParser
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
from chatmaild.database import Database
|
import pytest
|
||||||
from chatmaild.config import read_config, write_initial_config
|
from chatmaild.config import read_config, write_initial_config
|
||||||
|
from chatmaild.database import Database
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
@@ -14,7 +16,11 @@ def make_config(tmp_path):
|
|||||||
inipath = tmp_path.joinpath("chatmail.ini")
|
inipath = tmp_path.joinpath("chatmail.ini")
|
||||||
|
|
||||||
def make_conf(mail_domain):
|
def make_conf(mail_domain):
|
||||||
write_initial_config(inipath, mail_domain=mail_domain)
|
basedir = tmp_path.joinpath(f"vmail/{mail_domain}")
|
||||||
|
basedir.mkdir(parents=True, exist_ok=True)
|
||||||
|
passdb = tmp_path.joinpath("vmail/passdb.sqlite")
|
||||||
|
overrides = dict(mailboxes_dir=str(basedir), passdb_path=str(passdb))
|
||||||
|
write_initial_config(inipath, mail_domain, overrides=overrides)
|
||||||
return read_config(inipath)
|
return read_config(inipath)
|
||||||
|
|
||||||
return make_conf
|
return make_conf
|
||||||
@@ -57,11 +63,18 @@ def db(tmpdir):
|
|||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def maildata(request):
|
def maildata(request):
|
||||||
datadir = importlib.resources.files(__package__).joinpath("mail-data")
|
try:
|
||||||
|
datadir = importlib.resources.files(__package__).joinpath("mail-data")
|
||||||
|
except TypeError:
|
||||||
|
# in python3.9 or lower, the above doesn't work, so we get datadir this way:
|
||||||
|
datadir = Path(os.getcwd()).joinpath("chatmaild/src/chatmaild/tests/mail-data")
|
||||||
|
|
||||||
assert datadir.exists(), datadir
|
assert datadir.exists(), datadir
|
||||||
|
|
||||||
def maildata(name, from_addr, to_addr):
|
def maildata(name, from_addr, to_addr):
|
||||||
data = datadir.joinpath(name).read_text()
|
# Using `.read_bytes().decode()` instead of `.read_text()` to preserve newlines.
|
||||||
|
data = datadir.joinpath(name).read_bytes().decode()
|
||||||
|
|
||||||
text = data.format(from_addr=from_addr, to_addr=to_addr)
|
text = data.format(from_addr=from_addr, to_addr=to_addr)
|
||||||
return BytesParser(policy=policy.default).parsebytes(text.encode())
|
return BytesParser(policy=policy.default).parsebytes(text.encode())
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import pytest
|
||||||
from chatmaild.config import read_config
|
from chatmaild.config import read_config
|
||||||
|
|
||||||
|
|
||||||
@@ -24,9 +25,37 @@ def test_read_config_testrun(make_config):
|
|||||||
assert config.postfix_reinject_port == 10025
|
assert config.postfix_reinject_port == 10025
|
||||||
assert config.max_user_send_per_minute == 60
|
assert config.max_user_send_per_minute == 60
|
||||||
assert config.max_mailbox_size == "100M"
|
assert config.max_mailbox_size == "100M"
|
||||||
assert config.delete_mails_after == "40"
|
assert config.delete_mails_after == "20"
|
||||||
assert config.username_min_length == 9
|
assert config.username_min_length == 9
|
||||||
assert config.username_max_length == 9
|
assert config.username_max_length == 9
|
||||||
assert config.password_min_length == 9
|
assert config.password_min_length == 9
|
||||||
assert "privacy@testrun.org" in config.passthrough_recipients
|
assert "privacy@testrun.org" in config.passthrough_recipients
|
||||||
assert config.passthrough_senders == []
|
assert config.passthrough_senders == []
|
||||||
|
|
||||||
|
|
||||||
|
def test_config_userstate_paths(make_config, tmp_path):
|
||||||
|
config = make_config("something.testrun.org")
|
||||||
|
mailboxes_dir = config.mailboxes_dir
|
||||||
|
passdb_path = config.passdb_path
|
||||||
|
assert mailboxes_dir.name == "something.testrun.org"
|
||||||
|
assert passdb_path.name == "passdb.sqlite"
|
||||||
|
assert passdb_path.is_relative_to(tmp_path)
|
||||||
|
assert config.mail_domain == "something.testrun.org"
|
||||||
|
path = config.get_user_maildir("user1@something.testrun.org")
|
||||||
|
assert not path.exists()
|
||||||
|
assert path == mailboxes_dir.joinpath("user1@something.testrun.org")
|
||||||
|
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
config.get_user_maildir("")
|
||||||
|
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
config.get_user_maildir(None)
|
||||||
|
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
config.get_user_maildir("../some@something.testrun.org")
|
||||||
|
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
config.get_user_maildir("..")
|
||||||
|
|
||||||
|
with pytest.raises(ValueError):
|
||||||
|
config.get_user_maildir(".")
|
||||||
|
|||||||
51
chatmaild/src/chatmaild/tests/test_delete_inactive_users.py
Normal file
51
chatmaild/src/chatmaild/tests/test_delete_inactive_users.py
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
import time
|
||||||
|
|
||||||
|
from chatmaild.delete_inactive_users import delete_inactive_users
|
||||||
|
from chatmaild.doveauth import lookup_passdb
|
||||||
|
|
||||||
|
|
||||||
|
def test_remove_stale_users(db, example_config):
|
||||||
|
new = time.time()
|
||||||
|
old = new - (example_config.delete_inactive_users_after * 86400) - 1
|
||||||
|
|
||||||
|
def create_user(addr, last_login):
|
||||||
|
lookup_passdb(db, example_config, addr, "q9mr3faue", last_login=last_login)
|
||||||
|
md = example_config.get_user_maildir(addr)
|
||||||
|
md.mkdir(parents=True)
|
||||||
|
md.joinpath("cur").mkdir()
|
||||||
|
md.joinpath("cur", "something").mkdir()
|
||||||
|
|
||||||
|
# create some stale and some new accounts
|
||||||
|
to_remove = []
|
||||||
|
for i in range(150):
|
||||||
|
addr = f"oldold{i:03}@chat.example.org"
|
||||||
|
create_user(addr, last_login=old)
|
||||||
|
with db.read_connection() as conn:
|
||||||
|
assert conn.get_user(addr)
|
||||||
|
to_remove.append(addr)
|
||||||
|
|
||||||
|
remain = []
|
||||||
|
for i in range(5):
|
||||||
|
addr = f"newnew{i:03}@chat.example.org"
|
||||||
|
create_user(addr, last_login=new)
|
||||||
|
remain.append(addr)
|
||||||
|
|
||||||
|
# check pre and post-conditions for delete_inactive_users()
|
||||||
|
|
||||||
|
for addr in to_remove:
|
||||||
|
assert example_config.get_user_maildir(addr).exists()
|
||||||
|
|
||||||
|
delete_inactive_users(db, example_config)
|
||||||
|
|
||||||
|
for p in example_config.mailboxes_dir.iterdir():
|
||||||
|
assert not p.name.startswith("old")
|
||||||
|
|
||||||
|
for addr in to_remove:
|
||||||
|
assert not example_config.get_user_maildir(addr).exists()
|
||||||
|
with db.read_connection() as conn:
|
||||||
|
assert not conn.get_user(addr)
|
||||||
|
|
||||||
|
for addr in remain:
|
||||||
|
assert example_config.get_user_maildir(addr).exists()
|
||||||
|
with db.read_connection() as conn:
|
||||||
|
assert conn.get_user(addr)
|
||||||
@@ -1,23 +1,27 @@
|
|||||||
import io
|
import io
|
||||||
import json
|
import json
|
||||||
import pytest
|
|
||||||
import queue
|
import queue
|
||||||
import threading
|
import threading
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
import chatmaild.doveauth
|
import chatmaild.doveauth
|
||||||
|
import pytest
|
||||||
|
from chatmaild.database import DBError
|
||||||
from chatmaild.doveauth import (
|
from chatmaild.doveauth import (
|
||||||
get_user_data,
|
get_user_data,
|
||||||
lookup_passdb,
|
|
||||||
handle_dovecot_request,
|
|
||||||
handle_dovecot_protocol,
|
handle_dovecot_protocol,
|
||||||
|
handle_dovecot_request,
|
||||||
|
is_allowed_to_create,
|
||||||
|
iter_userdb,
|
||||||
|
iter_userdb_lastlogin_before,
|
||||||
|
lookup_passdb,
|
||||||
)
|
)
|
||||||
from chatmaild.database import DBError
|
from chatmaild.newemail import create_newemail_dict
|
||||||
|
|
||||||
|
|
||||||
def test_basic(db, example_config):
|
def test_basic(db, example_config):
|
||||||
lookup_passdb(db, example_config, "asdf12345@chat.example.org", "q9mr3faue")
|
lookup_passdb(db, example_config, "asdf12345@chat.example.org", "q9mr3faue")
|
||||||
data = get_user_data(db, "asdf12345@chat.example.org")
|
data = get_user_data(db, example_config, "asdf12345@chat.example.org")
|
||||||
assert data
|
assert data
|
||||||
data2 = lookup_passdb(
|
data2 = lookup_passdb(
|
||||||
db, example_config, "asdf12345@chat.example.org", "q9mr3jewvadsfaue"
|
db, example_config, "asdf12345@chat.example.org", "q9mr3jewvadsfaue"
|
||||||
@@ -25,6 +29,49 @@ def test_basic(db, example_config):
|
|||||||
assert data == data2
|
assert data == data2
|
||||||
|
|
||||||
|
|
||||||
|
def test_iterate_addresses(db, example_config):
|
||||||
|
addresses = []
|
||||||
|
|
||||||
|
for i in range(10):
|
||||||
|
addresses.append(f"asdf1234{i}@chat.example.org")
|
||||||
|
lookup_passdb(db, example_config, addresses[-1], "q9mr3faue")
|
||||||
|
res = iter_userdb(db)
|
||||||
|
assert res == addresses
|
||||||
|
|
||||||
|
|
||||||
|
def test_iterate_addresses_lastlogin_before(db, example_config):
|
||||||
|
addresses = []
|
||||||
|
|
||||||
|
cutoff_date = 1000
|
||||||
|
for i in range(10):
|
||||||
|
addr = f"oldold{i:03}@chat.example.org"
|
||||||
|
lookup_passdb(
|
||||||
|
db, example_config, addr, "q9mr3faue", last_login=cutoff_date - 10
|
||||||
|
)
|
||||||
|
addresses.append(addr)
|
||||||
|
|
||||||
|
for i in range(5):
|
||||||
|
addr = f"newnew{i:03}@chat.example.org"
|
||||||
|
lookup_passdb(db, example_config, addr, "q9mr3faue", last_login=cutoff_date + i)
|
||||||
|
|
||||||
|
res = iter_userdb_lastlogin_before(db, cutoff_date)
|
||||||
|
assert sorted(res) == sorted(addresses)
|
||||||
|
|
||||||
|
|
||||||
|
def test_invalid_username_length(example_config):
|
||||||
|
config = example_config
|
||||||
|
config.username_min_length = 6
|
||||||
|
config.username_max_length = 10
|
||||||
|
password = create_newemail_dict(config)["password"]
|
||||||
|
assert not is_allowed_to_create(config, f"a1234@{config.mail_domain}", password)
|
||||||
|
assert is_allowed_to_create(config, f"012345@{config.mail_domain}", password)
|
||||||
|
assert is_allowed_to_create(config, f"0123456@{config.mail_domain}", password)
|
||||||
|
assert is_allowed_to_create(config, f"0123456789@{config.mail_domain}", password)
|
||||||
|
assert not is_allowed_to_create(
|
||||||
|
config, f"0123456789x@{config.mail_domain}", password
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def test_dont_overwrite_password_on_wrong_login(db, example_config):
|
def test_dont_overwrite_password_on_wrong_login(db, example_config):
|
||||||
"""Test that logging in with a different password doesn't create a new user"""
|
"""Test that logging in with a different password doesn't create a new user"""
|
||||||
res = lookup_passdb(
|
res = lookup_passdb(
|
||||||
@@ -43,7 +90,7 @@ def test_nocreate_file(db, monkeypatch, tmpdir, example_config):
|
|||||||
lookup_passdb(
|
lookup_passdb(
|
||||||
db, example_config, "newuser12@chat.example.org", "zequ0Aimuchoodaechik"
|
db, example_config, "newuser12@chat.example.org", "zequ0Aimuchoodaechik"
|
||||||
)
|
)
|
||||||
assert not get_user_data(db, "newuser12@chat.example.org")
|
assert not get_user_data(db, example_config, "newuser12@chat.example.org")
|
||||||
|
|
||||||
|
|
||||||
def test_db_version(db):
|
def test_db_version(db):
|
||||||
@@ -67,14 +114,19 @@ def test_handle_dovecot_request(db, example_config):
|
|||||||
assert res
|
assert res
|
||||||
assert res[0] == "O" and res.endswith("\n")
|
assert res[0] == "O" and res.endswith("\n")
|
||||||
userdata = json.loads(res[1:].strip())
|
userdata = json.loads(res[1:].strip())
|
||||||
assert (
|
assert userdata["home"].endswith("chat.example.org/some42123@chat.example.org")
|
||||||
userdata["home"]
|
|
||||||
== "/home/vmail/mail/chat.example.org/some42123@chat.example.org"
|
|
||||||
)
|
|
||||||
assert userdata["uid"] == userdata["gid"] == "vmail"
|
assert userdata["uid"] == userdata["gid"] == "vmail"
|
||||||
assert userdata["password"].startswith("{SHA512-CRYPT}")
|
assert userdata["password"].startswith("{SHA512-CRYPT}")
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_dovecot_protocol_hello_is_skipped(db, example_config, caplog):
|
||||||
|
rfile = io.BytesIO(b"H3\t2\t0\t\tauth\n")
|
||||||
|
wfile = io.BytesIO()
|
||||||
|
handle_dovecot_protocol(rfile, wfile, db, example_config)
|
||||||
|
assert wfile.getvalue() == b""
|
||||||
|
assert not caplog.messages
|
||||||
|
|
||||||
|
|
||||||
def test_handle_dovecot_protocol(db, example_config):
|
def test_handle_dovecot_protocol(db, example_config):
|
||||||
rfile = io.BytesIO(
|
rfile = io.BytesIO(
|
||||||
b"H3\t2\t0\t\tauth\nLshared/userdb/foobar@chat.example.org\tfoobar@chat.example.org\n"
|
b"H3\t2\t0\t\tauth\nLshared/userdb/foobar@chat.example.org\tfoobar@chat.example.org\n"
|
||||||
@@ -84,6 +136,18 @@ def test_handle_dovecot_protocol(db, example_config):
|
|||||||
assert wfile.getvalue() == b"N\n"
|
assert wfile.getvalue() == b"N\n"
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_dovecot_protocol_iterate(db, gencreds, example_config):
|
||||||
|
lookup_passdb(db, example_config, "asdf00000@chat.example.org", "q9mr3faue")
|
||||||
|
lookup_passdb(db, example_config, "asdf11111@chat.example.org", "q9mr3faue")
|
||||||
|
rfile = io.BytesIO(b"H3\t2\t0\t\tauth\nI0\t0\tshared/userdb/")
|
||||||
|
wfile = io.BytesIO()
|
||||||
|
handle_dovecot_protocol(rfile, wfile, db, example_config)
|
||||||
|
lines = wfile.getvalue().decode("ascii").split("\n")
|
||||||
|
assert lines[0] == "Oshared/userdb/asdf00000@chat.example.org\t"
|
||||||
|
assert lines[1] == "Oshared/userdb/asdf11111@chat.example.org\t"
|
||||||
|
assert not lines[2]
|
||||||
|
|
||||||
|
|
||||||
def test_50_concurrent_lookups_different_accounts(db, gencreds, example_config):
|
def test_50_concurrent_lookups_different_accounts(db, gencreds, example_config):
|
||||||
num_threads = 50
|
num_threads = 50
|
||||||
req_per_thread = 5
|
req_per_thread = 5
|
||||||
|
|||||||
19
chatmaild/src/chatmaild/tests/test_filedict.py
Normal file
19
chatmaild/src/chatmaild/tests/test_filedict.py
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
from chatmaild.filedict import FileDict
|
||||||
|
|
||||||
|
|
||||||
|
def test_basic(tmp_path):
|
||||||
|
fdict = FileDict(tmp_path.joinpath("metadata"))
|
||||||
|
assert fdict.read() == {}
|
||||||
|
with fdict.modify() as d:
|
||||||
|
d["devicetoken"] = [1, 2, 3]
|
||||||
|
d["456"] = 4.2
|
||||||
|
new = fdict.read()
|
||||||
|
assert new["devicetoken"] == [1, 2, 3]
|
||||||
|
assert new["456"] == 4.2
|
||||||
|
|
||||||
|
|
||||||
|
def test_bad_marshal_file(tmp_path, caplog):
|
||||||
|
fdict1 = FileDict(tmp_path.joinpath("metadata"))
|
||||||
|
fdict1.path.write_bytes(b"l12k3l12k3l")
|
||||||
|
assert fdict1.read() == {}
|
||||||
|
assert "corrupt" in caplog.records[0].msg
|
||||||
@@ -1,12 +1,11 @@
|
|||||||
|
import pytest
|
||||||
from chatmaild.filtermail import (
|
from chatmaild.filtermail import (
|
||||||
check_encrypted,
|
|
||||||
BeforeQueueHandler,
|
BeforeQueueHandler,
|
||||||
SendRateLimiter,
|
SendRateLimiter,
|
||||||
check_mdn,
|
check_armored_payload,
|
||||||
|
check_encrypted,
|
||||||
)
|
)
|
||||||
|
|
||||||
import pytest
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def maildomain():
|
def maildomain():
|
||||||
@@ -63,34 +62,19 @@ def test_filtermail_encryption_detection(maildata):
|
|||||||
assert not check_encrypted(msg)
|
assert not check_encrypted(msg)
|
||||||
|
|
||||||
|
|
||||||
def test_filtermail_is_mdn(maildata, gencreds, handler):
|
def test_filtermail_no_literal_packets(maildata):
|
||||||
|
"""Test that literal OpenPGP packet is not considered an encrypted mail."""
|
||||||
|
msg = maildata("literal.eml", from_addr="1@example.org", to_addr="2@example.org")
|
||||||
|
assert not check_encrypted(msg)
|
||||||
|
|
||||||
|
|
||||||
|
def test_filtermail_unencrypted_mdn(maildata, gencreds):
|
||||||
|
"""Unencrypted MDNs should not pass."""
|
||||||
from_addr = gencreds()[0]
|
from_addr = gencreds()[0]
|
||||||
to_addr = gencreds()[0] + ".other"
|
to_addr = gencreds()[0] + ".other"
|
||||||
msg = maildata("mdn.eml", from_addr, to_addr)
|
msg = maildata("mdn.eml", from_addr, to_addr)
|
||||||
|
|
||||||
class env:
|
assert not check_encrypted(msg)
|
||||||
mail_from = from_addr
|
|
||||||
rcpt_tos = [to_addr]
|
|
||||||
content = msg.as_bytes()
|
|
||||||
|
|
||||||
assert check_mdn(msg, env)
|
|
||||||
print(msg.as_string())
|
|
||||||
|
|
||||||
assert not handler.check_DATA(env)
|
|
||||||
|
|
||||||
|
|
||||||
def test_filtermail_to_multiple_recipients_no_mdn(maildata, gencreds):
|
|
||||||
from_addr = gencreds()[0]
|
|
||||||
to_addr = gencreds()[0] + ".other"
|
|
||||||
thirdaddr = gencreds()[0]
|
|
||||||
msg = maildata("mdn.eml", from_addr, to_addr)
|
|
||||||
|
|
||||||
class env:
|
|
||||||
mail_from = from_addr
|
|
||||||
rcpt_tos = [to_addr, thirdaddr]
|
|
||||||
content = msg.as_bytes()
|
|
||||||
|
|
||||||
assert not check_mdn(msg, env)
|
|
||||||
|
|
||||||
|
|
||||||
def test_send_rate_limiter():
|
def test_send_rate_limiter():
|
||||||
@@ -143,3 +127,59 @@ def test_passthrough_senders(gencreds, handler, maildata):
|
|||||||
|
|
||||||
# assert that None/no error is returned
|
# assert that None/no error is returned
|
||||||
assert not handler.check_DATA(envelope=env)
|
assert not handler.check_DATA(envelope=env)
|
||||||
|
|
||||||
|
|
||||||
|
def test_check_armored_payload():
|
||||||
|
payload = """-----BEGIN PGP MESSAGE-----\r
|
||||||
|
\r
|
||||||
|
wU4DSqFx0d1yqAoSAQdAYkX/ZN/Az4B0k7X47zKyWrXxlDEdS3WOy0Yf2+GJTFgg\r
|
||||||
|
Zk5ql0mLG8Ze+ZifCS0XMO4otlemSyJ0K1ZPdFMGzUDBTgNqzkFabxXoXRIBB0AM\r
|
||||||
|
755wlX41X6Ay3KhnwBq7yEqSykVH6F3x11iHPKraLCAGZoaS8bKKNy/zg5slda1X\r
|
||||||
|
pt14b4aC1VwtSnYhcRRELNLD/wE2TFif+g7poMmFY50VyMPLYjVP96Z5QCT4+z4H\r
|
||||||
|
Ikh/pRRN8S3JNMrRJHc6prooSJmLcx47Y5un7VFy390MsJ+LiUJuQMDdYWRAinfs\r
|
||||||
|
Ebm89Ezjm7F03qbFPXE0X4ZNzVXS/eKO0uhJQdiov/vmbn41rNtHmNpqjaO0vi5+\r
|
||||||
|
sS9tR7yDUrIXiCUCN78eBLVioxtktsPZm5cDORbQWzv+7nmCEz9/JowCUcBVdCGn\r
|
||||||
|
1ofOaH82JCAX/cRx08pLaDNj6iolVBsi56Dd+2bGxJOZOG2AMcEyz0pXY0dOAJCD\r
|
||||||
|
iUThcQeGIdRnU3j8UBcnIEsjLu2+C+rrwMZQESMWKnJ0rnqTk0pK5kXScr6F/L0L\r
|
||||||
|
UE49ccIexNm3xZvYr5drszr6wz3Tv5fdue87P4etBt90gF/Vzknck+g1LLlkzZkp\r
|
||||||
|
d8dI0k2tOSPjUbDPnSy1x+X73WGpPZmj0kWT+RGvq0nH6UkJj3AQTG2qf1T8jK+3\r
|
||||||
|
rTp3LR9vDkMwDjX4R8SA9c0wdnUzzr79OYQC9lTnzcx+fM6BBmgQ2GrS33jaFLp7\r
|
||||||
|
L6/DFpCl5zhnPjM/2dKvMkw/Kd6XS/vjwsO405FQdjSDiQEEAZA+ZvAfcjdccbbU\r
|
||||||
|
yCO+x0QNdeBsufDVnh3xvzuWy4CICdTQT4s1AWRPCzjOj+SGmx5WqCLWfsd8Ma0+\r
|
||||||
|
w/C7SfTYu1FDQILLM+llpq1M/9GPley4QZ8JQjo262AyPXsPF/OW48uuZz0Db1xT\r
|
||||||
|
Yh4iHBztj4VSdy7l2+IyaIf7cnL4EEBFxv/MwmVDXvDlxyvfAfIsd3D9SvJESzKZ\r
|
||||||
|
VWDYwaocgeCN+ojKu1p885lu1EfRbX3fr3YO02K5/c2JYDkc0Py0W3wUP/J1XUax\r
|
||||||
|
pbKpzwlkxEgtmzsGqsOfMJqBV3TNDrOA2uBsa+uBqP5MGYLZ49S/4v/bW9I01Cr1\r
|
||||||
|
D2ZkV510Y1Vgo66WlP8mRqOTyt/5WRhPD+MxXdk67BNN/PmO6tMlVoJDuk+XwWPR\r
|
||||||
|
t2TvNaND/yabT9eYI55Og4fzKD6RIjouUX8DvKLkm+7aXxVs2uuLQ3Jco3O82z55\r
|
||||||
|
dbShU1jYsrw9oouXUz06MHPbkdhNbF/2hfhZ2qA31sNeovJw65iUv7sDKX3LVWgJ\r
|
||||||
|
10jlywcDwqlU8CO7WC9lGixYTbnOkYZpXCGEl8e6Jbs79l42YFo4ogYpFK1NXFhV\r
|
||||||
|
kOXRmDf/wmfj+c/ld3L2PkvwlgofhCudOQknZbo3ub1gjiTn7L+lMGHIj/3suMIl\r
|
||||||
|
ID4EUxAXScIM1ZEz2fjtW5jATlqYcLjLTbf/olw6HFyPNH+9IssqXeZNKnGwPUB9\r
|
||||||
|
3lTXsg0tpzl+x7F/2WjEw1DSNhjC0KnHt1vEYNMkUGDGFdN9y3ERLqX/FIgiASUb\r
|
||||||
|
bTvAVupnAK3raBezGmhrs6LsQtLS9P0VvQiLU3uDhMqw8Z4SISLpcD+NnVBHzQqm\r
|
||||||
|
6W5Qn/8xsCL6av18yUVTi2G3igt3QCNoYx9evt2ZcIkNoyyagUVjfZe5GHXh8Dnz\r
|
||||||
|
GaBXW/hg3HlXLRGaQu4RYCzBMJILcO25OhZOg6jbkCLiEexQlm2e9krB5cXR49Al\r
|
||||||
|
UN4fiB0KR9JyG2ayUdNJVkXZSZLnHyRgiaadlpUo16LVvw==\r
|
||||||
|
=b5Kp\r
|
||||||
|
-----END PGP MESSAGE-----\r
|
||||||
|
\r
|
||||||
|
"""
|
||||||
|
|
||||||
|
assert check_armored_payload(payload) == True
|
||||||
|
|
||||||
|
payload = """-----BEGIN PGP MESSAGE-----\r
|
||||||
|
\r
|
||||||
|
HELLOWORLD
|
||||||
|
-----END PGP MESSAGE-----\r
|
||||||
|
\r
|
||||||
|
"""
|
||||||
|
assert check_armored_payload(payload) == False
|
||||||
|
|
||||||
|
payload = """-----BEGIN PGP MESSAGE-----\r
|
||||||
|
\r
|
||||||
|
=njUN
|
||||||
|
-----END PGP MESSAGE-----\r
|
||||||
|
\r
|
||||||
|
"""
|
||||||
|
assert check_armored_payload(payload) == False
|
||||||
|
|||||||
312
chatmaild/src/chatmaild/tests/test_metadata.py
Normal file
312
chatmaild/src/chatmaild/tests/test_metadata.py
Normal file
@@ -0,0 +1,312 @@
|
|||||||
|
import io
|
||||||
|
import time
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
import requests
|
||||||
|
from chatmaild.metadata import (
|
||||||
|
Metadata,
|
||||||
|
handle_dovecot_protocol,
|
||||||
|
handle_dovecot_request,
|
||||||
|
)
|
||||||
|
from chatmaild.notifier import (
|
||||||
|
Notifier,
|
||||||
|
NotifyThread,
|
||||||
|
PersistentQueueItem,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def notifier(metadata):
|
||||||
|
queue_dir = metadata.vmail_dir.joinpath("pending_notifications")
|
||||||
|
queue_dir.mkdir()
|
||||||
|
return Notifier(queue_dir)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def metadata(tmp_path):
|
||||||
|
vmail_dir = tmp_path.joinpath("vmaildir")
|
||||||
|
vmail_dir.mkdir()
|
||||||
|
return Metadata(vmail_dir)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def testaddr():
|
||||||
|
return "user.name@example.org"
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def testaddr2():
|
||||||
|
return "user2@example.org"
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def token():
|
||||||
|
return "01234"
|
||||||
|
|
||||||
|
|
||||||
|
def get_mocked_requests(statuslist):
|
||||||
|
class ReqMock:
|
||||||
|
requests = []
|
||||||
|
|
||||||
|
def post(self, url, data, timeout):
|
||||||
|
self.requests.append((url, data, timeout))
|
||||||
|
res = statuslist.pop(0)
|
||||||
|
if isinstance(res, Exception):
|
||||||
|
raise res
|
||||||
|
|
||||||
|
class Result:
|
||||||
|
status_code = res
|
||||||
|
|
||||||
|
return Result()
|
||||||
|
|
||||||
|
return ReqMock()
|
||||||
|
|
||||||
|
|
||||||
|
def test_metadata_persistence(tmp_path, testaddr, testaddr2):
|
||||||
|
metadata1 = Metadata(tmp_path)
|
||||||
|
metadata2 = Metadata(tmp_path)
|
||||||
|
assert not metadata1.get_tokens_for_addr(testaddr)
|
||||||
|
assert not metadata2.get_tokens_for_addr(testaddr)
|
||||||
|
|
||||||
|
metadata1.add_token_to_addr(testaddr, "01234")
|
||||||
|
metadata1.add_token_to_addr(testaddr2, "456")
|
||||||
|
assert metadata2.get_tokens_for_addr(testaddr) == ["01234"]
|
||||||
|
assert metadata2.get_tokens_for_addr(testaddr2) == ["456"]
|
||||||
|
metadata2.remove_token_from_addr(testaddr, "01234")
|
||||||
|
assert not metadata1.get_tokens_for_addr(testaddr)
|
||||||
|
assert metadata1.get_tokens_for_addr(testaddr2) == ["456"]
|
||||||
|
|
||||||
|
|
||||||
|
def test_remove_nonexisting(metadata, tmp_path, testaddr):
|
||||||
|
metadata.add_token_to_addr(testaddr, "123")
|
||||||
|
metadata.remove_token_from_addr(testaddr, "1l23k1l2k3")
|
||||||
|
assert metadata.get_tokens_for_addr(testaddr) == ["123"]
|
||||||
|
|
||||||
|
|
||||||
|
def test_notifier_remove_without_set(metadata, testaddr):
|
||||||
|
metadata.remove_token_from_addr(testaddr, "123")
|
||||||
|
assert not metadata.get_tokens_for_addr(testaddr)
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_dovecot_request_lookup_fails(notifier, metadata, testaddr):
|
||||||
|
res = handle_dovecot_request(
|
||||||
|
f"Lpriv/123/chatmail\t{testaddr}", {}, notifier, metadata
|
||||||
|
)
|
||||||
|
assert res == "N\n"
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_dovecot_request_happy_path(notifier, metadata, testaddr, token):
|
||||||
|
transactions = {}
|
||||||
|
|
||||||
|
# set device token in a transaction
|
||||||
|
tx = "1111"
|
||||||
|
msg = f"B{tx}\t{testaddr}"
|
||||||
|
res = handle_dovecot_request(msg, transactions, notifier, metadata)
|
||||||
|
assert not res and not metadata.get_tokens_for_addr(testaddr)
|
||||||
|
assert transactions == {tx: dict(addr=testaddr, res="O\n")}
|
||||||
|
|
||||||
|
msg = f"S{tx}\tpriv/guid00/devicetoken\t{token}"
|
||||||
|
res = handle_dovecot_request(msg, transactions, notifier, metadata)
|
||||||
|
assert not res
|
||||||
|
assert len(transactions) == 1
|
||||||
|
assert metadata.get_tokens_for_addr(testaddr) == [token]
|
||||||
|
|
||||||
|
msg = f"C{tx}"
|
||||||
|
res = handle_dovecot_request(msg, transactions, notifier, metadata)
|
||||||
|
assert res == "O\n"
|
||||||
|
assert len(transactions) == 0
|
||||||
|
assert metadata.get_tokens_for_addr(testaddr) == [token]
|
||||||
|
|
||||||
|
# trigger notification for incoming message
|
||||||
|
tx2 = "2222"
|
||||||
|
assert (
|
||||||
|
handle_dovecot_request(f"B{tx2}\t{testaddr}", transactions, notifier, metadata)
|
||||||
|
is None
|
||||||
|
)
|
||||||
|
msg = f"S{tx2}\tpriv/guid00/messagenew"
|
||||||
|
assert handle_dovecot_request(msg, transactions, notifier, metadata) is None
|
||||||
|
queue_item = notifier.retry_queues[0].get()[1]
|
||||||
|
assert queue_item.token == token
|
||||||
|
assert handle_dovecot_request(f"C{tx2}", transactions, notifier, metadata) == "O\n"
|
||||||
|
assert not transactions
|
||||||
|
assert queue_item.path.exists()
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_dovecot_protocol_set_devicetoken(metadata, notifier):
|
||||||
|
rfile = io.BytesIO(
|
||||||
|
b"\n".join(
|
||||||
|
[
|
||||||
|
b"HELLO",
|
||||||
|
b"Btx00\tuser@example.org",
|
||||||
|
b"Stx00\tpriv/guid00/devicetoken\t01234",
|
||||||
|
b"Ctx00",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
wfile = io.BytesIO()
|
||||||
|
handle_dovecot_protocol(rfile, wfile, notifier, metadata)
|
||||||
|
assert wfile.getvalue() == b"O\n"
|
||||||
|
assert metadata.get_tokens_for_addr("user@example.org") == ["01234"]
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_dovecot_protocol_set_get_devicetoken(metadata, notifier):
|
||||||
|
rfile = io.BytesIO(
|
||||||
|
b"\n".join(
|
||||||
|
[
|
||||||
|
b"HELLO",
|
||||||
|
b"Btx00\tuser@example.org",
|
||||||
|
b"Stx00\tpriv/guid00/devicetoken\t01234",
|
||||||
|
b"Ctx00",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
wfile = io.BytesIO()
|
||||||
|
handle_dovecot_protocol(rfile, wfile, notifier, metadata)
|
||||||
|
assert metadata.get_tokens_for_addr("user@example.org") == ["01234"]
|
||||||
|
assert wfile.getvalue() == b"O\n"
|
||||||
|
|
||||||
|
rfile = io.BytesIO(
|
||||||
|
b"\n".join([b"HELLO", b"Lpriv/0123/devicetoken\tuser@example.org"])
|
||||||
|
)
|
||||||
|
wfile = io.BytesIO()
|
||||||
|
handle_dovecot_protocol(rfile, wfile, notifier, metadata)
|
||||||
|
assert wfile.getvalue() == b"O01234\n"
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_dovecot_protocol_iterate(metadata, notifier):
|
||||||
|
rfile = io.BytesIO(
|
||||||
|
b"\n".join(
|
||||||
|
[
|
||||||
|
b"H",
|
||||||
|
b"I9\t0\tpriv/5cbe730f146fea6535be0d003dd4fc98/\tci-2dzsrs@nine.testrun.org",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
wfile = io.BytesIO()
|
||||||
|
handle_dovecot_protocol(rfile, wfile, notifier, metadata)
|
||||||
|
assert wfile.getvalue() == b"\n"
|
||||||
|
|
||||||
|
|
||||||
|
def test_notifier_thread_deletes_persistent_file(metadata, notifier, testaddr):
|
||||||
|
reqmock = get_mocked_requests([200])
|
||||||
|
metadata.add_token_to_addr(testaddr, "01234")
|
||||||
|
notifier.new_message_for_addr(testaddr, metadata)
|
||||||
|
NotifyThread(notifier, 0, None).retry_one(reqmock)
|
||||||
|
url, data, timeout = reqmock.requests[0]
|
||||||
|
assert data == "01234"
|
||||||
|
assert metadata.get_tokens_for_addr(testaddr) == ["01234"]
|
||||||
|
notifier.requeue_persistent_queue_items()
|
||||||
|
assert notifier.retry_queues[0].qsize() == 0
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("status", [requests.exceptions.RequestException(), 404, 500])
|
||||||
|
def test_notifier_thread_connection_failures(
|
||||||
|
metadata, notifier, testaddr, status, caplog
|
||||||
|
):
|
||||||
|
"""test that tokens keep getting retried until they are given up."""
|
||||||
|
metadata.add_token_to_addr(testaddr, "01234")
|
||||||
|
notifier.new_message_for_addr(testaddr, metadata)
|
||||||
|
notifier.NOTIFICATION_RETRY_DELAY = 5
|
||||||
|
max_tries = len(notifier.retry_queues)
|
||||||
|
for i in range(max_tries):
|
||||||
|
caplog.clear()
|
||||||
|
reqmock = get_mocked_requests([status])
|
||||||
|
sleep_calls = []
|
||||||
|
NotifyThread(notifier, i, None).retry_one(reqmock, sleep=sleep_calls.append)
|
||||||
|
assert notifier.retry_queues[i].qsize() == 0
|
||||||
|
assert "request failed" in caplog.records[0].msg
|
||||||
|
if i > 0:
|
||||||
|
assert len(sleep_calls) == 1
|
||||||
|
if i + 1 < max_tries:
|
||||||
|
assert notifier.retry_queues[i + 1].qsize() == 1
|
||||||
|
assert len(caplog.records) == 1
|
||||||
|
else:
|
||||||
|
assert len(caplog.records) == 2
|
||||||
|
assert "deadline" in caplog.records[1].msg
|
||||||
|
notifier.requeue_persistent_queue_items()
|
||||||
|
assert notifier.retry_queues[0].qsize() == 0
|
||||||
|
|
||||||
|
|
||||||
|
def test_requeue_removes_tmp_files(notifier, metadata, testaddr, caplog):
|
||||||
|
metadata.add_token_to_addr(testaddr, "01234")
|
||||||
|
notifier.new_message_for_addr(testaddr, metadata)
|
||||||
|
p = notifier.queue_dir.joinpath("1203981203.tmp")
|
||||||
|
p.touch()
|
||||||
|
notifier2 = notifier.__class__(notifier.queue_dir)
|
||||||
|
notifier2.requeue_persistent_queue_items()
|
||||||
|
assert "spurious" in caplog.records[0].msg
|
||||||
|
assert not p.exists()
|
||||||
|
assert notifier2.retry_queues[0].qsize() == 1
|
||||||
|
when, queue_item = notifier2.retry_queues[0].get()
|
||||||
|
assert when <= int(time.time())
|
||||||
|
assert queue_item.addr == testaddr
|
||||||
|
|
||||||
|
|
||||||
|
def test_start_and_stop_notification_threads(notifier, testaddr):
|
||||||
|
threads = notifier.start_notification_threads(None)
|
||||||
|
for retry_num, threadlist in threads.items():
|
||||||
|
for t in threadlist:
|
||||||
|
t.stop()
|
||||||
|
t.join()
|
||||||
|
|
||||||
|
|
||||||
|
def test_multi_device_notifier(metadata, notifier, testaddr):
|
||||||
|
metadata.add_token_to_addr(testaddr, "01234")
|
||||||
|
metadata.add_token_to_addr(testaddr, "56789")
|
||||||
|
notifier.new_message_for_addr(testaddr, metadata)
|
||||||
|
reqmock = get_mocked_requests([200, 200])
|
||||||
|
NotifyThread(notifier, 0, None).retry_one(reqmock)
|
||||||
|
NotifyThread(notifier, 0, None).retry_one(reqmock)
|
||||||
|
assert notifier.retry_queues[0].qsize() == 0
|
||||||
|
assert notifier.retry_queues[1].qsize() == 0
|
||||||
|
url, data, timeout = reqmock.requests[0]
|
||||||
|
assert data == "01234"
|
||||||
|
url, data, timeout = reqmock.requests[1]
|
||||||
|
assert data == "56789"
|
||||||
|
assert metadata.get_tokens_for_addr(testaddr) == ["01234", "56789"]
|
||||||
|
|
||||||
|
|
||||||
|
def test_notifier_thread_run_gone_removes_token(metadata, notifier, testaddr):
|
||||||
|
metadata.add_token_to_addr(testaddr, "01234")
|
||||||
|
metadata.add_token_to_addr(testaddr, "45678")
|
||||||
|
notifier.new_message_for_addr(testaddr, metadata)
|
||||||
|
|
||||||
|
reqmock = get_mocked_requests([410, 200])
|
||||||
|
NotifyThread(notifier, 0, metadata.remove_token_from_addr).retry_one(reqmock)
|
||||||
|
NotifyThread(notifier, 0, None).retry_one(reqmock)
|
||||||
|
url, data, timeout = reqmock.requests[0]
|
||||||
|
assert data == "01234"
|
||||||
|
url, data, timeout = reqmock.requests[1]
|
||||||
|
assert data == "45678"
|
||||||
|
assert metadata.get_tokens_for_addr(testaddr) == ["45678"]
|
||||||
|
assert notifier.retry_queues[0].qsize() == 0
|
||||||
|
assert notifier.retry_queues[1].qsize() == 0
|
||||||
|
|
||||||
|
|
||||||
|
def test_persistent_queue_items(tmp_path, testaddr, token):
|
||||||
|
queue_item = PersistentQueueItem.create(tmp_path, testaddr, 432, token)
|
||||||
|
assert queue_item.addr == testaddr
|
||||||
|
assert queue_item.start_ts == 432
|
||||||
|
assert queue_item.token == token
|
||||||
|
item2 = PersistentQueueItem.read_from_path(queue_item.path)
|
||||||
|
assert item2.addr == testaddr
|
||||||
|
assert item2.start_ts == 432
|
||||||
|
assert item2.token == token
|
||||||
|
assert item2 == queue_item
|
||||||
|
item2.delete()
|
||||||
|
assert not item2.path.exists()
|
||||||
|
assert not queue_item < item2 and not item2 < queue_item
|
||||||
|
|
||||||
|
|
||||||
|
def test_iroh_relay(metadata):
|
||||||
|
rfile = io.BytesIO(
|
||||||
|
b"\n".join(
|
||||||
|
[
|
||||||
|
b"H",
|
||||||
|
b"Lshared/0123/vendor/vendor.dovecot/pvt/server/vendor/deltachat/irohrelay\tuser@example.org",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
wfile = io.BytesIO()
|
||||||
|
handle_dovecot_protocol(rfile, wfile, notifier, metadata, "https://example.org/")
|
||||||
|
assert wfile.getvalue() == b"Ohttps://example.org/\n"
|
||||||
@@ -8,9 +8,10 @@ def test_main(tmp_path, capsys):
|
|||||||
out, _ = capsys.readouterr()
|
out, _ = capsys.readouterr()
|
||||||
d = {}
|
d = {}
|
||||||
for line in out.split("\n"):
|
for line in out.split("\n"):
|
||||||
if line.strip():
|
if line.strip() and not line.startswith("#"):
|
||||||
name, num, _ = line.split()
|
name, num = line.split()
|
||||||
d[name] = int(num)
|
d[name] = int(num)
|
||||||
|
|
||||||
assert d["accounts"] == 4
|
assert d["accounts"] == 4
|
||||||
assert d["ci_accounts"] == 3
|
assert d["ci_accounts"] == 3
|
||||||
|
assert d["nonci_accounts"] == 1
|
||||||
|
|||||||
@@ -16,9 +16,10 @@ dependencies = [
|
|||||||
"build",
|
"build",
|
||||||
"tox",
|
"tox",
|
||||||
"ruff",
|
"ruff",
|
||||||
"black",
|
|
||||||
"pytest",
|
"pytest",
|
||||||
"pytest-xdist",
|
"pytest-xdist",
|
||||||
|
"execnet",
|
||||||
|
"imap_tools",
|
||||||
]
|
]
|
||||||
|
|
||||||
[project.scripts]
|
[project.scripts]
|
||||||
@@ -30,3 +31,13 @@ cmdeploy = "cmdeploy.cmdeploy:main"
|
|||||||
|
|
||||||
[tool.pytest.ini_options]
|
[tool.pytest.ini_options]
|
||||||
addopts = "-v -ra --strict-markers"
|
addopts = "-v -ra --strict-markers"
|
||||||
|
|
||||||
|
[tool.ruff]
|
||||||
|
lint.select = [
|
||||||
|
"F", # Pyflakes
|
||||||
|
"I", # isort
|
||||||
|
|
||||||
|
"PLC", # Pylint Convention
|
||||||
|
"PLE", # Pylint Error
|
||||||
|
"PLW", # Pylint Warning
|
||||||
|
]
|
||||||
|
|||||||
@@ -1,20 +1,21 @@
|
|||||||
"""
|
"""
|
||||||
Chat Mail pyinfra deploy.
|
Chat Mail pyinfra deploy.
|
||||||
"""
|
"""
|
||||||
import sys
|
|
||||||
import importlib.resources
|
import importlib.resources
|
||||||
import subprocess
|
|
||||||
import shutil
|
|
||||||
import io
|
import io
|
||||||
|
import shutil
|
||||||
|
import subprocess
|
||||||
|
import sys
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
|
from chatmaild.config import Config, read_config
|
||||||
from pyinfra import host
|
from pyinfra import host
|
||||||
from pyinfra.operations import apt, files, server, systemd, pip
|
|
||||||
from pyinfra.facts.files import File
|
from pyinfra.facts.files import File
|
||||||
from pyinfra.facts.systemd import SystemdEnabled
|
from pyinfra.facts.systemd import SystemdEnabled
|
||||||
from .acmetool import deploy_acmetool
|
from pyinfra.operations import apt, files, pip, server, systemd
|
||||||
|
|
||||||
from chatmaild.config import read_config, Config
|
from .acmetool import deploy_acmetool
|
||||||
|
|
||||||
|
|
||||||
def _build_chatmaild(dist_dir) -> None:
|
def _build_chatmaild(dist_dir) -> None:
|
||||||
@@ -91,7 +92,7 @@ def _install_remote_venv_with_chatmaild(config) -> None:
|
|||||||
group="root",
|
group="root",
|
||||||
mode="644",
|
mode="644",
|
||||||
config={
|
config={
|
||||||
"mail_domain": config.mail_domain,
|
"mailboxes_dir": config.mailboxes_dir,
|
||||||
"execpath": f"{remote_venv_dir}/bin/chatmail-metrics",
|
"execpath": f"{remote_venv_dir}/bin/chatmail-metrics",
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
@@ -101,13 +102,17 @@ def _install_remote_venv_with_chatmaild(config) -> None:
|
|||||||
"doveauth",
|
"doveauth",
|
||||||
"filtermail",
|
"filtermail",
|
||||||
"echobot",
|
"echobot",
|
||||||
|
"chatmail-metadata",
|
||||||
):
|
):
|
||||||
params = dict(
|
params = dict(
|
||||||
execpath=f"{remote_venv_dir}/bin/{fn}",
|
execpath=f"{remote_venv_dir}/bin/{fn}",
|
||||||
config_path=remote_chatmail_inipath,
|
config_path=remote_chatmail_inipath,
|
||||||
remote_venv_dir=remote_venv_dir,
|
remote_venv_dir=remote_venv_dir,
|
||||||
|
mail_domain=config.mail_domain,
|
||||||
|
)
|
||||||
|
source_path = importlib.resources.files(__package__).joinpath(
|
||||||
|
"service", f"{fn}.service.f"
|
||||||
)
|
)
|
||||||
source_path = importlib.resources.files("chatmaild").joinpath(f"{fn}.service.f")
|
|
||||||
content = source_path.read_text().format(**params).encode()
|
content = source_path.read_text().format(**params).encode()
|
||||||
|
|
||||||
files.put(
|
files.put(
|
||||||
@@ -130,20 +135,6 @@ def _configure_opendkim(domain: str, dkim_selector: str = "dkim") -> bool:
|
|||||||
"""Configures OpenDKIM"""
|
"""Configures OpenDKIM"""
|
||||||
need_restart = False
|
need_restart = False
|
||||||
|
|
||||||
server.group(name="Create opendkim group", group="opendkim", system=True)
|
|
||||||
server.user(
|
|
||||||
name="Create opendkim user",
|
|
||||||
user="opendkim",
|
|
||||||
groups=["opendkim"],
|
|
||||||
system=True,
|
|
||||||
)
|
|
||||||
server.user(
|
|
||||||
name="Add postfix user to opendkim group for socket access",
|
|
||||||
user="postfix",
|
|
||||||
groups=["opendkim"],
|
|
||||||
system=True,
|
|
||||||
)
|
|
||||||
|
|
||||||
main_config = files.template(
|
main_config = files.template(
|
||||||
src=importlib.resources.files(__package__).joinpath("opendkim/opendkim.conf"),
|
src=importlib.resources.files(__package__).joinpath("opendkim/opendkim.conf"),
|
||||||
dest="/etc/opendkim.conf",
|
dest="/etc/opendkim.conf",
|
||||||
@@ -303,9 +294,7 @@ def _configure_postfix(config: Config, debug: bool = False) -> bool:
|
|||||||
|
|
||||||
# Login map that 1:1 maps email address to login.
|
# Login map that 1:1 maps email address to login.
|
||||||
login_map = files.put(
|
login_map = files.put(
|
||||||
src=importlib.resources.files(__package__).joinpath(
|
src=importlib.resources.files(__package__).joinpath("postfix/login_map"),
|
||||||
"postfix/login_map"
|
|
||||||
),
|
|
||||||
dest="/etc/postfix/login_map",
|
dest="/etc/postfix/login_map",
|
||||||
user="root",
|
user="root",
|
||||||
group="root",
|
group="root",
|
||||||
@@ -338,6 +327,16 @@ def _configure_dovecot(config: Config, debug: bool = False) -> bool:
|
|||||||
mode="644",
|
mode="644",
|
||||||
)
|
)
|
||||||
need_restart |= auth_config.changed
|
need_restart |= auth_config.changed
|
||||||
|
lua_push_notification_script = files.put(
|
||||||
|
src=importlib.resources.files(__package__).joinpath(
|
||||||
|
"dovecot/push_notification.lua"
|
||||||
|
),
|
||||||
|
dest="/etc/dovecot/push_notification.lua",
|
||||||
|
user="root",
|
||||||
|
group="root",
|
||||||
|
mode="644",
|
||||||
|
)
|
||||||
|
need_restart |= lua_push_notification_script.changed
|
||||||
|
|
||||||
files.template(
|
files.template(
|
||||||
src=importlib.resources.files(__package__).joinpath("dovecot/expunge.cron.j2"),
|
src=importlib.resources.files(__package__).joinpath("dovecot/expunge.cron.j2"),
|
||||||
@@ -427,8 +426,11 @@ def check_config(config):
|
|||||||
mail_domain = config.mail_domain
|
mail_domain = config.mail_domain
|
||||||
if mail_domain != "testrun.org" and not mail_domain.endswith(".testrun.org"):
|
if mail_domain != "testrun.org" and not mail_domain.endswith(".testrun.org"):
|
||||||
blocked_words = "merlinux schmieder testrun.org".split()
|
blocked_words = "merlinux schmieder testrun.org".split()
|
||||||
for value in config.__dict__.values():
|
for key in config.__dict__:
|
||||||
if any(x in str(value) for x in blocked_words):
|
value = config.__dict__[key]
|
||||||
|
if key.startswith("privacy") and any(
|
||||||
|
x in str(value) for x in blocked_words
|
||||||
|
):
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
f"please set your own privacy contacts/addresses in {config._inipath}"
|
f"please set your own privacy contacts/addresses in {config._inipath}"
|
||||||
)
|
)
|
||||||
@@ -446,9 +448,47 @@ def deploy_chatmail(config_path: Path) -> None:
|
|||||||
|
|
||||||
from .www import build_webpages
|
from .www import build_webpages
|
||||||
|
|
||||||
apt.update(name="apt update", cache_time=24 * 3600)
|
|
||||||
server.group(name="Create vmail group", group="vmail", system=True)
|
server.group(name="Create vmail group", group="vmail", system=True)
|
||||||
server.user(name="Create vmail user", user="vmail", group="vmail", system=True)
|
server.user(name="Create vmail user", user="vmail", group="vmail", system=True)
|
||||||
|
server.group(name="Create opendkim group", group="opendkim", system=True)
|
||||||
|
server.user(
|
||||||
|
name="Create opendkim user",
|
||||||
|
user="opendkim",
|
||||||
|
groups=["opendkim"],
|
||||||
|
system=True,
|
||||||
|
)
|
||||||
|
server.user(
|
||||||
|
name="Add postfix user to opendkim group for socket access",
|
||||||
|
user="postfix",
|
||||||
|
groups=["opendkim"],
|
||||||
|
system=True,
|
||||||
|
)
|
||||||
|
server.user(name="Create echobot user", user="echobot", system=True)
|
||||||
|
|
||||||
|
# Add our OBS repository for dovecot_no_delay
|
||||||
|
files.put(
|
||||||
|
name="Add Deltachat OBS GPG key to apt keyring",
|
||||||
|
src=importlib.resources.files(__package__).joinpath("obs-home-deltachat.gpg"),
|
||||||
|
dest="/etc/apt/keyrings/obs-home-deltachat.gpg",
|
||||||
|
user="root",
|
||||||
|
group="root",
|
||||||
|
mode="644",
|
||||||
|
)
|
||||||
|
|
||||||
|
files.line(
|
||||||
|
name="Add DeltaChat OBS home repository to sources.list",
|
||||||
|
path="/etc/apt/sources.list",
|
||||||
|
line="deb [signed-by=/etc/apt/keyrings/obs-home-deltachat.gpg] https://download.opensuse.org/repositories/home:/deltachat/Debian_12/ ./",
|
||||||
|
escape_regex_characters=True,
|
||||||
|
ensure_newline=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
apt.update(name="apt update", cache_time=24 * 3600)
|
||||||
|
|
||||||
|
apt.packages(
|
||||||
|
name="Install rsync",
|
||||||
|
packages=["rsync"],
|
||||||
|
)
|
||||||
|
|
||||||
# Run local DNS resolver `unbound`.
|
# Run local DNS resolver `unbound`.
|
||||||
# `resolvconf` takes care of setting up /etc/resolv.conf
|
# `resolvconf` takes care of setting up /etc/resolv.conf
|
||||||
@@ -473,10 +513,15 @@ def deploy_chatmail(config_path: Path) -> None:
|
|||||||
|
|
||||||
# Deploy acmetool to have TLS certificates.
|
# Deploy acmetool to have TLS certificates.
|
||||||
deploy_acmetool(
|
deploy_acmetool(
|
||||||
nginx_hook=True,
|
|
||||||
domains=[mail_domain, f"mta-sts.{mail_domain}", f"www.{mail_domain}"],
|
domains=[mail_domain, f"mta-sts.{mail_domain}", f"www.{mail_domain}"],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
apt.packages(
|
||||||
|
# required for setfacl for echobot
|
||||||
|
name="Install acl",
|
||||||
|
packages="acl",
|
||||||
|
)
|
||||||
|
|
||||||
apt.packages(
|
apt.packages(
|
||||||
name="Install Postfix",
|
name="Install Postfix",
|
||||||
packages="postfix",
|
packages="postfix",
|
||||||
@@ -489,7 +534,7 @@ def deploy_chatmail(config_path: Path) -> None:
|
|||||||
|
|
||||||
apt.packages(
|
apt.packages(
|
||||||
name="Install nginx",
|
name="Install nginx",
|
||||||
packages=["nginx"],
|
packages=["nginx", "libnginx-mod-stream"],
|
||||||
)
|
)
|
||||||
|
|
||||||
apt.packages(
|
apt.packages(
|
||||||
@@ -531,14 +576,9 @@ def deploy_chatmail(config_path: Path) -> None:
|
|||||||
restarted=mta_sts_need_restart,
|
restarted=mta_sts_need_restart,
|
||||||
)
|
)
|
||||||
|
|
||||||
systemd.service(
|
# Dovecot should be started before Postfix
|
||||||
name="Start and enable Postfix",
|
# because it creates authentication socket
|
||||||
service="postfix.service",
|
# required by Postfix.
|
||||||
running=True,
|
|
||||||
enabled=True,
|
|
||||||
restarted=postfix_need_restart,
|
|
||||||
)
|
|
||||||
|
|
||||||
systemd.service(
|
systemd.service(
|
||||||
name="Start and enable Dovecot",
|
name="Start and enable Dovecot",
|
||||||
service="dovecot.service",
|
service="dovecot.service",
|
||||||
@@ -547,6 +587,14 @@ def deploy_chatmail(config_path: Path) -> None:
|
|||||||
restarted=dovecot_need_restart,
|
restarted=dovecot_need_restart,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
systemd.service(
|
||||||
|
name="Start and enable Postfix",
|
||||||
|
service="postfix.service",
|
||||||
|
running=True,
|
||||||
|
enabled=True,
|
||||||
|
restarted=postfix_need_restart,
|
||||||
|
)
|
||||||
|
|
||||||
systemd.service(
|
systemd.service(
|
||||||
name="Start and enable nginx",
|
name="Start and enable nginx",
|
||||||
service="nginx.service",
|
service="nginx.service",
|
||||||
@@ -575,5 +623,10 @@ def deploy_chatmail(config_path: Path) -> None:
|
|||||||
service="systemd-journald.service",
|
service="systemd-journald.service",
|
||||||
running=True,
|
running=True,
|
||||||
enabled=True,
|
enabled=True,
|
||||||
restarted=journald_conf,
|
restarted=journald_conf.changed,
|
||||||
|
)
|
||||||
|
|
||||||
|
apt.packages(
|
||||||
|
name="Ensure cron is installed",
|
||||||
|
packages=["cron"],
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
import importlib.resources
|
import importlib.resources
|
||||||
|
|
||||||
from pyinfra.operations import apt, files, systemd, server
|
|
||||||
from pyinfra import host
|
from pyinfra import host
|
||||||
from pyinfra.facts.systemd import SystemdStatus
|
from pyinfra.facts.systemd import SystemdStatus
|
||||||
|
from pyinfra.operations import apt, files, server, systemd
|
||||||
|
|
||||||
|
|
||||||
def deploy_acmetool(nginx_hook=False, email="", domains=[]):
|
def deploy_acmetool(email="", domains=[]):
|
||||||
"""Deploy acmetool."""
|
"""Deploy acmetool."""
|
||||||
apt.packages(
|
apt.packages(
|
||||||
name="Install acmetool",
|
name="Install acmetool",
|
||||||
@@ -20,16 +20,13 @@ def deploy_acmetool(nginx_hook=False, email="", domains=[]):
|
|||||||
mode="644",
|
mode="644",
|
||||||
)
|
)
|
||||||
|
|
||||||
if nginx_hook:
|
files.put(
|
||||||
files.put(
|
src=importlib.resources.files(__package__).joinpath("acmetool.hook").open("rb"),
|
||||||
src=importlib.resources.files(__package__)
|
dest="/usr/lib/acme/hooks/nginx",
|
||||||
.joinpath("acmetool.hook")
|
user="root",
|
||||||
.open("rb"),
|
group="root",
|
||||||
dest="/usr/lib/acme/hooks/nginx",
|
mode="744",
|
||||||
user="root",
|
)
|
||||||
group="root",
|
|
||||||
mode="744",
|
|
||||||
)
|
|
||||||
|
|
||||||
files.template(
|
files.template(
|
||||||
src=importlib.resources.files(__package__).joinpath("response-file.yaml.j2"),
|
src=importlib.resources.files(__package__).joinpath("response-file.yaml.j2"),
|
||||||
@@ -74,5 +71,5 @@ def deploy_acmetool(nginx_hook=False, email="", domains=[]):
|
|||||||
|
|
||||||
server.shell(
|
server.shell(
|
||||||
name=f"Request certificate for: { ', '.join(domains) }",
|
name=f"Request certificate for: { ', '.join(domains) }",
|
||||||
commands=[f"acmetool want { ' '.join(domains)}"],
|
commands=[f"acmetool want --xlog.severity=debug { ' '.join(domains)}"],
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
SHELL=/bin/sh
|
SHELL=/bin/sh
|
||||||
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin
|
PATH=/bin:/sbin:/usr/bin:/usr/sbin:/usr/local/bin:/usr/local/sbin
|
||||||
MAILTO=root
|
MAILTO=root
|
||||||
20 16 * * * root /usr/bin/acmetool --batch reconcile
|
20 16 * * * root /usr/bin/acmetool --batch reconcile && systemctl reload dovecot && systemctl reload postfix && systemctl reload nginx
|
||||||
|
|||||||
@@ -3,3 +3,5 @@ set -e
|
|||||||
EVENT_NAME="$1"
|
EVENT_NAME="$1"
|
||||||
[ "$EVENT_NAME" = "live-updated" ] || exit 42
|
[ "$EVENT_NAME" = "live-updated" ] || exit 42
|
||||||
systemctl restart nginx.service
|
systemctl restart nginx.service
|
||||||
|
systemctl reload dovecot.service
|
||||||
|
systemctl reload postfix.service
|
||||||
|
|||||||
@@ -1,2 +1,2 @@
|
|||||||
"acme-enter-email": "{{ email }}"
|
"acme-enter-email": "{{ email }}"
|
||||||
"acme-agreement:https://letsencrypt.org/documents/LE-SA-v1.3-September-21-2022.pdf": true
|
"acme-agreement:https://letsencrypt.org/documents/LE-SA-v1.4-April-3-2024.pdf": true
|
||||||
|
|||||||
@@ -1,15 +0,0 @@
|
|||||||
{chatmail_domain}. A {ipv4}
|
|
||||||
{chatmail_domain}. AAAA {ipv6}
|
|
||||||
{chatmail_domain}. MX 10 {chatmail_domain}.
|
|
||||||
_submission._tcp.{chatmail_domain}. SRV 0 1 587 {chatmail_domain}.
|
|
||||||
_submissions._tcp.{chatmail_domain}. SRV 0 1 465 {chatmail_domain}.
|
|
||||||
_imap._tcp.{chatmail_domain}. SRV 0 1 143 {chatmail_domain}.
|
|
||||||
_imaps._tcp.{chatmail_domain}. SRV 0 1 993 {chatmail_domain}.
|
|
||||||
{chatmail_domain}. CAA 128 issue "letsencrypt.org;accounturi={acme_account_url}"
|
|
||||||
{chatmail_domain}. TXT "v=spf1 a:{chatmail_domain} -all"
|
|
||||||
_dmarc.{chatmail_domain}. TXT "v=DMARC1;p=reject;adkim=s;aspf=s"
|
|
||||||
_mta-sts.{chatmail_domain}. TXT "v=STSv1; id={sts_id}"
|
|
||||||
mta-sts.{chatmail_domain}. CNAME {chatmail_domain}.
|
|
||||||
www.{chatmail_domain}. CNAME {chatmail_domain}.
|
|
||||||
{dkim_entry}
|
|
||||||
_adsp._domainkey.{chatmail_domain}. TXT "dkim=discardable"
|
|
||||||
21
cmdeploy/src/cmdeploy/chatmail.zone.j2
Normal file
21
cmdeploy/src/cmdeploy/chatmail.zone.j2
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
{% if ipv4 %}
|
||||||
|
{{ chatmail_domain }}. A {{ ipv4 }}
|
||||||
|
{% endif %}
|
||||||
|
{% if ipv6 %}
|
||||||
|
{{ chatmail_domain }}. AAAA {{ ipv6 }}
|
||||||
|
{% endif %}
|
||||||
|
{{ chatmail_domain }}. MX 10 {{ chatmail_domain }}.
|
||||||
|
_submission._tcp.{{ chatmail_domain }}. SRV 0 1 587 {{ chatmail_domain }}.
|
||||||
|
_submissions._tcp.{{ chatmail_domain }}. SRV 0 1 465 {{ chatmail_domain }}.
|
||||||
|
_imap._tcp.{{ chatmail_domain }}. SRV 0 1 143 {{ chatmail_domain }}.
|
||||||
|
_imaps._tcp.{{ chatmail_domain }}. SRV 0 1 993 {{ chatmail_domain }}.
|
||||||
|
{% if acme_account_url %}
|
||||||
|
{{ chatmail_domain }}. CAA 128 issue "letsencrypt.org;accounturi={{ acme_account_url }}"
|
||||||
|
{% endif %}
|
||||||
|
{{ chatmail_domain }}. TXT "v=spf1 a:{{ chatmail_domain }} ~all"
|
||||||
|
_dmarc.{{ chatmail_domain }}. TXT "v=DMARC1;p=reject;adkim=s;aspf=s"
|
||||||
|
_mta-sts.{{ chatmail_domain }}. TXT "v=STSv1; id={{ sts_id }}"
|
||||||
|
mta-sts.{{ chatmail_domain }}. CNAME {{ chatmail_domain }}.
|
||||||
|
www.{{ chatmail_domain }}. CNAME {{ chatmail_domain }}.
|
||||||
|
{{ dkim_entry }}
|
||||||
|
_adsp._domainkey.{{ chatmail_domain }}. TXT "dkim=discardable"
|
||||||
@@ -2,20 +2,21 @@
|
|||||||
Provides the `cmdeploy` entry point function,
|
Provides the `cmdeploy` entry point function,
|
||||||
along with command line option and subcommand parsing.
|
along with command line option and subcommand parsing.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
import argparse
|
import argparse
|
||||||
import shutil
|
|
||||||
import subprocess
|
|
||||||
import importlib.resources
|
import importlib.resources
|
||||||
import importlib.util
|
import importlib.util
|
||||||
import os
|
import os
|
||||||
|
import shutil
|
||||||
|
import subprocess
|
||||||
import sys
|
import sys
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
|
|
||||||
from termcolor import colored
|
|
||||||
from chatmaild.config import read_config, write_initial_config
|
from chatmaild.config import read_config, write_initial_config
|
||||||
from cmdeploy.dns import show_dns, check_necessary_dns
|
from termcolor import colored
|
||||||
|
|
||||||
|
from . import dns, remote_funcs
|
||||||
|
from .sshexec import SSHExec
|
||||||
|
|
||||||
#
|
#
|
||||||
# cmdeploy sub commands and options
|
# cmdeploy sub commands and options
|
||||||
@@ -35,13 +36,10 @@ def init_cmd(args, out):
|
|||||||
mail_domain = args.chatmail_domain
|
mail_domain = args.chatmail_domain
|
||||||
if args.inipath.exists():
|
if args.inipath.exists():
|
||||||
print(f"Path exists, not modifying: {args.inipath}")
|
print(f"Path exists, not modifying: {args.inipath}")
|
||||||
|
return 1
|
||||||
else:
|
else:
|
||||||
write_initial_config(args.inipath, mail_domain)
|
write_initial_config(args.inipath, mail_domain, overrides={})
|
||||||
out.green(f"created config file for {mail_domain} in {args.inipath}")
|
out.green(f"created config file for {mail_domain} in {args.inipath}")
|
||||||
check_necessary_dns(
|
|
||||||
out,
|
|
||||||
mail_domain,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def run_cmd_options(parser):
|
def run_cmd_options(parser):
|
||||||
@@ -55,12 +53,10 @@ def run_cmd_options(parser):
|
|||||||
|
|
||||||
def run_cmd(args, out):
|
def run_cmd(args, out):
|
||||||
"""Deploy chatmail services on the remote server."""
|
"""Deploy chatmail services on the remote server."""
|
||||||
mail_domain = args.config.mail_domain
|
|
||||||
if not check_necessary_dns(
|
remote_data = dns.get_initial_remote_data(args, out)
|
||||||
out,
|
if not remote_data:
|
||||||
mail_domain,
|
return 1
|
||||||
):
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
env = os.environ.copy()
|
env = os.environ.copy()
|
||||||
env["CHATMAIL_INI"] = args.inipath
|
env["CHATMAIL_INI"] = args.inipath
|
||||||
@@ -68,8 +64,16 @@ def run_cmd(args, out):
|
|||||||
pyinf = "pyinfra --dry" if args.dry_run else "pyinfra"
|
pyinf = "pyinfra --dry" if args.dry_run else "pyinfra"
|
||||||
cmd = f"{pyinf} --ssh-user root {args.config.mail_domain} {deploy_path}"
|
cmd = f"{pyinf} --ssh-user root {args.config.mail_domain} {deploy_path}"
|
||||||
|
|
||||||
out.check_call(cmd, env=env)
|
retcode = out.check_call(cmd, env=env)
|
||||||
print("Deploy completed, call `cmdeploy dns` next.")
|
if retcode == 0:
|
||||||
|
out.green("Deploy completed, call `cmdeploy dns` next.")
|
||||||
|
elif not remote_data["acme_account_url"]:
|
||||||
|
out.red("Deploy completed but letsencrypt not configured")
|
||||||
|
out.red("Run 'cmdeploy run' again")
|
||||||
|
retcode = 0
|
||||||
|
else:
|
||||||
|
out.red("Deploy failed")
|
||||||
|
return retcode
|
||||||
|
|
||||||
|
|
||||||
def dns_cmd_options(parser):
|
def dns_cmd_options(parser):
|
||||||
@@ -81,15 +85,18 @@ def dns_cmd_options(parser):
|
|||||||
|
|
||||||
|
|
||||||
def dns_cmd(args, out):
|
def dns_cmd(args, out):
|
||||||
"""Generate dns zone file."""
|
"""Check DNS entries and optionally generate dns zone file."""
|
||||||
exit_code = show_dns(args, out)
|
remote_data = dns.get_initial_remote_data(args, out)
|
||||||
exit(exit_code)
|
if not remote_data:
|
||||||
|
return 1
|
||||||
|
retcode = dns.show_dns(args, out, remote_data)
|
||||||
|
return retcode
|
||||||
|
|
||||||
|
|
||||||
def status_cmd(args, out):
|
def status_cmd(args, out):
|
||||||
"""Display status for online chatmail instance."""
|
"""Display status for online chatmail instance."""
|
||||||
|
|
||||||
ssh = f"ssh root@{args.config.mail_domain}"
|
sshexec = args.get_sshexec()
|
||||||
|
|
||||||
out.green(f"chatmail domain: {args.config.mail_domain}")
|
out.green(f"chatmail domain: {args.config.mail_domain}")
|
||||||
if args.config.privacy_mail:
|
if args.config.privacy_mail:
|
||||||
@@ -97,10 +104,8 @@ def status_cmd(args, out):
|
|||||||
else:
|
else:
|
||||||
out.red("no privacy settings")
|
out.red("no privacy settings")
|
||||||
|
|
||||||
s1 = "systemctl --type=service --state=running"
|
for line in sshexec(remote_funcs.get_systemd_running):
|
||||||
for line in out.shell_output(f"{ssh} -- {s1}").split("\n"):
|
print(line)
|
||||||
if line.startswith(" "):
|
|
||||||
print(line)
|
|
||||||
|
|
||||||
|
|
||||||
def test_cmd_options(parser):
|
def test_cmd_options(parser):
|
||||||
@@ -129,7 +134,7 @@ def test_cmd(args, out):
|
|||||||
"-n4",
|
"-n4",
|
||||||
"-rs",
|
"-rs",
|
||||||
"-x",
|
"-x",
|
||||||
"-vrx",
|
"-v",
|
||||||
"--durations=5",
|
"--durations=5",
|
||||||
]
|
]
|
||||||
if args.slow:
|
if args.slow:
|
||||||
@@ -139,14 +144,6 @@ def test_cmd(args, out):
|
|||||||
|
|
||||||
|
|
||||||
def fmt_cmd_options(parser):
|
def fmt_cmd_options(parser):
|
||||||
parser.add_argument(
|
|
||||||
"--verbose",
|
|
||||||
"-v",
|
|
||||||
dest="verbose",
|
|
||||||
action="store_true",
|
|
||||||
help="provide information on invocations",
|
|
||||||
)
|
|
||||||
|
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"--check",
|
"--check",
|
||||||
"-c",
|
"-c",
|
||||||
@@ -156,27 +153,26 @@ def fmt_cmd_options(parser):
|
|||||||
|
|
||||||
|
|
||||||
def fmt_cmd(args, out):
|
def fmt_cmd(args, out):
|
||||||
"""Run formattting fixes (ruff and black) on all chatmail source code."""
|
"""Run formattting fixes on all chatmail source code."""
|
||||||
|
|
||||||
sources = [str(importlib.resources.files(x)) for x in ("chatmaild", "cmdeploy")]
|
sources = [str(importlib.resources.files(x)) for x in ("chatmaild", "cmdeploy")]
|
||||||
black_args = [shutil.which("black")]
|
format_args = [shutil.which("ruff"), "format"]
|
||||||
ruff_args = [shutil.which("ruff")]
|
check_args = [shutil.which("ruff"), "check"]
|
||||||
|
|
||||||
if args.check:
|
if args.check:
|
||||||
black_args.append("--check")
|
format_args.append("--diff")
|
||||||
else:
|
else:
|
||||||
ruff_args.append("--fix")
|
check_args.append("--fix")
|
||||||
|
|
||||||
if not args.verbose:
|
if not args.verbose:
|
||||||
black_args.append("-q")
|
check_args.append("--quiet")
|
||||||
ruff_args.append("-q")
|
format_args.append("--quiet")
|
||||||
|
|
||||||
black_args.extend(sources)
|
format_args.extend(sources)
|
||||||
ruff_args.extend(sources)
|
check_args.extend(sources)
|
||||||
|
|
||||||
out.check_call(" ".join(black_args), quiet=not args.verbose)
|
out.check_call(" ".join(format_args), quiet=not args.verbose)
|
||||||
out.check_call(" ".join(ruff_args), quiet=not args.verbose)
|
out.check_call(" ".join(check_args), quiet=not args.verbose)
|
||||||
return 0
|
|
||||||
|
|
||||||
|
|
||||||
def bench_cmd(args, out):
|
def bench_cmd(args, out):
|
||||||
@@ -212,16 +208,6 @@ class Out:
|
|||||||
color = "red" if red else ("green" if green else None)
|
color = "red" if red else ("green" if green else None)
|
||||||
print(colored(msg, color), file=file)
|
print(colored(msg, color), file=file)
|
||||||
|
|
||||||
def shell_output(self, arg, no_print=False, timeout=10):
|
|
||||||
if not no_print:
|
|
||||||
self(f"[$ {arg}]", file=sys.stderr)
|
|
||||||
output = subprocess.STDOUT
|
|
||||||
else:
|
|
||||||
output = subprocess.DEVNULL
|
|
||||||
return subprocess.check_output(
|
|
||||||
arg, shell=True, timeout=timeout, stderr=output
|
|
||||||
).decode()
|
|
||||||
|
|
||||||
def check_call(self, arg, env=None, quiet=False):
|
def check_call(self, arg, env=None, quiet=False):
|
||||||
if not quiet:
|
if not quiet:
|
||||||
self(f"[$ {arg}]", file=sys.stderr)
|
self(f"[$ {arg}]", file=sys.stderr)
|
||||||
@@ -231,7 +217,7 @@ class Out:
|
|||||||
if not quiet:
|
if not quiet:
|
||||||
cmdstring = " ".join(args)
|
cmdstring = " ".join(args)
|
||||||
self(f"[$ {cmdstring}]", file=sys.stderr)
|
self(f"[$ {cmdstring}]", file=sys.stderr)
|
||||||
proc = subprocess.run(args, env=env)
|
proc = subprocess.run(args, env=env, check=False)
|
||||||
return proc.returncode
|
return proc.returncode
|
||||||
|
|
||||||
|
|
||||||
@@ -244,6 +230,14 @@ def add_config_option(parser):
|
|||||||
type=Path,
|
type=Path,
|
||||||
help="path to the chatmail.ini file",
|
help="path to the chatmail.ini file",
|
||||||
)
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
"--verbose",
|
||||||
|
"-v",
|
||||||
|
dest="verbose",
|
||||||
|
action="store_true",
|
||||||
|
default=False,
|
||||||
|
help="provide verbose logging",
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def add_subcommand(subparsers, func):
|
def add_subcommand(subparsers, func):
|
||||||
@@ -283,11 +277,25 @@ def get_parser():
|
|||||||
|
|
||||||
|
|
||||||
def main(args=None):
|
def main(args=None):
|
||||||
"""Provide main entry point for 'xdcget' CLI invocation."""
|
"""Provide main entry point for 'cmdeploy' CLI invocation."""
|
||||||
parser = get_parser()
|
parser = get_parser()
|
||||||
args = parser.parse_args(args=args)
|
args = parser.parse_args(args=args)
|
||||||
if not hasattr(args, "func"):
|
if not hasattr(args, "func"):
|
||||||
return parser.parse_args(["-h"])
|
return parser.parse_args(["-h"])
|
||||||
|
|
||||||
|
ssh_exec_cache = []
|
||||||
|
|
||||||
|
def get_sshexec():
|
||||||
|
if not ssh_exec_cache:
|
||||||
|
print(f"[ssh] login to {args.config.mail_domain}")
|
||||||
|
ssh_exec = SSHExec(
|
||||||
|
args.config.mail_domain, remote_funcs, verbose=args.verbose
|
||||||
|
)
|
||||||
|
ssh_exec_cache.append(ssh_exec)
|
||||||
|
return ssh_exec_cache[0]
|
||||||
|
|
||||||
|
args.get_sshexec = get_sshexec
|
||||||
|
|
||||||
out = Out()
|
out = Out()
|
||||||
kwargs = {}
|
kwargs = {}
|
||||||
if args.func.__name__ not in ("init_cmd", "fmt_cmd"):
|
if args.func.__name__ not in ("init_cmd", "fmt_cmd"):
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
import os
|
|
||||||
import importlib.resources
|
import importlib.resources
|
||||||
|
import os
|
||||||
|
|
||||||
import pyinfra
|
import pyinfra
|
||||||
|
|
||||||
from cmdeploy import deploy_chatmail
|
from cmdeploy import deploy_chatmail
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,207 +1,74 @@
|
|||||||
import sys
|
|
||||||
|
|
||||||
import requests
|
|
||||||
import importlib
|
|
||||||
import subprocess
|
|
||||||
import datetime
|
import datetime
|
||||||
|
import importlib
|
||||||
|
|
||||||
|
from jinja2 import Template
|
||||||
|
|
||||||
|
from . import remote_funcs
|
||||||
|
|
||||||
|
|
||||||
class DNS:
|
def get_initial_remote_data(args, out):
|
||||||
def __init__(self, out, mail_domain):
|
sshexec = args.get_sshexec()
|
||||||
self.session = requests.Session()
|
|
||||||
self.out = out
|
|
||||||
self.ssh = f"ssh root@{mail_domain} -- "
|
|
||||||
try:
|
|
||||||
self.shell(f"unbound-control flush_zone {mail_domain}")
|
|
||||||
except subprocess.CalledProcessError:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def shell(self, cmd):
|
|
||||||
try:
|
|
||||||
return self.out.shell_output(f"{self.ssh}{cmd}", no_print=True)
|
|
||||||
except (subprocess.CalledProcessError, subprocess.TimeoutExpired) as e:
|
|
||||||
if "exit status 255" in str(e) or "timed out" in str(e):
|
|
||||||
self.out.red(f"Error: can't reach the server with: {self.ssh[:-4]}")
|
|
||||||
sys.exit(1)
|
|
||||||
else:
|
|
||||||
raise
|
|
||||||
|
|
||||||
def get_ipv4(self):
|
|
||||||
cmd = "ip a | grep 'inet ' | grep 'scope global' | grep -oE '[0-9]{1,3}.[0-9]{1,3}.[0-9]{1,3}.[0-9]{1,3}' | head -1"
|
|
||||||
return self.shell(cmd).strip()
|
|
||||||
|
|
||||||
def get_ipv6(self):
|
|
||||||
cmd = "ip a | grep inet6 | grep 'scope global' | sed -e 's#/64 scope global##' | sed -e 's#inet6##'"
|
|
||||||
return self.shell(cmd).strip()
|
|
||||||
|
|
||||||
def get(self, typ: str, domain: str) -> str | None:
|
|
||||||
"""Get a DNS entry"""
|
|
||||||
dig_result = self.shell(f"dig -r -q {domain} -t {typ} +short")
|
|
||||||
line = dig_result.partition("\n")[0]
|
|
||||||
if line:
|
|
||||||
return line
|
|
||||||
|
|
||||||
def check_ptr_record(self, ip: str, mail_domain) -> bool:
|
|
||||||
"""Check the PTR record for an IPv4 or IPv6 address."""
|
|
||||||
result = self.shell(f"dig -r -x {ip} +short").rstrip()
|
|
||||||
return result == f"{mail_domain}."
|
|
||||||
|
|
||||||
|
|
||||||
def show_dns(args, out) -> int:
|
|
||||||
"""Check existing DNS records, optionally write them to zone file, return exit code 0 or 1."""
|
|
||||||
template = importlib.resources.files(__package__).joinpath("chatmail.zone.f")
|
|
||||||
mail_domain = args.config.mail_domain
|
mail_domain = args.config.mail_domain
|
||||||
ssh = f"ssh root@{mail_domain}"
|
remote_data = sshexec.logged(
|
||||||
dns = DNS(out, mail_domain)
|
call=remote_funcs.perform_initial_checks, kwargs=dict(mail_domain=mail_domain)
|
||||||
|
)
|
||||||
|
|
||||||
def read_dkim_entries(entry):
|
if not remote_data["A"] and not remote_data["AAAA"]:
|
||||||
lines = []
|
out.red("Missing A and/or AAAA DNS records for {mail_domain}!")
|
||||||
for line in entry.split("\n"):
|
elif not remote_data["MTA_STS"]:
|
||||||
if line.startswith(";") or not line.strip():
|
out.red("Missing MTA_STS record:")
|
||||||
continue
|
out(f"{mail_domain}. CNAME {mail_domain}")
|
||||||
line = line.replace("\t", " ")
|
else:
|
||||||
lines.append(line)
|
return remote_data
|
||||||
return "\n".join(lines)
|
|
||||||
|
|
||||||
print("Checking your DKIM keys and DNS entries...")
|
|
||||||
try:
|
def show_dns(args, out, remote_data) -> int:
|
||||||
acme_account_url = out.shell_output(f"{ssh} -- acmetool account-url")
|
"""Check existing DNS records, optionally write them to zone file
|
||||||
except subprocess.CalledProcessError:
|
and return (exitcode, remote_data) tuple."""
|
||||||
print("Please run `cmdeploy run` first.")
|
|
||||||
|
sshexec = args.get_sshexec()
|
||||||
|
|
||||||
|
if not remote_data["acme_account_url"]:
|
||||||
|
out.red("could not get letsencrypt account url, please run 'cmdeploy run'")
|
||||||
return 1
|
return 1
|
||||||
dkim_entry = read_dkim_entries(out.shell_output(f"{ssh} -- opendkim-genzone -F"))
|
|
||||||
|
|
||||||
ipv6 = dns.get_ipv6()
|
if not remote_data["dkim_entry"]:
|
||||||
reverse_ipv6 = dns.check_ptr_record(ipv6, mail_domain)
|
out.red("could not determine dkim_entry, please run 'cmdeploy run'")
|
||||||
ipv4 = dns.get_ipv4()
|
return 1
|
||||||
reverse_ipv4 = dns.check_ptr_record(ipv4, mail_domain)
|
|
||||||
to_print = []
|
|
||||||
|
|
||||||
with open(template, "r") as f:
|
sts_id = remote_data.get("sts_id")
|
||||||
zonefile = (
|
if not sts_id:
|
||||||
f.read()
|
sts_id = datetime.datetime.now().strftime("%Y%m%d%H%M")
|
||||||
.format(
|
|
||||||
acme_account_url=acme_account_url,
|
|
||||||
sts_id=datetime.datetime.now().strftime("%Y%m%d%H%M"),
|
|
||||||
chatmail_domain=args.config.mail_domain,
|
|
||||||
dkim_entry=dkim_entry,
|
|
||||||
ipv6=ipv6,
|
|
||||||
ipv4=ipv4,
|
|
||||||
)
|
|
||||||
.strip()
|
|
||||||
)
|
|
||||||
try:
|
|
||||||
with open(args.zonefile, "w+") as zf:
|
|
||||||
zf.write(zonefile)
|
|
||||||
print(f"DNS records successfully written to: {args.zonefile}")
|
|
||||||
return 0
|
|
||||||
except TypeError:
|
|
||||||
pass
|
|
||||||
started_dkim_parsing = False
|
|
||||||
for line in zonefile.splitlines():
|
|
||||||
line = line.format(
|
|
||||||
acme_account_url=acme_account_url,
|
|
||||||
sts_id=datetime.datetime.now().strftime("%Y%m%d%H%M"),
|
|
||||||
chatmail_domain=args.config.mail_domain,
|
|
||||||
dkim_entry=dkim_entry,
|
|
||||||
ipv6=ipv6,
|
|
||||||
).strip()
|
|
||||||
for typ in ["A", "AAAA", "CNAME", "CAA"]:
|
|
||||||
if f" {typ} " in line:
|
|
||||||
domain, value = line.split(f" {typ} ")
|
|
||||||
current = dns.get(typ, domain.strip()[:-1])
|
|
||||||
if current != value.strip():
|
|
||||||
to_print.append(line)
|
|
||||||
if " MX " in line:
|
|
||||||
domain, typ, prio, value = line.split()
|
|
||||||
current = dns.get(typ, domain[:-1])
|
|
||||||
if not current:
|
|
||||||
to_print.append(line)
|
|
||||||
elif current.split()[1] != value:
|
|
||||||
print(line.replace(prio, str(int(current[0]) + 1)))
|
|
||||||
if " SRV " in line:
|
|
||||||
domain, typ, prio, weight, port, value = line.split()
|
|
||||||
current = dns.get("SRV", domain[:-1])
|
|
||||||
if current != f"{prio} {weight} {port} {value}":
|
|
||||||
to_print.append(line)
|
|
||||||
if " TXT " in line:
|
|
||||||
domain, value = line.split(" TXT ")
|
|
||||||
current = dns.get("TXT", domain.strip()[:-1])
|
|
||||||
if domain.startswith("_mta-sts."):
|
|
||||||
if current:
|
|
||||||
if current.split("id=")[0] == value.split("id=")[0]:
|
|
||||||
continue
|
|
||||||
if current != value:
|
|
||||||
to_print.append(line)
|
|
||||||
if "IN TXT ( " in line:
|
|
||||||
started_dkim_parsing = True
|
|
||||||
dkim_lines = [line]
|
|
||||||
if started_dkim_parsing and line.startswith('"'):
|
|
||||||
dkim_lines.append(" " + line)
|
|
||||||
domain, data = "\n".join(dkim_lines).split(" IN TXT ")
|
|
||||||
current = dns.get("TXT", domain.strip()[:-1])
|
|
||||||
if current:
|
|
||||||
current = "( %s )" % (current.replace('" "', '"\n "'))
|
|
||||||
if current.replace(";", "\\;") != data:
|
|
||||||
to_print.append(dkim_entry)
|
|
||||||
else:
|
|
||||||
to_print.append(dkim_entry)
|
|
||||||
|
|
||||||
exit_code = 0
|
template = importlib.resources.files(__package__).joinpath("chatmail.zone.j2")
|
||||||
if to_print:
|
content = template.read_text()
|
||||||
to_print.insert(
|
zonefile = Template(content).render(
|
||||||
0, "You should configure the following DNS entries at your provider:\n"
|
acme_account_url=remote_data.get("acme_account_url"),
|
||||||
)
|
dkim_entry=remote_data["dkim_entry"],
|
||||||
to_print.append(
|
ipv4=remote_data["A"],
|
||||||
"\nIf you already configured the DNS entries, wait a bit until the DNS entries propagate to the Internet."
|
ipv6=remote_data["AAAA"],
|
||||||
)
|
sts_id=sts_id,
|
||||||
print("\n".join(to_print))
|
chatmail_domain=args.config.mail_domain,
|
||||||
exit_code = 1
|
)
|
||||||
|
lines = [x.strip() for x in zonefile.split("\n") if x.strip()]
|
||||||
|
lines.append("")
|
||||||
|
zonefile = "\n".join(lines)
|
||||||
|
|
||||||
|
diff_records = sshexec.logged(
|
||||||
|
remote_funcs.check_zonefile, kwargs=dict(zonefile=zonefile)
|
||||||
|
)
|
||||||
|
|
||||||
|
if getattr(args, "zonefile", None):
|
||||||
|
with open(args.zonefile, "w+") as zf:
|
||||||
|
zf.write(zonefile)
|
||||||
|
out.green(f"DNS records successfully written to: {args.zonefile}")
|
||||||
|
return -1
|
||||||
|
|
||||||
|
if diff_records:
|
||||||
|
out.red("Please set the following DNS entries at your DNS provider:\n")
|
||||||
|
for line in diff_records:
|
||||||
|
out(line)
|
||||||
|
return 1
|
||||||
else:
|
else:
|
||||||
out.green("Great! All your DNS entries are correct.")
|
out.green("Great! All your DNS entries are verified and correct.")
|
||||||
|
return 0
|
||||||
to_print = []
|
|
||||||
if not reverse_ipv4:
|
|
||||||
to_print.append(f"\tIPv4:\t{ipv4}\t{args.config.mail_domain}")
|
|
||||||
if not reverse_ipv6:
|
|
||||||
to_print.append(f"\tIPv6:\t{ipv6}\t{args.config.mail_domain}")
|
|
||||||
if len(to_print) > 0:
|
|
||||||
if len(to_print) == 1:
|
|
||||||
warning = "You should add the following PTR/reverse DNS entry:"
|
|
||||||
else:
|
|
||||||
warning = "You should add the following PTR/reverse DNS entries:"
|
|
||||||
out.red(warning)
|
|
||||||
for entry in to_print:
|
|
||||||
print(entry)
|
|
||||||
print(
|
|
||||||
"You can do so at your hosting provider (maybe this isn't your DNS provider)."
|
|
||||||
)
|
|
||||||
exit_code = 1
|
|
||||||
return exit_code
|
|
||||||
|
|
||||||
|
|
||||||
def check_necessary_dns(out, mail_domain):
|
|
||||||
"""Check whether $mail_domain and mta-sts.$mail_domain resolve."""
|
|
||||||
dns = DNS(out, mail_domain)
|
|
||||||
ipv4 = dns.get("A", mail_domain)
|
|
||||||
ipv6 = dns.get("AAAA", mail_domain)
|
|
||||||
mta_entry = dns.get("CNAME", "mta-sts." + mail_domain)
|
|
||||||
www_entry = dns.get("CNAME", "www." + mail_domain)
|
|
||||||
to_print = []
|
|
||||||
if not (ipv4 or ipv6):
|
|
||||||
to_print.append(f"\t{mail_domain}.\t\t\tA<your server's IPv4 address>")
|
|
||||||
if mta_entry != mail_domain + ".":
|
|
||||||
to_print.append(f"\tmta-sts.{mail_domain}.\tCNAME\t{mail_domain}.")
|
|
||||||
if www_entry != mail_domain + ".":
|
|
||||||
to_print.append(f"\twww.{mail_domain}.\tCNAME\t{mail_domain}.")
|
|
||||||
if to_print:
|
|
||||||
to_print.insert(
|
|
||||||
0,
|
|
||||||
"\nFor chatmail to work, you need to configure this at your DNS provider:\n",
|
|
||||||
)
|
|
||||||
for line in to_print:
|
|
||||||
print(line)
|
|
||||||
print()
|
|
||||||
else:
|
|
||||||
dns.out.green("\nAll necessary DNS entries seem to be set.")
|
|
||||||
return True
|
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
uri = proxy:/run/dovecot/doveauth.socket:auth
|
uri = proxy:/run/doveauth/doveauth.socket:auth
|
||||||
iterate_disable = yes
|
iterate_disable = no
|
||||||
|
iterate_prefix = userdb/
|
||||||
|
|
||||||
default_pass_scheme = plain
|
default_pass_scheme = plain
|
||||||
# %E escapes characters " (double quote), ' (single quote) and \ (backslash) with \ (backslash).
|
# %E escapes characters " (double quote), ' (single quote) and \ (backslash) with \ (backslash).
|
||||||
# See <https://doc.dovecot.org/configuration_manual/config_file/config_variables/#modifiers>
|
# See <https://doc.dovecot.org/configuration_manual/config_file/config_variables/#modifiers>
|
||||||
|
|||||||
@@ -13,15 +13,44 @@ auth_cache_size = 100M
|
|||||||
mail_debug = yes
|
mail_debug = yes
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
|
# Prevent warnings similar to:
|
||||||
|
# config: Warning: service auth { client_limit=1000 } is lower than required under max. load (10200). Counted for protocol services with service_count != 1: service lmtp { process_limit=100 } + service imap-urlauth-login { process_limit=100 } + service imap-login { process_limit=10000 }
|
||||||
|
# config: Warning: service anvil { client_limit=1000 } is lower than required under max. load (10103). Counted with: service imap-urlauth-login { process_limit=100 } + service imap-login { process_limit=10000 } + service auth { process_limit=1 }
|
||||||
|
# master: Warning: service(stats): client_limit (1000) reached, client connections are being dropped
|
||||||
|
default_client_limit = 20000
|
||||||
|
|
||||||
|
# Increase number of logged in IMAP connections.
|
||||||
|
# Each connection is handled by a separate `imap` process.
|
||||||
|
# `imap` process should have `client_limit=1` as described in
|
||||||
|
# <https://doc.dovecot.org/configuration_manual/service_configuration/#service-limits>
|
||||||
|
# so each logged in IMAP session will need its own `imap` process.
|
||||||
|
#
|
||||||
|
# If this limit is reached,
|
||||||
|
# users will fail to LOGIN as `imap-login` process
|
||||||
|
# will accept them logging in but fail to transfer logged in
|
||||||
|
# connection to `imap` process until someone logs out and
|
||||||
|
# the following warning will be logged:
|
||||||
|
# Warning: service(imap): process_limit (1024) reached, client connections are being dropped
|
||||||
|
service imap {
|
||||||
|
process_limit = 50000
|
||||||
|
}
|
||||||
|
|
||||||
mail_server_admin = mailto:root@{{ config.mail_domain }}
|
mail_server_admin = mailto:root@{{ config.mail_domain }}
|
||||||
mail_server_comment = Chatmail server
|
mail_server_comment = Chatmail server
|
||||||
|
|
||||||
mail_plugins = quota
|
# `zlib` enables compressing messages stored in the maildir.
|
||||||
|
# See
|
||||||
|
# <https://doc.dovecot.org/configuration_manual/zlib_plugin/>
|
||||||
|
# for documentation.
|
||||||
|
#
|
||||||
|
# quota plugin documentation:
|
||||||
|
# <https://doc.dovecot.org/configuration_manual/quota_plugin/>
|
||||||
|
mail_plugins = zlib quota
|
||||||
|
|
||||||
# these are the capabilities Delta Chat cares about actually
|
# these are the capabilities Delta Chat cares about actually
|
||||||
# so let's keep the network overhead per login small
|
# so let's keep the network overhead per login small
|
||||||
# https://github.com/deltachat/deltachat-core-rust/blob/master/src/imap/capabilities.rs
|
# https://github.com/deltachat/deltachat-core-rust/blob/master/src/imap/capabilities.rs
|
||||||
imap_capability = IMAP4rev1 IDLE MOVE QUOTA CONDSTORE NOTIFY METADATA
|
imap_capability = IMAP4rev1 IDLE MOVE QUOTA CONDSTORE NOTIFY METADATA XDELTAPUSH XCHATMAIL
|
||||||
|
|
||||||
|
|
||||||
# Authentication for system users.
|
# Authentication for system users.
|
||||||
@@ -38,7 +67,7 @@ userdb {
|
|||||||
##
|
##
|
||||||
|
|
||||||
# Mailboxes are stored in the "mail" directory of the vmail user home.
|
# Mailboxes are stored in the "mail" directory of the vmail user home.
|
||||||
mail_location = maildir:/home/vmail/mail/%d/%u
|
mail_location = maildir:{{ config.mailboxes_dir }}/%u
|
||||||
|
|
||||||
namespace inbox {
|
namespace inbox {
|
||||||
inbox = yes
|
inbox = yes
|
||||||
@@ -71,7 +100,10 @@ mail_privileged_group = vmail
|
|||||||
## Mail processes
|
## Mail processes
|
||||||
##
|
##
|
||||||
|
|
||||||
# Enable IMAP COMPRESS (RFC 4978).
|
# Pass all IMAP METADATA requests to the server implementing Dovecot's dict protocol.
|
||||||
|
mail_attribute_dict = proxy:/run/chatmail-metadata/metadata.socket:metadata
|
||||||
|
|
||||||
|
# `imap_zlib` enables 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 imap_quota
|
||||||
@@ -79,7 +111,19 @@ protocol imap {
|
|||||||
}
|
}
|
||||||
|
|
||||||
protocol lmtp {
|
protocol lmtp {
|
||||||
mail_plugins = $mail_plugins quota
|
# notify plugin is a dependency of push_notification plugin:
|
||||||
|
# <https://doc.dovecot.org/settings/plugin/notify-plugin/>
|
||||||
|
#
|
||||||
|
# push_notification plugin documentation:
|
||||||
|
# <https://doc.dovecot.org/configuration_manual/push_notification/>
|
||||||
|
#
|
||||||
|
# mail_lua and push_notification_lua are needed for Lua push notification handler.
|
||||||
|
# <https://doc.dovecot.org/configuration_manual/push_notification/#configuration>
|
||||||
|
mail_plugins = $mail_plugins mail_lua notify push_notification push_notification_lua
|
||||||
|
}
|
||||||
|
|
||||||
|
plugin {
|
||||||
|
zlib_save = gz
|
||||||
}
|
}
|
||||||
|
|
||||||
plugin {
|
plugin {
|
||||||
@@ -95,7 +139,11 @@ plugin {
|
|||||||
# quota_over_flag_value = TRUE
|
# quota_over_flag_value = TRUE
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# push_notification configuration
|
||||||
|
plugin {
|
||||||
|
# <https://doc.dovecot.org/configuration_manual/push_notification/#lua-lua>
|
||||||
|
push_notification_driver = lua:file=/etc/dovecot/push_notification.lua
|
||||||
|
}
|
||||||
|
|
||||||
service lmtp {
|
service lmtp {
|
||||||
user=vmail
|
user=vmail
|
||||||
|
|||||||
@@ -1,10 +1,12 @@
|
|||||||
# delete all mails after {{ config.delete_mails_after }} days, in the Inbox
|
# delete all mails after {{ config.delete_mails_after }} days, in the Inbox
|
||||||
2 0 * * * dovecot find /home/vmail/mail/{{ config.mail_domain }} -path '*/cur/*' -mtime +{{ config.delete_mails_after }} -type f -delete
|
2 0 * * * vmail find {{ config.mailboxes_dir }} -path '*/cur/*' -mtime +{{ config.delete_mails_after }} -type f -delete
|
||||||
# or in any IMAP subfolder
|
# or in any IMAP subfolder
|
||||||
2 0 * * * dovecot find /home/vmail/mail/{{ config.mail_domain }} -path '*/.*/cur/*' -mtime +{{ config.delete_mails_after }} -type f -delete
|
2 0 * * * vmail find {{ config.mailboxes_dir }} -path '*/.*/cur/*' -mtime +{{ config.delete_mails_after }} -type f -delete
|
||||||
# even if they are unseen
|
# even if they are unseen
|
||||||
2 0 * * * dovecot find /home/vmail/mail/{{ config.mail_domain }} -path '*/new/*' -mtime +{{ config.delete_mails_after }} -type f -delete
|
2 0 * * * vmail find {{ config.mailboxes_dir }} -path '*/new/*' -mtime +{{ config.delete_mails_after }} -type f -delete
|
||||||
2 0 * * * dovecot find /home/vmail/mail/{{ config.mail_domain }} -path '*/.*/new/*' -mtime +{{ config.delete_mails_after }} -type f -delete
|
2 0 * * * vmail find {{ config.mailboxes_dir }} -path '*/.*/new/*' -mtime +{{ config.delete_mails_after }} -type f -delete
|
||||||
# or only temporary (but then they shouldn't be around after {{ config.delete_mails_after }} days anyway).
|
# or only temporary (but then they shouldn't be around after {{ config.delete_mails_after }} days anyway).
|
||||||
2 0 * * * dovecot find /home/vmail/mail/{{ config.mail_domain }} -path '*/tmp/*' -mtime +{{ config.delete_mails_after }} -type f -delete
|
2 0 * * * vmail find {{ config.mailboxes_dir }} -path '*/tmp/*' -mtime +{{ config.delete_mails_after }} -type f -delete
|
||||||
2 0 * * * dovecot find /home/vmail/mail/{{ config.mail_domain }} -path '*/.*/tmp/*' -mtime +{{ config.delete_mails_after }} -type f -delete
|
2 0 * * * vmail find {{ config.mailboxes_dir }} -path '*/.*/tmp/*' -mtime +{{ config.delete_mails_after }} -type f -delete
|
||||||
|
3 0 * * * vmail find {{ config.mailboxes_dir }} -name 'maildirsize' -type f -delete
|
||||||
|
4 0 * * * vmail /usr/local/lib/chatmaild/venv/bin/delete_inactive_users /usr/local/lib/chatmaild/chatmail.ini
|
||||||
|
|||||||
28
cmdeploy/src/cmdeploy/dovecot/push_notification.lua
Normal file
28
cmdeploy/src/cmdeploy/dovecot/push_notification.lua
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
function dovecot_lua_notify_begin_txn(user)
|
||||||
|
return user
|
||||||
|
end
|
||||||
|
|
||||||
|
function contains(v, needle)
|
||||||
|
for _, keyword in ipairs(v) do
|
||||||
|
if keyword == needle then
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
|
||||||
|
function dovecot_lua_notify_event_message_new(user, event)
|
||||||
|
local mbox = user:mailbox(event.mailbox)
|
||||||
|
mbox:sync()
|
||||||
|
|
||||||
|
if user.username ~= event.from_address then
|
||||||
|
-- Incoming message
|
||||||
|
-- Notify METADATA server about new message.
|
||||||
|
mbox:metadata_set("/private/messagenew", "")
|
||||||
|
end
|
||||||
|
|
||||||
|
mbox:free()
|
||||||
|
end
|
||||||
|
|
||||||
|
function dovecot_lua_notify_end_txn(ctx, success)
|
||||||
|
end
|
||||||
@@ -1,8 +1,9 @@
|
|||||||
import importlib
|
import importlib
|
||||||
import qrcode
|
|
||||||
import os
|
|
||||||
from PIL import ImageFont, ImageDraw, Image
|
|
||||||
import io
|
import io
|
||||||
|
import os
|
||||||
|
|
||||||
|
import qrcode
|
||||||
|
from PIL import Image, ImageDraw, ImageFont
|
||||||
|
|
||||||
|
|
||||||
def gen_qr_png_data(maildomain):
|
def gen_qr_png_data(maildomain):
|
||||||
|
|||||||
@@ -1,2 +1,3 @@
|
|||||||
[Journal]
|
[Journal]
|
||||||
MaxRetentionSec=3d
|
MaxRetentionSec=3d
|
||||||
|
Storage=volatile
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
*/5 * * * * root {{ config.execpath }} /home/vmail/mail/{{ config.mail_domain }} >/var/www/html/metrics
|
*/5 * * * * root {{ config.execpath }} {{ config.mailboxes_dir }} >/var/www/html/metrics
|
||||||
|
|||||||
@@ -19,6 +19,13 @@
|
|||||||
<authentication>password-cleartext</authentication>
|
<authentication>password-cleartext</authentication>
|
||||||
<username>%EMAILADDRESS%</username>
|
<username>%EMAILADDRESS%</username>
|
||||||
</incomingServer>
|
</incomingServer>
|
||||||
|
<incomingServer type="imap">
|
||||||
|
<hostname>{{ config.domain_name }}</hostname>
|
||||||
|
<port>443</port>
|
||||||
|
<socketType>SSL</socketType>
|
||||||
|
<authentication>password-cleartext</authentication>
|
||||||
|
<username>%EMAILADDRESS%</username>
|
||||||
|
</incomingServer>
|
||||||
<outgoingServer type="smtp">
|
<outgoingServer type="smtp">
|
||||||
<hostname>{{ config.domain_name }}</hostname>
|
<hostname>{{ config.domain_name }}</hostname>
|
||||||
<port>465</port>
|
<port>465</port>
|
||||||
@@ -33,5 +40,12 @@
|
|||||||
<authentication>password-cleartext</authentication>
|
<authentication>password-cleartext</authentication>
|
||||||
<username>%EMAILADDRESS%</username>
|
<username>%EMAILADDRESS%</username>
|
||||||
</outgoingServer>
|
</outgoingServer>
|
||||||
|
<outgoingServer type="smtp">
|
||||||
|
<hostname>{{ config.domain_name }}</hostname>
|
||||||
|
<port>443</port>
|
||||||
|
<socketType>SSL</socketType>
|
||||||
|
<authentication>password-cleartext</authentication>
|
||||||
|
<username>%EMAILADDRESS%</username>
|
||||||
|
</outgoingServer>
|
||||||
</emailProvider>
|
</emailProvider>
|
||||||
</clientConfig>
|
</clientConfig>
|
||||||
|
|||||||
@@ -1,13 +1,30 @@
|
|||||||
|
load_module modules/ngx_stream_module.so;
|
||||||
|
|
||||||
user www-data;
|
user www-data;
|
||||||
worker_processes auto;
|
worker_processes auto;
|
||||||
pid /run/nginx.pid;
|
pid /run/nginx.pid;
|
||||||
error_log /var/log/nginx/error.log;
|
error_log syslog:server=unix:/dev/log,facility=local3;
|
||||||
|
|
||||||
events {
|
events {
|
||||||
worker_connections 768;
|
worker_connections 768;
|
||||||
# multi_accept on;
|
# multi_accept on;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
stream {
|
||||||
|
map $ssl_preread_alpn_protocols $proxy {
|
||||||
|
default 127.0.0.1:8443;
|
||||||
|
~\bsmtp\b 127.0.0.1:submissions;
|
||||||
|
~\bimap\b 127.0.0.1:imaps;
|
||||||
|
}
|
||||||
|
|
||||||
|
server {
|
||||||
|
listen 443;
|
||||||
|
listen [::]:443;
|
||||||
|
proxy_pass $proxy;
|
||||||
|
ssl_preread on;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
http {
|
http {
|
||||||
sendfile on;
|
sendfile on;
|
||||||
tcp_nopush on;
|
tcp_nopush on;
|
||||||
@@ -26,8 +43,8 @@ http {
|
|||||||
gzip on;
|
gzip on;
|
||||||
|
|
||||||
server {
|
server {
|
||||||
listen 443 ssl default_server;
|
listen 8443 ssl default_server;
|
||||||
listen [::]:443 ssl default_server;
|
listen [::]:8443 ssl default_server;
|
||||||
|
|
||||||
root /var/www/html;
|
root /var/www/html;
|
||||||
|
|
||||||
@@ -35,6 +52,8 @@ http {
|
|||||||
|
|
||||||
server_name _;
|
server_name _;
|
||||||
|
|
||||||
|
access_log syslog:server=unix:/dev/log,facility=local7;
|
||||||
|
|
||||||
location / {
|
location / {
|
||||||
# First attempt to serve request as file, then
|
# First attempt to serve request as file, then
|
||||||
# as directory, then fall back to displaying a 404.
|
# as directory, then fall back to displaying a 404.
|
||||||
@@ -76,9 +95,10 @@ http {
|
|||||||
|
|
||||||
# Redirect www. to non-www
|
# Redirect www. to non-www
|
||||||
server {
|
server {
|
||||||
listen 443 ssl;
|
listen 8443 ssl;
|
||||||
listen [::]:443 ssl;
|
listen [::]:8443 ssl;
|
||||||
server_name www.{{ config.domain_name }};
|
server_name www.{{ config.domain_name }};
|
||||||
return 301 $scheme://{{ config.domain_name }}$request_uri;
|
return 301 $scheme://{{ config.domain_name }}$request_uri;
|
||||||
|
access_log syslog:server=unix:/dev/log,facility=local7;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
BIN
cmdeploy/src/cmdeploy/obs-home-deltachat.gpg
Normal file
BIN
cmdeploy/src/cmdeploy/obs-home-deltachat.gpg
Normal file
Binary file not shown.
@@ -25,7 +25,24 @@ KeyTable /etc/dkimkeys/KeyTable
|
|||||||
SigningTable refile:/etc/dkimkeys/SigningTable
|
SigningTable refile:/etc/dkimkeys/SigningTable
|
||||||
|
|
||||||
# Sign Autocrypt header in addition to the default specified in RFC 6376.
|
# Sign Autocrypt header in addition to the default specified in RFC 6376.
|
||||||
SignHeaders *,+autocrypt
|
#
|
||||||
|
# Default list is here:
|
||||||
|
# <https://github.com/trusteddomainproject/OpenDKIM/blob/5c539587561785a66c1f67f720f2fb741f320785/libopendkim/dkim.c#L221-L245>
|
||||||
|
SignHeaders *,+autocrypt,+content-type
|
||||||
|
|
||||||
|
# Prevent addition of second Content-Type header
|
||||||
|
# and other important headers that should not be added
|
||||||
|
# after signing the message.
|
||||||
|
# See
|
||||||
|
# <https://www.zone.eu/blog/2024/05/17/bimi-and-dmarc-cant-save-you/>
|
||||||
|
# and RFC 6376 (page 41) for reference.
|
||||||
|
#
|
||||||
|
# We don't use "l=" body length so the problem described in RFC 6376
|
||||||
|
# is not applicable, but adding e.g. a second "From" header
|
||||||
|
# or second "Autocrypt" header is better prevented in any case.
|
||||||
|
#
|
||||||
|
# Default is empty.
|
||||||
|
OversignHeaders from,reply-to,subject,date,to,cc,resent-date,resent-from,resent-sender,resent-to,resent-cc,in-reply-to,references,list-id,list-help,list-unsubscribe,list-subscribe,list-post,list-owner,list-archive,autocrypt
|
||||||
|
|
||||||
# Script to ignore signatures that do not correspond to the From: domain.
|
# Script to ignore signatures that do not correspond to the From: domain.
|
||||||
ScreenPolicyScript /etc/opendkim/screen.lua
|
ScreenPolicyScript /etc/opendkim/screen.lua
|
||||||
|
|||||||
@@ -77,3 +77,7 @@ mua_helo_restrictions = permit_mynetworks, reject_invalid_helo_hostname, reject_
|
|||||||
|
|
||||||
# 1:1 map MAIL FROM to SASL login name.
|
# 1:1 map MAIL FROM to SASL login name.
|
||||||
smtpd_sender_login_maps = regexp:/etc/postfix/login_map
|
smtpd_sender_login_maps = regexp:/etc/postfix/login_map
|
||||||
|
|
||||||
|
# Do not lookup SMTP client hostnames to reduce delays
|
||||||
|
# and avoid unnecessary DNS requests.
|
||||||
|
smtpd_peername_lookup = no
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ smtp inet n - y - - smtpd -v
|
|||||||
smtp inet n - y - - smtpd
|
smtp inet n - y - - smtpd
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
-o smtpd_milters=unix:opendkim/opendkim.sock
|
-o smtpd_milters=unix:opendkim/opendkim.sock
|
||||||
submission inet n - y - - smtpd
|
submission inet n - y - 5000 smtpd
|
||||||
-o syslog_name=postfix/submission
|
-o syslog_name=postfix/submission
|
||||||
-o smtpd_tls_security_level=encrypt
|
-o smtpd_tls_security_level=encrypt
|
||||||
-o smtpd_sasl_auth_enable=yes
|
-o smtpd_sasl_auth_enable=yes
|
||||||
@@ -32,7 +32,7 @@ submission inet n - y - - smtpd
|
|||||||
-o smtpd_client_connection_count_limit=1000
|
-o smtpd_client_connection_count_limit=1000
|
||||||
-o smtpd_proxy_filter=127.0.0.1:{{ config.filtermail_smtp_port }}
|
-o smtpd_proxy_filter=127.0.0.1:{{ config.filtermail_smtp_port }}
|
||||||
-o cleanup_service_name=authclean
|
-o cleanup_service_name=authclean
|
||||||
smtps inet n - y - - smtpd
|
smtps inet n - y - 5000 smtpd
|
||||||
-o syslog_name=postfix/smtps
|
-o syslog_name=postfix/smtps
|
||||||
-o smtpd_tls_wrappermode=yes
|
-o smtpd_tls_wrappermode=yes
|
||||||
-o smtpd_tls_security_level=encrypt
|
-o smtpd_tls_security_level=encrypt
|
||||||
|
|||||||
104
cmdeploy/src/cmdeploy/remote_funcs.py
Normal file
104
cmdeploy/src/cmdeploy/remote_funcs.py
Normal file
@@ -0,0 +1,104 @@
|
|||||||
|
"""
|
||||||
|
Pure python functions which execute remotely in a system Python interpreter.
|
||||||
|
|
||||||
|
All functions of this module
|
||||||
|
|
||||||
|
- need to get and and return Python builtin data types only,
|
||||||
|
|
||||||
|
- can only use standard library dependencies,
|
||||||
|
|
||||||
|
- can freely call each other.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import re
|
||||||
|
from subprocess import CalledProcessError, check_output
|
||||||
|
|
||||||
|
|
||||||
|
def shell(command, fail_ok=False):
|
||||||
|
print(f"$ {command}")
|
||||||
|
try:
|
||||||
|
return check_output(command, shell=True).decode().rstrip()
|
||||||
|
except CalledProcessError:
|
||||||
|
if not fail_ok:
|
||||||
|
raise
|
||||||
|
return ""
|
||||||
|
|
||||||
|
|
||||||
|
def get_systemd_running():
|
||||||
|
lines = shell("systemctl --type=service --state=running").split("\n")
|
||||||
|
return [line for line in lines if line.startswith(" ")]
|
||||||
|
|
||||||
|
|
||||||
|
def perform_initial_checks(mail_domain):
|
||||||
|
"""Collecting initial DNS zone content."""
|
||||||
|
A = query_dns("A", mail_domain)
|
||||||
|
AAAA = query_dns("AAAA", mail_domain)
|
||||||
|
MTA_STS = query_dns("CNAME", f"mta-sts.{mail_domain}")
|
||||||
|
|
||||||
|
res = dict(A=A, AAAA=AAAA, MTA_STS=MTA_STS)
|
||||||
|
if not MTA_STS or (not A and not AAAA):
|
||||||
|
return res
|
||||||
|
|
||||||
|
res["acme_account_url"] = shell("acmetool account-url", fail_ok=True)
|
||||||
|
if not shell("dig", fail_ok=True):
|
||||||
|
shell("apt-get install -y dnsutils")
|
||||||
|
shell(f"unbound-control flush_zone {mail_domain}", fail_ok=True)
|
||||||
|
res["dkim_entry"] = get_dkim_entry(mail_domain, dkim_selector="opendkim")
|
||||||
|
|
||||||
|
# parse out sts-id if exists, example: "v=STSv1; id=2090123"
|
||||||
|
parts = query_dns("TXT", f"_mta-sts.{mail_domain}").split("id=")
|
||||||
|
res["sts_id"] = parts[1].rstrip('"') if len(parts) == 2 else ""
|
||||||
|
return res
|
||||||
|
|
||||||
|
|
||||||
|
def get_dkim_entry(mail_domain, dkim_selector):
|
||||||
|
try:
|
||||||
|
dkim_pubkey = shell(
|
||||||
|
f"openssl rsa -in /etc/dkimkeys/{dkim_selector}.private "
|
||||||
|
"-pubout 2>/dev/null | awk '/-/{next}{printf(\"%s\",$0)}'"
|
||||||
|
)
|
||||||
|
except CalledProcessError:
|
||||||
|
return
|
||||||
|
dkim_value_raw = f"v=DKIM1;k=rsa;p={dkim_pubkey};s=email;t=s"
|
||||||
|
dkim_value = '" "'.join(re.findall(".{1,255}", dkim_value_raw))
|
||||||
|
return f'{dkim_selector}._domainkey.{mail_domain}. TXT "{dkim_value}"'
|
||||||
|
|
||||||
|
|
||||||
|
def query_dns(typ, domain):
|
||||||
|
res = shell(f"dig -r -q {domain} -t {typ} +short")
|
||||||
|
print(res)
|
||||||
|
if res:
|
||||||
|
return res.split("\n")[0]
|
||||||
|
|
||||||
|
|
||||||
|
def check_zonefile(zonefile):
|
||||||
|
"""Check all expected zone file entries."""
|
||||||
|
diff = []
|
||||||
|
|
||||||
|
for zf_line in zonefile.splitlines():
|
||||||
|
print("")
|
||||||
|
print(f"dns-checking {zf_line!r}")
|
||||||
|
zf_domain, zf_typ, zf_value = zf_line.split(maxsplit=2)
|
||||||
|
zf_domain = zf_domain.rstrip(".")
|
||||||
|
zf_value = zf_value.strip()
|
||||||
|
query_value = query_dns(zf_typ, zf_domain)
|
||||||
|
if zf_value != query_value:
|
||||||
|
assert zf_typ in ("A", "AAAA", "CNAME", "CAA", "SRV", "MX", "TXT"), zf_line
|
||||||
|
diff.append(zf_line)
|
||||||
|
|
||||||
|
return diff
|
||||||
|
|
||||||
|
|
||||||
|
# check if this module is executed remotely
|
||||||
|
# and setup a simple serialized function-execution loop
|
||||||
|
|
||||||
|
if __name__ == "__channelexec__":
|
||||||
|
|
||||||
|
def print(item):
|
||||||
|
channel.send(("log", item)) # noqa
|
||||||
|
|
||||||
|
while 1:
|
||||||
|
func_name, kwargs = channel.receive() # noqa
|
||||||
|
kwargs = kwargs if kwargs else {}
|
||||||
|
res = globals()[func_name](**kwargs) # noqa
|
||||||
|
channel.send(("finish", res)) # noqa
|
||||||
12
cmdeploy/src/cmdeploy/service/chatmail-metadata.service.f
Normal file
12
cmdeploy/src/cmdeploy/service/chatmail-metadata.service.f
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
[Unit]
|
||||||
|
Description=Chatmail dict proxy for IMAP METADATA
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
ExecStart={execpath} /run/chatmail-metadata/metadata.socket {config_path}
|
||||||
|
Restart=always
|
||||||
|
RestartSec=30
|
||||||
|
User=vmail
|
||||||
|
RuntimeDirectory=chatmail-metadata
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
@@ -2,9 +2,11 @@
|
|||||||
Description=Chatmail dict authentication proxy for dovecot
|
Description=Chatmail dict authentication proxy for dovecot
|
||||||
|
|
||||||
[Service]
|
[Service]
|
||||||
ExecStart={execpath} /run/dovecot/doveauth.socket vmail /home/vmail/passdb.sqlite {config_path}
|
ExecStart={execpath} /run/doveauth/doveauth.socket {config_path}
|
||||||
Restart=always
|
Restart=always
|
||||||
RestartSec=30
|
RestartSec=30
|
||||||
|
User=vmail
|
||||||
|
RuntimeDirectory=doveauth
|
||||||
|
|
||||||
[Install]
|
[Install]
|
||||||
WantedBy=multi-user.target
|
WantedBy=multi-user.target
|
||||||
67
cmdeploy/src/cmdeploy/service/echobot.service.f
Normal file
67
cmdeploy/src/cmdeploy/service/echobot.service.f
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
[Unit]
|
||||||
|
Description=Chatmail echo bot for testing it works
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
ExecStart={execpath} {config_path}
|
||||||
|
Environment="PATH={remote_venv_dir}:$PATH"
|
||||||
|
Restart=always
|
||||||
|
RestartSec=30
|
||||||
|
|
||||||
|
User=echobot
|
||||||
|
Group=echobot
|
||||||
|
|
||||||
|
# Create /var/lib/echobot
|
||||||
|
StateDirectory=echobot
|
||||||
|
|
||||||
|
# Create /run/echobot
|
||||||
|
#
|
||||||
|
# echobot stores /run/echobot/password
|
||||||
|
# with a password there, which doveauth then reads.
|
||||||
|
RuntimeDirectory=echobot
|
||||||
|
|
||||||
|
WorkingDirectory=/var/lib/echobot
|
||||||
|
|
||||||
|
# Apply security restrictions suggested by
|
||||||
|
# systemd-analyze security echobot.service
|
||||||
|
CapabilityBoundingSet=
|
||||||
|
LockPersonality=true
|
||||||
|
MemoryDenyWriteExecute=true
|
||||||
|
NoNewPrivileges=true
|
||||||
|
PrivateDevices=true
|
||||||
|
PrivateMounts=true
|
||||||
|
PrivateTmp=true
|
||||||
|
|
||||||
|
# We need to know about doveauth user to give it access to /run/echobot/password
|
||||||
|
PrivateUsers=false
|
||||||
|
|
||||||
|
ProtectClock=true
|
||||||
|
ProtectControlGroups=true
|
||||||
|
ProtectHostname=true
|
||||||
|
ProtectKernelLogs=true
|
||||||
|
ProtectKernelModules=true
|
||||||
|
ProtectKernelTunables=true
|
||||||
|
ProtectProc=noaccess
|
||||||
|
|
||||||
|
# Should be "strict", but we currently write /accounts folder in a protected path
|
||||||
|
ProtectSystem=full
|
||||||
|
|
||||||
|
RemoveIPC=true
|
||||||
|
RestrictAddressFamilies=AF_INET AF_INET6
|
||||||
|
RestrictNamespaces=true
|
||||||
|
RestrictRealtime=true
|
||||||
|
RestrictSUIDSGID=true
|
||||||
|
SystemCallArchitectures=native
|
||||||
|
SystemCallFilter=~@clock
|
||||||
|
SystemCallFilter=~@cpu-emulation
|
||||||
|
SystemCallFilter=~@debug
|
||||||
|
SystemCallFilter=~@module
|
||||||
|
SystemCallFilter=~@mount
|
||||||
|
SystemCallFilter=~@obsolete
|
||||||
|
SystemCallFilter=~@raw-io
|
||||||
|
SystemCallFilter=~@reboot
|
||||||
|
SystemCallFilter=~@resources
|
||||||
|
SystemCallFilter=~@swap
|
||||||
|
UMask=0077
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
WantedBy=multi-user.target
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
[Unit]
|
[Unit]
|
||||||
Description=Chatmail Postfix BeforeQeue filter
|
Description=Chatmail Postfix before queue filter
|
||||||
|
|
||||||
[Service]
|
[Service]
|
||||||
ExecStart={execpath} {config_path}
|
ExecStart={execpath} {config_path}
|
||||||
39
cmdeploy/src/cmdeploy/sshexec.py
Normal file
39
cmdeploy/src/cmdeploy/sshexec.py
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
import sys
|
||||||
|
|
||||||
|
import execnet
|
||||||
|
|
||||||
|
|
||||||
|
class SSHExec:
|
||||||
|
RemoteError = execnet.RemoteError
|
||||||
|
|
||||||
|
def __init__(self, host, remote_funcs, verbose=False, python="python3", timeout=60):
|
||||||
|
self.gateway = execnet.makegateway(f"ssh=root@{host}//python={python}")
|
||||||
|
self._remote_cmdloop_channel = self.gateway.remote_exec(remote_funcs)
|
||||||
|
self.timeout = timeout
|
||||||
|
self.verbose = verbose
|
||||||
|
|
||||||
|
def __call__(self, call, kwargs=None, log_callback=None):
|
||||||
|
self._remote_cmdloop_channel.send((call.__name__, kwargs))
|
||||||
|
while 1:
|
||||||
|
code, data = self._remote_cmdloop_channel.receive(timeout=self.timeout)
|
||||||
|
if log_callback is not None and code == "log":
|
||||||
|
log_callback(data)
|
||||||
|
elif code == "finish":
|
||||||
|
return data
|
||||||
|
|
||||||
|
def logged(self, call, kwargs):
|
||||||
|
def log_progress(data):
|
||||||
|
sys.stdout.write(".")
|
||||||
|
sys.stdout.flush()
|
||||||
|
|
||||||
|
title = call.__doc__
|
||||||
|
if not title:
|
||||||
|
title = call.__name__
|
||||||
|
if self.verbose:
|
||||||
|
print("[ssh] " + title)
|
||||||
|
return self(call, kwargs, log_callback=print)
|
||||||
|
else:
|
||||||
|
print(title, end="")
|
||||||
|
res = self(call, kwargs, log_callback=log_progress)
|
||||||
|
print()
|
||||||
|
return res
|
||||||
@@ -1,8 +1,10 @@
|
|||||||
import pytest
|
|
||||||
import threading
|
|
||||||
import queue
|
import queue
|
||||||
|
import socket
|
||||||
|
import threading
|
||||||
|
|
||||||
|
import pytest
|
||||||
from chatmaild.config import read_config
|
from chatmaild.config import read_config
|
||||||
|
|
||||||
from cmdeploy.cmdeploy import main
|
from cmdeploy.cmdeploy import main
|
||||||
|
|
||||||
|
|
||||||
@@ -13,6 +15,13 @@ def test_init(tmp_path, maildomain):
|
|||||||
assert config.mail_domain == maildomain
|
assert config.mail_domain == maildomain
|
||||||
|
|
||||||
|
|
||||||
|
def test_capabilities(imap):
|
||||||
|
imap.connect()
|
||||||
|
capas = imap.conn.capabilities
|
||||||
|
assert "XCHATMAIL" in capas
|
||||||
|
assert "XDELTAPUSH" in capas
|
||||||
|
|
||||||
|
|
||||||
def test_login_basic_functioning(imap_or_smtp, gencreds, lp):
|
def test_login_basic_functioning(imap_or_smtp, gencreds, lp):
|
||||||
"""Test a) that an initial login creates a user automatically
|
"""Test a) that an initial login creates a user automatically
|
||||||
and b) verify we can also login a second time with the same password
|
and b) verify we can also login a second time with the same password
|
||||||
@@ -78,3 +87,24 @@ def test_concurrent_logins_same_account(
|
|||||||
|
|
||||||
for _ in conns:
|
for _ in conns:
|
||||||
assert login_results.get()
|
assert login_results.get()
|
||||||
|
|
||||||
|
|
||||||
|
def test_no_vrfy(chatmail_config):
|
||||||
|
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
||||||
|
sock.connect((chatmail_config.mail_domain, 25))
|
||||||
|
banner = sock.recv(1024)
|
||||||
|
print(banner)
|
||||||
|
sock.send(b"VRFY wrongaddress@%s\r\n" % (chatmail_config.mail_domain.encode(),))
|
||||||
|
result = sock.recv(1024)
|
||||||
|
print(result)
|
||||||
|
sock.send(b"VRFY echo@%s\r\n" % (chatmail_config.mail_domain.encode(),))
|
||||||
|
result2 = sock.recv(1024)
|
||||||
|
print(result2)
|
||||||
|
assert result[0:10] == result2[0:10]
|
||||||
|
sock.send(b"VRFY wrongaddress\r\n")
|
||||||
|
result = sock.recv(1024)
|
||||||
|
print(result)
|
||||||
|
sock.send(b"VRFY echo\r\n")
|
||||||
|
result2 = sock.recv(1024)
|
||||||
|
print(result2)
|
||||||
|
assert result[0:10] == result2[0:10] == b"252 2.0.0 "
|
||||||
|
|||||||
@@ -1,6 +1,45 @@
|
|||||||
import smtplib
|
import smtplib
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
|
from cmdeploy import remote_funcs
|
||||||
|
from cmdeploy.sshexec import SSHExec
|
||||||
|
|
||||||
|
|
||||||
|
class TestSSHExecutor:
|
||||||
|
@pytest.fixture(scope="class")
|
||||||
|
def sshexec(self, sshdomain):
|
||||||
|
return SSHExec(sshdomain, remote_funcs)
|
||||||
|
|
||||||
|
def test_ls(self, sshexec):
|
||||||
|
out = sshexec(call=remote_funcs.shell, kwargs=dict(command="ls"))
|
||||||
|
out2 = sshexec(call=remote_funcs.shell, kwargs=dict(command="ls"))
|
||||||
|
assert out == out2
|
||||||
|
|
||||||
|
def test_perform_initial(self, sshexec, maildomain):
|
||||||
|
res = sshexec(
|
||||||
|
remote_funcs.perform_initial_checks, kwargs=dict(mail_domain=maildomain)
|
||||||
|
)
|
||||||
|
assert res["A"] or res["AAAA"]
|
||||||
|
|
||||||
|
def test_logged(self, sshexec, maildomain, capsys):
|
||||||
|
sshexec.logged(
|
||||||
|
remote_funcs.perform_initial_checks, kwargs=dict(mail_domain=maildomain)
|
||||||
|
)
|
||||||
|
out, err = capsys.readouterr()
|
||||||
|
assert out.startswith("Collecting")
|
||||||
|
assert out.endswith("....\n")
|
||||||
|
assert out.count("\n") == 1
|
||||||
|
|
||||||
|
sshexec.verbose = True
|
||||||
|
sshexec.logged(
|
||||||
|
remote_funcs.perform_initial_checks, kwargs=dict(mail_domain=maildomain)
|
||||||
|
)
|
||||||
|
out, err = capsys.readouterr()
|
||||||
|
lines = out.split("\n")
|
||||||
|
assert len(lines) > 4
|
||||||
|
assert remote_funcs.perform_initial_checks.__doc__ in lines[0]
|
||||||
|
|
||||||
|
|
||||||
def test_remote(remote, imap_or_smtp):
|
def test_remote(remote, imap_or_smtp):
|
||||||
lineproducer = remote.iter_output(imap_or_smtp.logcmd)
|
lineproducer = remote.iter_output(imap_or_smtp.logcmd)
|
||||||
@@ -83,3 +122,19 @@ def test_exceed_rate_limit(cmsetup, gencreds, maildata, chatmail_config):
|
|||||||
assert b"4.7.1: Too much mail from" in outcome[1]
|
assert b"4.7.1: Too much mail from" in outcome[1]
|
||||||
return
|
return
|
||||||
pytest.fail("Rate limit was not exceeded")
|
pytest.fail("Rate limit was not exceeded")
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.slow
|
||||||
|
def test_expunged(remote, chatmail_config):
|
||||||
|
outdated_days = int(chatmail_config.delete_mails_after) + 1
|
||||||
|
find_cmds = [
|
||||||
|
f"find {chatmail_config.mailboxes_dir} -path '*/cur/*' -mtime +{outdated_days} -type f",
|
||||||
|
f"find {chatmail_config.mailboxes_dir} -path '*/.*/cur/*' -mtime +{outdated_days} -type f",
|
||||||
|
f"find {chatmail_config.mailboxes_dir} -path '*/new/*' -mtime +{outdated_days} -type f",
|
||||||
|
f"find {chatmail_config.mailboxes_dir} -path '*/.*/new/*' -mtime +{outdated_days} -type f",
|
||||||
|
f"find {chatmail_config.mailboxes_dir} -path '*/tmp/*' -mtime +{outdated_days} -type f",
|
||||||
|
f"find {chatmail_config.mailboxes_dir} -path '*/.*/tmp/*' -mtime +{outdated_days} -type f",
|
||||||
|
]
|
||||||
|
for cmd in find_cmds:
|
||||||
|
for line in remote.iter_output(cmd):
|
||||||
|
assert not line
|
||||||
|
|||||||
@@ -1,10 +1,50 @@
|
|||||||
import time
|
import ipaddress
|
||||||
import re
|
|
||||||
import random
|
import random
|
||||||
|
import re
|
||||||
|
import time
|
||||||
|
|
||||||
|
import imap_tools
|
||||||
import pytest
|
import pytest
|
||||||
import requests
|
import requests
|
||||||
import ipaddress
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def imap_mailbox(cmfactory):
|
||||||
|
(ac1,) = cmfactory.get_online_accounts(1)
|
||||||
|
user = ac1.get_config("addr")
|
||||||
|
password = ac1.get_config("mail_pw")
|
||||||
|
mailbox = imap_tools.MailBox(user.split("@")[1])
|
||||||
|
mailbox.login(user, password)
|
||||||
|
return mailbox
|
||||||
|
|
||||||
|
|
||||||
|
class TestMetadataTokens:
|
||||||
|
"Tests that use Metadata extension for storing tokens"
|
||||||
|
|
||||||
|
def test_set_get_metadata(self, imap_mailbox):
|
||||||
|
"set and get metadata token for an account"
|
||||||
|
client = imap_mailbox.client
|
||||||
|
client.send(b'a01 SETMETADATA INBOX (/private/devicetoken "1111" )\n')
|
||||||
|
res = client.readline()
|
||||||
|
assert b"OK Setmetadata completed" in res
|
||||||
|
|
||||||
|
client.send(b"a02 GETMETADATA INBOX /private/devicetoken\n")
|
||||||
|
res = client.readline()
|
||||||
|
assert res[:1] == b"*"
|
||||||
|
res = client.readline().strip().rstrip(b")")
|
||||||
|
assert res == b"1111"
|
||||||
|
assert b"Getmetadata completed" in client.readline()
|
||||||
|
|
||||||
|
client.send(b'a01 SETMETADATA INBOX (/private/devicetoken "2222" )\n')
|
||||||
|
res = client.readline()
|
||||||
|
assert b"OK Setmetadata completed" in res
|
||||||
|
|
||||||
|
client.send(b"a02 GETMETADATA INBOX /private/devicetoken\n")
|
||||||
|
res = client.readline()
|
||||||
|
assert res[:1] == b"*"
|
||||||
|
res = client.readline().strip().rstrip(b")")
|
||||||
|
assert res == b"1111 2222"
|
||||||
|
assert b"Getmetadata completed" in client.readline()
|
||||||
|
|
||||||
|
|
||||||
class TestEndToEndDeltaChat:
|
class TestEndToEndDeltaChat:
|
||||||
@@ -63,7 +103,7 @@ class TestEndToEndDeltaChat:
|
|||||||
|
|
||||||
addr = ac2.get_config("addr").lower()
|
addr = ac2.get_config("addr").lower()
|
||||||
saved_ok = 0
|
saved_ok = 0
|
||||||
for line in remote.iter_output("journalctl -f -u dovecot"):
|
for line in remote.iter_output("journalctl -n0 -f -u dovecot"):
|
||||||
if addr not in line:
|
if addr not in line:
|
||||||
# print(line)
|
# print(line)
|
||||||
continue
|
continue
|
||||||
@@ -75,7 +115,10 @@ class TestEndToEndDeltaChat:
|
|||||||
)
|
)
|
||||||
lp.indent("good, message sending failed because quota was exceeded")
|
lp.indent("good, message sending failed because quota was exceeded")
|
||||||
return
|
return
|
||||||
if "saved mail to inbox" in line:
|
if (
|
||||||
|
"stored mail into mailbox 'inbox'" in line
|
||||||
|
or "saved mail to inbox" in line
|
||||||
|
):
|
||||||
saved_ok += 1
|
saved_ok += 1
|
||||||
print(f"{saved_ok}: {line}")
|
print(f"{saved_ok}: {line}")
|
||||||
if saved_ok >= num_to_send:
|
if saved_ok >= num_to_send:
|
||||||
@@ -112,7 +155,7 @@ class TestEndToEndDeltaChat:
|
|||||||
lp.sec("ac1 sends a message and ac2 marks it as seen")
|
lp.sec("ac1 sends a message and ac2 marks it as seen")
|
||||||
chat = ac1.create_chat(ac2)
|
chat = ac1.create_chat(ac2)
|
||||||
msg = chat.send_text("hi")
|
msg = chat.send_text("hi")
|
||||||
m = ac2.wait_next_incoming_message()
|
m = ac2._evtracker.wait_next_incoming_message()
|
||||||
m.mark_seen()
|
m.mark_seen()
|
||||||
# we can only indirectly wait for mark-seen to cause an smtp-error
|
# we can only indirectly wait for mark-seen to cause an smtp-error
|
||||||
lp.sec("try to wait for markseen to complete and check error states")
|
lp.sec("try to wait for markseen to complete and check error states")
|
||||||
@@ -132,7 +175,7 @@ def test_hide_senders_ip_address(cmfactory):
|
|||||||
chat = cmfactory.get_accepted_chat(user1, user2)
|
chat = cmfactory.get_accepted_chat(user1, user2)
|
||||||
|
|
||||||
chat.send_text("testing submission header cleanup")
|
chat.send_text("testing submission header cleanup")
|
||||||
user2.wait_next_incoming_message()
|
user2._evtracker.wait_next_incoming_message()
|
||||||
user2.direct_imap.select_folder("Inbox")
|
user2.direct_imap.select_folder("Inbox")
|
||||||
msg = user2.direct_imap.get_all_messages()[0]
|
msg = user2.direct_imap.get_all_messages()[0]
|
||||||
assert public_ip not in msg.obj.as_string()
|
assert public_ip not in msg.obj.as_string()
|
||||||
@@ -142,9 +185,9 @@ def test_echobot(cmfactory, chatmail_config, lp):
|
|||||||
ac = cmfactory.get_online_accounts(1)[0]
|
ac = cmfactory.get_online_accounts(1)[0]
|
||||||
|
|
||||||
lp.sec(f"Send message to echo@{chatmail_config.mail_domain}")
|
lp.sec(f"Send message to echo@{chatmail_config.mail_domain}")
|
||||||
chat = ac.create_chat(f'echo@{chatmail_config.mail_domain}')
|
chat = ac.create_chat(f"echo@{chatmail_config.mail_domain}")
|
||||||
text = "hi, I hope you text me back"
|
text = "hi, I hope you text me back"
|
||||||
chat.send_text(text)
|
chat.send_text(text)
|
||||||
lp.sec("Wait for reply from echobot")
|
lp.sec("Wait for reply from echobot")
|
||||||
reply = ac.wait_next_incoming_message()
|
reply = ac._evtracker.wait_next_incoming_message()
|
||||||
assert reply.text == text
|
assert reply.text == text
|
||||||
|
|||||||
@@ -1,17 +1,16 @@
|
|||||||
import os
|
|
||||||
import io
|
|
||||||
import time
|
|
||||||
import random
|
|
||||||
import subprocess
|
|
||||||
import imaplib
|
import imaplib
|
||||||
import smtplib
|
import io
|
||||||
import itertools
|
import itertools
|
||||||
|
import os
|
||||||
|
import random
|
||||||
|
import smtplib
|
||||||
|
import subprocess
|
||||||
|
import time
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from chatmaild.database import Database
|
|
||||||
from chatmaild.config import read_config
|
from chatmaild.config import read_config
|
||||||
|
from chatmaild.database import Database
|
||||||
|
|
||||||
conftestdir = Path(__file__).parent
|
conftestdir = Path(__file__).parent
|
||||||
|
|
||||||
@@ -36,7 +35,7 @@ def pytest_runtest_setup(item):
|
|||||||
pytest.skip("skipping slow test, use --slow to run")
|
pytest.skip("skipping slow test, use --slow to run")
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture(scope="session")
|
||||||
def chatmail_config(pytestconfig):
|
def chatmail_config(pytestconfig):
|
||||||
current = basedir = Path().resolve()
|
current = basedir = Path().resolve()
|
||||||
while 1:
|
while 1:
|
||||||
@@ -50,12 +49,12 @@ def chatmail_config(pytestconfig):
|
|||||||
pytest.skip(f"no chatmail.ini file found in {basedir} or parent dirs")
|
pytest.skip(f"no chatmail.ini file found in {basedir} or parent dirs")
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture(scope="session")
|
||||||
def maildomain(chatmail_config):
|
def maildomain(chatmail_config):
|
||||||
return chatmail_config.mail_domain
|
return chatmail_config.mail_domain
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture(scope="session")
|
||||||
def sshdomain(maildomain):
|
def sshdomain(maildomain):
|
||||||
return os.environ.get("CHATMAIL_SSH", maildomain)
|
return os.environ.get("CHATMAIL_SSH", maildomain)
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import os
|
import os
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from cmdeploy.cmdeploy import get_parser, main
|
from cmdeploy.cmdeploy import get_parser, main
|
||||||
|
|
||||||
|
|
||||||
@@ -20,8 +21,9 @@ class TestCmdline:
|
|||||||
run = parser.parse_args(["run"])
|
run = parser.parse_args(["run"])
|
||||||
assert init and run
|
assert init and run
|
||||||
|
|
||||||
@pytest.mark.xfail(reason="init doesn't exit anymore, check for CLI output instead")
|
def test_init_not_overwrite(self, capsys):
|
||||||
def test_init_not_overwrite(self):
|
assert main(["init", "chat.example.org"]) == 0
|
||||||
main(["init", "chat.example.org"])
|
capsys.readouterr()
|
||||||
with pytest.raises(SystemExit):
|
assert main(["init", "chat.example.org"]) == 1
|
||||||
main(["init", "chat.example.org"])
|
out, err = capsys.readouterr()
|
||||||
|
assert "path exists" in out.lower()
|
||||||
|
|||||||
@@ -1,13 +1,14 @@
|
|||||||
import importlib.resources
|
|
||||||
import webbrowser
|
|
||||||
import hashlib
|
import hashlib
|
||||||
|
import importlib.resources
|
||||||
import time
|
import time
|
||||||
import traceback
|
import traceback
|
||||||
|
import webbrowser
|
||||||
|
|
||||||
import markdown
|
import markdown
|
||||||
from jinja2 import Template
|
|
||||||
from .genqr import gen_qr_png_data
|
|
||||||
from chatmaild.config import read_config
|
from chatmaild.config import read_config
|
||||||
|
from jinja2 import Template
|
||||||
|
|
||||||
|
from .genqr import gen_qr_png_data
|
||||||
|
|
||||||
|
|
||||||
def snapshot_dir_stats(somedir):
|
def snapshot_dir_stats(somedir):
|
||||||
@@ -120,7 +121,8 @@ def main():
|
|||||||
print(f"watching {src_path} directory for changes")
|
print(f"watching {src_path} directory for changes")
|
||||||
|
|
||||||
changenum = 0
|
changenum = 0
|
||||||
for count in range(0, 1000000):
|
count = 0
|
||||||
|
while True:
|
||||||
newstats = snapshot_dir_stats(src_path)
|
newstats = snapshot_dir_stats(src_path)
|
||||||
if newstats == stats and count % 60 != 0:
|
if newstats == stats and count % 60 != 0:
|
||||||
count += 1
|
count += 1
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
#!/usr/bin/env bash
|
#!/bin/sh
|
||||||
#
|
#
|
||||||
# Wrapper for cmdelpoy to run it in activated virtualenv.
|
# Wrapper for cmdelpoy to run it in activated virtualenv.
|
||||||
set -e
|
set -e
|
||||||
|
|||||||
80
scripts/dovecot/README.md
Normal file
80
scripts/dovecot/README.md
Normal file
@@ -0,0 +1,80 @@
|
|||||||
|
## Introduction to custom Dovecot builds
|
||||||
|
|
||||||
|
Chatmail servers use a custom Debian build of the IMAP 'dovecot' server software because
|
||||||
|
|
||||||
|
a) Dovecot developers did not yet merge a [pull request](https://github.com/dovecot/core/pull/216)
|
||||||
|
which majorly speeds up message delivery by removing a hardcoded 0.5 second delay
|
||||||
|
on relaying incoming messages.
|
||||||
|
|
||||||
|
b) Even if merged, it would take years for it to reach Debian stable.
|
||||||
|
|
||||||
|
c) The modified dovecot has been successfully used since December 2023 without issues
|
||||||
|
and we see no noticeable downside (theoretically higher CPU usage but not measureable)
|
||||||
|
but a considerable upside as the delay-removal facilitates end-to-end message
|
||||||
|
delivery of 200 ms in real networks.
|
||||||
|
|
||||||
|
The modified forked dovecot code lives at
|
||||||
|
[https://github.com/chatmail/dovecot](https://github.com/chatmail/dovecot).
|
||||||
|
The remainder of this document describes the setup of the Debian repository
|
||||||
|
containing the patched dovecot version.
|
||||||
|
|
||||||
|
## Building Debian packages at build.opensuse.org
|
||||||
|
|
||||||
|
Delta Chat developers maintain an [account](https://build.opensuse.org/project/show/home:deltachat)
|
||||||
|
in the [Open Build Service (OBS)](https://openbuildservice.org/),
|
||||||
|
where the [resulting package](https://build.opensuse.org/package/show/home:deltachat/dovecot)
|
||||||
|
is now used in deploying chatmail servers.
|
||||||
|
|
||||||
|
The Open Build Service (OBS) is a platform for building and distributing software packages
|
||||||
|
across various operating systems and architectures.
|
||||||
|
It supports openSUSE, Fedora, Debian, Ubuntu and Arch.
|
||||||
|
It's [primary instance](https://build.opensuse.org/) is ran by the openSUSE project
|
||||||
|
and is part of the pipeline of the creation of SUSE Linux Enterprise.
|
||||||
|
|
||||||
|
The OBS provides a mercurial-like interface to create source repositories
|
||||||
|
that are then automatically built.
|
||||||
|
While in theory a package can be created entirely over the web interface,
|
||||||
|
the use of the cli-tool `osc` is more convenient and is described in the [official documentation](https://openbuildservice.org/help/manuals/obs-user-guide/art.obs.bg#sec.obsbg.obsconfig).
|
||||||
|
|
||||||
|
### How to build the dovecot debian package on the OBS via our script
|
||||||
|
|
||||||
|
In scripts/dovecot/ is a shell script that prepares the required files and pushes them to build.opensuse.org.
|
||||||
|
|
||||||
|
Before using the script, you should have osc set up as described in the [official documentation](https://openbuildservice.org/help/manuals/obs-user-guide/art.obs.bg#sec.obsbg.obsconfig).
|
||||||
|
|
||||||
|
The script assumes you are on Debian. It automatically installs any needed dependencies and creates the source package. To upload the resulting source package to the OBS you need to enter the username and password for deltachat on build.opensuse.org in the last step of the script.
|
||||||
|
|
||||||
|
Use `source build-obs.sh` to run it.
|
||||||
|
|
||||||
|
### Adding the resulting OBS repository to Debian 12
|
||||||
|
|
||||||
|
Our dovecot fork is automatically installed as part of the chatmail deployment. You can see it in cmdeploy/src/cmdeploy/__init__.py. If you want to add our fork manually to a system, you can do the following:
|
||||||
|
|
||||||
|
First add our signing key to your apt keyring:
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo cp cmdeploy/src/cmdeploy/obs-home-deltachat.gpg /etc/apt/keyrings/obs-home-deltachat.gpg`
|
||||||
|
```
|
||||||
|
|
||||||
|
Now add our repository and key to /etc/apt/sources.list with a text editor of your choice:
|
||||||
|
|
||||||
|
```
|
||||||
|
deb [signed-by=/etc/apt/keyrings/obs-home-deltachat.gpg] https://download.opensuse.org/repositories/home:/deltachat/Debian_12/ ./
|
||||||
|
```
|
||||||
|
|
||||||
|
You can now install dovecot like normal.
|
||||||
|
|
||||||
|
```
|
||||||
|
sudo apt update
|
||||||
|
sudo apt install dovecot-core
|
||||||
|
```
|
||||||
|
|
||||||
|
### Security concerns
|
||||||
|
|
||||||
|
The signing of the patched dovecot package is done in the OBS and
|
||||||
|
in theory SUSE could make changes to the package delivered.
|
||||||
|
It is probably reasonable to trust SUSE to not mess with the build
|
||||||
|
process because it would cause serious negative reputation damage for them
|
||||||
|
if they tried and someone finds out.
|
||||||
|
|
||||||
|
Our dovecot fork will receive the same security backports as the dovecot package in Debian Sid.
|
||||||
54
scripts/dovecot/build-obs.sh
Normal file
54
scripts/dovecot/build-obs.sh
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# Install dependencies
|
||||||
|
echo "Installing dependencies for this script:"
|
||||||
|
sudo apt install -y devscripts build-essential osc curl git debhelper-compat
|
||||||
|
|
||||||
|
# Define path of your local OBS repository
|
||||||
|
SCRIPT_DIR=$PWD
|
||||||
|
OBS_PATH=$SCRIPT_DIR/obs
|
||||||
|
REPO_PATH=$OBS_PATH/home:deltachat/dovecot/
|
||||||
|
|
||||||
|
# Download Debian Source Files
|
||||||
|
echo "Downloading precise files from Debian unstable repository..."
|
||||||
|
mkdir dovecot-build
|
||||||
|
cd dovecot-build
|
||||||
|
|
||||||
|
# taken May 6th 2024, from https://packages.debian.org/unstable/dovecot-core
|
||||||
|
curl http://deb.debian.org/debian/pool/main/d/dovecot/dovecot_2.3.21+dfsg1-3.debian.tar.xz -O
|
||||||
|
curl http://deb.debian.org/debian/pool/main/d/dovecot/dovecot_2.3.21+dfsg1.orig.tar.gz -O
|
||||||
|
curl http://deb.debian.org/debian/pool/main/d/dovecot/dovecot_2.3.21+dfsg1.orig-pigeonhole.tar.gz -O
|
||||||
|
|
||||||
|
# Clone the Chatmail Dovecot Repo
|
||||||
|
echo "Cloning the Chatmail Dovecot fork..."
|
||||||
|
git clone https://github.com/chatmail/dovecot.git
|
||||||
|
|
||||||
|
# Build the source package
|
||||||
|
echo "Building the source package"
|
||||||
|
cd dovecot
|
||||||
|
dpkg-source -b .
|
||||||
|
|
||||||
|
# Setting up OSC
|
||||||
|
echo "Setting up OBS home repository"
|
||||||
|
mkdir $OBS_PATH
|
||||||
|
cd $OBS_PATH
|
||||||
|
rm -rf home:deltachat/dovecot
|
||||||
|
osc checkout home:deltachat/dovecot
|
||||||
|
|
||||||
|
# Copy Files to Your Local OBS Repository,
|
||||||
|
echo "Copying files to your local OBS repository..."
|
||||||
|
cd $SCRIPT_DIR/dovecot-build
|
||||||
|
cp -rf dovecot_2.3.21+dfsg1-3.debian.tar.xz $REPO_PATH
|
||||||
|
cp -rf dovecot_2.3.21+dfsg1.orig.tar.gz $REPO_PATH
|
||||||
|
cp -rf dovecot_2.3.21+dfsg1.orig-pigeonhole.tar.gz $REPO_PATH
|
||||||
|
cp -rf dovecot_2.3.21+dfsg1-3.dsc $REPO_PATH
|
||||||
|
|
||||||
|
# Push Changes to OBS
|
||||||
|
echo "Pushing changes to OBS..."
|
||||||
|
cd $REPO_PATH
|
||||||
|
osc up
|
||||||
|
osc add dovecot_2.3.21+dfsg1-3.debian.tar.xz
|
||||||
|
osc add dovecot_2.3.21+dfsg1.orig.tar.gz
|
||||||
|
osc add dovecot_2.3.21+dfsg1.orig-pigeonhole.tar.gz
|
||||||
|
osc add dovecot_2.3.21+dfsg1-3.dsc
|
||||||
|
osc commit
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
#!/bin/bash
|
#!/bin/sh
|
||||||
set -e
|
set -e
|
||||||
python3 -m venv --upgrade-deps venv
|
python3 -m venv --upgrade-deps venv
|
||||||
|
|
||||||
|
|||||||
@@ -1,11 +1,17 @@
|
|||||||
|
|
||||||
<img class="banner" src="collage-top.png"/>
|
<img class="banner" src="collage-top.png"/>
|
||||||
|
|
||||||
## Dear [Delta Chat](https://get.delta.chat) users and newcomers,
|
## Dear [Delta Chat](https://get.delta.chat) users and newcomers ...
|
||||||
|
|
||||||
|
{% if config.mail_domain != "nine.testrun.org" %}
|
||||||
Welcome to instant, interoperable and [privacy-preserving](privacy.html) messaging :)
|
Welcome to instant, interoperable and [privacy-preserving](privacy.html) messaging :)
|
||||||
|
{% else %}
|
||||||
|
Welcome to the default onboarding server ({{ config.mail_domain }})
|
||||||
|
for Delta Chat users. For details how it avoids storing personal information
|
||||||
|
please see our [privacy policy](privacy.html).
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
👉 **Tap** or scan this QR code to get a random `@{{config.mail_domain}}` e-mail address
|
👉 **Tap** or scan this QR code to get a `@{{config.mail_domain}}` chat profile
|
||||||
|
|
||||||
<a href="DCACCOUNT:https://{{ config.mail_domain }}/new">
|
<a href="DCACCOUNT:https://{{ config.mail_domain }}/new">
|
||||||
<img width=300 style="float: none;" src="qr-chatmail-invite-{{config.mail_domain}}.png" /></a>
|
<img width=300 style="float: none;" src="qr-chatmail-invite-{{config.mail_domain}}.png" /></a>
|
||||||
|
|||||||
@@ -1,6 +1,4 @@
|
|||||||
|
|
||||||
<img class="banner" src="collage-info.png"/>
|
|
||||||
|
|
||||||
## More information
|
## More information
|
||||||
|
|
||||||
{{ config.mail_domain }} provides a low-maintenance, resource efficient and
|
{{ config.mail_domain }} provides a low-maintenance, resource efficient and
|
||||||
@@ -11,7 +9,7 @@ for the usage in chats, especially DeltaChat.
|
|||||||
### Choosing a chatmail address instead of using a random one
|
### Choosing a chatmail address instead of using a random one
|
||||||
|
|
||||||
In the Delta Chat account setup
|
In the Delta Chat account setup
|
||||||
you may tap `LOG INTO YOUR E-MAIL ACCOUNT`
|
you may tap `I already have a profile`
|
||||||
and fill the two fields like this:
|
and fill the two fields like this:
|
||||||
|
|
||||||
- `Address`: invent a word with
|
- `Address`: invent a word with
|
||||||
|
|||||||
@@ -1,21 +1,41 @@
|
|||||||
<img class="banner" src="collage-privacy.png"/>
|
|
||||||
|
|
||||||
# Privacy Policy for {{ config.mail_domain }}
|
# Privacy Policy for {{ config.mail_domain }}
|
||||||
|
|
||||||
We want to show you in a fair and transparent way
|
{% if config.mail_domain == "nine.testrun.org" %}
|
||||||
what personal data is processed by us.
|
Welcome to `{{config.mail_domain}}`, the default chatmail onboarding server for Delta Chat users.
|
||||||
We follow a strict privacy-by-design approach
|
It is operated on the side by a small sysops team employed by [merlinux](https://merlinux.eu),
|
||||||
and try to avoid processing your data in the first place,
|
an open-source R&D company also acting as the fiscal sponsor of Delta Chat app developments.
|
||||||
but as you may know,
|
See [other chatmail servers](https://delta.chat/en/chatmail) for alternative server operators.
|
||||||
the internet,
|
{% endif %}
|
||||||
and in particular sending e-mail messages,
|
|
||||||
does not work without data.
|
|
||||||
Still,
|
## Summary: No personal data asked or collected
|
||||||
it's only fair that you know at all times
|
|
||||||
what personal data is processed
|
This chatmail server neither asks for nor retains personal information.
|
||||||
when you use our service.
|
Chatmail servers exist to reliably transmit (store and deliver) end-to-end encrypted messages
|
||||||
|
between user's devices running the Delta Chat messenger app.
|
||||||
|
Technically, you may think of a Chatmail server as
|
||||||
|
an end-to-end encrypted "messaging router" at Internet-scale.
|
||||||
|
|
||||||
|
A chatmail server is very unlike classic e-mail servers (for example Google Mail servers)
|
||||||
|
that ask for personal data and permanently store messages.
|
||||||
|
A chatmail server behaves more like the Signal messaging server
|
||||||
|
but does not know about phone numbers and securely and automatically interoperates
|
||||||
|
with other chatmail and classic e-mail servers.
|
||||||
|
|
||||||
|
In particular, this chatmail server
|
||||||
|
|
||||||
|
- unconditionally removes messages after {{ config.delete_mails_after }} days,
|
||||||
|
|
||||||
|
- prohibits sending out un-encrypted messages,
|
||||||
|
|
||||||
|
- only has temporary log files used for debugging purposes.
|
||||||
|
|
||||||
|
Legally, authorities might still regard chatmail as a "classic e-mail" server
|
||||||
|
which collects and retains personal data.
|
||||||
|
We do not agree on this interpretation. Nevertheless, we provide more legal details below
|
||||||
|
to make life easier for data protection specialists and lawyers scrutinizing chatmail operations.
|
||||||
|
|
||||||
If you have any remaining questions about data protection, please contact us.
|
|
||||||
|
|
||||||
## 1. Name and contact information
|
## 1. Name and contact information
|
||||||
|
|
||||||
@@ -57,7 +77,7 @@ we process the following data and details:
|
|||||||
- Users can retrieve or delete all stored messages
|
- Users can retrieve or delete all stored messages
|
||||||
without intervention from the operators using standard IMAP client tools.
|
without intervention from the operators using standard IMAP client tools.
|
||||||
|
|
||||||
### 3.1 Account setup
|
### 2.1 Account setup
|
||||||
|
|
||||||
Creating an account happens in one of two ways on our mail servers:
|
Creating an account happens in one of two ways on our mail servers:
|
||||||
|
|
||||||
@@ -78,7 +98,7 @@ Art. 6 (1) lit. b GDPR,
|
|||||||
as you have a usage contract with us
|
as you have a usage contract with us
|
||||||
by using our services.
|
by using our services.
|
||||||
|
|
||||||
## 3.2 Processing of E-Mail-Messages
|
### 2.2 Processing of E-Mail-Messages
|
||||||
|
|
||||||
In addition,
|
In addition,
|
||||||
we will process data
|
we will process data
|
||||||
@@ -104,7 +124,7 @@ Therefore, limits are enforced:
|
|||||||
|
|
||||||
- message size limits
|
- message size limits
|
||||||
|
|
||||||
- any other limit neccessary for the whole server to function in a healthy way
|
- any other limit necessary for the whole server to function in a healthy way
|
||||||
and to prevent abuse.
|
and to prevent abuse.
|
||||||
|
|
||||||
The processing and use of the above permissions
|
The processing and use of the above permissions
|
||||||
@@ -178,8 +198,9 @@ for the purpose of drawing conclusions about your person.
|
|||||||
|
|
||||||
## 4. Transfer of Data
|
## 4. Transfer of Data
|
||||||
|
|
||||||
Your personal data
|
We do not retain any personal data but e-mail messages waiting to be delivered
|
||||||
will not be transferred to third parties
|
may contain personal data.
|
||||||
|
Any such residual personal data will not be transferred to third parties
|
||||||
for purposes other than those listed below:
|
for purposes other than those listed below:
|
||||||
|
|
||||||
a) you have given your express consent
|
a) you have given your express consent
|
||||||
|
|||||||
Reference in New Issue
Block a user