mirror of
				https://github.com/zulip/zulip.git
				synced 2025-11-03 21:43:21 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			219 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			219 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
from argparse import ArgumentParser
 | 
						|
from collections.abc import Callable
 | 
						|
from typing import Any
 | 
						|
 | 
						|
import orjson
 | 
						|
from django.conf import settings
 | 
						|
from django.db.models import Q, QuerySet
 | 
						|
from typing_extensions import override
 | 
						|
 | 
						|
from confirmation.models import one_click_unsubscribe_link
 | 
						|
from zerver.lib.management import ZulipBaseCommand
 | 
						|
from zerver.lib.send_email import send_custom_email, send_custom_server_email
 | 
						|
from zerver.models import Realm, UserProfile
 | 
						|
 | 
						|
if settings.ZILENCER_ENABLED:
 | 
						|
    from zilencer.models import RemoteZulipServer
 | 
						|
 | 
						|
 | 
						|
class Command(ZulipBaseCommand):
 | 
						|
    help = """
 | 
						|
    Send a custom email with Zulip branding to the specified users.
 | 
						|
 | 
						|
    Useful to send a notice to all users of a realm or server.
 | 
						|
 | 
						|
    The From and Subject headers can be provided in the body of the Markdown
 | 
						|
    document used to generate the email, or on the command line."""
 | 
						|
 | 
						|
    @override
 | 
						|
    def add_arguments(self, parser: ArgumentParser) -> None:
 | 
						|
        targets = parser.add_mutually_exclusive_group(required=True)
 | 
						|
        targets.add_argument(
 | 
						|
            "--entire-server", action="store_true", help="Send to every user on the server."
 | 
						|
        )
 | 
						|
        targets.add_argument(
 | 
						|
            "--marketing",
 | 
						|
            action="store_true",
 | 
						|
            help="Send to active users and realm owners with the enable_marketing_emails setting enabled.",
 | 
						|
        )
 | 
						|
        targets.add_argument(
 | 
						|
            "--remote-servers",
 | 
						|
            action="store_true",
 | 
						|
            help="Send to registered contact email addresses for remote Zulip servers.",
 | 
						|
        )
 | 
						|
        targets.add_argument(
 | 
						|
            "--all-sponsored-org-admins",
 | 
						|
            action="store_true",
 | 
						|
            help="Send to all organization administrators of sponsored organizations.",
 | 
						|
        )
 | 
						|
        self.add_user_list_args(
 | 
						|
            targets,
 | 
						|
            help="Email addresses of user(s) to send emails to.",
 | 
						|
            all_users_help="Send to every user on the realm.",
 | 
						|
        )
 | 
						|
        # Realm is only required for --users and --all-users, so it is
 | 
						|
        # not mutually exclusive with the rest of the above.
 | 
						|
        self.add_realm_args(parser)
 | 
						|
 | 
						|
        # This is an additional filter on the above.  It ideally would
 | 
						|
        # be in the mutually-exclusive set, but we would like to reuse
 | 
						|
        # it with --remote-servers
 | 
						|
        parser.add_argument(
 | 
						|
            "--json-file",
 | 
						|
            help="Load the JSON file, and send to the users whose ids are the keys in that dict; "
 | 
						|
            "the context for each email will be extended by each value in the dict.",
 | 
						|
        )
 | 
						|
 | 
						|
        # This is an additional filter which is composed with the above
 | 
						|
        parser.add_argument(
 | 
						|
            "--admins-only",
 | 
						|
            help="Filter recipients selected via other options to only organization administrators",
 | 
						|
            action="store_true",
 | 
						|
        )
 | 
						|
 | 
						|
        parser.add_argument(
 | 
						|
            "--markdown-template-path",
 | 
						|
            "--path",
 | 
						|
            required=True,
 | 
						|
            help="Path to a Markdown-format body for the email.",
 | 
						|
        )
 | 
						|
        parser.add_argument(
 | 
						|
            "--subject",
 | 
						|
            help="Subject for the email. It can be declared in Markdown file in headers",
 | 
						|
        )
 | 
						|
        parser.add_argument(
 | 
						|
            "--from-name",
 | 
						|
            help="From line for the email. It can be declared in Markdown file in headers",
 | 
						|
        )
 | 
						|
        parser.add_argument(
 | 
						|
            "--from-address",
 | 
						|
            help="From email address",
 | 
						|
        )
 | 
						|
        parser.add_argument("--reply-to", help="Optional reply-to line for the email")
 | 
						|
 | 
						|
        parser.add_argument(
 | 
						|
            "--dry-run",
 | 
						|
            action="store_true",
 | 
						|
            help="Prints emails of the recipients and text of the email.",
 | 
						|
        )
 | 
						|
 | 
						|
    @override
 | 
						|
    def handle(
 | 
						|
        self, *args: Any, dry_run: bool = False, admins_only: bool = False, **options: str
 | 
						|
    ) -> None:
 | 
						|
        users: QuerySet[UserProfile] = UserProfile.objects.none()
 | 
						|
        add_context: Callable[[dict[str, object], UserProfile], None] | None = None
 | 
						|
        distinct_email = False
 | 
						|
 | 
						|
        if options["remote_servers"]:
 | 
						|
            servers = RemoteZulipServer.objects.filter(deactivated=False)
 | 
						|
            add_server_context = None
 | 
						|
            if options["json_file"]:
 | 
						|
                with open(options["json_file"]) as f:
 | 
						|
                    server_data: dict[str, dict[str, object]] = orjson.loads(f.read())
 | 
						|
                servers = RemoteZulipServer.objects.filter(
 | 
						|
                    id__in=[int(server_id) for server_id in server_data]
 | 
						|
                )
 | 
						|
 | 
						|
                def add_server_context_from_dict(
 | 
						|
                    context: dict[str, object], server: RemoteZulipServer
 | 
						|
                ) -> None:
 | 
						|
                    context.update(server_data.get(str(server.id), {}))
 | 
						|
 | 
						|
                add_server_context = add_server_context_from_dict
 | 
						|
 | 
						|
            send_custom_server_email(
 | 
						|
                servers,
 | 
						|
                dry_run=dry_run,
 | 
						|
                options=options,
 | 
						|
                add_context=add_server_context,
 | 
						|
            )
 | 
						|
            if dry_run:
 | 
						|
                print("Would send the above email to:")
 | 
						|
                for server in servers:
 | 
						|
                    print(f"  {server.contact_email} ({server.hostname})")
 | 
						|
            return
 | 
						|
 | 
						|
        if options["entire_server"]:
 | 
						|
            users = UserProfile.objects.filter(
 | 
						|
                is_active=True, is_bot=False, is_mirror_dummy=False, realm__deactivated=False
 | 
						|
            )
 | 
						|
        elif options["marketing"]:
 | 
						|
            # Marketing email sent at most once to each email address
 | 
						|
            # for users who are recently active (!long_term_idle)
 | 
						|
            # users of the product, or who are admins/owners.
 | 
						|
            users = UserProfile.objects.filter(
 | 
						|
                is_active=True,
 | 
						|
                is_bot=False,
 | 
						|
                is_mirror_dummy=False,
 | 
						|
                realm__deactivated=False,
 | 
						|
                enable_marketing_emails=True,
 | 
						|
            ).filter(
 | 
						|
                Q(long_term_idle=False)
 | 
						|
                | Q(
 | 
						|
                    role__in=[
 | 
						|
                        UserProfile.ROLE_REALM_OWNER,
 | 
						|
                        UserProfile.ROLE_REALM_ADMINISTRATOR,
 | 
						|
                    ]
 | 
						|
                )
 | 
						|
            )
 | 
						|
            distinct_email = True
 | 
						|
 | 
						|
            def add_marketing_unsubscribe(context: dict[str, object], user: UserProfile) -> None:
 | 
						|
                context["unsubscribe_link"] = one_click_unsubscribe_link(user, "marketing")
 | 
						|
 | 
						|
            add_context = add_marketing_unsubscribe
 | 
						|
 | 
						|
        elif options["all_sponsored_org_admins"]:
 | 
						|
            # Sends at most one copy to each email address, even if it
 | 
						|
            # is an administrator in several organizations.
 | 
						|
            sponsored_realms = Realm.objects.filter(
 | 
						|
                plan_type=Realm.PLAN_TYPE_STANDARD_FREE, deactivated=False
 | 
						|
            )
 | 
						|
            admin_roles = [UserProfile.ROLE_REALM_ADMINISTRATOR, UserProfile.ROLE_REALM_OWNER]
 | 
						|
            users = UserProfile.objects.filter(
 | 
						|
                is_active=True,
 | 
						|
                is_bot=False,
 | 
						|
                is_mirror_dummy=False,
 | 
						|
                role__in=admin_roles,
 | 
						|
                realm__deactivated=False,
 | 
						|
                realm__in=sponsored_realms,
 | 
						|
            )
 | 
						|
            distinct_email = True
 | 
						|
        else:
 | 
						|
            realm = self.get_realm(options)
 | 
						|
            users = self.get_users(options, realm, is_bot=False)
 | 
						|
 | 
						|
        if options["json_file"]:
 | 
						|
            with open(options["json_file"]) as f:
 | 
						|
                user_data: dict[str, dict[str, object]] = orjson.loads(f.read())
 | 
						|
            users = users.filter(id__in=[int(user_id) for user_id in user_data])
 | 
						|
 | 
						|
            def add_context_from_dict(context: dict[str, object], user: UserProfile) -> None:
 | 
						|
                context.update(user_data.get(str(user.id), {}))
 | 
						|
 | 
						|
            add_context = add_context_from_dict
 | 
						|
 | 
						|
        if admins_only:
 | 
						|
            users = users.filter(
 | 
						|
                role__in=[UserProfile.ROLE_REALM_ADMINISTRATOR, UserProfile.ROLE_REALM_OWNER]
 | 
						|
            )
 | 
						|
 | 
						|
        # Only email users who've agreed to the terms of service.
 | 
						|
        if settings.TERMS_OF_SERVICE_VERSION is not None:
 | 
						|
            users = users.exclude(
 | 
						|
                Q(tos_version=None) | Q(tos_version=UserProfile.TOS_VERSION_BEFORE_FIRST_LOGIN)
 | 
						|
            )
 | 
						|
        users = send_custom_email(
 | 
						|
            users,
 | 
						|
            dry_run=dry_run,
 | 
						|
            options=options,
 | 
						|
            add_context=add_context,
 | 
						|
            distinct_email=distinct_email,
 | 
						|
        )
 | 
						|
 | 
						|
        if dry_run:
 | 
						|
            print("Would send the above email to:")
 | 
						|
            for user in users:
 | 
						|
                print(f"  {user.delivery_email} ({user.realm.string_id})")
 |