348 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
			
		
		
	
	
			348 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
| # -*- coding: utf-8 -*-
 | |
| # Copyright 2014-2016 OpenMarket Ltd
 | |
| #
 | |
| # Licensed under the Apache License, Version 2.0 (the "License");
 | |
| # you may not use this file except in compliance with the License.
 | |
| # You may obtain a copy of the License at
 | |
| #
 | |
| #     http://www.apache.org/licenses/LICENSE-2.0
 | |
| #
 | |
| # Unless required by applicable law or agreed to in writing, software
 | |
| # distributed under the License is distributed on an "AS IS" BASIS,
 | |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| # See the License for the specific language governing permissions and
 | |
| # limitations under the License.
 | |
| 
 | |
| """Tests REST events for /profile paths."""
 | |
| import json
 | |
| 
 | |
| from mock import Mock
 | |
| 
 | |
| from twisted.internet import defer
 | |
| 
 | |
| import synapse.types
 | |
| from synapse.api.errors import AuthError, SynapseError
 | |
| from synapse.rest import admin
 | |
| from synapse.rest.client.v1 import login, profile, room
 | |
| 
 | |
| from tests import unittest
 | |
| 
 | |
| from ....utils import MockHttpResource, setup_test_homeserver
 | |
| 
 | |
| myid = "@1234ABCD:test"
 | |
| PATH_PREFIX = "/_matrix/client/r0"
 | |
| 
 | |
| 
 | |
| class MockHandlerProfileTestCase(unittest.TestCase):
 | |
|     """ Tests rest layer of profile management.
 | |
| 
 | |
|     Todo: move these into ProfileTestCase
 | |
|     """
 | |
| 
 | |
|     @defer.inlineCallbacks
 | |
|     def setUp(self):
 | |
|         self.mock_resource = MockHttpResource(prefix=PATH_PREFIX)
 | |
|         self.mock_handler = Mock(
 | |
|             spec=[
 | |
|                 "get_displayname",
 | |
|                 "set_displayname",
 | |
|                 "get_avatar_url",
 | |
|                 "set_avatar_url",
 | |
|                 "check_profile_query_allowed",
 | |
|             ]
 | |
|         )
 | |
| 
 | |
|         self.mock_handler.get_displayname.return_value = defer.succeed(Mock())
 | |
|         self.mock_handler.set_displayname.return_value = defer.succeed(Mock())
 | |
|         self.mock_handler.get_avatar_url.return_value = defer.succeed(Mock())
 | |
|         self.mock_handler.set_avatar_url.return_value = defer.succeed(Mock())
 | |
|         self.mock_handler.check_profile_query_allowed.return_value = defer.succeed(
 | |
|             Mock()
 | |
|         )
 | |
| 
 | |
|         hs = yield setup_test_homeserver(
 | |
|             self.addCleanup,
 | |
|             "test",
 | |
|             http_client=None,
 | |
|             resource_for_client=self.mock_resource,
 | |
|             federation=Mock(),
 | |
|             federation_client=Mock(),
 | |
|             profile_handler=self.mock_handler,
 | |
|         )
 | |
| 
 | |
|         def _get_user_by_req(request=None, allow_guest=False):
 | |
|             return defer.succeed(synapse.types.create_requester(myid))
 | |
| 
 | |
|         hs.get_auth().get_user_by_req = _get_user_by_req
 | |
| 
 | |
|         profile.register_servlets(hs, self.mock_resource)
 | |
| 
 | |
|     @defer.inlineCallbacks
 | |
|     def test_get_my_name(self):
 | |
|         mocked_get = self.mock_handler.get_displayname
 | |
|         mocked_get.return_value = defer.succeed("Frank")
 | |
| 
 | |
|         (code, response) = yield self.mock_resource.trigger(
 | |
|             "GET", "/profile/%s/displayname" % (myid), None
 | |
|         )
 | |
| 
 | |
|         self.assertEquals(200, code)
 | |
|         self.assertEquals({"displayname": "Frank"}, response)
 | |
