# Copyright (C) 2009-2014 by the Free Software Foundation, Inc. # # This file is part of GNU Mailman. # # GNU Mailman 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. # # GNU Mailman 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 # GNU Mailman. If not, see . """The 'members' subcommand.""" from __future__ import absolute_import, print_function, unicode_literals __metaclass__ = type __all__ = [ 'Members', ] import sys import codecs from email.utils import formataddr, parseaddr from operator import attrgetter from passlib.utils import generate_password as generate from zope.component import getUtility from zope.interface import implementer from mailman.app.membership import add_member from mailman.config import config from mailman.core.i18n import _ from mailman.database.transaction import transactional from mailman.interfaces.command import ICLISubCommand from mailman.interfaces.listmanager import IListManager from mailman.interfaces.member import ( AlreadySubscribedError, DeliveryMode, DeliveryStatus) @implementer(ICLISubCommand) class Members: """Manage list memberships. With no arguments, list all members.""" name = 'members' def add(self, parser, command_parser): """See `ICLISubCommand`.""" self.parser = parser command_parser.add_argument( '-a', '--add', dest='input_filename', metavar='FILENAME', help=_("""\ Add all member addresses in FILENAME. FILENAME can be '-' to indicate standard input. Blank lines and lines That start with a '#' are ignored. Without this option, this command displays mailing list members.""")) command_parser.add_argument( '-o', '--output', dest='output_filename', metavar='FILENAME', help=_("""Display output to FILENAME instead of stdout. FILENAME can be '-' to indicate standard output.""")) command_parser.add_argument( '-r', '--regular', default=None, action='store_true', help=_('Display only regular delivery members.')) command_parser.add_argument( '-d', '--digest', default=None, metavar='KIND', # BAW 2010-01-23 summary digests are not really supported yet. choices=('any', 'plaintext', 'mime'), help=_("""Display only digest members of KIND. 'any' means any digest type, 'plaintext' means only plain text (RFC 1153) type digests, 'mime' means MIME type digests.""")) command_parser.add_argument( '-n', '--nomail', default=None, metavar='WHY', choices=('enabled', 'any', 'unknown' 'byadmin', 'byuser', 'bybounces'), help=_("""Display only members with a given delivery status. 'enabled' means all members whose delivery is enabled, 'any' means members whose delivery is disabled for any reason, 'byuser' means that the member disabled their own delivery, 'bybounces' means that delivery was disabled by the automated bounce processor, 'byadmin' means delivery was disabled by the list administrator or moderator, and 'unknown' means that delivery was disabled for unknown (legacy) reasons.""")) # Required positional argument. command_parser.add_argument( 'listname', metavar='LISTNAME', nargs=1, help=_("""\ The 'fully qualified list name', i.e. the posting address of the mailing list. It must be a valid email address and the domain must be registered with Mailman. List names are forced to lower case.""")) def process(self, args): """See `ICLISubCommand`.""" assert len(args.listname) == 1, 'Missing mailing list name' fqdn_listname = args.listname[0] mlist = getUtility(IListManager).get(fqdn_listname) if mlist is None: self.parser.error(_('No such list: $fqdn_listname')) if args.input_filename is None: self.display_members(mlist, args) else: self.add_members(mlist, args) def display_members(self, mlist, args): """Display the members of a mailing list. :param mlist: The mailing list to operate on. :type mlist: `IMailingList` :param args: The command line arguments. :type args: `argparse.Namespace` """ if args.output_filename == '-' or args.output_filename is None: fp = sys.stdout else: fp = codecs.open(args.output_filename, 'w', 'utf-8') if args.digest == 'any': digest_types = [DeliveryMode.plaintext_digests, DeliveryMode.mime_digests, DeliveryMode.summary_digests] elif args.digest is not None: digest_types = [DeliveryMode[args.digest + '_digests']] else: # Don't filter on digest type. pass if args.nomail is None: # Don't filter on delivery status. pass elif args.nomail == 'byadmin': status_types = [DeliveryStatus.by_moderator] elif args.nomail.startswith('by'): status_types = [DeliveryStatus['by_' + args.nomail[2:]]] elif args.nomail == 'enabled': status_types = [DeliveryStatus.enabled] elif args.nomail == 'unknown': status_types = [DeliveryStatus.unknown] elif args.nomail == 'any': status_types = [DeliveryStatus.by_user, DeliveryStatus.by_bounces, DeliveryStatus.by_moderator, DeliveryStatus.unknown] else: raise AssertionError('Unknown delivery status: %s' % args.nomail) try: addresses = list(mlist.members.addresses) if len(addresses) == 0: print(mlist.fqdn_listname, 'has no members', file=fp) return for address in sorted(addresses, key=attrgetter('email')): if args.regular: member = mlist.members.get_member(address.email) if member.delivery_mode != DeliveryMode.regular: continue if args.digest is not None: member = mlist.members.get_member(address.email) if member.delivery_mode not in digest_types: continue if args.nomail is not None: member = mlist.members.get_member(address.email) if member.delivery_status not in status_types: continue print( formataddr((address.display_name, address.original_email)), file=fp) finally: if fp is not sys.stdout: fp.close() @transactional def add_members(self, mlist, args): """Add the members in a file to a mailing list. :param mlist: The mailing list to operate on. :type mlist: `IMailingList` :param args: The command line arguments. :type args: `argparse.Namespace` """ if args.input_filename == '-': fp = sys.stdin else: fp = codecs.open(args.input_filename, 'r', 'utf-8') try: for line in fp: # Ignore blank lines and lines that start with a '#'. if line.startswith('#') or len(line.strip()) == 0: continue # Parse the line and ensure that the values are unicodes. display_name, email = parseaddr(line) # Give the user a default, user-friendly password. password = generate(int(config.passwords.password_length)) try: add_member(mlist, email, display_name, password, DeliveryMode.regular, mlist.preferred_language.code) except AlreadySubscribedError: # It's okay if the address is already subscribed, just # print a warning and continue. print('Already subscribed (skipping):', email, display_name) finally: if fp is not sys.stdin: fp.close()