filtering.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456
  1. # -*- coding: utf-8 -*-
  2. # Copyright 2015, 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 six import text_type
  16. import jsonschema
  17. from canonicaljson import json
  18. from jsonschema import FormatChecker
  19. from twisted.internet import defer
  20. from synapse.api.errors import SynapseError
  21. from synapse.storage.presence import UserPresenceState
  22. from synapse.types import RoomID, UserID
  23. FILTER_SCHEMA = {
  24. "additionalProperties": False,
  25. "type": "object",
  26. "properties": {
  27. "limit": {
  28. "type": "number"
  29. },
  30. "senders": {
  31. "$ref": "#/definitions/user_id_array"
  32. },
  33. "not_senders": {
  34. "$ref": "#/definitions/user_id_array"
  35. },
  36. # TODO: We don't limit event type values but we probably should...
  37. # check types are valid event types
  38. "types": {
  39. "type": "array",
  40. "items": {
  41. "type": "string"
  42. }
  43. },
  44. "not_types": {
  45. "type": "array",
  46. "items": {
  47. "type": "string"
  48. }
  49. }
  50. }
  51. }
  52. ROOM_FILTER_SCHEMA = {
  53. "additionalProperties": False,
  54. "type": "object",
  55. "properties": {
  56. "not_rooms": {
  57. "$ref": "#/definitions/room_id_array"
  58. },
  59. "rooms": {
  60. "$ref": "#/definitions/room_id_array"
  61. },
  62. "ephemeral": {
  63. "$ref": "#/definitions/room_event_filter"
  64. },
  65. "include_leave": {
  66. "type": "boolean"
  67. },
  68. "state": {
  69. "$ref": "#/definitions/room_event_filter"
  70. },
  71. "timeline": {
  72. "$ref": "#/definitions/room_event_filter"
  73. },
  74. "account_data": {
  75. "$ref": "#/definitions/room_event_filter"
  76. },
  77. }
  78. }
  79. ROOM_EVENT_FILTER_SCHEMA = {
  80. "additionalProperties": False,
  81. "type": "object",
  82. "properties": {
  83. "limit": {
  84. "type": "number"
  85. },
  86. "senders": {
  87. "$ref": "#/definitions/user_id_array"
  88. },
  89. "not_senders": {
  90. "$ref": "#/definitions/user_id_array"
  91. },
  92. "types": {
  93. "type": "array",
  94. "items": {
  95. "type": "string"
  96. }
  97. },
  98. "not_types": {
  99. "type": "array",
  100. "items": {
  101. "type": "string"
  102. }
  103. },
  104. "rooms": {
  105. "$ref": "#/definitions/room_id_array"
  106. },
  107. "not_rooms": {
  108. "$ref": "#/definitions/room_id_array"
  109. },
  110. "contains_url": {
  111. "type": "boolean"
  112. },
  113. "lazy_load_members": {
  114. "type": "boolean"
  115. },
  116. "include_redundant_members": {
  117. "type": "boolean"
  118. },
  119. }
  120. }
  121. USER_ID_ARRAY_SCHEMA = {
  122. "type": "array",
  123. "items": {
  124. "type": "string",
  125. "format": "matrix_user_id"
  126. }
  127. }
  128. ROOM_ID_ARRAY_SCHEMA = {
  129. "type": "array",
  130. "items": {
  131. "type": "string",
  132. "format": "matrix_room_id"
  133. }
  134. }
  135. USER_FILTER_SCHEMA = {
  136. "$schema": "http://json-schema.org/draft-04/schema#",
  137. "description": "schema for a Sync filter",
  138. "type": "object",
  139. "definitions": {
  140. "room_id_array": ROOM_ID_ARRAY_SCHEMA,
  141. "user_id_array": USER_ID_ARRAY_SCHEMA,
  142. "filter": FILTER_SCHEMA,
  143. "room_filter": ROOM_FILTER_SCHEMA,
  144. "room_event_filter": ROOM_EVENT_FILTER_SCHEMA
  145. },
  146. "properties": {
  147. "presence": {
  148. "$ref": "#/definitions/filter"
  149. },
  150. "account_data": {
  151. "$ref": "#/definitions/filter"
  152. },
  153. "room": {
  154. "$ref": "#/definitions/room_filter"
  155. },
  156. "event_format": {
  157. "type": "string",
  158. "enum": ["client", "federation"]
  159. },
  160. "event_fields": {
  161. "type": "array",
  162. "items": {
  163. "type": "string",
  164. # Don't allow '\\' in event field filters. This makes matching
  165. # events a lot easier as we can then use a negative lookbehind
  166. # assertion to split '\.' If we allowed \\ then it would
  167. # incorrectly split '\\.' See synapse.events.utils.serialize_event
  168. #
  169. # Note that because this is a regular expression, we have to escape
  170. # each backslash in the pattern.
  171. "pattern": r"^((?!\\\\).)*$"
  172. }
  173. }
  174. },
  175. "additionalProperties": False
  176. }
  177. @FormatChecker.cls_checks('matrix_room_id')
  178. def matrix_room_id_validator(room_id_str):
  179. return RoomID.from_string(room_id_str)
  180. @FormatChecker.cls_checks('matrix_user_id')
  181. def matrix_user_id_validator(user_id_str):
  182. return UserID.from_string(user_id_str)
  183. class Filtering(object):
  184. def __init__(self, hs):
  185. super(Filtering, self).__init__()
  186. self.store = hs.get_datastore()
  187. @defer.inlineCallbacks
  188. def get_user_filter(self, user_localpart, filter_id):
  189. result = yield self.store.get_user_filter(user_localpart, filter_id)
  190. defer.returnValue(FilterCollection(result))
  191. def add_user_filter(self, user_localpart, user_filter):
  192. self.check_valid_filter(user_filter)
  193. return self.store.add_user_filter(user_localpart, user_filter)
  194. # TODO(paul): surely we should probably add a delete_user_filter or
  195. # replace_user_filter at some point? There's no REST API specified for
  196. # them however
  197. def check_valid_filter(self, user_filter_json):
  198. """Check if the provided filter is valid.
  199. This inspects all definitions contained within the filter.
  200. Args:
  201. user_filter_json(dict): The filter
  202. Raises:
  203. SynapseError: If the filter is not valid.
  204. """
  205. # NB: Filters are the complete json blobs. "Definitions" are an
  206. # individual top-level key e.g. public_user_data. Filters are made of
  207. # many definitions.
  208. try:
  209. jsonschema.validate(user_filter_json, USER_FILTER_SCHEMA,
  210. format_checker=FormatChecker())
  211. except jsonschema.ValidationError as e:
  212. raise SynapseError(400, str(e))
  213. class FilterCollection(object):
  214. def __init__(self, filter_json):
  215. self._filter_json = filter_json
  216. room_filter_json = self._filter_json.get("room", {})
  217. self._room_filter = Filter({
  218. k: v for k, v in room_filter_json.items()
  219. if k in ("rooms", "not_rooms")
  220. })
  221. self._room_timeline_filter = Filter(room_filter_json.get("timeline", {}))
  222. self._room_state_filter = Filter(room_filter_json.get("state", {}))
  223. self._room_ephemeral_filter = Filter(room_filter_json.get("ephemeral", {}))
  224. self._room_account_data = Filter(room_filter_json.get("account_data", {}))
  225. self._presence_filter = Filter(filter_json.get("presence", {}))
  226. self._account_data = Filter(filter_json.get("account_data", {}))
  227. self.include_leave = filter_json.get("room", {}).get(
  228. "include_leave", False
  229. )
  230. self.event_fields = filter_json.get("event_fields", [])
  231. self.event_format = filter_json.get("event_format", "client")
  232. def __repr__(self):
  233. return "<FilterCollection %s>" % (json.dumps(self._filter_json),)
  234. def get_filter_json(self):
  235. return self._filter_json
  236. def timeline_limit(self):
  237. return self._room_timeline_filter.limit()
  238. def presence_limit(self):
  239. return self._presence_filter.limit()
  240. def ephemeral_limit(self):
  241. return self._room_ephemeral_filter.limit()
  242. def lazy_load_members(self):
  243. return self._room_state_filter.lazy_load_members()
  244. def include_redundant_members(self):
  245. return self._room_state_filter.include_redundant_members()
  246. def filter_presence(self, events):
  247. return self._presence_filter.filter(events)
  248. def filter_account_data(self, events):
  249. return self._account_data.filter(events)
  250. def filter_room_state(self, events):
  251. return self._room_state_filter.filter(self._room_filter.filter(events))
  252. def filter_room_timeline(self, events):
  253. return self._room_timeline_filter.filter(self._room_filter.filter(events))
  254. def filter_room_ephemeral(self, events):
  255. return self._room_ephemeral_filter.filter(self._room_filter.filter(events))
  256. def filter_room_account_data(self, events):
  257. return self._room_account_data.filter(self._room_filter.filter(events))
  258. def blocks_all_presence(self):
  259. return (
  260. self._presence_filter.filters_all_types() or
  261. self._presence_filter.filters_all_senders()
  262. )
  263. def blocks_all_room_ephemeral(self):
  264. return (
  265. self._room_ephemeral_filter.filters_all_types() or
  266. self._room_ephemeral_filter.filters_all_senders() or
  267. self._room_ephemeral_filter.filters_all_rooms()
  268. )
  269. def blocks_all_room_timeline(self):
  270. return (
  271. self._room_timeline_filter.filters_all_types() or
  272. self._room_timeline_filter.filters_all_senders() or
  273. self._room_timeline_filter.filters_all_rooms()
  274. )
  275. class Filter(object):
  276. def __init__(self, filter_json):
  277. self.filter_json = filter_json
  278. self.types = self.filter_json.get("types", None)
  279. self.not_types = self.filter_json.get("not_types", [])
  280. self.rooms = self.filter_json.get("rooms", None)
  281. self.not_rooms = self.filter_json.get("not_rooms", [])
  282. self.senders = self.filter_json.get("senders", None)
  283. self.not_senders = self.filter_json.get("not_senders", [])
  284. self.contains_url = self.filter_json.get("contains_url", None)
  285. def filters_all_types(self):
  286. return "*" in self.not_types
  287. def filters_all_senders(self):
  288. return "*" in self.not_senders
  289. def filters_all_rooms(self):
  290. return "*" in self.not_rooms
  291. def check(self, event):
  292. """Checks whether the filter matches the given event.
  293. Returns:
  294. bool: True if the event matches
  295. """
  296. # We usually get the full "events" as dictionaries coming through,
  297. # except for presence which actually gets passed around as its own
  298. # namedtuple type.
  299. if isinstance(event, UserPresenceState):
  300. sender = event.user_id
  301. room_id = None
  302. ev_type = "m.presence"
  303. contains_url = False
  304. else:
  305. sender = event.get("sender", None)
  306. if not sender:
  307. # Presence events had their 'sender' in content.user_id, but are
  308. # now handled above. We don't know if anything else uses this
  309. # form. TODO: Check this and probably remove it.
  310. content = event.get("content")
  311. # account_data has been allowed to have non-dict content, so
  312. # check type first
  313. if isinstance(content, dict):
  314. sender = content.get("user_id")
  315. room_id = event.get("room_id", None)
  316. ev_type = event.get("type", None)
  317. content = event.get("content", {})
  318. # check if there is a string url field in the content for filtering purposes
  319. contains_url = isinstance(content.get("url"), text_type)
  320. return self.check_fields(
  321. room_id,
  322. sender,
  323. ev_type,
  324. contains_url,
  325. )
  326. def check_fields(self, room_id, sender, event_type, contains_url):
  327. """Checks whether the filter matches the given event fields.
  328. Returns:
  329. bool: True if the event fields match
  330. """
  331. literal_keys = {
  332. "rooms": lambda v: room_id == v,
  333. "senders": lambda v: sender == v,
  334. "types": lambda v: _matches_wildcard(event_type, v)
  335. }
  336. for name, match_func in literal_keys.items():
  337. not_name = "not_%s" % (name,)
  338. disallowed_values = getattr(self, not_name)
  339. if any(map(match_func, disallowed_values)):
  340. return False
  341. allowed_values = getattr(self, name)
  342. if allowed_values is not None:
  343. if not any(map(match_func, allowed_values)):
  344. return False
  345. contains_url_filter = self.filter_json.get("contains_url")
  346. if contains_url_filter is not None:
  347. if contains_url_filter != contains_url:
  348. return False
  349. return True
  350. def filter_rooms(self, room_ids):
  351. """Apply the 'rooms' filter to a given list of rooms.
  352. Args:
  353. room_ids (list): A list of room_ids.
  354. Returns:
  355. list: A list of room_ids that match the filter
  356. """
  357. room_ids = set(room_ids)
  358. disallowed_rooms = set(self.filter_json.get("not_rooms", []))
  359. room_ids -= disallowed_rooms
  360. allowed_rooms = self.filter_json.get("rooms", None)
  361. if allowed_rooms is not None:
  362. room_ids &= set(allowed_rooms)
  363. return room_ids
  364. def filter(self, events):
  365. return list(filter(self.check, events))
  366. def limit(self):
  367. return self.filter_json.get("limit", 10)
  368. def lazy_load_members(self):
  369. return self.filter_json.get("lazy_load_members", False)
  370. def include_redundant_members(self):
  371. return self.filter_json.get("include_redundant_members", False)
  372. def _matches_wildcard(actual_value, filter_value):
  373. if filter_value.endswith("*"):
  374. type_prefix = filter_value[:-1]
  375. return actual_value.startswith(type_prefix)
  376. else:
  377. return actual_value == filter_value
  378. DEFAULT_FILTER_COLLECTION = FilterCollection({})