|         self.assertEquals(mocked_get.call_args[0][0].localpart, "1234ABCD")
 | |
| 
 | |
|     @defer.inlineCallbacks
 | |
|     def test_set_my_name(self):
 | |
|         mocked_set = self.mock_handler.set_displayname
 | |
|         mocked_set.return_value = defer.succeed(())
 | |
| 
 | |
|         (code, response) = yield self.mock_resource.trigger(
 | |
|             "PUT", "/profile/%s/displayname" % (myid), b'{"displayname": "Frank Jr."}'
 | |
|         )
 | |
| 
 | |
|         self.assertEquals(200, code)
 | |
|         self.assertEquals(mocked_set.call_args[0][0].localpart, "1234ABCD")
 | |
|         self.assertEquals(mocked_set.call_args[0][1].user.localpart, "1234ABCD")
 | |
|         self.assertEquals(mocked_set.call_args[0][2], "Frank Jr.")
 | |
| 
 | |
|     @defer.inlineCallbacks
 | |
|     def test_set_my_name_noauth(self):
 | |
|         mocked_set = self.mock_handler.set_displayname
 | |
|         mocked_set.side_effect = AuthError(400, "message")
 | |
| 
 | |
|         (code, response) = yield self.mock_resource.trigger(
 | |
|             "PUT",
 | |
|             "/profile/%s/displayname" % ("@4567:test"),
 | |
|             b'{"displayname": "Frank Jr."}',
 | |
|         )
 | |
| 
 | |
|         self.assertTrue(400 <= code < 499, msg="code %d is in the 4xx range" % (code))
 | |
| 
 | |
|     @defer.inlineCallbacks
 | |
|     def test_get_other_name(self):
 | |
|         mocked_get = self.mock_handler.get_displayname
 | |
|         mocked_get.return_value = defer.succeed("Bob")
 | |
| 
 | |
|         (code, response) = yield self.mock_resource.trigger(
 | |
|             "GET", "/profile/%s/displayname" % ("@opaque:elsewhere"), None
 | |
|         )
 | |
| 
 | |
|         self.assertEquals(200, code)
 | |
|         self.assertEquals({"displayname": "Bob"}, response)
 | |
| 
 | |
|     @defer.inlineCallbacks
 | |
|     def test_set_other_name(self):
 | |
|         mocked_set = self.mock_handler.set_displayname
 | |
|         mocked_set.side_effect = SynapseError(400, "message")
 | |
| 
 | |
|         (code, response) = yield self.mock_resource.trigger(
 | |
|             "PUT",
 | |
|             "/profile/%s/displayname" % ("@opaque:elsewhere"),
 | |
|             b'{"displayname":"bob"}',
 | |
|         )
 | |
| 
 | |
|         self.assertTrue(400 <= code <= 499, msg="code %d is in the 4xx range" % (code))
 | |
| 
 | |
|     @defer.inlineCallbacks
 | |
|     def test_get_my_avatar(self):
 | |
|         mocked_get = self.mock_handler.get_avatar_url
 | |
|         mocked_get.return_value = defer.succeed("http://my.server/me.png")
 | |
| 
 | |
|         (code, response) = yield self.mock_resource.trigger(
 | |
|             "GET", "/profile/%s/avatar_url" % (myid), None
 | |
|         )
 | |
| 
 | |
|         self.assertEquals(200, code)
 | |
|         self.assertEquals({"avatar_url": "http://my.server/me.png"}, response)
 | |
|         self.assertEquals(mocked_get.call_args[0][0].localpart, "1234ABCD")
 | |
| 
 | |
|     @defer.inlineCallbacks
 | |
|     def test_set_my_avatar(self):
 | |
|         mocked_set = self.mock_handler.set_avatar_url
 | |
|         mocked_set.return_value = defer.succeed(())
 | |
| 
 | |
|         (code, response) = yield self.mock_resource.trigger(
 | |
|             "PUT",
 | |
|             "/profile/%s/avatar_url" % (myid),
 | |
|             b'{"avatar_url": "http://my.server/pic.gif"}',
 | |
|         )
 | |
| 
 | |
