from unittest import mock import orjson import responses from django.core.signing import Signer from django.http import HttpResponseRedirect from typing_extensions import override from zerver.lib.test_classes import ZulipTestCase from zerver.lib.url_encoding import append_url_query_string class TestVideoCall(ZulipTestCase): @override def setUp(self) -> None: super().setUp() self.user = self.example_user("hamlet") self.login_user(self.user) # Signing for bbb self.signer = Signer() self.signed_bbb_a_object = self.signer.sign_object( { "meeting_id": "a", "name": "a", "password": "a", } ) def test_register_video_request_no_settings(self) -> None: with self.settings(VIDEO_ZOOM_CLIENT_ID=None): response = self.client_get("/calls/zoom/register") self.assert_json_error( response, "Zoom credentials have not been configured", ) def test_register_video_request(self) -> None: response = self.client_get("/calls/zoom/register") self.assertEqual(response.status_code, 302) @responses.activate def test_create_zoom_video_and_audio_links(self) -> None: responses.add( responses.POST, "https://zoom.us/oauth/token", json={"access_token": "oldtoken", "expires_in": -60}, ) response = self.client_get( "/calls/zoom/complete", {"code": "code", "state": '{"realm":"zulip","sid":""}'}, ) self.assertEqual(response.status_code, 200) # Test creating a video link responses.replace( responses.POST, "https://zoom.us/oauth/token", json={"access_token": "newtoken", "expires_in": 60}, ) responses.add( responses.POST, "https://api.zoom.us/v2/users/me/meetings", json={"join_url": "example.com"}, ) response = self.client_post("/json/calls/zoom/create", {"is_video_call": "true"}) self.assertEqual( responses.calls[-1].request.url, "https://api.zoom.us/v2/users/me/meetings", ) assert responses.calls[-1].request.body is not None self.assertEqual( orjson.loads(responses.calls[-1].request.body), { "settings": { "host_video": True, "participant_video": True, }, "default_password": True, }, ) self.assertEqual( responses.calls[-1].request.headers["Authorization"], "Bearer newtoken", ) json = self.assert_json_success(response) self.assertEqual(json["url"], "example.com") # Test creating an audio link responses.replace( responses.POST, "https://zoom.us/oauth/token", json={"access_token": "newtoken", "expires_in": 60}, ) responses.add( responses.POST, "https://api.zoom.us/v2/users/me/meetings", json={"join_url": "example.com"}, ) response = self.client_post("/json/calls/zoom/create", {"is_video_call": "false"}) self.assertEqual( responses.calls[-1].request.url, "https://api.zoom.us/v2/users/me/meetings", ) assert responses.calls[-1].request.body is not None self.assertEqual( orjson.loads(responses.calls[-1].request.body), { "settings": { "host_video": False, "participant_video": False, }, "default_password": True, }, ) self.assertEqual( responses.calls[-1].request.headers["Authorization"], "Bearer newtoken", ) json = self.assert_json_success(response) self.assertEqual(json["url"], "example.com") # Test for authentication error self.logout() self.login_user(self.user) response = self.client_post("/json/calls/zoom/create") self.assert_json_error(response, "Invalid Zoom access token") def test_create_video_realm_redirect(self) -> None: response = self.client_get( "/calls/zoom/complete", {"code": "code", "state": '{"realm":"zephyr","sid":"somesid"}'}, ) self.assertEqual(response.status_code, 302) self.assertIn("http://zephyr.testserver/", response["Location"]) self.assertIn("somesid", response["Location"]) def test_create_video_sid_error(self) -> None: response = self.client_get( "/calls/zoom/complete", {"code": "code", "state": '{"realm":"zulip","sid":"bad"}'}, ) self.assert_json_error(response, "Invalid Zoom session identifier") @responses.activate def test_create_video_credential_error(self) -> None: responses.add(responses.POST, "https://zoom.us/oauth/token", status=400) response = self.client_get( "/calls/zoom/complete", {"code": "code", "state": '{"realm":"zulip","sid":""}'}, ) self.assert_json_error(response, "Invalid Zoom credentials") @responses.activate def test_create_video_refresh_error(self) -> None: responses.add( responses.POST, "https://zoom.us/oauth/token", json={"access_token": "token", "expires_in": -60}, ) response = self.client_get( "/calls/zoom/complete", {"code": "code", "state": '{"realm":"zulip","sid":""}'}, ) self.assertEqual(response.status_code, 200) responses.replace(responses.POST, "https://zoom.us/oauth/token", status=400) response = self.client_post("/json/calls/zoom/create") self.assert_json_error(response, "Invalid Zoom access token") @responses.activate def test_create_video_request_error(self) -> None: responses.add( responses.POST, "https://zoom.us/oauth/token", json={"access_token": "token"}, ) responses.add( responses.POST, "https://api.zoom.us/v2/users/me/meetings", status=400, ) response = self.client_get( "/calls/zoom/complete", {"code": "code", "state": '{"realm":"zulip","sid":""}'}, ) self.assertEqual(response.status_code, 200) response = self.client_post("/json/calls/zoom/create") self.assert_json_error(response, "Failed to create Zoom call") responses.replace( responses.POST, "https://api.zoom.us/v2/users/me/meetings", status=401, ) response = self.client_post("/json/calls/zoom/create") self.assert_json_error(response, "Invalid Zoom access token") @responses.activate def test_deauthorize_zoom_user(self) -> None: response = self.client_post( "/calls/zoom/deauthorize", """\ { "event": "app_deauthorized", "payload": { "user_data_retention": "false", "account_id": "EabCDEFghiLHMA", "user_id": "z9jkdsfsdfjhdkfjQ", "signature": "827edc3452044f0bc86bdd5684afb7d1e6becfa1a767f24df1b287853cf73000", "deauthorization_time": "2019-06-17T13:52:28.632Z", "client_id": "ADZ9k9bTWmGUoUbECUKU_a" } } """, content_type="application/json", ) self.assert_json_success(response) def test_create_bigbluebutton_link(self) -> None: with ( mock.patch("zerver.views.video_calls.random.randint", return_value="1"), mock.patch("secrets.token_bytes", return_value=b"\x00" * 20), ): response = self.client_get( "/json/calls/bigbluebutton/create?meeting_name=general > meeting" ) response_dict = self.assert_json_success(response) self.assertEqual( response_dict["url"], append_url_query_string( "/calls/bigbluebutton/join", "bigbluebutton=" + self.signer.sign_object( { "meeting_id": "zulip-1", "name": "general > meeting", "password": "A" * 32, } ), ), ) @responses.activate def test_join_bigbluebutton_redirect(self) -> None: responses.add( responses.GET, "https://bbb.example.com/bigbluebutton/api/create?meetingID=a&name=a" "&moderatorPW=a&attendeePW=a&checksum=131bdec35f62fc63d5436e6f791d6d7aed7cf79ef256c03597e51d320d042823", "SUCCESS0", ) response = self.client_get( "/calls/bigbluebutton/join", {"bigbluebutton": self.signed_bbb_a_object} ) self.assertEqual(response.status_code, 302) self.assertEqual(isinstance(response, HttpResponseRedirect), True) self.assertEqual( response["Location"], "https://bbb.example.com/bigbluebutton/api/join?meetingID=a&" "password=a&fullName=King%20Hamlet&createTime=0&checksum=47ca959b4ff5c8047a5a56d6e99c07e17eac43dbf792afc0a2a9f6491ec0048b", ) @responses.activate def test_join_bigbluebutton_invalid_signature(self) -> None: responses.add( responses.GET, "https://bbb.example.com/bigbluebutton/api/create?meetingID=a&name=a" "&moderatorPW=a&attendeePW=a&checksum=131bdec35f62fc63d5436e6f791d6d7aed7cf79ef256c03597e51d320d042823", "SUCCESS0", ) response = self.client_get( "/calls/bigbluebutton/join", {"bigbluebutton": self.signed_bbb_a_object + "zoo"} ) self.assert_json_error(response, "Invalid signature.") @responses.activate def test_join_bigbluebutton_redirect_wrong_big_blue_button_checksum(self) -> None: responses.add( responses.GET, "https://bbb.example.com/bigbluebutton/api/create?meetingID=a&name=a&moderatorPW=a&attendeePW=a&checksum=131bdec35f62fc63d5436e6f791d6d7aed7cf79ef256c03597e51d320d042823", "FAILEDchecksumError" "You did not pass the checksum security check", ) response = self.client_get( "/calls/bigbluebutton/join", {"bigbluebutton": self.signed_bbb_a_object}, ) self.assert_json_error(response, "Error authenticating to the BigBlueButton server.") @responses.activate def test_join_bigbluebutton_redirect_server_error(self) -> None: # Simulate bbb server error responses.add( responses.GET, "https://bbb.example.com/bigbluebutton/api/create?meetingID=a&name=a&moderatorPW=a&attendeePW=a&checksum=131bdec35f62fc63d5436e6f791d6d7aed7cf79ef256c03597e51d320d042823", "", status=500, ) response = self.client_get( "/calls/bigbluebutton/join", {"bigbluebutton": self.signed_bbb_a_object}, ) self.assert_json_error(response, "Error connecting to the BigBlueButton server.") @responses.activate def test_join_bigbluebutton_redirect_error_by_server(self) -> None: # Simulate bbb server error responses.add( responses.GET, "https://bbb.example.com/bigbluebutton/api/create?meetingID=a&name=a&moderatorPW=a&attendeePW=a&checksum=131bdec35f62fc63d5436e6f791d6d7aed7cf79ef256c03597e51d320d042823", "FAILUREotherFailure", ) response = self.client_get( "/calls/bigbluebutton/join", {"bigbluebutton": self.signed_bbb_a_object}, ) self.assert_json_error(response, "BigBlueButton server returned an unexpected error.") def test_join_bigbluebutton_redirect_not_configured(self) -> None: with self.settings(BIG_BLUE_BUTTON_SECRET=None, BIG_BLUE_BUTTON_URL=None): response = self.client_get( "/calls/bigbluebutton/join", {"bigbluebutton": self.signed_bbb_a_object}, ) self.assert_json_error(response, "BigBlueButton is not configured.")