2018-06-30 19:23:10 +02:00
|
|
|
# This file is part of Gajim.
|
|
|
|
#
|
|
|
|
# Gajim is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published
|
|
|
|
# by the Free Software Foundation; version 3 only.
|
|
|
|
#
|
|
|
|
# Gajim 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 General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with Gajim. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
# XEP-0153: vCard-Based Avatars
|
|
|
|
|
|
|
|
import os
|
2018-09-01 20:34:56 +02:00
|
|
|
from pathlib import Path
|
2018-06-30 19:23:10 +02:00
|
|
|
|
|
|
|
import nbxmpp
|
2018-12-28 13:38:15 +01:00
|
|
|
from nbxmpp.structs import StanzaHandler
|
|
|
|
from nbxmpp.const import AvatarState
|
2018-06-30 19:23:10 +02:00
|
|
|
|
|
|
|
from gajim.common import app
|
|
|
|
from gajim.common import helpers
|
|
|
|
from gajim.common import configpaths
|
|
|
|
from gajim.common.const import RequestAvatar
|
2019-03-09 23:16:27 +01:00
|
|
|
from gajim.common.modules.base import BaseModule
|
2018-06-30 19:23:10 +02:00
|
|
|
|
|
|
|
|
2019-03-09 23:16:27 +01:00
|
|
|
class VCardAvatars(BaseModule):
|
2018-06-30 19:23:10 +02:00
|
|
|
def __init__(self, con):
|
2019-03-09 23:16:27 +01:00
|
|
|
BaseModule.__init__(self, con)
|
2018-06-30 19:23:10 +02:00
|
|
|
self._requested_shas = []
|
|
|
|
|
|
|
|
self.handlers = [
|
2018-12-28 13:38:15 +01:00
|
|
|
StanzaHandler(name='presence',
|
|
|
|
callback=self._presence_received,
|
|
|
|
ns=nbxmpp.NS_VCARD_UPDATE,
|
|
|
|
priority=51),
|
2018-06-30 19:23:10 +02:00
|
|
|
]
|
|
|
|
|
|
|
|
self.avatar_advertised = False
|
2018-09-01 20:34:56 +02:00
|
|
|
self._find_own_avatar()
|
|
|
|
|
|
|
|
def _find_own_avatar(self):
|
|
|
|
sha = app.config.get_per('accounts', self._account, 'avatar_sha')
|
|
|
|
if not sha:
|
|
|
|
return
|
|
|
|
path = Path(configpaths.get('AVATAR')) / sha
|
|
|
|
if not path.exists():
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.info('Missing own avatar, reset sha')
|
2018-09-01 20:34:56 +02:00
|
|
|
app.config.set_per('accounts', self._account, 'avatar_sha', '')
|
2018-06-30 19:23:10 +02:00
|
|
|
|
2018-12-28 13:38:15 +01:00
|
|
|
def _presence_received(self, _con, _stanza, properties):
|
2019-01-11 20:45:11 +01:00
|
|
|
if not properties.type.is_available:
|
|
|
|
return
|
|
|
|
|
2018-12-28 13:38:15 +01:00
|
|
|
if properties.avatar_state in (AvatarState.IGNORE,
|
|
|
|
AvatarState.NOT_READY):
|
2018-06-30 19:23:10 +02:00
|
|
|
return
|
|
|
|
|
2018-12-28 13:38:15 +01:00
|
|
|
if self._con.get_own_jid().bareMatch(properties.jid):
|
|
|
|
if self._con.get_own_jid() == properties.jid:
|
2018-09-01 20:34:56 +02:00
|
|
|
# Initial presence reflection
|
|
|
|
if self._con.avatar_conversion:
|
|
|
|
# XEP-0398: Tells us the current avatar sha on the
|
|
|
|
# inital presence reflection
|
2018-12-28 13:38:15 +01:00
|
|
|
self._self_update_received(properties)
|
2018-09-01 20:34:56 +02:00
|
|
|
else:
|
|
|
|
# Presence from another resource of ours
|
2018-12-28 13:38:15 +01:00
|
|
|
self._self_update_received(properties)
|
2018-06-30 19:23:10 +02:00
|
|
|
return
|
|
|
|
|
2018-12-28 13:38:15 +01:00
|
|
|
if properties.from_muc:
|
|
|
|
self._gc_update_received(properties)
|
2018-06-30 19:23:10 +02:00
|
|
|
else:
|
2018-12-28 13:38:15 +01:00
|
|
|
# Check if presence is from a MUC service
|
|
|
|
contact = app.contacts.get_groupchat_contact(self._account,
|
|
|
|
str(properties.jid))
|
|
|
|
self._update_received(properties, room=contact is not None)
|
|
|
|
|
|
|
|
def _self_update_received(self, properties):
|
|
|
|
jid = properties.jid.getBare()
|
|
|
|
if properties.avatar_state == AvatarState.EMPTY:
|
2018-06-30 19:23:10 +02:00
|
|
|
# Empty <photo/> tag, means no avatar is advertised
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.info('%s has no avatar published', properties.jid)
|
2018-09-01 20:34:56 +02:00
|
|
|
app.config.set_per('accounts', self._account, 'avatar_sha', '')
|
|
|
|
app.contacts.set_avatar(self._account, jid, None)
|
|
|
|
app.interface.update_avatar(self._account, jid)
|
2018-06-30 19:23:10 +02:00
|
|
|
return
|
|
|
|
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.info('Update: %s %s', jid, properties.avatar_sha)
|
2018-06-30 19:23:10 +02:00
|
|
|
current_sha = app.config.get_per(
|
|
|
|
'accounts', self._account, 'avatar_sha')
|
|
|
|
|
2018-12-28 13:38:15 +01:00
|
|
|
if properties.avatar_sha != current_sha:
|
|
|
|
path = Path(configpaths.get('AVATAR')) / properties.avatar_sha
|
2018-09-01 20:34:56 +02:00
|
|
|
if path.exists():
|
2018-12-28 13:38:15 +01:00
|
|
|
app.config.set_per('accounts', self._account,
|
|
|
|
'avatar_sha', properties.avatar_sha)
|
|
|
|
app.contacts.set_avatar(self._account,
|
|
|
|
jid,
|
|
|
|
properties.avatar_sha)
|
2018-09-01 20:34:56 +02:00
|
|
|
app.interface.update_avatar(self._account, jid)
|
|
|
|
else:
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.info('Request : %s', jid)
|
2018-09-01 20:34:56 +02:00
|
|
|
self._con.get_module('VCardTemp').request_vcard(
|
|
|
|
RequestAvatar.SELF)
|
2018-06-30 19:23:10 +02:00
|
|
|
else:
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.info('Avatar already known: %s %s',
|
|
|
|
jid, properties.avatar_sha)
|
2018-06-30 19:23:10 +02:00
|
|
|
|
2018-12-28 13:38:15 +01:00
|
|
|
def _update_received(self, properties, room=False):
|
|
|
|
jid = properties.jid.getBare()
|
|
|
|
if properties.avatar_state == AvatarState.EMPTY:
|
2018-06-30 19:23:10 +02:00
|
|
|
# Empty <photo/> tag, means no avatar is advertised
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.info('%s has no avatar published', properties.jid)
|
2018-06-30 19:23:10 +02:00
|
|
|
|
|
|
|
# Remove avatar
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.debug('Remove: %s', jid)
|
2018-06-30 19:23:10 +02:00
|
|
|
app.contacts.set_avatar(self._account, jid, None)
|
|
|
|
acc_jid = self._con.get_own_jid().getStripped()
|
|
|
|
if not room:
|
|
|
|
app.logger.set_avatar_sha(acc_jid, jid, None)
|
|
|
|
app.interface.update_avatar(
|
|
|
|
self._account, jid, room_avatar=room)
|
|
|
|
else:
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.info('Update: %s %s',
|
|
|
|
properties.jid, properties.avatar_sha)
|
2018-06-30 19:23:10 +02:00
|
|
|
current_sha = app.contacts.get_avatar_sha(self._account, jid)
|
|
|
|
|
2018-12-28 13:38:15 +01:00
|
|
|
if properties.avatar_sha == current_sha:
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.info('Avatar already known: %s %s',
|
|
|
|
jid, properties.avatar_sha)
|
2018-06-30 19:23:10 +02:00
|
|
|
return
|
|
|
|
|
|
|
|
if room:
|
|
|
|
# We dont save the room avatar hash in our DB, so check
|
|
|
|
# if we previously downloaded it
|
2018-12-28 13:38:15 +01:00
|
|
|
if app.interface.avatar_exists(properties.avatar_sha):
|
|
|
|
app.contacts.set_avatar(self._account,
|
|
|
|
jid,
|
|
|
|
properties.avatar_sha)
|
2018-06-30 19:23:10 +02:00
|
|
|
app.interface.update_avatar(
|
|
|
|
self._account, jid, room_avatar=room)
|
2018-07-06 23:54:29 +02:00
|
|
|
return
|
2018-06-30 19:23:10 +02:00
|
|
|
|
2018-12-28 13:38:15 +01:00
|
|
|
if properties.avatar_sha not in self._requested_shas:
|
|
|
|
self._requested_shas.append(properties.avatar_sha)
|
2018-06-30 19:23:10 +02:00
|
|
|
if room:
|
|
|
|
self._con.get_module('VCardTemp').request_vcard(
|
2018-12-28 13:38:15 +01:00
|
|
|
RequestAvatar.ROOM, jid, sha=properties.avatar_sha)
|
2018-06-30 19:23:10 +02:00
|
|
|
else:
|
|
|
|
self._con.get_module('VCardTemp').request_vcard(
|
2018-12-28 13:38:15 +01:00
|
|
|
RequestAvatar.USER, jid, sha=properties.avatar_sha)
|
2018-06-30 19:23:10 +02:00
|
|
|
|
2018-12-28 13:38:15 +01:00
|
|
|
def _gc_update_received(self, properties):
|
|
|
|
nick = properties.jid.getResource()
|
2018-06-30 19:23:10 +02:00
|
|
|
|
|
|
|
gc_contact = app.contacts.get_gc_contact(
|
2018-12-28 13:38:15 +01:00
|
|
|
self._account, properties.jid.getBare(), nick)
|
2018-06-30 19:23:10 +02:00
|
|
|
|
|
|
|
if gc_contact is None:
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.error('no gc contact found: %s', nick)
|
2018-06-30 19:23:10 +02:00
|
|
|
return
|
|
|
|
|
2018-12-28 13:38:15 +01:00
|
|
|
if properties.avatar_state == AvatarState.EMPTY:
|
2018-06-30 19:23:10 +02:00
|
|
|
# Empty <photo/> tag, means no avatar is advertised, remove avatar
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.info('%s has no avatar published', nick)
|
|
|
|
self._log.debug('Remove: %s', nick)
|
2018-06-30 19:23:10 +02:00
|
|
|
gc_contact.avatar_sha = None
|
|
|
|
app.interface.update_avatar(contact=gc_contact)
|
|
|
|
else:
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.info('Update: %s %s', nick, properties.avatar_sha)
|
2018-12-28 13:38:15 +01:00
|
|
|
path = os.path.join(configpaths.get('AVATAR'),
|
|
|
|
properties.avatar_sha)
|
2018-06-30 19:23:10 +02:00
|
|
|
if not os.path.isfile(path):
|
2018-12-28 13:38:15 +01:00
|
|
|
if properties.avatar_sha not in self._requested_shas:
|
2018-06-30 19:23:10 +02:00
|
|
|
app.log('avatar').info('Request: %s', nick)
|
2018-12-28 13:38:15 +01:00
|
|
|
self._requested_shas.append(properties.avatar_sha)
|
2018-06-30 19:23:10 +02:00
|
|
|
self._con.get_module('VCardTemp').request_vcard(
|
2018-12-28 13:38:15 +01:00
|
|
|
RequestAvatar.USER, str(properties.jid),
|
|
|
|
room=True, sha=properties.avatar_sha)
|
2018-06-30 19:23:10 +02:00
|
|
|
return
|
|
|
|
|
2018-12-28 13:38:15 +01:00
|
|
|
if gc_contact.avatar_sha != properties.avatar_sha:
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.info('%s changed their Avatar: %s',
|
|
|
|
nick, properties.avatar_sha)
|
2018-12-28 13:38:15 +01:00
|
|
|
gc_contact.avatar_sha = properties.avatar_sha
|
2018-06-30 19:23:10 +02:00
|
|
|
app.interface.update_avatar(contact=gc_contact)
|
|
|
|
else:
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.info('Avatar already known: %s', nick)
|
2018-06-30 19:23:10 +02:00
|
|
|
|
2018-09-01 20:34:56 +02:00
|
|
|
def send_avatar_presence(self, force=False, after_publish=False):
|
|
|
|
if self._con.avatar_conversion:
|
|
|
|
if not after_publish:
|
|
|
|
# XEP-0398: We only resend presence after we publish a
|
|
|
|
# new avatar
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
if self.avatar_advertised and not force:
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.debug('Avatar already advertised')
|
2018-09-01 20:34:56 +02:00
|
|
|
return
|
|
|
|
|
2018-06-30 19:23:10 +02:00
|
|
|
show = helpers.get_xmpp_show(app.SHOW_LIST[self._con.connected])
|
2018-09-01 17:07:30 +02:00
|
|
|
|
|
|
|
self._con.get_module('Presence').send_presence(
|
|
|
|
priority=self._con.priority,
|
|
|
|
show=show,
|
|
|
|
status=self._con.status)
|
|
|
|
|
2018-06-30 19:23:10 +02:00
|
|
|
self.avatar_advertised = True
|
|
|
|
|
|
|
|
def add_update_node(self, node):
|
|
|
|
update = node.setTag('x', namespace=nbxmpp.NS_VCARD_UPDATE)
|
|
|
|
if self._con.get_module('VCardTemp').own_vcard_received:
|
|
|
|
sha = app.config.get_per('accounts', self._account, 'avatar_sha')
|
|
|
|
own_jid = self._con.get_own_jid()
|
2019-03-09 23:16:27 +01:00
|
|
|
self._log.info('Send avatar presence to: %s %s',
|
|
|
|
node.getTo() or own_jid, sha or 'no sha advertised')
|
2018-06-30 19:23:10 +02:00
|
|
|
update.setTagData('photo', sha)
|
2018-07-07 13:52:44 +02:00
|
|
|
|
|
|
|
|
|
|
|
def get_instance(*args, **kwargs):
|
|
|
|
return VCardAvatars(*args, **kwargs), 'VCardAvatars'
|