|         self.assertEquals(200, code)
 | |
|         self.assertEquals(mocked_set.call_args[0][0].localpart, "1234ABCD")
 | |
|         self.assertEquals(mocked_set.call_args[0][1].user.localpart, "1234ABCD")
 | |
|         self.assertEquals(mocked_set.call_args[0][2], "http://my.server/pic.gif")
 | |
| 
 | |
| 
 | |
| class ProfileTestCase(unittest.HomeserverTestCase):
 | |
| 
 | |
|     servlets = [
 | |
|         admin.register_servlets_for_client_rest_resource,
 | |
|         login.register_servlets,
 | |
|         profile.register_servlets,
 | |
|     ]
 | |
| 
 | |
|     def make_homeserver(self, reactor, clock):
 | |
|         self.hs = self.setup_test_homeserver()
 | |
|         return self.hs
 | |
| 
 | |
|     def prepare(self, reactor, clock, hs):
 | |
|         self.owner = self.register_user("owner", "pass")
 | |
|         self.owner_tok = self.login("owner", "pass")
 | |
| 
 | |
|     def test_set_displayname(self):
 | |
|         request, channel = self.make_request(
 | |
|             "PUT",
 | |
|             "/profile/%s/displayname" % (self.owner,),
 | |
|             content=json.dumps({"displayname": "test"}),
 | |
|             access_token=self.owner_tok,
 | |
|         )
 | |
|         self.render(request)
 | |
|         self.assertEqual(channel.code, 200, channel.result)
 | |
| 
 | |
|         res = self.get_displayname()
 | |
|         self.assertEqual(res, "test")
 | |
| 
 | |
|     def test_set_displayname_too_long(self):
 | |
|         """Attempts to set a stupid displayname should get a 400"""
 | |
|         request, channel = self.make_request(
 | |
|             "PUT",
 | |
|             "/profile/%s/displayname" % (self.owner,),
 | |
|             content=json.dumps({"displayname": "test" * 100}),
 | |
|             access_token=self.owner_tok,
 | |
|         )
 | |
|         self.render(request)
 | |
|         self.assertEqual(channel.code, 400, channel.result)
 | |
| 
 | |
|         res = self.get_displayname()
 | |
|         self.assertEqual(res, "owner")
 | |
| 
 | |
|     def get_displayname(self):
 | |
|         request, channel = self.make_request(
 | |
|             "GET", "/profile/%s/displayname" % (self.owner,)
 | |
|         )
 | |
|         self.render(request)
 | |
|         self.assertEqual(channel.code, 200, channel.result)
 | |
|         return channel.json_body["displayname"]
 | |
| 
 | |
| 
 | |
| class ProfilesRestrictedTestCase(unittest.HomeserverTestCase):
 | |
| 
 | |
|     servlets = [
 | |
|         admin.register_servlets_for_client_rest_resource,
 | |
|         login.register_servlets,
 | |
|         profile.register_servlets,
 | |
|         room.register_servlets,
 | |
|     ]
 | |
| 
 | |
|     def make_homeserver(self, reactor, clock):
 | |
| 
 | |
|         config = self.default_config()
 | |
|         config["require_auth_for_profile_requests"] = True
 | |
|         config["limit_profile_requests_to_users_who_share_rooms"] = True
 | |
|         self.hs = self.setup_test_homeserver(config=config)
 | |
| 
 | |
|         return self.hs
 | |
| 
 | |
|     def prepare(self, reactor, clock, hs):
 | |
|         # User owning the requested profile.
 | |
|         self.owner = self.register_user("owner", "pass")
 | |
|         self.owner_tok = self.login("owner", "pass")
 | |
|         self.profile_url = "/profile/%s" % (self.owner)
 | |
| 
 | |
|         # User requesting the profile.
 | |
|         self.requester = self.register_user("requester", "pass")
 | |
|         self.requester_tok = self.login("requester", "pass")
 | |
| 
 | |
|         self.room_id = self.helper.create_room_as(self.owner, tok=self.owner_tok)
 | |
| 
 | |
|     def test_no_auth(self):
 | |
|         self.try_fetch_profile(401)
 | |
| 
 | |
|     def test_not_in_shared_room(self):
 | |
|         self.ensure_requester_left_room()
 | |
| 
 | |
|         self.try_fetch_profile(403, access_token=self.requester_tok)
 | |
| 
 | |
|     def test_in_shared_room(self):
 | |
|         self.ensure_requester_left_room()
 | |
| 
 | |
|         self.helper.join(room=self.room_id, user=self.requester, tok=self.requester_tok)
 | |
| 
 | |
|         self.try_fetch_profile(200, self.requester_tok)
 | |
| 
 | |
|     def try_fetch_profile(self, expected_code, access_token=None):
 | |
|         self.request_profile(expected_code, access_token=access_token)
 | |
| 
 | |
|         self.request_profile(
 | |
|             expected_code, url_suffix="/displayname", access_token=access_token
 | |
|         )
 | |
| 
 | |
|         self.request_profile(
 | |
|             expected_code, url_suffix="/avatar_url", access_token=access_token
 | |
|         )
 | |
| 
 | |
|     def request_profile(self, expected_code, url_suffix="", access_token=None):
 | |
|         request, channel = self.make_request(
 | |
|             "GET", self.profile_url + url_suffix, access_token=access_token
 | |
|         )
 | |
|         self.render(request)
 | |
|         self.assertEqual(channel.code, expected_code, channel.result)
 | |
| 
 | |
|     def ensure_requester_left_room(self):
 | |
|         try:
 | |
|             self.helper.leave(
 | |
|                 room=self.room_id, user=self.requester, tok=self.requester_tok
 | |
|             )
 | |
|         except AssertionError:
 | |
|             # We don't care whether the leave request didn't return a 200 (e.g.
 | |
|             # if the user isn't already in the room), because we only want to
 | |
|             # make sure the user isn't in the room.
 | |
|             pass
 | |
| 
 | |
| 
 | |
| class OwnProfileUnrestrictedTestCase(unittest.HomeserverTestCase):
 | |
| 
 | |
|     servlets = [
 | |
|         admin.register_servlets_for_client_rest_resource,
 | |
|         login.register_servlets,
 | |
|         profile.register_servlets,
 | |
|     ]
 | |
| 
 | |
|     def make_homeserver(self, reactor, clock):
 | |
|         config = self.default_config()
 | |
|         config["require_auth_for_profile_requests"] = True
 | |
|         config["limit_profile_requests_to_users_who_share_rooms"] = True
 | |
|         self.hs = self.setup_test_homeserver(config=config)
 | |
| 
 | |
|         return self.hs
 | |
| 
 | |
|     def prepare(self, reactor, clock, hs):
 | |
|         # User requesting the profile.
 | |
|         self.requester = self.register_user("requester", "pass")
 | |
|         self.requester_tok = self.login("requester", "pass")
 | |
| 
 | |
|     def test_can_lookup_own_profile(self):
 | |
|         """Tests that a user can lookup their own profile without having to be in a room
 | |
|         if 'require_auth_for_profile_requests' is set to true in the server's config.
 | |
|         """
 | |
|         request, channel = self.make_request(
 | |
|             "GET", "/profile/" + self.requester, access_token=self.requester_tok
 | |
|         )
 | |
|         self.render(request)
 | |
|         self.assertEqual(channel.code, 200, channel.result)
 | |
| 
 | |
|         request, channel = self.make_request(
 | |
|             "GET",
 | |
|             "/profile/" + self.requester + "/displayname",
 | |
|             access_token=self.requester_tok,
 | |
|         )
 | |
|         self.render(request)
 | |
|         self.assertEqual(channel.code, 200, channel.result)
 | |
| 
 | |
|         request, channel = self.make_request(
 | |
|             "GET",
 | |
|             "/profile/" + self.requester + "/avatar_url",
 | |
|             access_token=self.requester_tok,
 | |
|         )
 | |
|         self.render(request)
 | |
|         self.assertEqual(channel.code, 200, channel.result)
 |