Compare commits
	
		
			6 Commits
		
	
	
		
			py38
			...
			f3e80c43ae
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| f3e80c43ae | |||
| 829be8413b | |||
| 121a02b8ae | |||
| e18f1c7a96 | |||
| cc5ad89977 | |||
| a1fd586dbd | 
							
								
								
									
										13
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								Makefile
									
									
									
									
									
								
							| @@ -1,8 +1,10 @@ | ||||
| # Needs python3 >= 3.9, sed, git for build | ||||
| # Needs python3 >= 3.9, sed, git for build, docker for tests | ||||
| build: clean | ||||
| 	python3 -m pip install -r requirements.txt --no-compile --target build | ||||
| 	cp -r mail4one/ build/ | ||||
| 	sed -i "s/DEVELOMENT/$(shell scripts/get_version.sh)/" build/mail4one/version.py | ||||
| 	find build -name "*.pyi" -o -name "py.typed" | xargs -I typefile rm typefile | ||||
| 	rm -rf build/bin | ||||
| 	rm -rf build/mail4one/__pycache__ | ||||
| 	rm -rf build/*.dist-info | ||||
| 	python3 -m zipapp \ | ||||
| @@ -18,17 +20,19 @@ clean: | ||||
| docker-tests: | ||||
| 	docker run --pull=always -v `pwd`:/app -w /app --rm python:3.11-alpine sh scripts/runtests.sh | ||||
| 	docker run --pull=always -v `pwd`:/app -w /app --rm python:3.10-alpine sh scripts/runtests.sh | ||||
| 	docker run --pull=always -v `pwd`:/app -w /app --rm python:3.12        sh scripts/runtests.sh | ||||
| 	docker run --pull=always -v `pwd`:/app -w /app --rm python:3.11        sh scripts/runtests.sh | ||||
| 	docker run --pull=always -v `pwd`:/app -w /app --rm python:3.10        sh scripts/runtests.sh | ||||
| 	docker run --pull=always -v `pwd`:/app -w /app --rm python:3.9         sh scripts/runtests.sh | ||||
|  | ||||
| # ============================================================================ | ||||
| # Below targets for devs. Need pipenv, black installed | ||||
|  | ||||
| requirements.txt: Pipfile.lock | ||||
| 	pipenv requirements > requirements.txt | ||||
|  | ||||
| format: | ||||
| 	black mail4one/*py | ||||
| 	black mail4one/*py tests/*py | ||||
|  | ||||
| build-dev: requirements.txt build | ||||
|  | ||||
| @@ -38,6 +42,11 @@ setup: | ||||
| cleanup: | ||||
| 	pipenv --rm | ||||
|  | ||||
| update: | ||||
| 	rm requirements.txt Pipfile.lock | ||||
| 	pipenv update | ||||
| 	pipenv requirements > requirements.txt | ||||
|  | ||||
| shell: | ||||
| 	MYPYPATH=`pipenv --venv`/lib/python3.11/site-packages pipenv shell | ||||
| 	 | ||||
|   | ||||
							
								
								
									
										8
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										8
									
								
								Pipfile.lock
									
									
									
										generated
									
									
									
								
							| @@ -22,6 +22,7 @@ | ||||
|                 "sha256:f9243b7dfe00aaf567da8728d891752426b51392174a34d2cf5c18053b63dcbc" | ||||
|             ], | ||||
|             "index": "pypi", | ||||
|             "markers": "python_version ~= '3.7'", | ||||
|             "version": "==1.4.4.post2" | ||||
|         }, | ||||
|         "atpublic": { | ||||
| @@ -34,11 +35,11 @@ | ||||
|         }, | ||||
|         "attrs": { | ||||
|             "hashes": [ | ||||
|                 "sha256:1f28b4522cdc2fb4256ac1a020c78acf9cba2c6b461ccd2c126f3aa8e8335d04", | ||||
|                 "sha256:6279836d581513a26f1bf235f9acd333bc9115683f14f7e8fae46c98fc50e015" | ||||
|                 "sha256:935dc3b529c262f6cf76e50877d35a4bd3c1de194fd41f47a2b7ae8f19971f30", | ||||
|                 "sha256:99b87a485a5820b23b879f04c2305b44b951b502fd64be915879d77a7e8fc6f1" | ||||
|             ], | ||||
|             "markers": "python_version >= '3.7'", | ||||
|             "version": "==23.1.0" | ||||
|             "version": "==23.2.0" | ||||
|         }, | ||||
|         "python-jata": { | ||||
|             "hashes": [ | ||||
| @@ -46,6 +47,7 @@ | ||||
|                 "sha256:ff4cd7ca75c9a8306b69ef6e878c296a5602f3279c6f9a82b6105b8eba764760" | ||||
|             ], | ||||
|             "index": "pypi", | ||||
|             "markers": "python_version >= '3.8'", | ||||
|             "version": "==1.2" | ||||
|         } | ||||
|     }, | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| # Mail4one | ||||
|  | ||||
| Personal mail server for a single user or a small family. Written in pure python with minimal dependencies. | ||||
| Personal mail server for a single user or a small family. Written in pure python with [minimal dependencies](Pipfile). | ||||
| Designed for dynamic alias based workflow where a different alias is used for each purpose. | ||||
|  | ||||
| # Getting started | ||||
| @@ -23,7 +23,7 @@ Mail4one only takes care of receiving and serving email. For sending email, use | ||||
|  | ||||
| Most of them have generous free tier which is more than enough for personal use. | ||||
|  | ||||
| Sending email is tricky. Even if everything is correctly setup (DMARC, DKIM, SPF), popular email vendors like google, microsoft may mark emails sent from your IP as spam for no reason. | ||||
| Sending email is tricky. Even if everything is correctly setup (DMARC, DKIM, SPF), popular email vendors like google, microsoft may mark emails sent from your IP as spam for no reason. Hence using a dedicated service is the only reliable way to send emails. | ||||
|  | ||||
| # Community | ||||
|  | ||||
| @@ -62,7 +62,7 @@ This should generate `mail4one.pyz` in current folder. This is a [executable pyt | ||||
| * Write dedicated documentation | ||||
| * Test with more email clients ([Thunderbird](https://www.thunderbird.net/) and [k9mail](https://k9mail.app/) are tested now) | ||||
| * IMAP support | ||||
| * Web UI for editing config | ||||
| * Web UI for editing config ([WIP](https://github.com/mail4one/mail4one/tree/webform)) | ||||
| * Support email submission from client to forward to other senders or direct delivery | ||||
| * Optional SPAM filtering | ||||
| * Optional DMARC,SPF,DKIM verification | ||||
|   | ||||
| @@ -2,13 +2,11 @@ | ||||
|  | ||||
| #  certbot deploy hook to copy certificates to mail4one when renewed. | ||||
| #  Initial setup, Install certbot(https://certbot.eff.org/) and run `certbot certonly` as root | ||||
| #  Doc: https://eff-certbot.readthedocs.io/en/latest/using.html#renewing-certificates | ||||
| # | ||||
| #  This file is supposed to be copied to /etc/letsencrypt/renewal-hooks/deploy/ | ||||
| #  Change the mail domain to the one on MX record | ||||
|  | ||||
| set -x | ||||
|  | ||||
|  | ||||
| if [ "$RENEWED_DOMAINS" = "mail.mydomain.com" ] | ||||
| then | ||||
| 		mkdir -p /var/lib/mail4one/certs | ||||
| @@ -17,4 +15,5 @@ then | ||||
| 		cp "$RENEWED_LINEAGE/fullchain.pem" /var/lib/mail4one/certs/ | ||||
| 		cp "$RENEWED_LINEAGE/privkey.pem" /var/lib/mail4one/certs/ | ||||
| 		systemctl restart mail4one.service | ||||
| 		echo "$(date) Renewed and deployed certificates for mail4one" >> /var/log/mail4one-cert-renew.log | ||||
| fi | ||||
|   | ||||
| @@ -386,13 +386,14 @@ def debug_main(): | ||||
|     logging.basicConfig(level=logging.DEBUG) | ||||
|  | ||||
|     import sys | ||||
|     from .pwhash import gen_pwhash | ||||
|  | ||||
|     _, mails_path, port, password = sys.argv | ||||
|     _, mails_path, mbox = sys.argv | ||||
|  | ||||
|     mails_path = Path(mails_path) | ||||
|     port = int(port) | ||||
|     users = [User(username="dummy", password_hash=gen_pwhash("dummy"), mbox=mbox)] | ||||
|  | ||||
|     asyncio.run(a_main(mails_path, port, password_hash=password_hash)) | ||||
|     asyncio.run(a_main("127.0.0.1", 1101, mails_path, users=users)) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| -i https://pypi.org/simple | ||||
| aiosmtpd==1.4.4.post2 | ||||
| atpublic==4.0 ; python_version >= '3.8' | ||||
| attrs==23.1.0 ; python_version >= '3.7' | ||||
| python-jata==1.2 | ||||
| aiosmtpd==1.4.4.post2; python_version ~= '3.7' | ||||
| atpublic==4.0; python_version >= '3.8' | ||||
| attrs==23.2.0; python_version >= '3.7' | ||||
| python-jata==1.2; python_version >= '3.8' | ||||
|   | ||||
| @@ -8,7 +8,7 @@ then | ||||
| 		tag_val=$(git describe --dirty=DIRTY --exact-match) | ||||
| 		case "$tag_val" in  | ||||
| 				*DIRTY) | ||||
| 						echo "git=$commit-changes" | ||||
| 						echo "git-$commit-changes" | ||||
| 						;; | ||||
| 				v*) # Only consider tags starting with v | ||||
| 						echo "$tag_val" | ||||
|   | ||||
| @@ -67,10 +67,11 @@ class TestConfig(unittest.TestCase): | ||||
|     def test_get_mboxes(self) -> None: | ||||
|         cfg = config.Config(TEST_CONFIG) | ||||
|         rules = config.parse_checkers(cfg) | ||||
|         self.assertEqual(config.get_mboxes("foo@bar.com", rules), ['spam']) | ||||
|         self.assertEqual(config.get_mboxes("foo@mydomain.com", rules), ['all']) | ||||
|         self.assertEqual(config.get_mboxes("first.last@mydomain.com", rules), | ||||
|                          ['important', 'all']) | ||||
|         self.assertEqual(config.get_mboxes("foo@bar.com", rules), ["spam"]) | ||||
|         self.assertEqual(config.get_mboxes("foo@mydomain.com", rules), ["all"]) | ||||
|         self.assertEqual( | ||||
|             config.get_mboxes("first.last@mydomain.com", rules), ["important", "all"] | ||||
|         ) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|   | ||||
| @@ -8,14 +8,16 @@ from mail4one.pop3 import create_pop_server | ||||
| from mail4one.config import User | ||||
| from pathlib import Path | ||||
|  | ||||
| TEST_HASH = "".join(c for c in """ | ||||
| TEST_HASH = "".join( | ||||
|     """ | ||||
| AFTY5EVN7AX47ZL7UMH3BETYWFBTAV3XHR73CEFAJBPN2NIHPWD | ||||
| ZHV2UQSMSPHSQQ2A2BFQBNC77VL7F2UKATQNJZGYLCSU6C43UQD | ||||
| AQXWXSWNGAEPGIMG2F3QDKBXL3MRHY6K2BPID64ZR6LABLPVSF | ||||
| """ if not c.isspace()) | ||||
| """.split() | ||||
| ) | ||||
|  | ||||
| TEST_USER = 'foobar' | ||||
| TEST_MBOX = 'foobar_mails' | ||||
| TEST_USER = "foobar" | ||||
| TEST_MBOX = "foobar_mails" | ||||
|  | ||||
| USERS = [User(username=TEST_USER, password_hash=TEST_HASH, mbox=TEST_MBOX)] | ||||
|  | ||||
| @@ -38,7 +40,8 @@ Hello bro\r | ||||
| IlzVOJqu9Zp7twFAtzcV\r | ||||
| yQVk36B0mGU2gtWxXLr\r | ||||
| PeF0RtbI0mAuVPLQDHCi\r | ||||
| \r\n""" | ||||
| \r | ||||
| """ | ||||
|  | ||||
|  | ||||
| def setUpModule() -> None: | ||||
| @@ -48,11 +51,11 @@ def setUpModule() -> None: | ||||
|     unittest.addModuleCleanup(td.cleanup) | ||||
|     MAILS_PATH = Path(td.name) | ||||
|     os.mkdir(MAILS_PATH / TEST_MBOX) | ||||
|     for md in ('new', 'cur', 'tmp'): | ||||
|     for md in ("new", "cur", "tmp"): | ||||
|         os.mkdir(MAILS_PATH / TEST_MBOX / md) | ||||
|     with open(MAILS_PATH / TEST_MBOX/ 'new/msg1.eml', 'wb') as f: | ||||
|     with open(MAILS_PATH / TEST_MBOX / "new/msg1.eml", "wb") as f: | ||||
|         f.write(TESTMAIL) | ||||
|     with open(MAILS_PATH / TEST_MBOX/ 'new/msg2.eml', 'wb') as f: | ||||
|     with open(MAILS_PATH / TEST_MBOX / "new/msg2.eml", "wb") as f: | ||||
|         f.write(TESTMAIL) | ||||
|     logging.debug(MAILS_PATH) | ||||
|  | ||||
| @@ -65,13 +68,11 @@ class TestPop3(unittest.IsolatedAsyncioTestCase): | ||||
|  | ||||
|     async def asyncSetUp(self) -> None: | ||||
|         logging.debug("at asyncSetUp") | ||||
|         pop_server = await create_pop_server(host='127.0.0.1', | ||||
|                                              port=7995, | ||||
|                                              mails_path=MAILS_PATH, | ||||
|                                              users=USERS) | ||||
|         pop_server = await create_pop_server( | ||||
|             host="127.0.0.1", port=7995, mails_path=MAILS_PATH, users=USERS | ||||
|         ) | ||||
|         self.task = asyncio.create_task(pop_server.serve_forever()) | ||||
|         self.reader, self.writer = await asyncio.open_connection( | ||||
|             '127.0.0.1', 7995) | ||||
|         self.reader, self.writer = await asyncio.open_connection("127.0.0.1", 7995) | ||||
|  | ||||
|     async def test_QUIT(self) -> None: | ||||
|         dialog = """ | ||||
| @@ -115,8 +116,8 @@ class TestPop3(unittest.IsolatedAsyncioTestCase): | ||||
|         await self.dialog_checker(dialog) | ||||
|  | ||||
|     async def test_dupe_AUTH(self) -> None: | ||||
|         r1, w1 = await asyncio.open_connection('127.0.0.1', 7995) | ||||
|         r2, w2 = await asyncio.open_connection('127.0.0.1', 7995) | ||||
|         r1, w1 = await asyncio.open_connection("127.0.0.1", 7995) | ||||
|         r2, w2 = await asyncio.open_connection("127.0.0.1", 7995) | ||||
|         dialog = """ | ||||
|         S: +OK Server Ready | ||||
|         C: USER foobar | ||||
| @@ -206,9 +207,9 @@ class TestPop3(unittest.IsolatedAsyncioTestCase): | ||||
|     async def dialog_checker(self, dialog: str) -> None: | ||||
|         await self.dialog_checker_impl(self.reader, self.writer, dialog) | ||||
|  | ||||
|     async def dialog_checker_impl(self, reader: asyncio.StreamReader, | ||||
|                                   writer: asyncio.StreamWriter, | ||||
|                                   dialog: str) -> None: | ||||
|     async def dialog_checker_impl( | ||||
|         self, reader: asyncio.StreamReader, writer: asyncio.StreamWriter, dialog: str | ||||
|     ) -> None: | ||||
|         for line in dialog.splitlines(): | ||||
|             line = line.strip() | ||||
|             if not line: | ||||
| @@ -222,5 +223,5 @@ class TestPop3(unittest.IsolatedAsyncioTestCase): | ||||
|                 self.assertEqual(data, resp) | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
| if __name__ == "__main__": | ||||
|     unittest.main() | ||||
|   | ||||
| @@ -10,27 +10,31 @@ class TestPWHash(unittest.TestCase): | ||||
|         pwinfo = parse_hash(pwhash) | ||||
|         self.assertEqual(len(pwinfo.salt), SALT_LEN) | ||||
|         self.assertEqual(len(pwinfo.scrypt_hash), KEY_LEN) | ||||
|         self.assertTrue(check_pass(password, pwinfo), | ||||
|                         "check pass with correct password") | ||||
|         self.assertFalse(check_pass("foobar", pwinfo), | ||||
|                          "check pass with wrong password") | ||||
|         self.assertTrue( | ||||
|             check_pass(password, pwinfo), "check pass with correct password" | ||||
|         ) | ||||
|         self.assertFalse(check_pass("foobar", pwinfo), "check pass with wrong password") | ||||
|  | ||||
|     def test_hardcoded_hash(self): | ||||
|         test_hash = "".join(c for c in """ | ||||
|         test_hash = "".join( | ||||
|             c | ||||
|             for c in """ | ||||
|         AFTY5EVN7AX47ZL7UMH3BETYWFBTAV3XHR73CEFAJBPN2NIHPWD | ||||
|         ZHV2UQSMSPHSQQ2A2BFQBNC77VL7F2UKATQNJZGYLCSU6C43UQD | ||||
|         AQXWXSWNGAEPGIMG2F3QDKBXL3MRHY6K2BPID64ZR6LABLPVSF | ||||
|         """ if not c.isspace()) | ||||
|         """ | ||||
|             if not c.isspace() | ||||
|         ) | ||||
|         pwinfo = parse_hash(test_hash) | ||||
|         self.assertTrue(check_pass("helloworld", pwinfo), | ||||
|                         "check pass with correct password") | ||||
|         self.assertFalse(check_pass("foobar", pwinfo), | ||||
|                          "check pass with wrong password") | ||||
|         self.assertTrue( | ||||
|             check_pass("helloworld", pwinfo), "check pass with correct password" | ||||
|         ) | ||||
|         self.assertFalse(check_pass("foobar", pwinfo), "check pass with wrong password") | ||||
|  | ||||
|     def test_invalid_hash(self): | ||||
|         with self.assertRaises(Exception): | ||||
|             parse_hash("sdlfkjdsklfjdsk") | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
| if __name__ == "__main__": | ||||
|     unittest.main() | ||||
|   | ||||
| @@ -10,7 +10,7 @@ from pathlib import Path | ||||
|  | ||||
| from mail4one.smtp import create_smtp_server | ||||
|  | ||||
| TEST_MBOX = 'foobar_mails' | ||||
| TEST_MBOX = "foobar_mails" | ||||
| MAILS_PATH: Path | ||||
|  | ||||
|  | ||||
| @@ -21,7 +21,7 @@ def setUpModule() -> None: | ||||
|     unittest.addModuleCleanup(td.cleanup) | ||||
|     MAILS_PATH = Path(td.name) | ||||
|     os.mkdir(MAILS_PATH / TEST_MBOX) | ||||
|     for md in ('new', 'cur', 'tmp'): | ||||
|     for md in ("new", "cur", "tmp"): | ||||
|         os.mkdir(MAILS_PATH / TEST_MBOX / md) | ||||
|  | ||||
|  | ||||
| @@ -32,7 +32,8 @@ class TestSMTP(unittest.IsolatedAsyncioTestCase): | ||||
|             host="127.0.0.1", | ||||
|             port=7996, | ||||
|             mails_path=MAILS_PATH, | ||||
|             mbox_finder=lambda addr: [TEST_MBOX]) | ||||
|             mbox_finder=lambda addr: [TEST_MBOX], | ||||
|         ) | ||||
|         self.task = asyncio.create_task(smtp_server.serve_forever()) | ||||
|  | ||||
|     async def test_send_mail(self) -> None: | ||||
| @@ -45,8 +46,9 @@ class TestSMTP(unittest.IsolatedAsyncioTestCase): | ||||
|         msg = b"".join(l.strip() + b"\r\n" for l in msg.splitlines()) | ||||
|  | ||||
|         def send_mail(): | ||||
|             with contextlib.closing(smtplib.SMTP(host="127.0.0.1", | ||||
|                                                  port=7996)) as client: | ||||
|             with contextlib.closing( | ||||
|                 smtplib.SMTP(host="127.0.0.1", port=7996) | ||||
|             ) as client: | ||||
|                 client.sendmail("foo@sender.com", "foo@bar.com", msg) | ||||
|                 _, local_port = client.sock.getsockname() | ||||
|                 return local_port | ||||
| @@ -62,7 +64,7 @@ class TestSMTP(unittest.IsolatedAsyncioTestCase): | ||||
|         Byee | ||||
|         """ | ||||
|         expected = "".join(l.strip() + "\r\n" for l in expected.splitlines()) | ||||
|         mails = list((MAILS_PATH / TEST_MBOX / 'new').glob("*")) | ||||
|         mails = list((MAILS_PATH / TEST_MBOX / "new").glob("*")) | ||||
|         self.assertEqual(len(mails), 1) | ||||
|         self.assertEqual(mails[0].read_bytes(), expected.encode()) | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user