aboutsummaryrefslogtreecommitdiff
path: root/src/mailman_pgp/archivers
diff options
context:
space:
mode:
Diffstat (limited to 'src/mailman_pgp/archivers')
-rw-r--r--src/mailman_pgp/archivers/local.py32
-rw-r--r--src/mailman_pgp/archivers/local_maildir.py72
-rw-r--r--src/mailman_pgp/archivers/local_mbox.py72
-rw-r--r--src/mailman_pgp/archivers/tests/__init__.py0
-rw-r--r--src/mailman_pgp/archivers/tests/test_maildir.py82
-rw-r--r--src/mailman_pgp/archivers/tests/test_mbox.py82
6 files changed, 308 insertions, 32 deletions
diff --git a/src/mailman_pgp/archivers/local.py b/src/mailman_pgp/archivers/local.py
deleted file mode 100644
index 04d9a22..0000000
--- a/src/mailman_pgp/archivers/local.py
+++ /dev/null
@@ -1,32 +0,0 @@
-# Copyright (C) 2017 Jan Jancar
-#
-# This file is a part of the Mailman PGP plugin.
-#
-# This program is free software; you can redistribute it and/or modify it under
-# the terms of the GNU General Public License as published by the Free
-# Software Foundation, either version 3 of the License, or (at your option)
-# any later version.
-#
-# This program is distributed in the hope that it will be useful, but WITHOUT
-# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
-# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
-# more details.
-#
-# You should have received a copy of the GNU General Public License along with
-# this program. If not, see <http://www.gnu.org/licenses/>.
-
-"""
-Archives messages locally, encrypted (TBD how),
-similar to Mailman's prototype archiver.
-"""
-
-from mailman.interfaces.archiver import IArchiver
-from public import public
-from zope.interface import implementer
-
-
-@public
-@implementer(IArchiver)
-class LocalArchiver:
- """Local PGP enabled archiver."""
- pass
diff --git a/src/mailman_pgp/archivers/local_maildir.py b/src/mailman_pgp/archivers/local_maildir.py
new file mode 100644
index 0000000..4b7e75d
--- /dev/null
+++ b/src/mailman_pgp/archivers/local_maildir.py
@@ -0,0 +1,72 @@
+# Copyright (C) 2017 Jan Jancar
+#
+# This file is a part of the Mailman PGP plugin.
+#
+# This program is free software; you can redistribute it and/or modify it under
+# the terms of the GNU General Public License as published by the Free
+# Software Foundation, either version 3 of the License, or (at your option)
+# any later version.
+#
+# This program is distributed in the hope that it will be useful, but WITHOUT
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
+# more details.
+#
+# You should have received a copy of the GNU General Public License along with
+# this program. If not, see <http://www.gnu.org/licenses/>.
+
+"""
+Archives messages to a maildir, locally, encrypted (TBD how),
+similar to Mailman's prototype archiver.
+"""
+import os
+from mailbox import Maildir
+
+from flufl.lock import Lock
+from mailman.interfaces.archiver import IArchiver
+from public import public
+from zope.interface import implementer
+
+from mailman_pgp.config import config, mm_config
+from mailman_pgp.model.list import PGPMailingList
+from mailman_pgp.pgp.mime import MIMEWrapper
+
+
+@public
+@implementer(IArchiver)
+class LocalMaildirArchiver:
+ """Local PGP enabled archiver."""
+
+ name = 'pgp-maildir-local'
+ is_enabled = False
+
+ @staticmethod
+ def list_url(mlist):
+ """See `IArchiver`."""
+ return None
+
+ @staticmethod
+ def permalink(mlist, msg):
+ """See `IArchiver`."""
+ return None
+
+ @staticmethod
+ def archive_message(mlist, msg):
+ """See `IArchiver`."""
+ pgp_list = PGPMailingList.for_list(mlist)
+ if not pgp_list:
+ return None
+ maildir_dir = config.get_value('archiving', 'maildir_dir')
+ maildir_dir.mkdir(parents=True, exist_ok=True)
+
+ list_dir = maildir_dir.joinpath(mlist.fqdn_listname)
+ maildir = Maildir(str(list_dir))
+ lock_file = os.path.join(mm_config.LOCK_DIR,
+ '{}-{}.lock'.format(mlist.fqdn_listname,
+ LocalMaildirArchiver.name)
+ )
+ wrapped = MIMEWrapper(msg)
+ encrypted = wrapped.encrypt(pgp_list.pubkey)
+ with Lock(lock_file):
+ maildir.add(encrypted)
+ return None
diff --git a/src/mailman_pgp/archivers/local_mbox.py b/src/mailman_pgp/archivers/local_mbox.py
new file mode 100644
index 0000000..99264c2
--- /dev/null
+++ b/src/mailman_pgp/archivers/local_mbox.py
@@ -0,0 +1,72 @@
+# Copyright (C) 2017 Jan Jancar
+#
+# This file is a part of the Mailman PGP plugin.
+#
+# This program is free software; you can redistribute it and/or modify it under
+# the terms of the GNU General Public License as published by the Free
+# Software Foundation, either version 3 of the License, or (at your option)
+# any later version.
+#
+# This program is distributed in the hope that it will be useful, but WITHOUT
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
+# more details.
+#
+# You should have received a copy of the GNU General Public License along with
+# this program. If not, see <http://www.gnu.org/licenses/>.
+
+"""
+Archives messages to a mbox, locally, encrypted (TBD how),
+similar to Mailman's prototype archiver.
+"""
+import os
+from mailbox import mbox
+
+from flufl.lock import Lock
+from mailman.interfaces.archiver import IArchiver
+from public import public
+from zope.interface import implementer
+
+from mailman_pgp.config import config, mm_config
+from mailman_pgp.model.list import PGPMailingList
+from mailman_pgp.pgp.mime import MIMEWrapper
+
+
+@public
+@implementer(IArchiver)
+class LocalMailboxArchiver:
+ """Local PGP enabled archiver."""
+
+ name = 'pgp-mbox-local'
+ is_enabled = False
+
+ @staticmethod
+ def list_url(mlist):
+ """See `IArchiver`."""
+ return None
+
+ @staticmethod
+ def permalink(mlist, msg):
+ """See `IArchiver`."""
+ return None
+
+ @staticmethod
+ def archive_message(mlist, msg):
+ """See `IArchiver`."""
+ pgp_list = PGPMailingList.for_list(mlist)
+ if not pgp_list:
+ return None
+ mailbox_dir = config.get_value('archiving', 'mailbox_dir')
+ mailbox_dir.mkdir(parents=True, exist_ok=True)
+
+ list_dir = mailbox_dir.joinpath(mlist.fqdn_listname)
+ mailbox = mbox(str(list_dir))
+ lock_file = os.path.join(mm_config.LOCK_DIR,
+ '{}-{}.lock'.format(mlist.fqdn_listname,
+ LocalMailboxArchiver.name)
+ )
+ wrapped = MIMEWrapper(msg)
+ encrypted = wrapped.encrypt(pgp_list.pubkey)
+ with Lock(lock_file):
+ mailbox.add(encrypted)
+ return None
diff --git a/src/mailman_pgp/archivers/tests/__init__.py b/src/mailman_pgp/archivers/tests/__init__.py
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/src/mailman_pgp/archivers/tests/__init__.py
diff --git a/src/mailman_pgp/archivers/tests/test_maildir.py b/src/mailman_pgp/archivers/tests/test_maildir.py
new file mode 100644
index 0000000..bc810a7
--- /dev/null
+++ b/src/mailman_pgp/archivers/tests/test_maildir.py
@@ -0,0 +1,82 @@
+# Copyright (C) 2017 Jan Jancar
+#
+# This file is a part of the Mailman PGP plugin.
+#
+# This program is free software; you can redistribute it and/or modify it under
+# the terms of the GNU General Public License as published by the Free
+# Software Foundation, either version 3 of the License, or (at your option)
+# any later version.
+#
+# This program is distributed in the hope that it will be useful, but WITHOUT
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
+# more details.
+#
+# You should have received a copy of the GNU General Public License along with
+# this program. If not, see <http://www.gnu.org/licenses/>.
+
+""""""
+import os
+import unittest
+from mailbox import Maildir
+from tempfile import TemporaryDirectory
+
+from mailman.app.lifecycle import create_list
+from mailman.testing.helpers import specialized_message_from_string as mfs
+
+from mailman_pgp.archivers.local_maildir import LocalMaildirArchiver
+from mailman_pgp.config import config
+from mailman_pgp.database import mm_transaction
+from mailman_pgp.model.list import PGPMailingList
+from mailman_pgp.pgp.mime import MIMEWrapper
+from mailman_pgp.testing.layers import PGPConfigLayer
+from mailman_pgp.testing.pgp import load_key
+
+
+class TestPGPMaildirArchiver(unittest.TestCase):
+ layer = PGPConfigLayer
+
+ def setUp(self):
+ self.msg = mfs("""\
+To: test@example.com
+From: anne@example.com
+Subject: Testing the test list
+Message-ID: <ant>
+Message-ID-Hash: MS6QLWERIJLGCRF44J7USBFDELMNT2BW
+
+Tests are better than no tests
+but the water deserves to be swum.
+""")
+ with mm_transaction():
+ self.mlist = create_list('test@example.com',
+ style_name='pgp-default')
+ self.pgp_list = PGPMailingList.for_list(self.mlist)
+ self.list_key = load_key('ecc_p256.priv.asc')
+ self.pgp_list.key = self.list_key
+
+ def test_no_links(self):
+ self.assertIsNone(LocalMaildirArchiver.list_url(self.mlist))
+ self.assertIsNone(LocalMaildirArchiver.permalink(self.mlist, self.msg))
+
+ def test_no_pgp_list(self):
+ with mm_transaction():
+ ordinary = create_list('ordinary@example.com')
+
+ LocalMaildirArchiver.archive_message(ordinary, self.msg)
+
+ def test_archives(self):
+ with TemporaryDirectory() as maildir_dir:
+ config.set('archiving', 'maildir_dir', maildir_dir)
+ LocalMaildirArchiver.archive_message(self.mlist, self.msg)
+
+ list_dir = os.path.join(maildir_dir, self.mlist.fqdn_listname)
+
+ maildir = Maildir(list_dir)
+ messages = maildir.values()
+ self.assertEqual(len(messages), 1)
+
+ message = messages[0]
+ wrapped = MIMEWrapper(message)
+ self.assertTrue(wrapped.is_encrypted())
+ decrypted = wrapped.decrypt(self.list_key)
+ self.assertTrue(self.msg.as_string(), decrypted.as_string())
diff --git a/src/mailman_pgp/archivers/tests/test_mbox.py b/src/mailman_pgp/archivers/tests/test_mbox.py
new file mode 100644
index 0000000..fde0e50
--- /dev/null
+++ b/src/mailman_pgp/archivers/tests/test_mbox.py
@@ -0,0 +1,82 @@
+# Copyright (C) 2017 Jan Jancar
+#
+# This file is a part of the Mailman PGP plugin.
+#
+# This program is free software; you can redistribute it and/or modify it under
+# the terms of the GNU General Public License as published by the Free
+# Software Foundation, either version 3 of the License, or (at your option)
+# any later version.
+#
+# This program is distributed in the hope that it will be useful, but WITHOUT
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
+# more details.
+#
+# You should have received a copy of the GNU General Public License along with
+# this program. If not, see <http://www.gnu.org/licenses/>.
+
+""""""
+import os
+import unittest
+from mailbox import mbox
+from tempfile import TemporaryDirectory
+
+from mailman.app.lifecycle import create_list
+from mailman.testing.helpers import specialized_message_from_string as mfs
+
+from mailman_pgp.archivers.local_mbox import LocalMailboxArchiver
+from mailman_pgp.config import config
+from mailman_pgp.database import mm_transaction
+from mailman_pgp.model.list import PGPMailingList
+from mailman_pgp.pgp.mime import MIMEWrapper
+from mailman_pgp.testing.layers import PGPConfigLayer
+from mailman_pgp.testing.pgp import load_key
+
+
+class TestPGPMboxArchiver(unittest.TestCase):
+ layer = PGPConfigLayer
+
+ def setUp(self):
+ self.msg = mfs("""\
+To: test@example.com
+From: anne@example.com
+Subject: Testing the test list
+Message-ID: <ant>
+Message-ID-Hash: MS6QLWERIJLGCRF44J7USBFDELMNT2BW
+
+Tests are better than no tests
+but the water deserves to be swum.
+""")
+ with mm_transaction():
+ self.mlist = create_list('test@example.com',
+ style_name='pgp-default')
+ self.pgp_list = PGPMailingList.for_list(self.mlist)
+ self.list_key = load_key('ecc_p256.priv.asc')
+ self.pgp_list.key = self.list_key
+
+ def test_no_links(self):
+ self.assertIsNone(LocalMailboxArchiver.list_url(self.mlist))
+ self.assertIsNone(LocalMailboxArchiver.permalink(self.mlist, self.msg))
+
+ def test_no_pgp_list(self):
+ with mm_transaction():
+ ordinary = create_list('ordinary@example.com')
+
+ LocalMailboxArchiver.archive_message(ordinary, self.msg)
+
+ def test_archives(self):
+ with TemporaryDirectory() as mailbox_dir:
+ config.set('archiving', 'mailbox_dir', mailbox_dir)
+ LocalMailboxArchiver.archive_message(self.mlist, self.msg)
+
+ list_dir = os.path.join(mailbox_dir, self.mlist.fqdn_listname)
+
+ mailbox = mbox(list_dir)
+ messages = mailbox.values()
+ self.assertEqual(len(messages), 1)
+
+ message = messages[0]
+ wrapped = MIMEWrapper(message)
+ self.assertTrue(wrapped.is_encrypted())
+ decrypted = wrapped.decrypt(self.list_key)
+ self.assertTrue(self.msg.as_string(), decrypted.as_string())