# Copyright (C) 2009 Stephan Erb # Copyright (C) 2018 Philipp Hörist # # 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 . # XEP-0115: Entity Capabilities import logging import nbxmpp from nbxmpp.structs import StanzaHandler from gajim.common import caps_cache from gajim.common import app from gajim.common.nec import NetworkEvent log = logging.getLogger('gajim.c.m.caps') class Caps: def __init__(self, con): self._con = con self._account = con.name self.handlers = [ StanzaHandler(name='presence', callback=self._entity_caps, ns=nbxmpp.NS_CAPS, priority=51), ] self._capscache = caps_cache.capscache self._create_suitable_client_caps = caps_cache.create_suitable_client_caps def _entity_caps(self, _con, _stanza, properties): if properties.type.is_error or properties.type.is_unavailable: return if properties.is_self_presence: return jid = str(properties.jid) hash_method = properties.entity_caps.hash node = properties.entity_caps.node caps_hash = properties.entity_caps.ver log.info('Received from %s, type: %s, method: %s, node: %s, hash: %s', jid, properties.type, hash_method, node, caps_hash) client_caps = self._create_suitable_client_caps( node, caps_hash, hash_method, jid) # Type is None means 'available' if properties.type.is_available and client_caps._hash_method == 'no': self._capscache.forget_caps(client_caps) client_caps = self._create_suitable_client_caps( node, caps_hash, hash_method) else: self._capscache.query_client_of_jid_if_unknown( self._con, jid, client_caps) self._update_client_caps_of_contact(properties.jid, client_caps) app.nec.push_incoming_event( NetworkEvent('caps-update', conn=self._con, fjid=jid, jid=properties.jid.getBare())) def _update_client_caps_of_contact(self, from_, client_caps): contact = self._get_contact_or_gc_contact_for_jid(from_) if contact is not None: contact.client_caps = client_caps else: log.info('Received Caps from unknown contact %s', from_) def _get_contact_or_gc_contact_for_jid(self, from_): contact = app.contacts.get_contact_from_full_jid(self._account, str(from_)) if contact is None: room_jid, resource = from_.getStripped(), from_.getResource() contact = app.contacts.get_gc_contact( self._account, room_jid, resource) return contact def contact_info_received(self, from_, identities, features, data, node): """ callback to update our caps cache with queried information after we have retrieved an unknown caps hash via a disco """ bare_jid = from_.getStripped() contact = self._get_contact_or_gc_contact_for_jid(from_) if not contact: log.info('Received Disco from unknown contact %s', from_) return lookup = contact.client_caps.get_cache_lookup_strategy() cache_item = lookup(self._capscache) if cache_item.is_valid(): # we already know that the hash is fine and have already cached # the identities and features return validate = contact.client_caps.get_hash_validation_strategy() hash_is_valid = validate(identities, features, data) if hash_is_valid: cache_item.set_and_store(identities, features) else: node = caps_hash = hash_method = None contact.client_caps = self._create_suitable_client_caps( node, caps_hash, hash_method) log.warning( 'Computed and retrieved caps hash differ. Ignoring ' 'caps of contact %s', contact.get_full_jid()) app.nec.push_incoming_event( NetworkEvent('caps-update', conn=self._con, fjid=str(from_), jid=bare_jid)) def get_instance(*args, **kwargs): return Caps(*args, **kwargs), 'Caps'