slixmpp/sleekxmpp/plugins/xep_0153/vcard_avatar.py

153 lines
5.3 KiB
Python
Raw Normal View History

2012-04-10 01:41:59 +00:00
"""
SleekXMPP: The Sleek XMPP Library
Copyright (C) 2012 Nathanael C. Fritz, Lance J.T. Stout
This file is part of SleekXMPP.
See the file LICENSE for copying permission.
"""
import hashlib
import logging
2012-07-25 08:36:31 +00:00
import threading
2012-04-10 01:41:59 +00:00
from sleekxmpp.stanza import Presence
from sleekxmpp.exceptions import XMPPError
2012-04-10 01:41:59 +00:00
from sleekxmpp.xmlstream import register_stanza_plugin
from sleekxmpp.plugins.base import BasePlugin
from sleekxmpp.plugins.xep_0153 import stanza, VCardTempUpdate
log = logging.getLogger(__name__)
class XEP_0153(BasePlugin):
name = 'xep_0153'
description = 'XEP-0153: vCard-Based Avatars'
dependencies = set(['xep_0054'])
stanza = stanza
def plugin_init(self):
self._hashes = {}
2012-07-25 08:36:31 +00:00
self._allow_advertising = threading.Event()
2012-04-10 01:41:59 +00:00
register_stanza_plugin(Presence, VCardTempUpdate)
self.xmpp.add_filter('out', self._update_presence)
self.xmpp.add_event_handler('session_start', self._start)
2012-07-25 08:36:31 +00:00
self.xmpp.add_event_handler('session_end', self._end)
2012-04-10 01:41:59 +00:00
self.xmpp.add_event_handler('presence_available', self._recv_presence)
self.xmpp.add_event_handler('presence_dnd', self._recv_presence)
self.xmpp.add_event_handler('presence_xa', self._recv_presence)
self.xmpp.add_event_handler('presence_chat', self._recv_presence)
self.xmpp.add_event_handler('presence_away', self._recv_presence)
self.api.register(self._set_hash, 'set_hash', default=True)
self.api.register(self._get_hash, 'get_hash', default=True)
2012-07-25 08:36:31 +00:00
self.api.register(self._reset_hash, 'reset_hash', default=True)
2012-04-10 01:41:59 +00:00
def plugin_end(self):
self.xmpp.del_filter('out', self._update_presence)
self.xmpp.del_event_handler('session_start', self._start)
2012-07-25 08:36:31 +00:00
self.xmpp.del_event_handler('session_end', self._end)
self.xmpp.del_event_handler('presence_available', self._recv_presence)
self.xmpp.del_event_handler('presence_dnd', self._recv_presence)
self.xmpp.del_event_handler('presence_xa', self._recv_presence)
self.xmpp.del_event_handler('presence_chat', self._recv_presence)
self.xmpp.del_event_handler('presence_away', self._recv_presence)
2012-06-19 08:29:48 +00:00
def set_avatar(self, jid=None, avatar=None, mtype=None, block=True,
2012-04-10 01:41:59 +00:00
timeout=None, callback=None):
2012-07-25 08:36:31 +00:00
if jid is None:
jid = self.xmpp.boundjid.bare
2012-04-10 01:41:59 +00:00
vcard = self.xmpp['xep_0054'].get_vcard(jid, cached=True)
vcard = vcard['vcard_temp']
vcard['PHOTO']['TYPE'] = mtype
vcard['PHOTO']['BINVAL'] = avatar
2012-07-25 08:36:31 +00:00
2012-04-10 01:41:59 +00:00
self.xmpp['xep_0054'].publish_vcard(jid=jid, vcard=vcard)
2012-07-25 08:36:31 +00:00
self.api['reset_hash'](jid)
self.xmpp.roster[jid].send_last_presence()
2012-04-10 01:41:59 +00:00
def _start(self, event):
2013-01-17 06:32:40 +00:00
try:
vcard = self.xmpp['xep_0054'].get_vcard(self.xmpp.boundjid.bare)
2013-01-17 06:32:40 +00:00
data = vcard['vcard_temp']['PHOTO']['BINVAL']
if not data:
new_hash = ''
else:
new_hash = hashlib.sha1(data).hexdigest()
self.api['set_hash'](self.xmpp.boundjid, args=new_hash)
self._allow_advertising.set()
2013-01-17 06:32:40 +00:00
except XMPPError:
2013-01-17 07:06:13 +00:00
log.debug('Could not retrieve vCard for %s' % self.xmpp.boundjid.bare)
2013-01-17 06:32:40 +00:00
2012-07-25 08:36:31 +00:00
def _end(self, event):
self._allow_advertising.clear()
2012-04-10 01:41:59 +00:00
def _update_presence(self, stanza):
if not isinstance(stanza, Presence):
return stanza
if stanza['type'] not in ('available', 'dnd', 'chat', 'away', 'xa'):
return stanza
2012-04-10 01:41:59 +00:00
current_hash = self.api['get_hash'](stanza['from'])
stanza['vcard_temp_update']['photo'] = current_hash
return stanza
2012-07-25 08:36:31 +00:00
def _reset_hash(self, jid, node, ifrom, args):
own_jid = (jid.bare == self.xmpp.boundjid.bare)
2012-04-10 01:41:59 +00:00
if self.xmpp.is_component:
own_jid = (jid.domain == self.xmpp.boundjid.domain)
2012-06-19 08:29:48 +00:00
2012-04-10 01:41:59 +00:00
self.api['set_hash'](jid, args=None)
if own_jid:
self.xmpp.roster[jid].send_last_presence()
try:
iq = self.xmpp['xep_0054'].get_vcard(jid=jid.bare, ifrom=ifrom)
2012-07-25 08:36:31 +00:00
data = iq['vcard_temp']['PHOTO']['BINVAL']
if not data:
new_hash = ''
else:
new_hash = hashlib.sha1(data).hexdigest()
2012-07-25 08:36:31 +00:00
self.api['set_hash'](jid, args=new_hash)
except XMPPError:
log.debug('Could not retrieve vCard for %s' % jid)
2012-04-10 01:41:59 +00:00
def _recv_presence(self, pres):
try:
if pres['muc']['affiliation']:
# Don't process vCard avatars for MUC occupants
# since they all share the same bare JID.
return
except: pass
2012-04-10 01:41:59 +00:00
if not pres.match('presence/vcard_temp_update'):
self.api['set_hash'](pres['from'], args=None)
return
2012-07-25 08:36:31 +00:00
2012-04-10 01:41:59 +00:00
data = pres['vcard_temp_update']['photo']
if data is None:
return
elif data == '' or data != self.api['get_hash'](pres['from']):
ifrom = pres['to'] if self.xmpp.is_component else None
self.api['reset_hash'](pres['from'], ifrom=ifrom)
2012-07-21 01:07:27 +00:00
self.xmpp.event('vcard_avatar_update', pres)
2012-04-10 01:41:59 +00:00
# =================================================================
def _get_hash(self, jid, node, ifrom, args):
return self._hashes.get(jid.bare, None)
def _set_hash(self, jid, node, ifrom, args):
self._hashes[jid.bare] = args