mirror of
				https://github.com/zulip/zulip.git
				synced 2025-10-24 16:43:57 +00:00 
			
		
		
		
	The subdomain and host parts are already used within zulip_url, api_url, this commit adds them as context properties themselves, so that they can be used in email addresses and URLs without the URI scheme, increasing uniformity. The new context properties `display_host` and `display_subdomain` are usable in the Help, API docs and Policies. Occurrences of `*.zulipchat.com` have been updated to use them.
		
			
				
	
	
		
			406 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			406 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import os
 | |
| import random
 | |
| import re
 | |
| from collections import OrderedDict
 | |
| from collections.abc import Callable
 | |
| from dataclasses import dataclass
 | |
| from typing import Any
 | |
| 
 | |
| from django.conf import settings
 | |
| from django.http import HttpRequest, HttpResponse, HttpResponseNotFound
 | |
| from django.template import loader
 | |
| from django.template.response import TemplateResponse
 | |
| from django.views.generic import TemplateView
 | |
| from lxml import html
 | |
| from lxml.etree import Element, SubElement, XPath, _Element
 | |
| from markupsafe import Markup
 | |
| from typing_extensions import override
 | |
| 
 | |
| from zerver.context_processors import zulip_default_context
 | |
| from zerver.decorator import add_google_analytics_context
 | |
| from zerver.lib.html_to_text import get_content_description
 | |
| from zerver.lib.integrations import (
 | |
|     CATEGORIES,
 | |
|     INTEGRATIONS,
 | |
|     META_CATEGORY,
 | |
|     HubotIntegration,
 | |
|     PythonAPIIntegration,
 | |
|     WebhookIntegration,
 | |
|     get_all_event_types_for_integration,
 | |
| )
 | |
| from zerver.lib.subdomains import get_subdomain
 | |
| from zerver.lib.templates import render_markdown_path
 | |
| from zerver.lib.typed_endpoint import PathOnly, typed_endpoint
 | |
| from zerver.models import Realm
 | |
| from zerver.openapi.openapi import get_endpoint_from_operationid, get_openapi_summary
 | |
| 
 | |
| 
 | |
| @dataclass
 | |
| class DocumentationArticle:
 | |
|     article_path: str
 | |
|     article_http_status: int
 | |
|     endpoint_path: str | None
 | |
|     endpoint_method: str | None
 | |
| 
 | |
| 
 | |
| def add_api_url_context(context: dict[str, Any], request: HttpRequest) -> None:
 | |
|     context.update(zulip_default_context(request))
 | |
| 
 | |
|     subdomain = get_subdomain(request)
 | |
|     if subdomain != Realm.SUBDOMAIN_FOR_ROOT_DOMAIN or not settings.ROOT_DOMAIN_LANDING_PAGE:
 | |
|         display_subdomain = subdomain
 | |
|         html_settings_links = True
 | |
|     else:
 | |
|         display_subdomain = "your-org"
 | |
|         html_settings_links = False
 | |
| 
 | |
|     display_host = Realm.host_for_subdomain(display_subdomain)
 | |
|     api_url_scheme_relative = display_host + "/api"
 | |
|     api_url = settings.EXTERNAL_URI_SCHEME + api_url_scheme_relative
 | |
|     zulip_url = settings.EXTERNAL_URI_SCHEME + display_host
 | |
| 
 | |
|     context["display_subdomain"] = display_subdomain
 | |
|     context["display_host"] = display_host
 | |
|     context["external_url_scheme"] = settings.EXTERNAL_URI_SCHEME
 | |
|     context["api_url"] = api_url
 | |
|     context["api_url_scheme_relative"] = api_url_scheme_relative
 | |
|     context["zulip_url"] = zulip_url
 | |
| 
 | |
|     context["html_settings_links"] = html_settings_links
 | |
| 
 | |
| 
 | |
| class ApiURLView(TemplateView):
 | |
|     @override
 | |
|     def get_context_data(self, **kwargs: Any) -> dict[str, str]:
 | |
|         context = super().get_context_data(**kwargs)
 | |
|         add_api_url_context(context, self.request)
 | |
|         return context
 | |
| 
 | |
