mirror of
				https://github.com/zulip/zulip.git
				synced 2025-11-04 14:03:30 +00:00 
			
		
		
		
	Since do_create_realm also creates general and core team streams, we rename general to verona right after the realm is created. Mostly because we dont really want two additional streams and this might probably make it easy to review things. There are puppeteer test changes because, we have a new "core team" stream in tests as well as there is a new default notification stream "Verona". Because of this tests in message-basics for example have to be changed since the newly added core team affects the order in which we navigate through the streams using arrow keys. The extra await for selector was added in subscriptions test to make the tests wait. Without the await the tests were passing ocassionally and failing in some other times. Fixes #6967
		
			
				
	
	
		
			336 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			336 lines
		
	
	
		
			11 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 functools import wraps
 | 
						|
from typing import Any, Callable, Dict, List, Optional, Set, Tuple
 | 
						|
 | 
						|
from django.utils.timezone import now as timezone_now
 | 
						|
 | 
						|
from zerver.lib.actions import (
 | 
						|
    do_add_linkifier,
 | 
						|
    do_add_reaction,
 | 
						|
    do_add_realm_playground,
 | 
						|
    do_create_user,
 | 
						|
    update_user_presence,
 | 
						|
)
 | 
						|
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.users import get_api_key
 | 
						|
from zerver.models import Client, Message, UserGroup, UserPresence, get_realm, get_user
 | 
						|
 | 
						|
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 genarator 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,
 | 
						|
    params: List[Dict[str, Any]],
 | 
						|
    request_body: Optional[Dict[str, Any]] = None,
 | 
						|
) -> Tuple[List[Dict[str, object]], Optional[Dict[str, object]]]:
 | 
						|
    if entry not in GENERATOR_FUNCTIONS:
 | 
						|
        return params, request_body
 | 
						|
    func = GENERATOR_FUNCTIONS[entry]
 | 
						|
    realm_example_values: Dict[str, object] = func()
 | 
						|
 | 
						|
    for param in params:
 | 
						|
        param_name = param["name"]
 | 
						|
        if param_name in realm_example_values:
 | 
						|
            if "content" in param:
 | 
						|
                param["content"]["application/json"]["example"] = realm_example_values[param_name]
 | 
						|
            else:
 | 
						|
                param["example"] = realm_example_values[param_name]
 | 
						|
 | 
						|
    if request_body is not None:
 | 
						|
        properties = request_body["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 params, 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]:
 | 
						|
    return {
 | 
						|
        "message_id": helpers.send_stream_message(helpers.example_user("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")
 | 
						|
 | 
						|
    # from OpenAPI format data in zulip.yaml
 | 
						|
    message_id = 43
 | 
						|
    emoji_name = "octopus"
 | 
						|
    emoji_code = "1f419"
 | 
						|
    reaction_type = "unicode_emoji"
 | 
						|
 | 
						|
    message = Message.objects.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 = []
 | 
						|
    for _ in range(3):
 | 
						|
        messages.append(helpers.send_stream_message(helpers.example_user("iago"), stream_name))
 | 
						|
    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, 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, 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.ACTIVE, False)
 | 
						|
    return {}
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/users:post"])
 | 
						|
def create_user() -> Dict[str, object]:
 | 
						|
    return {
 | 
						|
        "email": helpers.nonreg_email("test"),
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@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", "/user_groups/{user_group_id}:delete"]
 | 
						|
)
 | 
						|
def get_temp_user_group_id() -> Dict[str, object]:
 | 
						|
    user_group, _ = UserGroup.objects.get_or_create(name="temp", realm=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)s"
 | 
						|
    )
 | 
						|
    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_prefix": "https://python2.example.com",
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
@openapi_param_value_generator(["/realm/playgrounds/{playground_id}:delete"])
 | 
						|
def remove_realm_playground() -> Dict[str, object]:
 | 
						|
    playground_info = dict(
 | 
						|
        name="Python playground",
 | 
						|
        pygments_language="Python",
 | 
						|
        url_prefix="https://python.example.com",
 | 
						|
    )
 | 
						|
    playground_id = do_add_realm_playground(get_realm("zulip"), **playground_info)
 | 
						|
    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 = get_api_key(test_user)
 | 
						|
    # change authentication line to allow test_client to delete itself.
 | 
						|
    AUTHENTICATION_LINE[0] = f"{deactivate_test_user.email}:{test_user_api_key}"
 | 
						|
    return {}
 |