Compare commits

..

No commits in common. "848c98c456ba5c7df703d67f4468818ef55edebb" and "d8284f4475d9a1658a451ed3435c89d2b0f91d4e" have entirely different histories.

2 changed files with 70 additions and 164 deletions

View File

@ -1,149 +1,90 @@
#!/usr/bin/env python3 #!/usr/bin/env python3
""" """
Update Mailman 2 lists via a json API of the form: Update Mailman 2 lists via a json API of the form {"LIST": ["ADDRESS", ...]}
{
"LIST": {
"real_name": "REAL_NAME",
"moderator": ["ADDRESS", ...],
"subject_prefix": "PREFIX",
"reply_to_address": "REPLY_TO_ADDRESS",
"members": ["ADDRESS", ...]
},
...
}
""" """
import argparse import argparse
from dataclasses import dataclass
import os import os
from pathlib import Path from pathlib import Path
import secrets
import string
import subprocess import subprocess
import tempfile import tempfile
import requests import requests
PASSWORD_CHARS = string.ascii_letters + string.digits + string.punctuation
PASSWORD_LEN = 18
def config_list(mailman_bin: Path, mailing_list: str):
config_changes = """
# TODO: set real_name, moderator, subject prefix, and reply-to address
from_is_list = 1
anonymous_list = 1
first_strip_reply_to = 1
reply_goes_to_list = 2
@dataclass
class ListManager:
mailman_bin: Path
list_name: str
dry_run: bool
def _call_script(self, script: str, args: list[str], **kwargs):
output = subprocess.run(
[self.mailman_bin / script, *args],
encoding="ascii",
capture_output=True,
check=True,
**kwargs,
)
for line in output.stdout.splitlines():
print(f"[{script} {self.list_name}] {line}")
def newlist(self, urlhost: str, emailhost: str, admin: str):
password = "".join(secrets.choice(PASSWORD_CHARS) for i in range(PASSWORD_LEN))
self._call_script(
"newlist",
[
"--quiet",
f"--urlhost={urlhost}",
f"--emailhost={emailhost}",
self.list_name,
admin,
password,
],
)
def config_list(
self,
emailhost: str,
real_name: str,
moderator: list[str],
subject_prefix: str,
reply_to_address: str,
):
config_changes = "\n".join(
[
# have to modify mlist directly to allow removing suffix
"mlist.real_name = " + repr(real_name),
"host_name = " + repr(emailhost),
"moderator = " + repr(moderator),
"subject_prefix = " + repr(subject_prefix),
"from_is_list = 1",
"anonymous_list = 1",
"first_strip_reply_to = 1",
"reply_to_address = " + repr(reply_to_address),
# Use explicit address for Reply-To
"reply_goes_to_list = 2",
# quiet member management # quiet member management
"send_reminders = 0", send_reminders = 0
"send_welcome_msg = 0", send_welcome_msg = 0
"send_goodbye_msg = 0", send_goodbye_msg = 0
# ConcealSubscription | DontReceiveDuplicates
"new_member_options = 272", new_member_options = 272
"advertised = 0", advertised = 0
"subscribe_policy = 3", # require approval to join private_roster = 2 # only admins can view the roster
"unsubscribe_policy = 1", # require approval to unsubscribe default_member_moderation = 1
"private_roster = 2", # only admins can view the roster
"default_member_moderation = 1", generic_nonmember_action = 3 # discard non-member emails
"generic_nonmember_action = 3", # discard non-member emails forward_auto_discards = 0 # don't notify admin about discards
"forward_auto_discards = 0", # don't notify admin about discards """
"archive_private = 1",
]
)
with tempfile.NamedTemporaryFile("w", suffix=".py") as config_file: with tempfile.NamedTemporaryFile("w", suffix=".py") as config_file:
config_file.write(config_changes) config_file.write(config_changes)
config_file.flush() config_file.flush()
output = subprocess.run(
args = [ [
mailman_bin / "config_list",
"--inputfile", "--inputfile",
config_file.name, config_file.name,
self.list_name, mailing_list,
] ],
if self.dry_run: encoding="ascii",
args.append("--checkonly") capture_output=True,
check=True,
)
for line in output.stdout.splitlines():
print(f"[Configuring {mailing_list}] {line}")
self._call_script("config_list", args)
def sync_members(self, members: list[str]): def sync_members(
args = [ mailman_bin: Path, mailing_list: str, members: list[str], dry_run: bool
):
command = [
mailman_bin / "sync_members",
"--welcome-msg=no", "--welcome-msg=no",
"--goodbye-msg=no", "--goodbye-msg=no",
"--notifyadmin=no", "--notifyadmin=no",
"--file", "--file",
"-", "-",
self.list_name, mailing_list,
] ]
if self.dry_run: if dry_run:
args.append("--no-change") command.append("--no-change")
members_data = "\n".join(members) members_data = "\n".join(members)
self._call_script("sync_members", args, input=members_data) output = subprocess.run(
command,
input=members_data,
encoding="ascii",
capture_output=True,
check=True,
)
for line in output.stdout.splitlines():
print(f"[Syncing {mailing_list}] {line}")
def main( def main(mailman_bin: Path, api: str, api_auth: str, list_suffix: str, dry_run: bool):
mailman_bin: Path,
api: str,
api_auth: str,
list_suffix: str,
dry_run: bool,
urlhost: str,
emailhost: str,
admin: str,
):
r = requests.get(api, headers={"Authorization": api_auth}) r = requests.get(api, headers={"Authorization": api_auth})
if not r.ok: if not r.ok:
print(f"Failed to get mailing list data from api: {r.status_code} {r.text}") print(f"Failed to get mailing list data from api: {r.status_code} {r.text}")
return return
expected_lists = r.json()
existing_lists = subprocess.run( existing_lists = subprocess.run(
[mailman_bin / "list_lists", "-b"], [mailman_bin / "list_lists", "-b"],
@ -151,28 +92,18 @@ def main(
capture_output=True, capture_output=True,
check=True, check=True,
).stdout.split("\n") ).stdout.split("\n")
for name, props in expected_lists.items(): certification_lists = r.json()
for name, members in certification_lists.items():
list_name = name + list_suffix list_name = name + list_suffix
list_manager = ListManager(mailman_bin, list_name, dry_run) if list_name in existing_lists:
if list_name.lower() not in existing_lists:
if dry_run:
print(f"Skipping non-existing list {list_name} in dry run mode")
continue
else:
list_manager.newlist(urlhost, emailhost, admin)
print(f"Configuring/syncing {list_name}...") print(f"Configuring/syncing {list_name}...")
list_manager.config_list( config_list(mailman_bin, list_name)
emailhost, sync_members(mailman_bin, list_name, members, dry_run)
props["real_name"], else:
props["moderator"], print(f"Skipping {list_name}, as it does not exist in Mailman")
props["subject_prefix"],
props["reply_to_address"],
)
list_manager.sync_members(props["members"])
def parse_arguments(): if __name__ == "__main__":
argp = argparse.ArgumentParser(description=__doc__) argp = argparse.ArgumentParser(description=__doc__)
argp.add_argument( argp.add_argument(
"--bin", "--bin",
@ -181,29 +112,14 @@ def parse_arguments():
help="Path to Mailman site admin scripts (default %(default)s)", help="Path to Mailman site admin scripts (default %(default)s)",
) )
argp.add_argument("--api", required=True, help="API endpoint to retrieve JSON from") argp.add_argument("--api", required=True, help="API endpoint to retrieve JSON from")
argp.add_argument("--list-suffix", help="Suffix for mailing lists", default="") argp.add_argument("--list-suffix", help="Suffix for mailing lists")
argp.add_argument(
"--urlhost", help="Urlhost to use when creating new lists", required=True
)
argp.add_argument(
"--emailhost", help="Emailhost to use when creating new lists", required=True
)
argp.add_argument(
"--admin",
help="Admin email address to use when creating new lists",
required=True,
)
argp.add_argument( argp.add_argument(
"-n", "-n",
"--dry-run", "--dry-run",
action="store_true", action="store_true",
help="Don't make changes, just print what would happen", help="Don't make changes, just print what would happen",
) )
return argp.parse_args() args = argp.parse_args()
if __name__ == "__main__":
args = parse_arguments()
if "API_AUTH" in os.environ: if "API_AUTH" in os.environ:
api_auth = os.environ.get("API_AUTH") api_auth = os.environ.get("API_AUTH")
@ -212,16 +128,7 @@ if __name__ == "__main__":
exit(-1) exit(-1)
try: try:
main( main(args.bin, args.api, api_auth, args.list_suffix, args.dry_run)
args.bin,
args.api,
api_auth,
args.list_suffix,
args.dry_run,
args.urlhost,
args.emailhost,
args.admin,
)
except subprocess.CalledProcessError as e: except subprocess.CalledProcessError as e:
print(e.stderr) print(e.stderr)
raise raise

View File

@ -1 +0,0 @@
[tool.black]