| 
 | |
| sidebar_headings = XPath("//*[self::h1 or self::h2 or self::h3 or self::h4]")
 | |
| sidebar_links = XPath("//a[@href=$url]")
 | |
| 
 | |
| 
 | |
| class MarkdownDirectoryView(ApiURLView):
 | |
|     path_template = ""
 | |
|     policies_view = False
 | |
|     help_view = False
 | |
|     api_doc_view = False
 | |
| 
 | |
|     def __init__(self, **kwargs: Any) -> None:
 | |
|         super().__init__(**kwargs)
 | |
|         self._post_render_callbacks: list[Callable[[HttpResponse], HttpResponse | None]] = []
 | |
| 
 | |
|     def add_post_render_callback(
 | |
|         self, callback: Callable[[HttpResponse], HttpResponse | None]
 | |
|     ) -> None:
 | |
|         self._post_render_callbacks.append(callback)
 | |
| 
 | |
|     def get_path(self, article: str) -> DocumentationArticle:
 | |
|         http_status = 200
 | |
|         if article == "":
 | |
|             article = "index"
 | |
|         elif article == "include/sidebar_index":
 | |
|             pass
 | |
|         elif article == "api-doc-template":
 | |
|             # This markdown template shouldn't be accessed directly.
 | |
|             article = "missing"
 | |
|             http_status = 404
 | |
|         elif "/" in article:
 | |
|             article = "missing"
 | |
|             http_status = 404
 | |
|         elif len(article) > 100 or not re.match(r"^[0-9a-zA-Z_-]+$", article):
 | |
|             article = "missing"
 | |
|             http_status = 404
 | |
| 
 | |
|         path = self.path_template % (article,)
 | |
|         endpoint_name = None
 | |
|         endpoint_method = None
 | |
| 
 | |
|         if not self.path_template.startswith("/"):
 | |
|             # Relative paths only used for policies documentation
 | |
|             # when it is not configured or in the dev environment
 | |
|             assert self.policies_view
 | |
| 
 | |
|             try:
 | |
|                 loader.get_template(path)
 | |
|                 return DocumentationArticle(
 | |
|                     article_path=path,
 | |
|                     article_http_status=http_status,
 | |
|                     endpoint_path=endpoint_name,
 | |
|                     endpoint_method=endpoint_method,
 | |
|                 )
 | |
|             except loader.TemplateDoesNotExist:
 | |
|                 return DocumentationArticle(
 | |
|                     article_path=self.path_template % ("missing",),
 | |
|                     article_http_status=404,
 | |
|                     endpoint_path=None,
 | |
|                     endpoint_method=None,
 | |
|                 )
 | |
| 
 | |
|         if not os.path.exists(path):
 | |
|             if self.api_doc_view:
 | |
|                 try:
 | |
|                     # API endpoints documented in zerver/openapi/zulip.yaml
 | |
|                     endpoint_name, endpoint_method = get_endpoint_from_operationid(article)
 | |
|                     path = self.path_template % ("api-doc-template",)
 | |
|                 except AssertionError:
 | |
|                     return DocumentationArticle(
 | |
|                         article_path=self.path_template % ("missing",),
 | |
|                         article_http_status=404,
 | |
|                         endpoint_path=None,
 | |
|                         endpoint_method=None,
 | |
|                     )
 | |
|             elif self.help_view or self.policies_view:
 | |
|                 article = "missing"
 | |
|                 http_status = 404
 | |
|                 path = self.path_template % (article,)
 | |
|             else:
 | |
|                 raise AssertionError("Invalid documentation view type")
 | |
| 
 | |
|         return DocumentationArticle(
 | |
|             article_path=path,
 | |
|             article_http_status=http_status,
 | |
|             endpoint_path=endpoint_name,
 | |
|             endpoint_method=endpoint_method,
 | |
|         )
 | |
| 
 | |
|     @override
 | |
|     def get_context_data(self, **kwargs: Any) -> dict[str, Any]:
 | |
|         article = kwargs["article"]
 | |
|         context: dict[str, Any] = super().get_context_data()
 | |
| 
 | |
|         documentation_article = self.get_path(article)
 | |
|         context["article"] = documentation_article.article_path
 | |
|         not_index_page = not context["article"].endswith("/index.md")
 | |
| 
 | |
|         if documentation_article.article_path.startswith("/") and os.path.exists(
 | |
|             documentation_article.article_path
 | |
|         ):
 | |
|             # Absolute path case
 | |
|             article_absolute_path = documentation_article.article_path
 | |
|         else:
 | |
|             # Relative path case
 | |
|             article_absolute_path = os.path.join(
 | |
|                 settings.DEPLOY_ROOT, "templates", documentation_article.article_path
 | |
|             )
 | |
| 
 | |
|         if self.help_view:
 | |
|             context["page_is_help_center"] = True
 | |
|             context["doc_root"] = "/help/"
 | |
|             context["doc_root_title"] = "Help center"
 | |
|             sidebar_article = self.get_path("include/sidebar_index")
 | |
|             sidebar_index = sidebar_article.article_path
 | |
|             title_base = "Zulip help center"
 | |
|         elif self.policies_view:
 | |
|             context["page_is_policy_center"] = True
 | |
|             context["doc_root"] = "/policies/"
 | |
|             context["doc_root_title"] = "Terms and policies"
 | |
|             sidebar_article = self.get_path("sidebar_index")
 | |
|             if sidebar_article.article_http_status == 200:
 | |
|                 sidebar_index = sidebar_article.article_path
 | |
|             else:
 | |
|                 sidebar_index = None
 | |
|             title_base = "Zulip terms and policies"
 | |
|         elif self.api_doc_view:
 | |
|             context["page_is_api_center"] = True
 | |
|             context["doc_root"] = "/api/"
 | |
|             context["doc_root_title"] = "API documentation"
 | |
|             sidebar_article = self.get_path("sidebar_index")
 | |
|             sidebar_index = sidebar_article.article_path
 | |
|             title_base = "Zulip API documentation"
 | |
|         else:
 | |
|             raise AssertionError("Invalid documentation view type")
 | |
| 
 | |
|         # The following is a somewhat hacky approach to extract titles from articles.
 | |
|         endpoint_name = None
 | |
|         endpoint_method = None
 | |
|         if os.path.exists(article_absolute_path):
 | |
|             with open(article_absolute_path) as article_file:
 | |
|                 first_line = article_file.readlines()[0]
 | |
|             if self.api_doc_view and context["article"].endswith("api-doc-template.md"):
 | |
|                 endpoint_name, endpoint_method = (
 | |
|                     documentation_article.endpoint_path,
 | |
|                     documentation_article.endpoint_method,
 | |
|                 )
 | |
|                 assert endpoint_name is not None
 | |
|                 assert endpoint_method is not None
 | |
|                 article_title = get_openapi_summary(endpoint_name, endpoint_method)
 | |
|             elif self.api_doc_view and "{generate_api_header(" in first_line:
 | |
|                 api_operation = context["PAGE_METADATA_URL"].split("/api/")[1]
 | |
|                 endpoint_name, endpoint_method = get_endpoint_from_operationid(api_operation)
 | |
|                 article_title = get_openapi_summary(endpoint_name, endpoint_method)
 | |
|             else:
 | |
|                 # Strip the header and then use the first line to get the article title
 | |
|                 article_title = first_line.lstrip("#").strip()
 | |
|                 endpoint_name = endpoint_method = None
 | |
|             if not_index_page:
 | |
|                 context["PAGE_TITLE"] = f"{article_title} | {title_base}"
 | |
|             else:
 | |
|                 context["PAGE_TITLE"] = title_base
 | |
|             placeholder_open_graph_description = (
 | |
|                 f"REPLACEMENT_PAGE_DESCRIPTION_{int(2**24 * random.random())}"
 | |
|             )
 | |
|             context["PAGE_DESCRIPTION"] = placeholder_open_graph_description
 | |
| 
 | |
|             def update_description(response: HttpResponse) -> None:
 | |
|                 if placeholder_open_graph_description.encode() in response.content:
 | |
|                     first_paragraph_text = get_content_description(
 | |
|                         response.content, context["PAGE_METADATA_URL"]
 | |
|                     )
 | |
|                     response.content = response.content.replace(
 | |
|                         placeholder_open_graph_description.encode(),
 | |
|                         first_paragraph_text.encode(),
 | |
|                     )
 | |
| 
 | |
|             self.add_post_render_callback(update_description)
 | |
| 
 | |
|         # An "article" might require the api_url_context to be rendered
 | |
|         api_url_context: dict[str, Any] = {}
 | |
|         add_api_url_context(api_url_context, self.request)
 | |
|         api_url_context["run_content_validators"] = True
 | |
|         context["api_url_context"] = api_url_context
 | |
|         if endpoint_name and endpoint_method:
 | |
|             context["api_url_context"]["API_ENDPOINT_NAME"] = endpoint_name + ":" + endpoint_method
 | |
| 
 | |
|         if sidebar_index is not None:
 | |
|             sidebar_html = render_markdown_path(sidebar_index)
 | |
|         else:
 | |
|             sidebar_html = ""
 | |
|         tree = html.fragment_fromstring(sidebar_html, create_parent=True)
 | |
|         if not context.get("page_is_policy_center", False):
 | |
|             home_h1 = Element("h1")
 | |
|             home_link = SubElement(home_h1, "a")
 | |
|             home_link.attrib["class"] = "no-underline"
 | |
|             home_link.attrib["href"] = context["doc_root"]
 | |
|             home_link.text = context["doc_root_title"] + " home"
 | |
|             tree.insert(0, home_h1)
 | |
|         url = context["doc_root"] + article
 | |
|         # Remove ID attributes from sidebar headings so they don't conflict with index page headings
 | |
|         headings = sidebar_headings(tree)
 | |
|         assert isinstance(headings, list)
 | |
|         for h in headings:
 | |
|             assert isinstance(h, _Element)
 | |
|             h.attrib.pop("id", "")
 | |
|         # Highlight current article link
 | |
|         links = sidebar_links(tree, url=url)
 | |
|         assert isinstance(links, list)
 | |
|         for a in links:
 | |
|             assert isinstance(a, _Element)
 | |
|             old_class = a.attrib.get("class", "")
 | |
|             assert isinstance(old_class, str)
 | |
|             a.attrib["class"] = old_class + " highlighted"
 | |
|         context["sidebar_html"] = Markup().join(
 | |
|             Markup(html.tostring(child, encoding="unicode")) for child in tree
 | |
|         )
 | |
| 
 | |
|         add_google_analytics_context(context)
 | |
|         return context
 | |
| 
 | |
|     @override
 | |
|     def get(
 | |
|         self, request: HttpRequest, *args: object, article: str = "", **kwargs: object
 | |
|     ) -> HttpResponse:
 | |
|         # Hack: It's hard to reinitialize urls.py from tests, and so
 | |
|         # we want to defer the use of settings.POLICIES_DIRECTORY to
 | |
|         # runtime.
 | |
|         if self.policies_view:
 | |
|             self.path_template = f"{settings.POLICIES_DIRECTORY}/%s.md"
 | |
| 
 | |
|         documentation_article = self.get_path(article)
 | |
|         http_status = documentation_article.article_http_status
 | |
|         result = super().get(request, article=article)
 | |
|         assert isinstance(result, TemplateResponse)
 | |
|         for callback in self._post_render_callbacks:
 | |
|             result.add_post_render_callback(callback)
 | |
|         if http_status != 200:
 | |
|             result.status_code = http_status
 | |
|         return result
 | |
| 
 | |
| 
 | |
| def add_integrations_context(context: dict[str, Any]) -> None:
 | |
|     alphabetical_sorted_categories = OrderedDict(sorted(CATEGORIES.items()))
 | |
|     alphabetical_sorted_integration = OrderedDict(sorted(INTEGRATIONS.items()))
 | |
|     enabled_integrations_count = sum(v.is_enabled() for v in INTEGRATIONS.values())
 | |
|     # Subtract 1 so saying "Over X integrations" is correct. Then,
 | |
|     # round down to the nearest multiple of 10.
 | |
|     integrations_count_display = ((enabled_integrations_count - 1) // 10) * 10
 | |
|     context["categories_dict"] = alphabetical_sorted_categories
 | |
|     context["integrations_dict"] = alphabetical_sorted_integration
 | |
|     context["integrations_count_display"] = integrations_count_display
 | |
| 
 | |
| 
 | |
| def add_integrations_open_graph_context(context: dict[str, Any], request: HttpRequest) -> None:
 | |
|     path_name = request.path.rstrip("/").split("/")[-1]
 | |
|     description = (
 | |
|         "Zulip comes with over a hundred native integrations out of the box, "
 | |
|         "and integrates with Zapier and IFTTT to provide hundreds more. "
 | |
|         "Connect the apps you use every day to Zulip."
 | |
|     )
 | |
| 
 | |
|     if path_name in INTEGRATIONS:
 | |
|         integration = INTEGRATIONS[path_name]
 | |
|         context["PAGE_TITLE"] = f"{integration.display_name} | Zulip integrations"
 | |
|         context["PAGE_DESCRIPTION"] = description
 | |
| 
 | |
|     elif path_name in CATEGORIES:
 | |
|         category = CATEGORIES[path_name]
 | |
|         if path_name in META_CATEGORY:
 | |
|             context["PAGE_TITLE"] = f"{category} | Zulip integrations"
 | |
|         else:
 | |
|             context["PAGE_TITLE"] = f"{category} tools | Zulip integrations"
 | |
|         context["PAGE_DESCRIPTION"] = description
 | |
| 
 | |
|     elif path_name == "integrations":
 | |
|         context["PAGE_TITLE"] = "Zulip integrations"
 | |
|         context["PAGE_DESCRIPTION"] = description
 | |
| 
 | |
| 
 | |
| class IntegrationView(ApiURLView):
 | |
|     template_name = "zerver/integrations/index.html"
 | |
| 
 | |
|     @override
 | |
|     def get_context_data(self, **kwargs: Any) -> dict[str, Any]:
 | |
|         context: dict[str, Any] = super().get_context_data(**kwargs)
 | |
|         add_integrations_context(context)
 | |
|         add_integrations_open_graph_context(context, self.request)
 | |
|         add_google_analytics_context(context)
 | |
|         return context
 | |
| 
 | |
| 
 | |
| @typed_endpoint
 | |
| def integration_doc(request: HttpRequest, *, integration_name: PathOnly[str]) -> HttpResponse:
 | |
|     # FIXME: This check is jQuery-specific.
 | |
|     if request.headers.get("x-requested-with") != "XMLHttpRequest":
 | |
|         return HttpResponseNotFound()
 | |
| 
 | |
|     try:
 | |
|         integration = INTEGRATIONS[integration_name]
 | |
|     except KeyError:
 | |
|         return HttpResponseNotFound()
 | |
| 
 | |
|     context: dict[str, Any] = {}
 | |
|     add_api_url_context(context, request)
 | |
| 
 | |
|     context["integration_name"] = integration.name
 | |
|     context["integration_display_name"] = integration.display_name
 | |
|     if isinstance(integration, WebhookIntegration):
 | |
|         assert integration.url.startswith("api/")
 | |
|         context["integration_url"] = integration.url.removeprefix("api")
 | |
|         all_event_types = get_all_event_types_for_integration(integration)
 | |
|         if all_event_types is not None:
 | |
|             context["all_event_types"] = all_event_types
 | |
|     if isinstance(integration, HubotIntegration):
 | |
|         context["hubot_docs_url"] = integration.hubot_docs_url
 | |
|     if isinstance(integration, PythonAPIIntegration):
 | |
|         context["config_file_path"] = (
 | |
|             f"/usr/local/share/zulip/integrations/{integration.directory_name}/zulip_{integration.directory_name}_config.py"
 | |
|         )
 | |
|         context["integration_path"] = (
 | |
|             f"/usr/local/share/zulip/integrations/{integration.directory_name}"
 | |
|         )
 | |
| 
 | |
|     doc_html_str = render_markdown_path(integration.doc, context, integration_doc=True)
 | |
| 
 | |
|     return HttpResponse(doc_html_str)
 |