profile.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296
  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. import logging
  16. from twisted.internet import defer
  17. from synapse.api.errors import AuthError, CodeMessageException, SynapseError
  18. from synapse.types import UserID, get_domain_from_id
  19. from ._base import BaseHandler
  20. logger = logging.getLogger(__name__)
  21. class ProfileHandler(BaseHandler):
  22. PROFILE_UPDATE_MS = 60 * 1000
  23. PROFILE_UPDATE_EVERY_MS = 24 * 60 * 60 * 1000
  24. def __init__(self, hs):
  25. super(ProfileHandler, self).__init__(hs)
  26. self.federation = hs.get_federation_client()
  27. hs.get_federation_registry().register_query_handler(
  28. "profile", self.on_profile_query
  29. )
  30. self.user_directory_handler = hs.get_user_directory_handler()
  31. if hs.config.worker_app is None:
  32. self.clock.looping_call(
  33. self._update_remote_profile_cache, self.PROFILE_UPDATE_MS,
  34. )
  35. @defer.inlineCallbacks
  36. def get_profile(self, user_id):
  37. target_user = UserID.from_string(user_id)
  38. if self.hs.is_mine(target_user):
  39. displayname = yield self.store.get_profile_displayname(
  40. target_user.localpart
  41. )
  42. avatar_url = yield self.store.get_profile_avatar_url(
  43. target_user.localpart
  44. )
  45. defer.returnValue({
  46. "displayname": displayname,
  47. "avatar_url": avatar_url,
  48. })
  49. else:
  50. try:
  51. result = yield self.federation.make_query(
  52. destination=target_user.domain,
  53. query_type="profile",
  54. args={
  55. "user_id": user_id,
  56. },
  57. ignore_backoff=True,
  58. )
  59. defer.returnValue(result)
  60. except CodeMessageException as e:
  61. if e.code != 404:
  62. logger.exception("Failed to get displayname")
  63. raise
  64. @defer.inlineCallbacks
  65. def get_profile_from_cache(self, user_id):
  66. """Get the profile information from our local cache. If the user is
  67. ours then the profile information will always be corect. Otherwise,
  68. it may be out of date/missing.
  69. """
  70. target_user = UserID.from_string(user_id)
  71. if self.hs.is_mine(target_user):
  72. displayname = yield self.store.get_profile_displayname(
  73. target_user.localpart
  74. )
  75. avatar_url = yield self.store.get_profile_avatar_url(
  76. target_user.localpart
  77. )
  78. defer.returnValue({
  79. "displayname": displayname,
  80. "avatar_url": avatar_url,
  81. })
  82. else:
  83. profile = yield self.store.get_from_remote_profile_cache(user_id)
  84. defer.returnValue(profile or {})
  85. @defer.inlineCallbacks
  86. def get_displayname(self, target_user):
  87. if self.hs.is_mine(target_user):
  88. displayname = yield self.store.get_profile_displayname(
  89. target_user.localpart
  90. )
  91. defer.returnValue(displayname)
  92. else:
  93. try:
  94. result = yield self.federation.make_query(
  95. destination=target_user.domain,
  96. query_type="profile",
  97. args={
  98. "user_id": target_user.to_string(),
  99. "field": "displayname",
  100. },
  101. ignore_backoff=True,
  102. )
  103. except CodeMessageException as e:
  104. if e.code != 404:
  105. logger.exception("Failed to get displayname")
  106. raise
  107. except Exception:
  108. logger.exception("Failed to get displayname")
  109. else:
  110. defer.returnValue(result["displayname"])
  111. @defer.inlineCallbacks
  112. def set_displayname(self, target_user, requester, new_displayname, by_admin=False):
  113. """target_user is the user whose displayname is to be changed;
  114. auth_user is the user attempting to make this change."""
  115. if not self.hs.is_mine(target_user):
  116. raise SynapseError(400, "User is not hosted on this Home Server")
  117. if not by_admin and target_user != requester.user:
  118. raise AuthError(400, "Cannot set another user's displayname")
  119. if new_displayname == '':
  120. new_displayname = None
  121. yield self.store.set_profile_displayname(
  122. target_user.localpart, new_displayname
  123. )
  124. if self.hs.config.user_directory_search_all_users:
  125. profile = yield self.store.get_profileinfo(target_user.localpart)
  126. yield self.user_directory_handler.handle_local_profile_change(
  127. target_user.to_string(), profile
  128. )
  129. yield self._update_join_states(requester, target_user)
  130. @defer.inlineCallbacks
  131. def get_avatar_url(self, target_user):
  132. if self.hs.is_mine(target_user):
  133. avatar_url = yield self.store.get_profile_avatar_url(
  134. target_user.localpart
  135. )
  136. defer.returnValue(avatar_url)
  137. else:
  138. try:
  139. result = yield self.federation.make_query(
  140. destination=target_user.domain,
  141. query_type="profile",
  142. args={
  143. "user_id": target_user.to_string(),
  144. "field": "avatar_url",
  145. },
  146. ignore_backoff=True,
  147. )
  148. except CodeMessageException as e:
  149. if e.code != 404:
  150. logger.exception("Failed to get avatar_url")
  151. raise
  152. except Exception:
  153. logger.exception("Failed to get avatar_url")
  154. defer.returnValue(result["avatar_url"])
  155. @defer.inlineCallbacks
  156. def set_avatar_url(self, target_user, requester, new_avatar_url, by_admin=False):
  157. """target_user is the user whose avatar_url is to be changed;
  158. auth_user is the user attempting to make this change."""
  159. if not self.hs.is_mine(target_user):
  160. raise SynapseError(400, "User is not hosted on this Home Server")
  161. if not by_admin and target_user != requester.user:
  162. raise AuthError(400, "Cannot set another user's avatar_url")
  163. yield self.store.set_profile_avatar_url(
  164. target_user.localpart, new_avatar_url
  165. )
  166. if self.hs.config.user_directory_search_all_users:
  167. profile = yield self.store.get_profileinfo(target_user.localpart)
  168. yield self.user_directory_handler.handle_local_profile_change(
  169. target_user.to_string(), profile
  170. )
  171. yield self._update_join_states(requester, target_user)
  172. @defer.inlineCallbacks
  173. def on_profile_query(self, args):
  174. user = UserID.from_string(args["user_id"])
  175. if not self.hs.is_mine(user):
  176. raise SynapseError(400, "User is not hosted on this Home Server")
  177. just_field = args.get("field", None)
  178. response = {}
  179. if just_field is None or just_field == "displayname":
  180. response["displayname"] = yield self.store.get_profile_displayname(
  181. user.localpart
  182. )
  183. if just_field is None or just_field == "avatar_url":
  184. response["avatar_url"] = yield self.store.get_profile_avatar_url(
  185. user.localpart
  186. )
  187. defer.returnValue(response)
  188. @defer.inlineCallbacks
  189. def _update_join_states(self, requester, target_user):
  190. if not self.hs.is_mine(target_user):
  191. return
  192. yield self.ratelimit(requester)
  193. room_ids = yield self.store.get_rooms_for_user(
  194. target_user.to_string(),
  195. )
  196. for room_id in room_ids:
  197. handler = self.hs.get_room_member_handler()
  198. try:
  199. # Assume the target_user isn't a guest,
  200. # because we don't let guests set profile or avatar data.
  201. yield handler.update_membership(
  202. requester,
  203. target_user,
  204. room_id,
  205. "join", # We treat a profile update like a join.
  206. ratelimit=False, # Try to hide that these events aren't atomic.
  207. )
  208. except Exception as e:
  209. logger.warn(
  210. "Failed to update join event for room %s - %s",
  211. room_id, str(e.message)
  212. )
  213. def _update_remote_profile_cache(self):
  214. """Called periodically to check profiles of remote users we haven't
  215. checked in a while.
  216. """
  217. entries = yield self.store.get_remote_profile_cache_entries_that_expire(
  218. last_checked=self.clock.time_msec() - self.PROFILE_UPDATE_EVERY_MS
  219. )
  220. for user_id, displayname, avatar_url in entries:
  221. is_subscribed = yield self.store.is_subscribed_remote_profile_for_user(
  222. user_id,
  223. )
  224. if not is_subscribed:
  225. yield self.store.maybe_delete_remote_profile_cache(user_id)
  226. continue
  227. try:
  228. profile = yield self.federation.make_query(
  229. destination=get_domain_from_id(user_id),
  230. query_type="profile",
  231. args={
  232. "user_id": user_id,
  233. },
  234. ignore_backoff=True,
  235. )
  236. except Exception:
  237. logger.exception("Failed to get avatar_url")
  238. yield self.store.update_remote_profile_cache(
  239. user_id, displayname, avatar_url
  240. )
  241. continue
  242. new_name = profile.get("displayname")
  243. new_avatar = profile.get("avatar_url")
  244. # We always hit update to update the last_check timestamp
  245. yield self.store.update_remote_profile_cache(
  246. user_id, new_name, new_avatar
  247. )