mirror of
				https://github.com/zulip/zulip.git
				synced 2025-11-03 21:43:21 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			392 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			392 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
# Zulip's OpenAPI-based API documentation system is documented at
 | 
						|
#   https://zulip.readthedocs.io/en/latest/documentation/api.html
 | 
						|
#
 | 
						|
# This file contains helper functions for generating cURL examples
 | 
						|
# based on Zulip's OpenAPI definitions, as well as test setup and
 | 
						|
# fetching of appropriate parameter values to use when running the
 | 
						|
# cURL examples as part of the tools/test-api test suite.
 | 
						|
from collections.abc import Callable
 | 
						|
from functools import wraps
 | 
						|
from typing import Any
 | 
						|
 | 
						|
from django.utils.timezone import now as timezone_now
 | 
						|
 | 
						|
from zerver.actions.create_user import do_create_user
 | 
						|
from zerver.actions.presence import update_user_presence
 | 
						|
from zerver.actions.reactions import do_add_reaction
 | 
						|
from zerver.actions.realm_linkifiers import do_add_linkifier
 | 
						|
from zerver.actions.realm_playgrounds import check_add_realm_playground
 | 
						|
from zerver.lib.events import do_events_register
 | 
						|
from zerver.lib.initial_password import initial_password
 | 
						|
from zerver.lib.test_classes import ZulipTestCase
 | 
						|
from zerver.lib.upload import upload_message_attachment
 | 
						|
from zerver.models import Client, Message, NamedUserGroup, UserPresence
 | 
						|
from zerver.models.realms import get_realm
 | 
						|
from zerver.models.users import UserProfile, get_user
 | 
						|
from zerver.openapi.openapi import Parameter
 | 
						|
 | 
						|
GENERATOR_FUNCTIONS: dict[str, Callable[[], dict[str, object]]] = {}
 | 
						|
REGISTERED_GENERATOR_FUNCTIONS: set[str] = set()
 | 
						|
CALLED_GENERATOR_FUNCTIONS: set[str] = set()
 | 
						|
# This is a List rather than just a string in order to make it easier
 | 
						|
# to write to it from another module.
 | 
						|
AUTHENTICATION_LINE: list[str] = [""]
 | 
						|
 | 
						|
helpers = ZulipTestCase()
 | 
						|
 | 
						|
 | 
						|
def openapi_param_value_generator(
 | 
						|
    endpoints: list[str],
 | 
						|
) -> Callable[[Callable[[], dict[str, object]]], Callable[[], dict[str, object]]]:
 | 
						|
    """This decorator is used to register OpenAPI param value generator functions
 | 
						|
    with endpoints. Example usage:
 | 
						|
 | 
						|
    @openapi_param_value_generator(["/messages/render:post"])
 | 
						|
    def ...
 | 
						|
    """
 | 
						|
 | 
						|
    def wrapper(generator_func: Callable[[], dict[str, object]]) -> Callable[[], dict[str, object]]:
 | 
						|
        @wraps(generator_func)
 | 
						|
        def _record_calls_wrapper() -> dict[str, object]:
 | 
						|
            CALLED_GENERATOR_FUNCTIONS.add(generator_func.__name__)
 | 
						|
            return generator_func()
 | 
						|
 | 
						|
        REGISTERED_GENERATOR_FUNCTIONS.add(generator_func.__name__)
 | 
						|
        for endpoint in endpoints:
 | 
						|
            GENERATOR_FUNCTIONS[endpoint] = _record_calls_wrapper
 | 
						|
 | 
						|
        return _record_calls_wrapper
 | 
						|
 | 
						|
    return wrapper
 | 
						|
 | 
						|
 | 
						|
def assert_all_helper_functions_called() -> None:
 | 
						|
    """Throws an exception if any registered helpers were not called by tests"""
 | 
						|
    if REGISTERED_GENERATOR_FUNCTIONS == CALLED_GENERATOR_FUNCTIONS:
 | 
						|
        return
 | 
						|
 | 
						|
    uncalled_functions = str(REGISTERED_GENERATOR_FUNCTIONS - CALLED_GENERATOR_FUNCTIONS)
 | 
						|
 | 
						|
    raise Exception(f"Registered curl API generators were not called: {uncalled_functions}")
 | 
						|
 | 
						|
 | 
						|
