mirror of
https://github.com/zulip/zulip.git
synced 2025-10-23 04:52:12 +00:00
For get and filter queries of NamedUserGroup, realm_for_sharding field is used instead of realm field, as directly using realm_for_sharding field on NamedUserGroup makes the query faster than using realm present on the base UserGroup table.
294 lines
11 KiB
Python
Executable File
294 lines
11 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
"""Create or update a message thread screenshot using a thread file."""
|
|
|
|
import argparse
|
|
import os
|
|
import subprocess
|
|
import sys
|
|
from datetime import datetime, timezone
|
|
|
|
from django.conf import settings
|
|
from pydantic import BaseModel, ConfigDict
|
|
|
|
SCREENSHOTS_DIR = os.path.abspath(os.path.dirname(__file__))
|
|
TOOLS_DIR = os.path.abspath(os.path.dirname(SCREENSHOTS_DIR))
|
|
ROOT_DIR = os.path.dirname(TOOLS_DIR)
|
|
sys.path.insert(0, ROOT_DIR)
|
|
|
|
# check for the venv
|
|
from tools.lib import sanity_check
|
|
|
|
sanity_check.check_venv(__file__)
|
|
|
|
from scripts.lib.setup_path import setup_path
|
|
|
|
setup_path()
|
|
|
|
os.environ["DJANGO_SETTINGS_MODULE"] = "zproject.settings"
|
|
import django
|
|
|
|
django.setup()
|
|
|
|
import json
|
|
|
|
from tools.lib.test_script import prepare_puppeteer_run
|
|
from zerver.actions.create_user import do_create_user
|
|
from zerver.actions.message_edit import check_update_message
|
|
from zerver.actions.message_flags import do_update_message_flags
|
|
from zerver.actions.message_send import do_send_messages, internal_prep_stream_message
|
|
from zerver.actions.reactions import do_add_reaction
|
|
from zerver.actions.streams import bulk_add_subscriptions, do_change_subscription_property
|
|
from zerver.actions.user_groups import check_add_user_group
|
|
from zerver.actions.user_settings import do_change_avatar_fields
|
|
from zerver.lib.emoji import get_emoji_data
|
|
from zerver.lib.message import SendMessageRequest, access_message
|
|
from zerver.lib.stream_subscription import get_active_subscriptions_for_stream_id
|
|
from zerver.lib.streams import access_stream_by_id, ensure_stream
|
|
from zerver.lib.timestamp import datetime_to_timestamp
|
|
from zerver.lib.upload import upload_avatar_image
|
|
from zerver.lib.url_encoding import topic_narrow_url
|
|
from zerver.models import Message, UserProfile
|
|
from zerver.models.groups import NamedUserGroup
|
|
from zerver.models.realms import get_realm
|
|
from zerver.models.users import get_system_bot, get_user_by_delivery_email
|
|
|
|
realm = get_realm("zulip")
|
|
realm.message_content_edit_limit_seconds = None
|
|
realm.save()
|
|
|
|
DEFAULT_USER = get_user_by_delivery_email("iago@zulip.com", realm)
|
|
NOTIFICATION_BOT = get_system_bot(settings.NOTIFICATION_BOT, realm.id)
|
|
message_thread_ids: list[int] = []
|
|
|
|
USER_AVATARS_MAP = {
|
|
"Ariella Drake": "tools/screenshots/user_avatars/AriellaDrake.png",
|
|
"Elena García": "tools/screenshots/user_avatars/ElenaGarcia.jpg",
|
|
"Kevin Lin": "tools/screenshots/user_avatars/KevinLin.png",
|
|
"Zoe Davis": "tools/screenshots/user_avatars/ZoeDavis.png",
|
|
"Bo Williams": "tools/screenshots/user_avatars/BoWilliams.png",
|
|
"James Williams": "tools/screenshots/user_avatars/JamesWilliams.png",
|
|
"Manvir Singh": "tools/screenshots/user_avatars/ManvirSingh.png",
|
|
"Dal Kim": "tools/screenshots/user_avatars/DalKim.jpg",
|
|
"John Lin": "tools/screenshots/user_avatars/JohnLin.png",
|
|
"Maxy Stert": "tools/screenshots/user_avatars/MaxyStert.jpg",
|
|
}
|
|
|
|
|
|
class MessageThread(BaseModel):
|
|
model_config = ConfigDict(frozen=True)
|
|
|
|
sender: str
|
|
content: str
|
|
starred: bool
|
|
edited: bool
|
|
reactions: dict[str, list[str]]
|
|
date: dict[str, int]
|
|
|
|
|
|
def create_user(full_name: str, avatar_filename: str | None) -> None:
|
|
email = f"{full_name.replace(' ', '').replace('í', 'i')}@zulip.com"
|
|
try:
|
|
user = UserProfile.objects.get(realm=realm, full_name=full_name)
|
|
except UserProfile.DoesNotExist:
|
|
user = do_create_user(email, "password", realm, full_name, acting_user=DEFAULT_USER)
|
|
|
|
if avatar_filename is not None:
|
|
set_avatar(user, avatar_filename)
|
|
|
|
|
|
def set_avatar(user: UserProfile, filename: str) -> None:
|
|
with open(filename, "rb") as f:
|
|
upload_avatar_image(f, user)
|
|
do_change_avatar_fields(user, UserProfile.AVATAR_FROM_USER, acting_user=DEFAULT_USER)
|
|
|
|
|
|
def create_and_subscribe_stream(
|
|
stream_name: str, users: list[str], color: str | None = None, invite_only: bool = False
|
|
) -> None:
|
|
stream = ensure_stream(realm, stream_name, invite_only=invite_only, acting_user=DEFAULT_USER)
|
|
bulk_add_subscriptions(
|
|
realm,
|
|
[stream],
|
|
list(UserProfile.objects.filter(realm=realm, full_name__in=users)),
|
|
acting_user=DEFAULT_USER,
|
|
)
|
|
|
|
(stream, sub) = access_stream_by_id(DEFAULT_USER, stream.id)
|
|
assert sub is not None
|
|
if color is not None:
|
|
do_change_subscription_property(
|
|
DEFAULT_USER, sub, stream, "color", color, acting_user=DEFAULT_USER
|
|
)
|
|
|
|
|
|
def send_stream_messages(
|
|
stream_name: str, topic: str, staged_messages_data: list[MessageThread]
|
|
) -> list[int]:
|
|
staged_messages = [dict(staged_message) for staged_message in staged_messages_data]
|
|
stream = ensure_stream(realm, stream_name, acting_user=DEFAULT_USER)
|
|
subscribers_query = get_active_subscriptions_for_stream_id(
|
|
stream.id, include_deactivated_users=False
|
|
).values_list("user_profile", flat=True)
|
|
|
|
subscribers: dict[str, UserProfile] = {}
|
|
for subscriber_id in subscribers_query:
|
|
subscriber = UserProfile.objects.get(realm=realm, id=subscriber_id)
|
|
subscribers[subscriber.full_name] = subscriber
|
|
|
|
subscribers["Notification Bot"] = NOTIFICATION_BOT
|
|
|
|
messages: list[SendMessageRequest | None] = []
|
|
|
|
for message in staged_messages:
|
|
date_sent = message["date"]
|
|
|
|
message_request = internal_prep_stream_message(
|
|
subscribers[message["sender"]],
|
|
stream,
|
|
topic,
|
|
message["content"],
|
|
forged=True,
|
|
forged_timestamp=datetime_to_timestamp(
|
|
datetime(
|
|
date_sent["year"],
|
|
date_sent["month"],
|
|
date_sent["day"],
|
|
date_sent["hour"],
|
|
date_sent["minute"],
|
|
tzinfo=timezone.utc,
|
|
)
|
|
),
|
|
)
|
|
messages.append(message_request)
|
|
|
|
message_ids = [
|
|
sent_message_result.message_id for sent_message_result in do_send_messages(messages)
|
|
]
|
|
global message_thread_ids
|
|
message_thread_ids += message_ids
|
|
|
|
for message, message_id in zip(staged_messages, message_ids, strict=False):
|
|
if message.get("reactions") is not None:
|
|
reactions = message["reactions"]
|
|
for reaction, user_names in reactions.items():
|
|
users = [subscribers[user_name] for user_name in user_names]
|
|
add_message_reactions(message_id, reaction, users)
|
|
|
|
if message.get("starred"):
|
|
do_update_message_flags(DEFAULT_USER, "add", "starred", [message_id])
|
|
|
|
if message.get("edited"):
|
|
sender = UserProfile.objects.get(realm=realm, full_name=message["sender"])
|
|
updated_content = message["content"] + " "
|
|
check_update_message(sender, message_id, content=updated_content)
|
|
|
|
return message_ids
|
|
|
|
|
|
def add_message_reactions(message_id: int, emoji: str, users: list[UserProfile]) -> None:
|
|
preview_message = access_message(
|
|
user_profile=DEFAULT_USER, message_id=message_id, is_modifying_message=False
|
|
)
|
|
emoji_data = get_emoji_data(realm.id, emoji)
|
|
for user in users:
|
|
do_add_reaction(
|
|
user, preview_message, emoji, emoji_data.emoji_code, emoji_data.reaction_type
|
|
)
|
|
|
|
|
|
def create_user_group(group_name: str, members: list[str]) -> None:
|
|
member_profiles = [
|
|
UserProfile.objects.get(realm=realm, full_name=member_name) for member_name in members
|
|
]
|
|
member_profiles.append(DEFAULT_USER)
|
|
check_add_user_group(realm, group_name, member_profiles, acting_user=DEFAULT_USER)
|
|
|
|
|
|
def capture_streams_narrow_screenshot(
|
|
image_path: str, stream_name: str, topic: str, unread_msg_id: int
|
|
) -> None:
|
|
stream = ensure_stream(realm, stream_name, acting_user=DEFAULT_USER)
|
|
|
|
narrow_uri = topic_narrow_url(realm=realm, stream=stream, topic_name=topic)
|
|
narrow = f"{stream_name}/{topic}"
|
|
screenshot_script = os.path.join(SCREENSHOTS_DIR, "thread-screenshot.ts")
|
|
subprocess.check_call(
|
|
["node", screenshot_script, narrow_uri, narrow, str(unread_msg_id), image_path, realm.url]
|
|
)
|
|
|
|
|
|
DESCRIPTION = """
|
|
Generate screenshots of messages for corporate pages.
|
|
|
|
This script takes a json file with conversation details, and runs
|
|
tools/thread-screenshot.ts to take cropped screenshots of the
|
|
generated messages with puppeteer.
|
|
|
|
Make sure you have the dev environment up and running in a separate
|
|
terminal window when you run the script.
|
|
|
|
Note that you will often want to update the content of existing
|
|
json files (e.g., tools/screenshots/for-events.json) when you are
|
|
taking updated screenshots. For example, updating any dates for
|
|
the current year is recommended.
|
|
|
|
Also, note that you may need to adjust the viewport width in the
|
|
puppeteer code and/or update the channel or topic names in the json
|
|
content so that message header bars don't have truncated (...)
|
|
channel or topic names due to web app UI changes.
|
|
"""
|
|
|
|
parser = argparse.ArgumentParser(
|
|
description=DESCRIPTION, formatter_class=argparse.RawTextHelpFormatter
|
|
)
|
|
group = parser.add_mutually_exclusive_group(required=True)
|
|
|
|
group.add_argument(
|
|
"--thread",
|
|
nargs="+",
|
|
type=str,
|
|
help="Path of the file where the thread for screenshot is present",
|
|
)
|
|
fixture_group = parser.add_argument_group("thread")
|
|
|
|
options = parser.parse_args()
|
|
prepare_puppeteer_run()
|
|
|
|
try:
|
|
realm = get_realm("zulip")
|
|
with open(options.thread[0]) as f:
|
|
threads = json.load(f)
|
|
for thread in threads:
|
|
for user in thread["users"]:
|
|
user_avatar = USER_AVATARS_MAP.get(user)
|
|
create_user(user, user_avatar)
|
|
|
|
if thread["recipient_type"] == "channel":
|
|
users = list(thread["users"].keys())
|
|
users.append("Iago")
|
|
|
|
if thread.get("user_groups") is not None:
|
|
for user_group in thread.get("user_groups"):
|
|
user_grp = NamedUserGroup.objects.filter(
|
|
name=user_group["group_name"], realm_for_sharding=realm
|
|
).first()
|
|
if user_grp is not None:
|
|
user_grp.delete()
|
|
|
|
create_user_group(user_group["group_name"], user_group["members"])
|
|
|
|
invite_only = (
|
|
False if thread.get("invite_only") is None else thread.get("invite_only")
|
|
)
|
|
create_and_subscribe_stream(thread["channel"], users, thread["color"], invite_only)
|
|
message_ids = send_stream_messages(
|
|
thread["channel"], thread["topic"], thread["messages"]
|
|
)
|
|
capture_streams_narrow_screenshot(
|
|
thread["screenshot"], thread["channel"], thread["topic"], min(message_ids)
|
|
)
|
|
|
|
|
|
finally:
|
|
Message.objects.filter(id__in=message_thread_ids).delete()
|