gajim-plural/src/common/caps.py

454 lines
13 KiB
Python
Raw Normal View History

# -*- coding:utf-8 -*-
## src/common/caps.py
2007-06-27 02:51:12 +02:00
##
## Copyright (C) 2007 Tomasz Melcer <liori AT exroot.org>
## Travis Shirk <travis AT pobox.com>
## Copyright (C) 2007-2008 Yann Leboulanger <asterix AT lagaule.org>
## Copyright (C) 2008 Brendan Taylor <whateley AT gmail.com>
## Jonathan Schleifer <js-gajim AT webkeks.org>
## Copyright (C) 2008-2009 Stephan Erb <steve-e AT h3c.de>
2007-06-27 02:51:12 +02:00
##
## This file is part of Gajim.
##
## Gajim is free software; you can redistribute it and/or modify
2007-06-27 02:51:12 +02:00
## it under the terms of the GNU General Public License as published
## by the Free Software Foundation; version 3 only.
2007-06-27 02:51:12 +02:00
##
## Gajim is distributed in the hope that it will be useful,
2007-06-27 02:51:12 +02:00
## but WITHOUT ANY WARRANTY; without even the implied warranty of
## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
2007-06-27 02:51:12 +02:00
## 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/>.
##
2007-06-27 02:51:12 +02:00
2009-11-26 11:21:43 +01:00
"""
Module containing all XEP-115 (Entity Capabilities) related classes
Basic Idea:
CapsCache caches features to hash relationships. The cache is queried
2009-11-26 11:21:43 +01:00
through ClientCaps objects which are hold by contact instances.
"""
2007-07-09 23:24:47 +02:00
import gajim
import helpers
import base64
import hashlib
2007-06-27 02:51:12 +02:00
from common.xmpp import NS_XHTML_IM, NS_RECEIPTS, NS_ESESSION, NS_CHATSTATES
from common.xmpp import NS_JINGLE_ICE_UDP, NS_JINGLE_RTP_AUDIO
from common.xmpp import NS_JINGLE_RTP_VIDEO
# Features where we cannot safely assume that the other side supports them
FEATURE_BLACKLIST = [NS_CHATSTATES, NS_XHTML_IM, NS_RECEIPTS, NS_ESESSION,
NS_JINGLE_ICE_UDP, NS_JINGLE_RTP_AUDIO, NS_JINGLE_RTP_VIDEO]
# Query entry status codes
NEW = 0
QUERIED = 1
CACHED = 2 # got the answer
2009-10-27 22:48:47 +01:00
################################################################################
### Public API of this module
################################################################################
capscache = None
def initialize(logger):
2009-11-26 11:21:43 +01:00
"""
Initialize this module
"""
global capscache
capscache = CapsCache(logger)
def client_supports(client_caps, requested_feature):
lookup_item = client_caps.get_cache_lookup_strategy()
cache_item = lookup_item(capscache)
supported_features = cache_item.features
if requested_feature in supported_features:
return True
elif supported_features == [] and cache_item.status in (NEW, QUERIED):
# assume feature is supported, if we don't know yet, what the client
# is capable of
return requested_feature not in FEATURE_BLACKLIST
else:
return False
def compute_caps_hash(identities, features, dataforms=[], hash_method='sha-1'):
2009-11-26 11:21:43 +01:00
"""
Compute caps hash according to XEP-0115, V1.5
dataforms are xmpp.DataForms objects as common.dataforms don't allow several
2009-11-26 11:21:43 +01:00
values without a field type list-multi
"""
def sort_identities_func(i1, i2):
cat1 = i1['category']
cat2 = i2['category']
if cat1 < cat2:
return -1
if cat1 > cat2:
return 1
type1 = i1.get('type', '')
type2 = i2.get('type', '')
if type1 < type2:
return -1
if type1 > type2:
return 1
lang1 = i1.get('xml:lang', '')
lang2 = i2.get('xml:lang', '')
if lang1 < lang2:
return -1
if lang1 > lang2:
return 1
return 0
2009-11-26 11:21:43 +01:00
def sort_dataforms_func(d1, d2):
f1 = d1.getField('FORM_TYPE')
f2 = d2.getField('FORM_TYPE')
if f1 and f2 and (f1.getValue() < f2.getValue()):
return -1
return 1
2009-11-26 11:21:43 +01:00
S = ''
identities.sort(cmp=sort_identities_func)
for i in identities:
c = i['category']
type_ = i.get('type', '')
lang = i.get('xml:lang', '')
name = i.get('name', '')
S += '%s/%s/%s/%s<' % (c, type_, lang, name)
features.sort()
for f in features:
S += '%s<' % f
dataforms.sort(cmp=sort_dataforms_func)
for dataform in dataforms:
# fields indexed by var
fields = {}
for f in dataform.getChildren():
fields[f.getVar()] = f
form_type = fields.get('FORM_TYPE')
if form_type:
S += form_type.getValue() + '<'
del fields['FORM_TYPE']
for var in sorted(fields.keys()):
S += '%s<' % var
values = sorted(fields[var].getValues())
for value in values:
S += '%s<' % value
if hash_method == 'sha-1':
hash_ = hashlib.sha1(S)
elif hash_method == 'md5':
hash_ = hashlib.md5(S)
else:
return ''
return base64.b64encode(hash_.digest())
2009-11-26 11:21:43 +01:00
################################################################################
### Internal classes of this module
################################################################################
class AbstractClientCaps(object):
2009-11-26 11:21:43 +01:00
"""
Base class representing a client and its capabilities as advertised by a
caps tag in a presence
"""
def __init__(self, caps_hash, node):
self._hash = caps_hash
self._node = node
2009-11-26 11:21:43 +01:00
def get_discover_strategy(self):
return self._discover
2009-11-26 11:21:43 +01:00
def _discover(self, connection, jid):
2009-11-26 11:21:43 +01:00
"""
To be implemented by subclassess
"""
raise NotImplementedError
def get_cache_lookup_strategy(self):
return self._lookup_in_cache
2009-11-26 11:21:43 +01:00
def _lookup_in_cache(self, caps_cache):
2009-11-26 11:21:43 +01:00
"""
To be implemented by subclassess
"""
raise NotImplementedError
def get_hash_validation_strategy(self):
return self._is_hash_valid
2009-11-26 11:21:43 +01:00
def _is_hash_valid(self, identities, features, dataforms):
2009-11-26 11:21:43 +01:00
"""
To be implemented by subclassess
"""
raise NotImplementedError
class ClientCaps(AbstractClientCaps):
2009-11-26 11:21:43 +01:00
"""
The current XEP-115 implementation
"""
def __init__(self, caps_hash, node, hash_method):
AbstractClientCaps.__init__(self, caps_hash, node)
assert hash_method != 'old'
self._hash_method = hash_method
2009-11-26 11:21:43 +01:00
def _lookup_in_cache(self, caps_cache):
return caps_cache[(self._hash_method, self._hash)]
2009-11-26 11:21:43 +01:00
def _discover(self, connection, jid):
connection.discoverInfo(jid, '%s#%s' % (self._node, self._hash))
2009-11-26 11:21:43 +01:00
def _is_hash_valid(self, identities, features, dataforms):
computed_hash = compute_caps_hash(identities, features,
dataforms=dataforms, hash_method=self._hash_method)
2009-11-26 11:21:43 +01:00
return computed_hash == self._hash
class OldClientCaps(AbstractClientCaps):
2009-11-26 11:21:43 +01:00
"""
Old XEP-115 implemtation. Kept around for background competability
"""
def __init__(self, caps_hash, node):
AbstractClientCaps.__init__(self, caps_hash, node)
def _lookup_in_cache(self, caps_cache):
return caps_cache[('old', self._node + '#' + self._hash)]
2009-11-26 11:21:43 +01:00
def _discover(self, connection, jid):
connection.discoverInfo(jid)
2009-11-26 11:21:43 +01:00
def _is_hash_valid(self, identities, features, dataforms):
2009-11-26 11:21:43 +01:00
return True
class NullClientCaps(AbstractClientCaps):
2009-11-26 11:21:43 +01:00
"""
This is a NULL-Object to streamline caps handling if a client has not
2009-11-26 11:21:43 +01:00
advertised any caps or has advertised them in an improper way
Assumes (almost) everything is supported.
2009-11-26 11:21:43 +01:00
"""
def __init__(self):
AbstractClientCaps.__init__(self, None, None)
2009-11-26 11:21:43 +01:00
def _lookup_in_cache(self, caps_cache):
# lookup something which does not exist to get a new CacheItem created
cache_item = caps_cache[('dummy', '')]
assert cache_item.status != CACHED
return cache_item
2009-11-26 11:21:43 +01:00
def _discover(self, connection, jid):
pass
def _is_hash_valid(self, identities, features, dataforms):
2009-11-26 11:21:43 +01:00
return False
2009-10-27 22:48:47 +01:00
2007-06-27 02:51:12 +02:00
class CapsCache(object):
2009-11-26 11:21:43 +01:00
"""
This object keeps the mapping between caps data and real disco features they
represent, and provides simple way to query that info
"""
2007-06-27 02:51:12 +02:00
def __init__(self, logger=None):
# our containers:
# __cache is a dictionary mapping: pair of hash method and hash maps
2007-06-27 02:51:12 +02:00
# to CapsCacheItem object
# __CacheItem is a class that stores data about particular
# client (hash method/hash pair)
2007-06-27 02:51:12 +02:00
self.__cache = {}
class CacheItem(object):
2008-04-21 16:18:19 +02:00
# __names is a string cache; every string long enough is given
# another object, and we will have plenty of identical long
# strings. therefore we can cache them
__names = {}
2009-11-26 11:21:43 +01:00
2009-10-25 21:17:32 +01:00
def __init__(self, hash_method, hash_, logger):
2007-06-27 02:51:12 +02:00
# cached into db
self.hash_method = hash_method
self.hash = hash_
self._features = []
self._identities = []
2009-10-25 21:17:32 +01:00
self._logger = logger
2007-06-27 02:51:12 +02:00
self.status = NEW
self._recently_seen = False
2007-06-27 02:51:12 +02:00
def _get_features(self):
return self._features
2008-12-03 18:16:04 +01:00
def _set_features(self, value):
self._features = []
2008-04-21 14:57:34 +02:00
for feature in value:
2009-10-25 21:17:32 +01:00
self._features.append(self.__names.setdefault(feature, feature))
2009-11-26 11:21:43 +01:00
2008-04-21 16:16:20 +02:00
features = property(_get_features, _set_features)
2008-04-21 14:57:34 +02:00
def _get_identities(self):
list_ = []
for i in self._identities:
# transforms it back in a dict
d = dict()
d['category'] = i[0]
if i[1]:
d['type'] = i[1]
if i[2]:
d['xml:lang'] = i[2]
if i[3]:
d['name'] = i[3]
list_.append(d)
return list_
2009-11-26 11:21:43 +01:00
def _set_identities(self, value):
self._identities = []
2008-04-21 14:57:34 +02:00
for identity in value:
# dict are not hashable, so transform it into a tuple
2008-12-03 18:16:04 +01:00
t = (identity['category'], identity.get('type'),
identity.get('xml:lang'), identity.get('name'))
self._identities.append(self.__names.setdefault(t, t))
2009-11-26 11:21:43 +01:00
2008-04-21 16:16:20 +02:00
identities = property(_get_identities, _set_identities)
2008-04-21 14:57:34 +02:00
def set_and_store(self, identities, features):
self.identities = identities
self.features = features
2009-10-25 21:17:32 +01:00
self._logger.add_caps_entry(self.hash_method, self.hash,
2007-07-09 23:24:47 +02:00
identities, features)
self.status = CACHED
2009-11-26 11:21:43 +01:00
def update_last_seen(self):
if not self._recently_seen:
self._recently_seen = True
self._logger.update_caps_time(self.hash_method, self.hash)
2007-07-09 23:24:47 +02:00
2007-06-27 02:51:12 +02:00
self.__CacheItem = CacheItem
2007-07-09 23:24:47 +02:00
self.logger = logger
2007-06-27 02:51:12 +02:00
2009-10-25 21:17:32 +01:00
def initialize_from_db(self):
self._remove_outdated_caps()
for hash_method, hash_, identities, features in \
self.logger.iter_caps_data():
x = self[(hash_method, hash_)]
x.identities = identities
x.features = features
x.status = CACHED
2009-11-26 11:21:43 +01:00
def _remove_outdated_caps(self):
2009-11-26 11:21:43 +01:00
"""
Remove outdated values from the db
"""
self.logger.clean_caps_table()
2007-06-27 02:51:12 +02:00
def __getitem__(self, caps):
if caps in self.__cache:
return self.__cache[caps]
2008-12-03 18:16:04 +01:00
hash_method, hash_ = caps
2009-10-25 21:17:32 +01:00
x = self.__CacheItem(hash_method, hash_, self.logger)
2008-12-03 18:16:04 +01:00
self.__cache[(hash_method, hash_)] = x
2007-06-27 02:51:12 +02:00
return x
def query_client_of_jid_if_unknown(self, connection, jid, client_caps):
2009-11-26 11:21:43 +01:00
"""
Start a disco query to determine caps (node, ver, exts). Won't query if
the data is already in cache
"""
lookup_cache_item = client_caps.get_cache_lookup_strategy()
2009-11-26 11:21:43 +01:00
q = lookup_cache_item(self)
if q.status == NEW:
# do query for bare node+hash pair
2007-06-27 02:51:12 +02:00
# this will create proper object
q.status = QUERIED
discover = client_caps.get_discover_strategy()
discover(connection, jid)
2009-11-26 11:21:43 +01:00
else:
q.update_last_seen()
################################################################################
### Caps network coding
################################################################################
2007-06-27 02:51:12 +02:00
class ConnectionCaps(object):
2009-11-26 11:21:43 +01:00
"""
This class highly depends on that it is a part of Connection class
"""
2007-06-27 02:51:12 +02:00
def _capsPresenceCB(self, con, presence):
2009-11-26 11:21:43 +01:00
"""
Handle incoming presence stanzas... This is a callback for xmpp
registered in connection_handlers.py
2009-11-26 11:21:43 +01:00
"""
# we will put these into proper Contact object and ask
# for disco... so that disco will learn how to interpret
# these caps
pm_ctrl = None
2009-04-30 17:20:37 +02:00
try:
jid = helpers.get_full_jid_from_iq(presence)
except:
# Bad jid
return
contact = gajim.contacts.get_contact_from_full_jid(self.name, jid)
if contact is None:
room_jid, nick = gajim.get_room_and_nick_from_fjid(jid)
contact = gajim.contacts.get_gc_contact(
self.name, room_jid, nick)
pm_ctrl = gajim.interface.msg_win_mgr.get_control(jid, self.name)
if contact is None:
# TODO: a way to put contact not-in-roster
# into Contacts
2008-12-03 22:56:12 +01:00
return
caps_tag = presence.getTag('c')
if not caps_tag:
# presence did not contain caps_tag
client_caps = NullClientCaps()
else:
hash_method, node, caps_hash = caps_tag['hash'], caps_tag['node'], caps_tag['ver']
if not node or not caps_hash:
# improper caps in stanza, ignore client capabilities.
client_caps = NullClientCaps()
elif not hash_method:
client_caps = OldClientCaps(caps_hash, node)
else:
client_caps = ClientCaps(caps_hash, node, hash_method)
capscache.query_client_of_jid_if_unknown(self, jid, client_caps)
contact.client_caps = client_caps
2007-06-27 02:51:12 +02:00
if pm_ctrl:
pm_ctrl.update_contact()
2009-11-26 11:21:43 +01:00
def _capsDiscoCB(self, jid, node, identities, features, dataforms):
contact = gajim.contacts.get_contact_from_full_jid(self.name, jid)
if not contact:
room_jid, nick = gajim.get_room_and_nick_from_fjid(jid)
contact = gajim.contacts.get_gc_contact(self.name, room_jid, nick)
if contact is None:
return
lookup = contact.client_caps.get_cache_lookup_strategy()
2009-11-26 11:21:43 +01:00
cache_item = lookup(capscache)
if cache_item.status == CACHED:
return
else:
validate = contact.client_caps.get_hash_validation_strategy()
hash_is_valid = validate(identities, features, dataforms)
2009-11-26 11:21:43 +01:00
if hash_is_valid:
cache_item.set_and_store(identities, features)
else:
contact.client_caps = NullClientCaps()
# vim: se ts=3: