gajim-plural/src/common/caps.py

284 lines
8.8 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>
## 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
2007-07-09 23:24:47 +02:00
import gajim
import helpers
2007-06-27 02:51:12 +02:00
class CapsCache(object):
''' This object keeps the mapping between caps data and real disco
features they represent, and provides simple way to query that info.
2009-10-25 21:17:32 +01:00
2007-06-27 02:51:12 +02:00
It is application-wide, that is there's one object for all
connections.
2009-10-25 21:17:32 +01:00
2007-06-27 02:51:12 +02:00
Goals:
* handle storing/retrieving info from database
* cache info in memory
* expose simple interface
2009-10-25 21:17:32 +01:00
2007-06-27 02:51:12 +02:00
Properties:
* one object for all connections (move to logger.py?)
* store info efficiently (a set() of urls -- we can assume there won't be
too much of these, ensure that (X,Y,Z1) and (X,Y,Z2) has different
features.
'''
def __init__(self, logger=None):
''' Create a cache for entity capabilities. '''
# 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):
''' TODO: logging data into db '''
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
# TODO: maybe put all known xmpp namespace strings here
# (strings given in xmpppy)?
__names = {}
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
# not cached into db:
# have we sent the query?
# 0 == not queried
# 1 == queried
# 2 == got the answer
self.queried = 0
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))
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_
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))
2008-04-21 16:16:20 +02:00
identities = property(_get_identities, _set_identities)
2008-04-21 14:57:34 +02:00
def update(self, identities, features):
2007-07-09 23:24:47 +02:00
# NOTE: self refers to CapsCache object, not to CacheItem
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)
2007-06-27 02:51:12 +02:00
self.__CacheItem = CacheItem
# prepopulate data which we are sure of; note: we do not log these info
for account in gajim.connections:
2008-12-02 16:10:31 +01:00
gajimcaps = self[('sha-1', gajim.caps_hash[account])]
gajimcaps.identities = [gajim.gajim_identity]
gajimcaps.features = gajim.gajim_common_features + \
gajim.gajim_optional_features[account]
2007-06-27 02:51:12 +02:00
# start logging data from the net
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):
2007-06-27 02:51:12 +02:00
# get data from logger...
2007-07-09 23:24:47 +02:00
if self.logger is not None:
2008-12-03 18:16:04 +01:00
for hash_method, hash_, identities, features in \
self.logger.iter_caps_data():
2008-12-03 18:16:04 +01:00
x = self[(hash_method, hash_)]
x.identities = identities
x.features = features
x.queried = 2
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 preload(self, con, jid, node, hash_method, hash_):
2007-06-27 02:51:12 +02:00
''' Preload data about (node, ver, exts) caps using disco
query to jid using proper connection. Don't query if
the data is already in cache. '''
if hash_method == 'old':
q = self[(hash_method, node + '#' + hash_)]
else:
q = self[(hash_method, hash_)]
2007-06-27 02:51:12 +02:00
if q.queried==0:
# do query for bare node+hash pair
2007-06-27 02:51:12 +02:00
# this will create proper object
q.queried=1
if hash_method == 'old':
con.discoverInfo(jid)
else:
con.discoverInfo(jid, '%s#%s' % (node, hash_))
def is_supported(self, contact, feature):
if not contact:
return False
# Unfortunately, if all resources are offline, the contact
# includes the last resource that was online. Check for its
# show, so we can be sure it's existant. Otherwise, we still
# return caps for a contact that has no resources left.
if contact.show == 'offline':
return False
# FIXME: We assume everything is supported if we got no caps.
# This is the "Asterix way", after 0.12 release, I will
# likely implement a fallback to disco (could be disabled
# for mobile users who pay for traffic)
if contact.caps_hash_method == 'old':
features = self[(contact.caps_hash_method, contact.caps_node + '#' + \
contact.caps_hash)].features
else:
features = self[(contact.caps_hash_method, contact.caps_hash)].features
if feature in features or features == []:
return True
return False
2007-07-09 23:24:47 +02:00
gajim.capscache = CapsCache(gajim.logger)
2007-06-27 02:51:12 +02:00
class ConnectionCaps(object):
''' This class highly depends on that it is a part of Connection class. '''
def _capsPresenceCB(self, con, presence):
''' Handle incoming presence stanzas... This is a callback
for xmpp registered in connection_handlers.py'''
# 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
2007-06-27 02:51:12 +02:00
# get the caps element
caps = presence.getTag('c')
if not caps:
contact.caps_node = None
contact.caps_hash = None
contact.caps_hash_method = None
return
2007-06-27 02:51:12 +02:00
2008-12-03 18:16:04 +01:00
hash_method, node, hash_ = caps['hash'], caps['node'], caps['ver']
2008-12-03 18:16:04 +01:00
if hash_method is None and node and hash_:
# Old XEP-115 implentation
hash_method = 'old'
2008-12-03 18:16:04 +01:00
if hash_method is None or node is None or hash_ is None:
2007-06-27 02:51:12 +02:00
# improper caps in stanza, ignoring
contact.caps_node = None
contact.caps_hash = None
contact.hash_method = None
2007-06-27 02:51:12 +02:00
return
# start disco query...
2008-12-03 18:16:04 +01:00
gajim.capscache.preload(self, jid, node, hash_method, hash_)
2007-06-27 02:51:12 +02:00
# overwriting old data
contact.caps_node = node
contact.caps_hash_method = hash_method
2008-12-03 18:16:04 +01:00
contact.caps_hash = hash_
if pm_ctrl:
pm_ctrl.update_contact()
2008-04-21 14:57:34 +02: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
if not contact.caps_node:
return # we didn't asked for that?
if contact.caps_hash_method != 'old':
computed_hash = helpers.compute_caps_hash(identities, features,
dataforms=dataforms, hash_method=contact.caps_hash_method)
if computed_hash != contact.caps_hash:
# wrong hash, forget it
contact.caps_node = ''
contact.caps_hash_method = ''
contact.caps_hash = ''
return
# if we don't have this info already...
caps = gajim.capscache[(contact.caps_hash_method, contact.caps_hash)]
else:
# if we don't have this info already...
caps = gajim.capscache[(contact.caps_hash_method, contact.caps_node + \
'#' + contact.caps_hash)]
if caps.queried == 2:
return
2007-07-09 23:24:47 +02:00
caps.update(identities, features)
# vim: se ts=3: