2017-11-09 18:53:02 +01:00
|
|
|
import time
|
2019-02-11 07:03:20 +01:00
|
|
|
from typing import List
|
2017-03-08 12:36:07 +01:00
|
|
|
|
2019-04-24 03:18:00 +02:00
|
|
|
from bs4 import BeautifulSoup
|
2020-02-11 16:06:29 +01:00
|
|
|
from django.conf import settings
|
2017-11-10 11:57:02 +01:00
|
|
|
from django.test import override_settings
|
2017-11-09 18:53:02 +01:00
|
|
|
from unittest.mock import Mock, patch
|
2019-04-26 15:02:16 +02:00
|
|
|
from zerver.lib.realm_icon import get_realm_icon_url
|
2020-03-24 14:47:41 +01:00
|
|
|
from zerver.lib.streams import create_stream_if_needed
|
2017-11-09 18:53:02 +01:00
|
|
|
from zerver.lib.test_classes import ZulipTestCase
|
2019-04-24 04:30:15 +02:00
|
|
|
from zerver.middleware import is_slow_query, write_log_line
|
2020-02-11 16:06:29 +01:00
|
|
|
from zerver.models import get_realm, get_system_bot
|
2017-03-08 12:36:07 +01:00
|
|
|
|
2017-11-09 18:53:02 +01:00
|
|
|
class SlowQueryTest(ZulipTestCase):
|
2017-11-10 11:57:02 +01:00
|
|
|
SLOW_QUERY_TIME = 10
|
|
|
|
log_data = {'extra': '[transport=websocket]',
|
|
|
|
'time_started': 0,
|
|
|
|
'bugdown_requests_start': 0,
|
|
|
|
'bugdown_time_start': 0,
|
|
|
|
'remote_cache_time_start': 0,
|
|
|
|
'remote_cache_requests_start': 0}
|
|
|
|
|
2017-11-05 10:51:25 +01:00
|
|
|
def test_is_slow_query(self) -> None:
|
2017-03-08 12:36:07 +01:00
|
|
|
self.assertFalse(is_slow_query(1.1, '/some/random/url'))
|
|
|
|
self.assertTrue(is_slow_query(2, '/some/random/url'))
|
|
|
|
self.assertTrue(is_slow_query(5.1, '/activity'))
|
|
|
|
self.assertFalse(is_slow_query(2, '/activity'))
|
2017-10-16 22:07:19 +02:00
|
|
|
self.assertFalse(is_slow_query(2, '/json/report/error'))
|
2017-03-08 12:36:07 +01:00
|
|
|
self.assertFalse(is_slow_query(2, '/api/v1/deployments/report_error'))
|
|
|
|
self.assertFalse(is_slow_query(2, '/realm_activity/whatever'))
|
|
|
|
self.assertFalse(is_slow_query(2, '/user_activity/whatever'))
|
|
|
|
self.assertFalse(is_slow_query(9, '/accounts/webathena_kerberos_login/'))
|
|
|
|
self.assertTrue(is_slow_query(11, '/accounts/webathena_kerberos_login/'))
|
2017-11-09 18:53:02 +01:00
|
|
|
|
2018-07-31 02:37:24 +02:00
|
|
|
@override_settings(SLOW_QUERY_LOGS_STREAM="logs")
|
2017-11-09 18:53:02 +01:00
|
|
|
@patch('logging.info')
|
2017-11-05 10:51:25 +01:00
|
|
|
def test_slow_query_log(self, mock_logging_info: Mock) -> None:
|
2020-02-11 16:06:29 +01:00
|
|
|
error_bot = get_system_bot(settings.ERROR_BOT)
|
|
|
|
create_stream_if_needed(error_bot.realm, settings.SLOW_QUERY_LOGS_STREAM)
|
|
|
|
|
2017-11-10 11:57:02 +01:00
|
|
|
self.log_data['time_started'] = time.time() - self.SLOW_QUERY_TIME
|
|
|
|
write_log_line(self.log_data, path='/socket/open', method='SOCKET',
|
2020-03-09 11:39:20 +01:00
|
|
|
remote_ip='123.456.789.012', requestor_for_logs='unknown', client_name='?')
|
2017-11-09 18:53:02 +01:00
|
|
|
last_message = self.get_last_message()
|
|
|
|
self.assertEqual(last_message.sender.email, "error-bot@zulip.com")
|
|
|
|
self.assertIn("logs", str(last_message.recipient))
|
|
|
|
self.assertEqual(last_message.topic_name(), "testserver: slow queries")
|
|
|
|
self.assertRegexpMatches(last_message.content,
|
2018-07-02 00:05:24 +02:00
|
|
|
r"123\.456\.789\.012 SOCKET 200 10\.\ds .*")
|
2017-11-10 11:57:02 +01:00
|
|
|
|
|
|
|
@override_settings(ERROR_BOT=None)
|
|
|
|
@patch('logging.info')
|
2020-02-11 16:06:29 +01:00
|
|
|
@patch('zerver.lib.actions.internal_send_stream_message')
|
|
|
|
def test_slow_query_log_without_error_bot(self,
|
|
|
|
mock_internal_send_stream_message: Mock,
|
2017-11-05 10:51:25 +01:00
|
|
|
mock_logging_info: Mock) -> None:
|
2017-11-10 11:57:02 +01:00
|
|
|
self.log_data['time_started'] = time.time() - self.SLOW_QUERY_TIME
|
|
|
|
write_log_line(self.log_data, path='/socket/open', method='SOCKET',
|
2020-03-09 11:39:20 +01:00
|
|
|
remote_ip='123.456.789.012', requestor_for_logs='unknown', client_name='?')
|
2020-02-11 16:06:29 +01:00
|
|
|
mock_internal_send_stream_message.assert_not_called()
|
2018-12-14 23:28:00 +01:00
|
|
|
|
|
|
|
class OpenGraphTest(ZulipTestCase):
|
2019-02-11 07:03:20 +01:00
|
|
|
def check_title_and_description(self, path: str, title: str,
|
|
|
|
in_description: List[str],
|
2019-04-24 03:33:40 +02:00
|
|
|
not_in_description: List[str],
|
|
|
|
status_code: int=200) -> None:
|
2018-12-14 23:28:00 +01:00
|
|
|
response = self.client_get(path)
|
2019-04-24 03:33:40 +02:00
|
|
|
self.assertEqual(response.status_code, status_code)
|
2019-02-11 07:03:20 +01:00
|
|
|
decoded = response.content.decode('utf-8')
|
2019-04-24 03:18:00 +02:00
|
|
|
bs = BeautifulSoup(decoded, features='lxml')
|
|
|
|
open_graph_title = bs.select_one('meta[property="og:title"]').get('content')
|
|
|
|
self.assertEqual(open_graph_title, title)
|
2019-02-11 07:03:20 +01:00
|
|
|
|
2019-04-24 03:18:00 +02:00
|
|
|
open_graph_description = bs.select_one('meta[property="og:description"]').get('content')
|
2019-02-11 07:03:20 +01:00
|
|
|
for substring in in_description:
|
|
|
|
self.assertIn(substring, open_graph_description)
|
|
|
|
for substring in not_in_description:
|
|
|
|
self.assertNotIn(substring, open_graph_description)
|
2018-12-14 23:28:00 +01:00
|
|
|
|
|
|
|
def test_admonition_and_link(self) -> None:
|
|
|
|
# disable-message-edit-history starts with an {!admin-only.md!}, and has a link
|
|
|
|
# in the first paragraph.
|
|
|
|
self.check_title_and_description(
|
|
|
|
'/help/disable-message-edit-history',
|
|
|
|
"Disable message edit history (Zulip Help Center)",
|
2019-02-11 07:03:20 +01:00
|
|
|
["By default, Zulip displays messages",
|
2019-05-02 02:35:20 +02:00
|
|
|
"users can view the edit history of a message. | To remove the",
|
2019-02-11 07:03:20 +01:00
|
|
|
"best to delete the message entirely. "],
|
|
|
|
["Disable message edit history", "feature is only available", "Related articles",
|
|
|
|
"Restrict message editing"]
|
|
|
|
)
|
2018-12-14 23:28:00 +01:00
|
|
|
|
2019-04-24 03:37:34 +02:00
|
|
|
def test_double_quotes(self) -> None:
|
|
|
|
# night-mode has a quoted string "night mode"
|
|
|
|
self.check_title_and_description(
|
|
|
|
'/help/night-mode',
|
|
|
|
"Night mode (Zulip Help Center)",
|
|
|
|
['By default, Zulip has a white background. ',
|
|
|
|
'Zulip also provides a "night mode", which is great for working in a dark space.'],
|
|
|
|
[]
|
|
|
|
)
|
|
|
|
|
2018-12-14 23:28:00 +01:00
|
|
|
def test_settings_tab(self) -> None:
|
|
|
|
# deactivate-your-account starts with {settings_tab|your-account}
|
|
|
|
self.check_title_and_description(
|
|
|
|
'/help/deactivate-your-account',
|
|
|
|
"Deactivate your account (Zulip Help Center)",
|
2019-05-02 02:35:20 +02:00
|
|
|
["Any bots that you maintain will be disabled. | Deactivating "],
|
2019-02-11 08:22:05 +01:00
|
|
|
["Confirm by clicking", " ", "\n"])
|
2018-12-14 23:28:00 +01:00
|
|
|
|
|
|
|
def test_tabs(self) -> None:
|
|
|
|
# logging-out starts with {start_tabs}
|
|
|
|
self.check_title_and_description(
|
|
|
|
'/help/logging-out',
|
|
|
|
"Logging out (Zulip Help Center)",
|
|
|
|
# Ideally we'd do something better here
|
2019-04-24 03:18:00 +02:00
|
|
|
["We're here to help! Email us at zulip-admin@example.com with questions, feedback, or " +
|
2019-02-11 07:03:55 +01:00
|
|
|
"feature requests."],
|
|
|
|
["Click on the gear"])
|
2018-12-14 23:28:00 +01:00
|
|
|
|
|
|
|
def test_index_pages(self) -> None:
|
|
|
|
self.check_title_and_description(
|
|
|
|
'/help/',
|
|
|
|
"Zulip Help Center",
|
2019-02-11 07:03:20 +01:00
|
|
|
[("Zulip is a group chat app. Its most distinctive characteristic is that "
|
|
|
|
"conversation within an organization is divided into “streams” and further ")], [])
|
2018-12-14 23:28:00 +01:00
|
|
|
|
|
|
|
self.check_title_and_description(
|
|
|
|
'/api/',
|
|
|
|
"Zulip API Documentation",
|
2019-04-24 03:18:00 +02:00
|
|
|
[("Zulip's APIs allow you to integrate other services with Zulip. This "
|
2019-02-11 07:03:20 +01:00
|
|
|
"guide should help you find the API you need:")], [])
|
2018-12-14 23:28:00 +01:00
|
|
|
|
|
|
|
def test_nonexistent_page(self) -> None:
|
2019-04-24 03:33:40 +02:00
|
|
|
self.check_title_and_description(
|
|
|
|
'/help/not-a-real-page',
|
2018-12-14 23:28:00 +01:00
|
|
|
# Probably we should make this "Zulip Help Center"
|
2019-04-24 03:33:40 +02:00
|
|
|
"No such article. (Zulip Help Center)",
|
2019-05-02 02:35:20 +02:00
|
|
|
["No such article. | We're here to help!",
|
2019-04-24 03:33:40 +02:00
|
|
|
"Email us at zulip-admin@example.com with questions, feedback, or feature requests."],
|
|
|
|
[],
|
|
|
|
# Test that our open graph logic doesn't throw a 500
|
|
|
|
404)
|
2019-04-24 04:30:15 +02:00
|
|
|
|
|
|
|
def test_login_page_simple_description(self) -> None:
|
|
|
|
name = 'Zulip Dev'
|
|
|
|
description = "The Zulip development environment default organization. It's great for testing!"
|
|
|
|
|
|
|
|
self.check_title_and_description(
|
|
|
|
'/login/',
|
|
|
|
name,
|
|
|
|
[description],
|
|
|
|
[])
|
2019-04-24 04:10:56 +02:00
|
|
|
|
|
|
|
def test_login_page_markdown_description(self) -> None:
|
|
|
|
realm = get_realm('zulip')
|
|
|
|
description = ("Welcome to **Clojurians Zulip** - the place where the Clojure community meets.\n\n"
|
|
|
|
"Before you signup/login:\n\n"
|
|
|
|
"* note-1\n"
|
|
|
|
"* note-2\n"
|
|
|
|
"* note-3\n\n"
|
|
|
|
"Enjoy!")
|
|
|
|
realm.description = description
|
|
|
|
realm.save(update_fields=['description'])
|
|
|
|
|
|
|
|
self.check_title_and_description(
|
|
|
|
'/login/',
|
|
|
|
'Zulip Dev',
|
|
|
|
['Welcome to Clojurians Zulip - the place where the Clojure community meets',
|
2019-05-02 02:35:20 +02:00
|
|
|
'* note-1 * note-2 * note-3 | Enjoy!'],
|
2019-04-24 04:10:56 +02:00
|
|
|
[])
|
2019-04-26 15:02:16 +02:00
|
|
|
|
|
|
|
def test_login_page_realm_icon(self) -> None:
|
|
|
|
realm = get_realm('zulip')
|
|
|
|
realm.icon_source = 'U'
|
|
|
|
realm.save(update_fields=['icon_source'])
|
|
|
|
realm_icon = get_realm_icon_url(realm)
|
|
|
|
|
|
|
|
response = self.client_get('/login/')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
decoded = response.content.decode('utf-8')
|
|
|
|
bs = BeautifulSoup(decoded, features='lxml')
|
|
|
|
open_graph_image = bs.select_one('meta[property="og:image"]').get('content')
|
2019-05-16 14:07:12 +02:00
|
|
|
self.assertEqual(open_graph_image, '%s%s' % (realm.uri, realm_icon))
|
|
|
|
|
|
|
|
def test_login_page_realm_icon_absolute_url(self) -> None:
|
|
|
|
realm = get_realm('zulip')
|
|
|
|
realm.icon_source = 'U'
|
|
|
|
realm.save(update_fields=['icon_source'])
|
|
|
|
icon_url = "https://foo.s3.amazonaws.com/%s/realm/icon.png?version=%s" % (realm.id, 1)
|
|
|
|
with patch('zerver.lib.realm_icon.upload_backend.get_realm_icon_url', return_value=icon_url):
|
|
|
|
response = self.client_get('/login/')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
decoded = response.content.decode('utf-8')
|
|
|
|
bs = BeautifulSoup(decoded, features='lxml')
|
|
|
|
open_graph_image = bs.select_one('meta[property="og:image"]').get('content')
|
|
|
|
self.assertEqual(open_graph_image, icon_url)
|
2019-04-29 14:02:16 +02:00
|
|
|
|
|
|
|
def test_no_realm_api_page_og_url(self) -> None:
|
|
|
|
response = self.client_get('/api/', subdomain='')
|
|
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
|
|
|
|
decoded = response.content.decode('utf-8')
|
|
|
|
bs = BeautifulSoup(decoded, features='lxml')
|
|
|
|
open_graph_url = bs.select_one('meta[property="og:url"]').get('content')
|
|
|
|
|
|
|
|
self.assertTrue(open_graph_url.endswith('/api/'))
|