test_redaction.py 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. # -*- coding: utf-8 -*-
  2. # Copyright 2014-2016 OpenMarket Ltd
  3. #
  4. # Licensed under the Apache License, Version 2.0 (the "License");
  5. # you may not use this file except in compliance with the License.
  6. # You may obtain a copy of the License at
  7. #
  8. # http://www.apache.org/licenses/LICENSE-2.0
  9. #
  10. # Unless required by applicable law or agreed to in writing, software
  11. # distributed under the License is distributed on an "AS IS" BASIS,
  12. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. # See the License for the specific language governing permissions and
  14. # limitations under the License.
  15. from tests import unittest
  16. from twisted.internet import defer
  17. from synapse.api.constants import EventTypes, Membership
  18. from synapse.types import UserID, RoomID
  19. from tests.utils import setup_test_homeserver
  20. from mock import Mock
  21. class RedactionTestCase(unittest.TestCase):
  22. @defer.inlineCallbacks
  23. def setUp(self):
  24. hs = yield setup_test_homeserver(
  25. resource_for_federation=Mock(),
  26. http_client=None,
  27. )
  28. self.store = hs.get_datastore()
  29. self.event_builder_factory = hs.get_event_builder_factory()
  30. self.handlers = hs.get_handlers()
  31. self.message_handler = self.handlers.message_handler
  32. self.u_alice = UserID.from_string("@alice:test")
  33. self.u_bob = UserID.from_string("@bob:test")
  34. self.room1 = RoomID.from_string("!abc123:test")
  35. self.depth = 1
  36. @defer.inlineCallbacks
  37. def inject_room_member(self, room, user, membership, replaces_state=None,
  38. extra_content={}):
  39. content = {"membership": membership}
  40. content.update(extra_content)
  41. builder = self.event_builder_factory.new({
  42. "type": EventTypes.Member,
  43. "sender": user.to_string(),
  44. "state_key": user.to_string(),
  45. "room_id": room.to_string(),
  46. "content": content,
  47. })
  48. event, context = yield self.message_handler._create_new_client_event(
  49. builder
  50. )
  51. yield self.store.persist_event(event, context)
  52. defer.returnValue(event)
  53. @defer.inlineCallbacks
  54. def inject_message(self, room, user, body):
  55. self.depth += 1
  56. builder = self.event_builder_factory.new({
  57. "type": EventTypes.Message,
  58. "sender": user.to_string(),
  59. "state_key": user.to_string(),
  60. "room_id": room.to_string(),
  61. "content": {"body": body, "msgtype": u"message"},
  62. })
  63. event, context = yield self.message_handler._create_new_client_event(
  64. builder
  65. )
  66. yield self.store.persist_event(event, context)
  67. defer.returnValue(event)
  68. @defer.inlineCallbacks
  69. def inject_redaction(self, room, event_id, user, reason):
  70. builder = self.event_builder_factory.new({
  71. "type": EventTypes.Redaction,
  72. "sender": user.to_string(),
  73. "state_key": user.to_string(),
  74. "room_id": room.to_string(),
  75. "content": {"reason": reason},
  76. "redacts": event_id,
  77. })
  78. event, context = yield self.message_handler._create_new_client_event(
  79. builder
  80. )
  81. yield self.store.persist_event(event, context)
  82. @defer.inlineCallbacks
  83. def test_redact(self):
  84. yield self.inject_room_member(
  85. self.room1, self.u_alice, Membership.JOIN
  86. )
  87. start = yield self.store.get_room_events_max_id()
  88. msg_event = yield self.inject_message(self.room1, self.u_alice, u"t")
  89. end = yield self.store.get_room_events_max_id()
  90. results, _ = yield self.store.get_room_events_stream(
  91. self.u_alice.to_string(),
  92. start,
  93. end,
  94. )
  95. self.assertEqual(1, len(results))
  96. # Check event has not been redacted:
  97. event = results[0]
  98. self.assertObjectHasAttributes(
  99. {
  100. "type": EventTypes.Message,
  101. "user_id": self.u_alice.to_string(),
  102. "content": {"body": "t", "msgtype": "message"},
  103. },
  104. event,
  105. )
  106. self.assertFalse("redacted_because" in event.unsigned)
  107. # Redact event
  108. reason = "Because I said so"
  109. yield self.inject_redaction(
  110. self.room1, msg_event.event_id, self.u_alice, reason
  111. )
  112. results, _ = yield self.store.get_room_events_stream(
  113. self.u_alice.to_string(),
  114. start,
  115. end,
  116. )
  117. self.assertEqual(1, len(results))
  118. # Check redaction
  119. event = results[0]
  120. self.assertEqual(msg_event.event_id, event.event_id)
  121. self.assertTrue("redacted_because" in event.unsigned)
  122. self.assertObjectHasAttributes(
  123. {
  124. "type": EventTypes.Message,
  125. "user_id": self.u_alice.to_string(),
  126. "content": {},
  127. },
  128. event,
  129. )
  130. self.assertObjectHasAttributes(
  131. {
  132. "type": EventTypes.Redaction,
  133. "user_id": self.u_alice.to_string(),
  134. "content": {"reason": reason},
  135. },
  136. event.unsigned["redacted_because"],
  137. )
  138. @defer.inlineCallbacks
  139. def test_redact_join(self):
  140. yield self.inject_room_member(
  141. self.room1, self.u_alice, Membership.JOIN
  142. )
  143. start = yield self.store.get_room_events_max_id()
  144. msg_event = yield self.inject_room_member(
  145. self.room1, self.u_bob, Membership.JOIN,
  146. extra_content={"blue": "red"},
  147. )
  148. end = yield self.store.get_room_events_max_id()
  149. results, _ = yield self.store.get_room_events_stream(
  150. self.u_alice.to_string(),
  151. start,
  152. end,
  153. )
  154. self.assertEqual(1, len(results))
  155. # Check event has not been redacted:
  156. event = results[0]
  157. self.assertObjectHasAttributes(
  158. {
  159. "type": EventTypes.Member,
  160. "user_id": self.u_bob.to_string(),
  161. "content": {"membership": Membership.JOIN, "blue": "red"},
  162. },
  163. event,
  164. )
  165. self.assertFalse(hasattr(event, "redacted_because"))
  166. # Redact event
  167. reason = "Because I said so"
  168. yield self.inject_redaction(
  169. self.room1, msg_event.event_id, self.u_alice, reason
  170. )
  171. results, _ = yield self.store.get_room_events_stream(
  172. self.u_alice.to_string(),
  173. start,
  174. end,
  175. )
  176. self.assertEqual(1, len(results))
  177. # Check redaction
  178. event = results[0]
  179. self.assertTrue("redacted_because" in event.unsigned)
  180. self.assertObjectHasAttributes(
  181. {
  182. "type": EventTypes.Member,
  183. "user_id": self.u_bob.to_string(),
  184. "content": {"membership": Membership.JOIN},
  185. },
  186. event,
  187. )
  188. self.assertObjectHasAttributes(
  189. {
  190. "type": EventTypes.Redaction,
  191. "user_id": self.u_alice.to_string(),
  192. "content": {"reason": reason},
  193. },
  194. event.unsigned["redacted_because"],
  195. )