def patch_openapi_example_values(
 | 
						|
    entry: str,
 | 
						|
    parameters: list[Parameter],
 | 
						|
    request_body: dict[str, Any] | None = None,
 | 
						|
) -> tuple[list[Parameter], dict[str, object] | None]:
 | 
						|
    if entry not in GENERATOR_FUNCTIONS:
 | 
						|
        return parameters, request_body
 | 
						|
    func = GENERATOR_FUNCTIONS[entry]
 | 
						|
    realm_example_values: dict[str, object] = func()
 | 
						|
 | 
						|
    for parameter in parameters:
 | 
						|
        if parameter.name in realm_example_values:
 | 
						|
            parameter.example = realm_example_values[parameter.name]
 | 
						|
 | 
						|
    if request_body is not None and "multipart/form-data" in (content := request_body["content"]):
 | 
						|
        properties = content["multipart/form-data"]["schema"]["properties"]
 | 
						|
        for key, property in properties.items():
 | 
						|
            if key in realm_example_values:
 | 
						|
                property["example"] = realm_example_values[key]
 | 
						|
    return parameters, request_body
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/fetch_api_key:post"])
 | 
						|
def fetch_api_key() -> dict[str, object]:
 | 
						|
    email = helpers.example_email("iago")
 | 
						|
    password = initial_password(email)
 | 
						|
 | 
						|
    return {
 | 
						|
        "username": email,
 | 
						|
        "password": password,
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(
 | 
						|
    [
 | 
						|
        "/messages/{message_id}:get",
 | 
						|
        "/messages/{message_id}/history:get",
 | 
						|
        "/messages/{message_id}:patch",
 | 
						|
        "/messages/{message_id}:delete",
 | 
						|
    ]
 | 
						|
)
 | 
						|
def iago_message_id() -> dict[str, object]:
 | 
						|
    iago = helpers.example_user("iago")
 | 
						|
    helpers.subscribe(iago, "Denmark")
 | 
						|
    return {
 | 
						|
        "message_id": helpers.send_stream_message(iago, "Denmark"),
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/messages/{message_id}/reactions:delete"])
 | 
						|
def add_emoji_to_message() -> dict[str, object]:
 | 
						|
    user_profile = helpers.example_user("iago")
 | 
						|
 | 
						|
    # The message ID here is hardcoded based on the corresponding value
 | 
						|
    # for the example message IDs we use in zulip.yaml.
 | 
						|
    message_id = 48
 | 
						|
    emoji_name = "octopus"
 | 
						|
    emoji_code = "1f419"
 | 
						|
    reaction_type = "unicode_emoji"
 | 
						|
 | 
						|
    message = Message.objects.select_related(*Message.DEFAULT_SELECT_RELATED).get(id=message_id)
 | 
						|
    do_add_reaction(user_profile, message, emoji_name, emoji_code, reaction_type)
 | 
						|
 | 
						|
    return {}
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/messages/flags:post"])
 | 
						|
def update_flags_message_ids() -> dict[str, object]:
 | 
						|
    stream_name = "Venice"
 | 
						|
    helpers.subscribe(helpers.example_user("iago"), stream_name)
 | 
						|
 | 
						|
    messages = [
 | 
						|
        helpers.send_stream_message(helpers.example_user("iago"), stream_name) for _ in range(3)
 | 
						|
    ]
 | 
						|
    return {
 | 
						|
        "messages": messages,
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/mark_stream_as_read:post", "/users/me/{stream_id}/topics:get"])
 | 
						|
def get_venice_stream_id() -> dict[str, object]:
 | 
						|
    return {
 | 
						|
        "stream_id": helpers.get_stream_id("Venice"),
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/streams/{stream_id}:patch"])
 | 
						|
def update_stream() -> dict[str, object]:
 | 
						|
    stream = helpers.subscribe(helpers.example_user("iago"), "temp_stream 1")
 | 
						|
    return {
 | 
						|
        "stream_id": stream.id,
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/streams/{stream_id}:delete"])
 | 
						|
def create_temp_stream_and_get_id() -> dict[str, object]:
 | 
						|
    stream = helpers.subscribe(helpers.example_user("iago"), "temp_stream 2")
 | 
						|
    return {
 | 
						|
        "stream_id": stream.id,
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/mark_topic_as_read:post"])
 | 
						|
def get_denmark_stream_id_and_topic() -> dict[str, object]:
 | 
						|
    stream_name = "Denmark"
 | 
						|
    topic_name = "Tivoli Gardens"
 | 
						|
 | 
						|
    helpers.subscribe(helpers.example_user("iago"), stream_name)
 | 
						|
    helpers.send_stream_message(helpers.example_user("hamlet"), stream_name, topic_name=topic_name)
 | 
						|
 | 
						|
    return {
 | 
						|
        "stream_id": helpers.get_stream_id(stream_name),
 | 
						|
        "topic_name": topic_name,
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/users/me/subscriptions/properties:post"])
 | 
						|
def update_subscription_data() -> dict[str, object]:
 | 
						|
    profile = helpers.example_user("iago")
 | 
						|
    helpers.subscribe(profile, "Verona")
 | 
						|
    helpers.subscribe(profile, "social")
 | 
						|
    return {
 | 
						|
        "subscription_data": [
 | 
						|
            {"stream_id": helpers.get_stream_id("Verona"), "property": "pin_to_top", "value": True},
 | 
						|
            {"stream_id": helpers.get_stream_id("social"), "property": "color", "value": "#f00f00"},
 | 
						|
        ],
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/users/me/subscriptions:delete"])
 | 
						|
def delete_subscription_data() -> dict[str, object]:
 | 
						|
    iago = helpers.example_user("iago")
 | 
						|
    zoe = helpers.example_user("ZOE")
 | 
						|
    helpers.subscribe(iago, "Verona")
 | 
						|
    helpers.subscribe(iago, "social")
 | 
						|
    helpers.subscribe(zoe, "Verona")
 | 
						|
    helpers.subscribe(zoe, "social")
 | 
						|
    return {}
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/events:get"])
 | 
						|
def get_events() -> dict[str, object]:
 | 
						|
    profile = helpers.example_user("iago")
 | 
						|
    helpers.subscribe(profile, "Verona")
 | 
						|
    client = Client.objects.create(name="curl-test-client-1")
 | 
						|
    response = do_events_register(
 | 
						|
        profile, profile.realm, client, event_types=["message", "realm_emoji"]
 | 
						|
    )
 | 
						|
    helpers.send_stream_message(helpers.example_user("hamlet"), "Verona")
 | 
						|
    return {
 | 
						|
        "queue_id": response["queue_id"],
 | 
						|
        "last_event_id": response["last_event_id"],
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/events:delete"])
 | 
						|
def delete_event_queue() -> dict[str, object]:
 | 
						|
    profile = helpers.example_user("iago")
 | 
						|
    client = Client.objects.create(name="curl-test-client-2")
 | 
						|
    response = do_events_register(profile, profile.realm, client, event_types=["message"])
 | 
						|
    return {
 | 
						|
        "queue_id": response["queue_id"],
 | 
						|
        "last_event_id": response["last_event_id"],
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/users/{user_id_or_email}/presence:get"])
 | 
						|
def get_user_presence() -> dict[str, object]:
 | 
						|
    iago = helpers.example_user("iago")
 | 
						|
    client = Client.objects.create(name="curl-test-client-3")
 | 
						|
    update_user_presence(iago, client, timezone_now(), UserPresence.LEGACY_STATUS_ACTIVE_INT, False)
 | 
						|
    return {}
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/users:post"])
 | 
						|
def create_user() -> dict[str, object]:
 | 
						|
    return {
 | 
						|
        "email": helpers.nonreg_email("test"),
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/users/{email]:patch", "/users/{user_id}:patch"])
 | 
						|
def new_email_value() -> dict[str, object]:
 | 
						|
    count = 0
 | 
						|
    exists = True
 | 
						|
    while exists:
 | 
						|
        email = f"new{count}@zulip.com"
 | 
						|
        exists = UserProfile.objects.filter(delivery_email=email).exists()
 | 
						|
        count += 1
 | 
						|
    return {
 | 
						|
        "new_email": email,
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/user_groups/create:post"])
 | 
						|
def create_user_group_data() -> dict[str, object]:
 | 
						|
    return {
 | 
						|
        "members": [helpers.example_user("hamlet").id, helpers.example_user("othello").id],
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/user_groups/{user_group_id}:patch"])
 | 
						|
def get_temp_user_group_id() -> dict[str, object]:
 | 
						|
    user_group, _ = NamedUserGroup.objects.get_or_create(
 | 
						|
        name="temp",
 | 
						|
        realm=get_realm("zulip"),
 | 
						|
        can_add_members_group_id=11,
 | 
						|
        can_join_group_id=11,
 | 
						|
        can_leave_group_id=15,
 | 
						|
        can_manage_group_id=11,
 | 
						|
        can_mention_group_id=11,
 | 
						|
        can_remove_members_group_id=11,
 | 
						|
        realm_for_sharding=get_realm("zulip"),
 | 
						|
    )
 | 
						|
    return {
 | 
						|
        "user_group_id": user_group.id,
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/user_groups/{user_group_id}/deactivate:post"])
 | 
						|
def get_temp_user_group_id_for_deactivation() -> dict[str, object]:
 | 
						|
    print(NamedUserGroup.objects.all())
 | 
						|
    user_group, _ = NamedUserGroup.objects.get_or_create(
 | 
						|
        name="temp-deactivation",
 | 
						|
        realm=get_realm("zulip"),
 | 
						|
        can_add_members_group_id=11,
 | 
						|
        can_join_group_id=11,
 | 
						|
        can_leave_group_id=15,
 | 
						|
        can_manage_group_id=11,
 | 
						|
        can_mention_group_id=11,
 | 
						|
        can_remove_members_group_id=11,
 | 
						|
        realm_for_sharding=get_realm("zulip"),
 | 
						|
    )
 | 
						|
    return {
 | 
						|
        "user_group_id": user_group.id,
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/realm/filters/{filter_id}:delete"])
 | 
						|
def remove_realm_filters() -> dict[str, object]:
 | 
						|
    filter_id = do_add_linkifier(
 | 
						|
        get_realm("zulip"),
 | 
						|
        "#(?P<id>[0-9]{2,8})",
 | 
						|
        "https://github.com/zulip/zulip/pull/{id}",
 | 
						|
        acting_user=None,
 | 
						|
    )
 | 
						|
    return {
 | 
						|
        "filter_id": filter_id,
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/realm/emoji/{emoji_name}:post", "/user_uploads:post"])
 | 
						|
def upload_custom_emoji() -> dict[str, object]:
 | 
						|
    return {
 | 
						|
        "filename": "zerver/tests/images/animated_img.gif",
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/realm/playgrounds:post"])
 | 
						|
def add_realm_playground() -> dict[str, object]:
 | 
						|
    return {
 | 
						|
        "name": "Python2 playground",
 | 
						|
        "pygments_language": "Python2",
 | 
						|
        "url_template": "https://python2.example.com?code={code}",
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/realm/playgrounds/{playground_id}:delete"])
 | 
						|
def remove_realm_playground() -> dict[str, object]:
 | 
						|
    playground_id = check_add_realm_playground(
 | 
						|
        get_realm("zulip"),
 | 
						|
        acting_user=None,
 | 
						|
        name="Python playground",
 | 
						|
        pygments_language="Python",
 | 
						|
        url_template="https://python.example.com?code={code}",
 | 
						|
    )
 | 
						|
    return {
 | 
						|
        "playground_id": playground_id,
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/users/{user_id}:delete"])
 | 
						|
def deactivate_user() -> dict[str, object]:
 | 
						|
    user_profile = do_create_user(
 | 
						|
        email="testuser@zulip.com",
 | 
						|
        password=None,
 | 
						|
        full_name="test_user",
 | 
						|
        realm=get_realm("zulip"),
 | 
						|
        acting_user=None,
 | 
						|
    )
 | 
						|
    return {"user_id": user_profile.id}
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/users/me:delete"])
 | 
						|
def deactivate_own_user() -> dict[str, object]:
 | 
						|
    test_user_email = "delete-test@zulip.com"
 | 
						|
    deactivate_test_user = do_create_user(
 | 
						|
        test_user_email,
 | 
						|
        "secret",
 | 
						|
        get_realm("zulip"),
 | 
						|
        "Mr. Delete",
 | 
						|
        role=200,
 | 
						|
        acting_user=None,
 | 
						|
    )
 | 
						|
    realm = get_realm("zulip")
 | 
						|
    test_user = get_user(test_user_email, realm)
 | 
						|
    test_user_api_key = test_user.api_key
 | 
						|
    # change authentication line to allow test_client to delete itself.
 | 
						|
    AUTHENTICATION_LINE[0] = f"{deactivate_test_user.email}:{test_user_api_key}"
 | 
						|
    return {}
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/attachments/{attachment_id}:delete"])
 | 
						|
def remove_attachment() -> dict[str, object]:
 | 
						|
    user_profile = helpers.example_user("iago")
 | 
						|
    url = upload_message_attachment("dummy.txt", "text/plain", b"zulip!", user_profile)[0]
 | 
						|
    attachment_id = url.replace("/user_uploads/", "").split("/")[0]
 | 
						|
 | 
						|
    return {"attachment_id": attachment_id}
 |