Mercurial > libervia-backend
view sat/plugins/plugin_xep_0054.py @ 3254:6cf4bd6972c2
core, frontends: avatar refactoring:
/!\ huge commit
Avatar logic has been reworked around the IDENTITY plugin: plugins able to handle avatar
or other identity related metadata (like nicknames) register to IDENTITY plugin in the
same way as for other features like download/upload. Once registered, IDENTITY plugin will
call them when suitable in order of priority, and handle caching.
Methods to manage those metadata from frontend now use serialised data.
For now `avatar` and `nicknames` are handled:
- `avatar` is now a dict with `path` + metadata like `media_type`, instead of just a string
path
- `nicknames` is now a list of nicknames in order of priority. This list is never empty,
and `nicknames[0]` should be the preferred nickname to use by frontends in most cases.
In addition to contact specified nicknames, user set nickname (the one set in roster) is
used in priority when available.
Among the side changes done with this commit, there are:
- a new `contactGet` bridge method to get roster metadata for a single contact
- SatPresenceProtocol.send returns a Deferred to check when it has actually been sent
- memory's methods to handle entities data now use `client` as first argument
- metadata filter can be specified with `getIdentity`
- `getAvatar` and `setAvatar` are now part of the IDENTITY plugin instead of XEP-0054 (and
there signature has changed)
- `isRoom` and `getBareOrFull` are now part of XEP-0045 plugin
- jp avatar/get command uses `xdg-open` first when available for `--show` flag
- `--no-cache` has been added to jp avatar/get and identity/get
- jp identity/set has been simplified, explicit options (`--nickname` only for now) are
used instead of `--field`. `--field` may come back in the future if necessary for extra
data.
- QuickContactList `SetContact` now handle None as a value, and doesn't use it to delete the
metadata anymore
- improved cache handling for `metadata` and `nicknames` in quick frontend
- new `default` argument in QuickContactList `getCache`
author | Goffi <goffi@goffi.org> |
---|---|
date | Tue, 14 Apr 2020 21:00:33 +0200 |
parents | 5afd7416ca2d |
children | 7aa01e262e05 |
line wrap: on
line source
#!/usr/bin/env python3 # SAT plugin for managing xep-0054 # Copyright (C) 2009-2020 Jérôme Poisson (goffi@goffi.org) # Copyright (C) 2014 Emmanuel Gil Peyrot (linkmauve@linkmauve.fr) # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU Affero General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU Affero General Public License for more details. # You should have received a copy of the GNU Affero General Public License # along with this program. If not, see <http://www.gnu.org/licenses/>. import io from base64 import b64decode, b64encode from hashlib import sha1 from pathlib import Path from zope.interface import implementer from twisted.internet import threads, defer from twisted.words.protocols.jabber import jid, error from twisted.words.xish import domish from twisted.python.failure import Failure from wokkel import disco, iwokkel from sat.core import exceptions from sat.core.i18n import _ from sat.core.constants import Const as C from sat.core.log import getLogger from sat.memory import persistent from sat.tools import image log = getLogger(__name__) try: from PIL import Image except: raise exceptions.MissingModule( "Missing module pillow, please download/install it from https://python-pillow.github.io" ) try: from twisted.words.protocols.xmlstream import XMPPHandler except ImportError: from wokkel.subprotocols import XMPPHandler PLUGIN_INFO = { C.PI_NAME: "XEP 0054 Plugin", C.PI_IMPORT_NAME: "XEP-0054", C.PI_TYPE: "XEP", C.PI_PROTOCOLS: ["XEP-0054", "XEP-0153"], C.PI_DEPENDENCIES: ["IDENTITY"], C.PI_RECOMMENDATIONS: [], C.PI_MAIN: "XEP_0054", C.PI_HANDLER: "yes", C.PI_DESCRIPTION: _("""Implementation of vcard-temp"""), } AVATAR_PATH = "avatars" AVATAR_DIM = (128, 128) IQ_GET = '/iq[@type="get"]' NS_VCARD = "vcard-temp" VCARD_REQUEST = IQ_GET + '/vCard[@xmlns="' + NS_VCARD + '"]' # TODO: manage requests PRESENCE = "/presence" NS_VCARD_UPDATE = "vcard-temp:x:update" VCARD_UPDATE = PRESENCE + '/x[@xmlns="' + NS_VCARD_UPDATE + '"]' HASH_SHA1_EMPTY = "da39a3ee5e6b4b0d3255bfef95601890afd80709" class XEP_0054(object): def __init__(self, host): log.info(_("Plugin XEP_0054 initialization")) self.host = host self._i = host.plugins['IDENTITY'] self._i.register('avatar', self.getAvatar, self.setAvatar) self._i.register('nicknames', self.getNicknames, self.setNicknames) host.trigger.add("presence_available", self.presenceAvailableTrigger) def getHandler(self, client): return XEP_0054_handler(self) def presenceAvailableTrigger(self, presence_elt, client): try: avatar_hash = client._xep_0054_avatar_hashes[client.jid.userhost()] except KeyError: log.info( _("No avatar in cache for {profile}") .format(profile=client.profile)) return True x_elt = domish.Element((NS_VCARD_UPDATE, "x")) x_elt.addElement("photo", content=avatar_hash) presence_elt.addChild(x_elt) return True async def profileConnecting(self, client): client._xep_0054_avatar_hashes = persistent.PersistentDict( NS_VCARD, client.profile) await client._xep_0054_avatar_hashes.load() def getCache(self, client, entity_jid, name): """return cached value for jid @param entity_jid(jid.JID): target contact @param name(unicode): name of the value ('nick' or 'avatar') @return(unicode, None): wanted value or None""" entity_jid = self._i.getIdentityJid(client, entity_jid) try: data = self.host.memory.getEntityData(client, entity_jid, [name]) except exceptions.UnknownEntityError: return None return data.get(name) def savePhoto(self, client, photo_elt, entity_jid): """Parse a <PHOTO> photo_elt and save the picture""" # XXX: this method is launched in a separate thread try: mime_type = str(next(photo_elt.elements(NS_VCARD, "TYPE"))) except StopIteration: mime_type = None else: if not mime_type: # MIME type not known, we'll try autodetection below mime_type = None elif mime_type == "image/x-png": # XXX: this old MIME type is still used by some clients mime_type = "image/png" try: buf = str(next(photo_elt.elements(NS_VCARD, "BINVAL"))) except StopIteration: log.warning("BINVAL element not found") raise Failure(exceptions.NotFound()) if not buf: log.warning("empty avatar for {jid}".format(jid=entity_jid.full())) raise Failure(exceptions.NotFound()) log.debug(_("Decoding binary")) decoded = b64decode(buf) del buf if mime_type is None: log.debug( f"no media type found specified for {entity_jid}'s avatar, trying to " f"guess") try: mime_type = image.guess_type(io.BytesIO(decoded)) except IOError as e: log.warning(f"Can't open avatar buffer: {e}") if mime_type is None: msg = f"Can't find media type for {entity_jid}'s avatar" log.warning(msg) raise Failure(exceptions.DataError(msg)) image_hash = sha1(decoded).hexdigest() with self.host.common_cache.cacheData( PLUGIN_INFO["import_name"], image_hash, mime_type, ) as f: f.write(decoded) return image_hash async def vCard2Dict(self, client, vcard_elt, entity_jid): """Convert a VCard_elt to a dict, and save binaries""" log.debug(("parsing vcard_elt")) vcard_dict = {} for elem in vcard_elt.elements(): if elem.name == "FN": vcard_dict["fullname"] = str(elem) elif elem.name == "NICKNAME": nickname = vcard_dict["nickname"] = str(elem) await self._i.update( client, "nicknames", [nickname], entity_jid ) elif elem.name == "URL": vcard_dict["website"] = str(elem) elif elem.name == "EMAIL": vcard_dict["email"] = str(elem) elif elem.name == "BDAY": vcard_dict["birthday"] = str(elem) elif elem.name == "PHOTO": # TODO: handle EXTVAL try: avatar_hash = await threads.deferToThread( self.savePhoto, client, elem, entity_jid ) except (exceptions.DataError, exceptions.NotFound): avatar_hash = "" vcard_dict["avatar"] = avatar_hash except Exception as e: log.error(f"avatar saving error: {e}") avatar_hash = None else: vcard_dict["avatar"] = avatar_hash if avatar_hash is not None: await client._xep_0054_avatar_hashes.aset( entity_jid.full(), avatar_hash) if avatar_hash: avatar_cache = self.host.common_cache.getMetadata(avatar_hash) await self._i.update( client, "avatar", { 'path': avatar_cache['path'], 'media_type': avatar_cache['mime_type'], 'cache_uid': avatar_hash }, entity_jid ) else: await self._i.update(client, "avatar", None, entity_jid) else: log.debug("FIXME: [{}] VCard_elt tag is not managed yet".format(elem.name)) return vcard_dict async def getVCardElement(self, client, entity_jid): """Retrieve domish.Element of a VCard @param entity_jid(jid.JID): entity from who we need the vCard @raise DataError: we got an invalid answer """ iq_elt = client.IQ("get") iq_elt["from"] = client.jid.full() iq_elt["to"] = entity_jid.full() iq_elt.addElement("vCard", NS_VCARD) iq_ret_elt = await iq_elt.send(entity_jid.full()) try: return next(iq_ret_elt.elements(NS_VCARD, "vCard")) except StopIteration: log.warning(_( "vCard element not found for {entity_jid}: {xml}" ).format(entity_jid=entity_jid, xml=iq_ret_elt.toXml())) raise exceptions.DataError(f"no vCard element found for {entity_jid}") async def updateVCardElt(self, client, entity_jid, to_replace): """Create a vcard element to replace some metadata @param to_replace(list[str]): list of vcard element names to remove """ try: # we first check if a vcard already exists, to keep data vcard_elt = await self.getVCardElement(client, entity_jid) except error.StanzaError as e: if e.condition == "item-not-found": vcard_elt = domish.Element((NS_VCARD, "vCard")) else: raise e except exceptions.DataError: vcard_elt = domish.Element((NS_VCARD, "vCard")) else: # the vcard exists, we need to remove elements that we'll replace for elt_name in to_replace: try: elt = next(vcard_elt.elements(NS_VCARD, elt_name)) except StopIteration: pass else: vcard_elt.children.remove(elt) return vcard_elt async def getCard(self, client, entity_jid): """Ask server for VCard @param entity_jid(jid.JID): jid from which we want the VCard @result(dict): vCard data """ entity_jid = self._i.getIdentityJid(client, entity_jid) log.debug(f"Asking for {entity_jid}'s VCard") try: vcard_elt = await self.getVCardElement(client, entity_jid) except exceptions.DataError: self._i.update(client, "avatar", None, entity_jid) except Exception as e: log.warning(_( "Can't get vCard for {entity_jid}: {e}" ).format(entity_jid=entity_jid, e=e)) else: log.debug(_("VCard found")) return await self.vCard2Dict(client, vcard_elt, entity_jid) async def getAvatar(self, client, entity_jid): """Get avatar full path or hash if avatar is not in local cache, it will be requested to the server @param entity(jid.JID): entity to get avatar from """ entity_jid = self._i.getIdentityJid(client, entity_jid) hashes_cache = client._xep_0054_avatar_hashes try: avatar_hash = hashes_cache[entity_jid.full()] except KeyError: log.debug(f"avatar for {entity_jid} is not in cache, we retrieve it") vcard = await self.getCard(client, entity_jid) if vcard is None: return None avatar_hash = hashes_cache[entity_jid.full()] if not avatar_hash: return None avatar_cache = self.host.common_cache.getMetadata(avatar_hash) if avatar_cache is None: log.debug("avatar is no more in cache, we re-download it") vcard = await self.getCard(client, entity_jid) if vcard is None: return None avatar_cache = self.host.common_cache.getMetadata(avatar_hash) return self._i.avatarBuildMetadata( avatar_cache['path'], avatar_cache['mime_type'], avatar_hash) def _buildSetAvatar(self, client, vcard_elt, avatar_data): # XXX: this method is executed in a separate thread try: img = Image.open(avatar_data['path']) except IOError as e: raise exceptions.DataError(f"Can't open image: {e}") if img.size != AVATAR_DIM: img.thumbnail(AVATAR_DIM) if img.size[0] != img.size[1]: # we need to crop first left, upper = (0, 0) right, lower = img.size offset = abs(right - lower) / 2 if right == min(img.size): upper += offset lower -= offset else: left += offset right -= offset img = img.crop((left, upper, right, lower)) img_buf = io.BytesIO() img.save(img_buf, "PNG") photo_elt = vcard_elt.addElement("PHOTO") photo_elt.addElement("TYPE", content="image/png") image_b64 = b64encode(img_buf.getvalue()).decode() photo_elt.addElement("BINVAL", content=image_b64) image_hash = sha1(img_buf.getvalue()).hexdigest() with self.host.common_cache.cacheData( PLUGIN_INFO["import_name"], image_hash, "image/png" ) as f: f.write(img_buf.getvalue()) avatar_data['path'] = Path(f.name) avatar_data['cache_uid'] = image_hash return image_hash async def setAvatar(self, client, avatar_data, entity): """Set avatar of the profile @param avatar_data(dict): data of the image to use as avatar, as built by IDENTITY plugin. @param entity(jid.JID): entity whose avatar must be changed """ vcard_elt = await self.updateVCardElt(client, entity, ['PHOTO']) iq_elt = client.IQ() iq_elt.addChild(vcard_elt) await threads.deferToThread( self._buildSetAvatar, client, vcard_elt, avatar_data ) # image is now at the right size/format await iq_elt.send() # FIXME: should send the current presence, not always "available" ! await client.presence.available() async def getNicknames(self, client, entity): """get nick from cache, or check vCard @param entity(jid.JID): entity to get nick from @return(list[str]): nicknames found """ vcard_data = await self.getCard(client, entity) try: return [vcard_data['nickname']] except (KeyError, TypeError): return [] async def setNicknames(self, client, nicknames, entity): """Update our vCard and set a nickname @param nicknames(list[str]): new nicknames to use only first item of this list will be used here """ nick = nicknames[0].strip() vcard_elt = await self.updateVCardElt(client, entity, ['NICKNAME']) if nick: vcard_elt.addElement((NS_VCARD, "NICKNAME"), content=nick) iq_elt = client.IQ() iq_elt.addChild(vcard_elt) await iq_elt.send() @implementer(iwokkel.IDisco) class XEP_0054_handler(XMPPHandler): def __init__(self, plugin_parent): self.plugin_parent = plugin_parent self.host = plugin_parent.host def connectionInitialized(self): self.xmlstream.addObserver(VCARD_UPDATE, self._update) def getDiscoInfo(self, requestor, target, nodeIdentifier=""): return [disco.DiscoFeature(NS_VCARD)] def getDiscoItems(self, requestor, target, nodeIdentifier=""): return [] def _checkAvatarHash(self, client, entity, given_hash): """Check that hash in cache (i.e. computed hash) is the same as given one""" # XXX: if they differ, the avatar will be requested on each connection # TODO: try to avoid re-requesting avatar in this case computed_hash = client._xep_0054_avatar_hashes[entity.full] if computed_hash != given_hash: log.warning( "computed hash differs from given hash for {entity}:\n" "computed: {computed}\ngiven: {given}".format( entity=entity, computed=computed_hash, given=given_hash ) ) async def update(self, presence): """Called on <presence/> stanza with vcard data Check for avatar information, and get VCard if needed @param presence(domish.Element): <presence/> stanza """ client = self.parent entity_jid = self.plugin_parent._i.getIdentityJid( client, jid.JID(presence["from"])) try: x_elt = next(presence.elements(NS_VCARD_UPDATE, "x")) except StopIteration: return try: photo_elt = next(x_elt.elements(NS_VCARD_UPDATE, "photo")) except StopIteration: return new_hash = str(photo_elt).strip() if new_hash == HASH_SHA1_EMPTY: new_hash = "" hashes_cache = client._xep_0054_avatar_hashes old_hash = hashes_cache.get(entity_jid.full()) if old_hash == new_hash: # no change, we can return… if new_hash: # …but we double check that avatar is in cache avatar_cache = self.host.common_cache.getMetadata(new_hash) if avatar_cache is None: log.debug( f"Avatar for [{entity_jid}] is known but not in cache, we get " f"it" ) # getCard will put the avatar in cache await self.plugin_parent.getCard(client, entity_jid) else: log.debug(f"avatar for {entity_jid} is already in cache") return if new_hash is None: # XXX: we use empty string to indicate that there is no avatar new_hash = "" await hashes_cache.aset(entity_jid.full(), new_hash) if not new_hash: await self.plugin_parent._i.update( client, "avatar", None, entity_jid) # the avatar has been removed, no need to go further return avatar_cache = self.host.common_cache.getMetadata(new_hash) if avatar_cache is not None: log.debug( f"New avatar found for [{entity_jid}], it's already in cache, we use it" ) await self.plugin_parent._i.update( client, "avatar", { 'path': avatar_cache['path'], 'media_type': avatar_cache['mime_type'], 'cache_uid': new_hash, }, entity_jid ) else: log.debug( "New avatar found for [{entity_jid}], requesting vcard" ) vcard = await self.plugin_parent.getCard(client, entity_jid) if vcard is None: log.warning(f"Unexpected empty vCard for {entity_jid}") return await self._checkAvatarHash(client, entity_jid, new_hash) def _update(self, presence): defer.ensureDeferred(self.update(presence))