Add support for XEP-0013: Flexible Offline Message Retrieval
This commit is contained in:
parent
b8f04983e1
commit
b5b1c932c7
1
setup.py
1
setup.py
@ -60,6 +60,7 @@ packages = [ 'sleekxmpp',
|
|||||||
'sleekxmpp/plugins/xep_0009',
|
'sleekxmpp/plugins/xep_0009',
|
||||||
'sleekxmpp/plugins/xep_0009/stanza',
|
'sleekxmpp/plugins/xep_0009/stanza',
|
||||||
'sleekxmpp/plugins/xep_0012',
|
'sleekxmpp/plugins/xep_0012',
|
||||||
|
'sleekxmpp/plugins/xep_0013',
|
||||||
'sleekxmpp/plugins/xep_0016',
|
'sleekxmpp/plugins/xep_0016',
|
||||||
'sleekxmpp/plugins/xep_0027',
|
'sleekxmpp/plugins/xep_0027',
|
||||||
'sleekxmpp/plugins/xep_0030',
|
'sleekxmpp/plugins/xep_0030',
|
||||||
|
@ -18,6 +18,7 @@ __all__ = [
|
|||||||
'xep_0004', # Data Forms
|
'xep_0004', # Data Forms
|
||||||
'xep_0009', # Jabber-RPC
|
'xep_0009', # Jabber-RPC
|
||||||
'xep_0012', # Last Activity
|
'xep_0012', # Last Activity
|
||||||
|
'xep_0013', # Flexible Offline Message Retrieval
|
||||||
'xep_0016', # Privacy Lists
|
'xep_0016', # Privacy Lists
|
||||||
'xep_0027', # Current Jabber OpenPGP Usage
|
'xep_0027', # Current Jabber OpenPGP Usage
|
||||||
'xep_0030', # Service Discovery
|
'xep_0030', # Service Discovery
|
||||||
|
15
sleekxmpp/plugins/xep_0013/__init__.py
Normal file
15
sleekxmpp/plugins/xep_0013/__init__.py
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
"""
|
||||||
|
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 permissio
|
||||||
|
"""
|
||||||
|
|
||||||
|
from sleekxmpp.plugins.base import register_plugin
|
||||||
|
|
||||||
|
from sleekxmpp.plugins.xep_0013.stanza import Offline
|
||||||
|
from sleekxmpp.plugins.xep_0013.offline import XEP_0013
|
||||||
|
|
||||||
|
|
||||||
|
register_plugin(XEP_0013)
|
134
sleekxmpp/plugins/xep_0013/offline.py
Normal file
134
sleekxmpp/plugins/xep_0013/offline.py
Normal file
@ -0,0 +1,134 @@
|
|||||||
|
"""
|
||||||
|
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 permissio
|
||||||
|
"""
|
||||||
|
|
||||||
|
import logging
|
||||||
|
|
||||||
|
import sleekxmpp
|
||||||
|
from sleekxmpp.stanza import Message, Iq
|
||||||
|
from sleekxmpp.exceptions import XMPPError
|
||||||
|
from sleekxmpp.xmlstream.handler import Collector
|
||||||
|
from sleekxmpp.xmlstream.matcher import StanzaPath
|
||||||
|
from sleekxmpp.xmlstream import register_stanza_plugin
|
||||||
|
from sleekxmpp.plugins import BasePlugin
|
||||||
|
from sleekxmpp.plugins.xep_0013 import stanza
|
||||||
|
|
||||||
|
|
||||||
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class XEP_0013(BasePlugin):
|
||||||
|
|
||||||
|
"""
|
||||||
|
XEP-0013 Flexible Offline Message Retrieval
|
||||||
|
"""
|
||||||
|
|
||||||
|
name = 'xep_0013'
|
||||||
|
description = 'XEP-0013: Flexible Offline Message Retrieval'
|
||||||
|
dependencies = set(['xep_0030'])
|
||||||
|
stanza = stanza
|
||||||
|
|
||||||
|
def plugin_init(self):
|
||||||
|
register_stanza_plugin(Iq, stanza.Offline)
|
||||||
|
register_stanza_plugin(Message, stanza.Offline)
|
||||||
|
|
||||||
|
def get_count(self, **kwargs):
|
||||||
|
return self.xmpp['xep_0030'].get_info(
|
||||||
|
node='http://jabber.org/protocol/offline',
|
||||||
|
local=False,
|
||||||
|
**kwargs)
|
||||||
|
|
||||||
|
def get_headers(self, **kwargs):
|
||||||
|
return self.xmpp['xep_0030'].get_items(
|
||||||
|
node='http://jabber.org/protocol/offline',
|
||||||
|
local=False,
|
||||||
|
**kwargs)
|
||||||
|
|
||||||
|
def view(self, nodes, ifrom=None, block=True, timeout=None, callback=None):
|
||||||
|
if not isinstance(nodes, (list, set)):
|
||||||
|
nodes = [nodes]
|
||||||
|
|
||||||
|
iq = self.xmpp.Iq()
|
||||||
|
iq['type'] = 'get'
|
||||||
|
iq['from'] = ifrom
|
||||||
|
offline = iq['offline']
|
||||||
|
for node in nodes:
|
||||||
|
item = stanza.Item()
|
||||||
|
item['node'] = node
|
||||||
|
item['action'] = 'view'
|
||||||
|
offline.append(item)
|
||||||
|
|
||||||
|
collector = Collector(
|
||||||
|
'Offline_Results_%s' % iq['id'],
|
||||||
|
StanzaPath('message/offline'))
|
||||||
|
self.xmpp.register_handler(collector)
|
||||||
|
|
||||||
|
if not block and callback is not None:
|
||||||
|
def wrapped_cb(iq):
|
||||||
|
results = collector.stop()
|
||||||
|
if iq['type'] == 'result':
|
||||||
|
iq['offline']['results'] = results
|
||||||
|
callback(iq)
|
||||||
|
return iq.send(block=block, timeout=timeout, callback=wrapped_cb)
|
||||||
|
else:
|
||||||
|
try:
|
||||||
|
resp = iq.send(block=block, timeout=timeout, callback=callback)
|
||||||
|
resp['offline']['results'] = collector.stop()
|
||||||
|
return resp
|
||||||
|
except XMPPError as e:
|
||||||
|
collector.stop()
|
||||||
|
raise e
|
||||||
|
|
||||||
|
def remove(self, nodes, ifrom=None, block=True, timeout=None, callback=None):
|
||||||
|
if not isinstance(nodes, (list, set)):
|
||||||
|
nodes = [nodes]
|
||||||
|
|
||||||
|
iq = self.xmpp.Iq()
|
||||||
|
iq['type'] = 'set'
|
||||||
|
iq['from'] = ifrom
|
||||||
|
offline = iq['offline']
|
||||||
|
for node in nodes:
|
||||||
|
item = stanza.Item()
|
||||||
|
item['node'] = node
|
||||||
|
item['action'] = 'remove'
|
||||||
|
offline.append(item)
|
||||||
|
|
||||||
|
return iq.send(block=block, timeout=timeout, callback=callback)
|
||||||
|
|
||||||
|
def fetch(self, ifrom=None, block=True, timeout=None, callback=None):
|
||||||
|
iq = self.xmpp.Iq()
|
||||||
|
iq['type'] = 'set'
|
||||||
|
iq['from'] = ifrom
|
||||||
|
iq['offline']['fetch'] = True
|
||||||
|
|
||||||
|
collector = Collector(
|
||||||
|
'Offline_Results_%s' % iq['id'],
|
||||||
|
StanzaPath('message/offline'))
|
||||||
|
self.xmpp.register_handler(collector)
|
||||||
|
|
||||||
|
if not block and callback is not None:
|
||||||
|
def wrapped_cb(iq):
|
||||||
|
results = collector.stop()
|
||||||
|
if iq['type'] == 'result':
|
||||||
|
iq['offline']['results'] = results
|
||||||
|
callback(iq)
|
||||||
|
return iq.send(block=block, timeout=timeout, callback=wrapped_cb)
|
||||||
|
else:
|
||||||
|
try:
|
||||||
|
resp = iq.send(block=block, timeout=timeout, callback=callback)
|
||||||
|
resp['offline']['results'] = collector.stop()
|
||||||
|
return resp
|
||||||
|
except XMPPError as e:
|
||||||
|
collector.stop()
|
||||||
|
raise e
|
||||||
|
|
||||||
|
def purge(self, ifrom=None, block=True, timeout=None, callback=None):
|
||||||
|
iq = self.xmpp.Iq()
|
||||||
|
iq['type'] = 'set'
|
||||||
|
iq['from'] = ifrom
|
||||||
|
iq['offline']['purge'] = True
|
||||||
|
return iq.send(block=block, timeout=timeout, callback=callback)
|
53
sleekxmpp/plugins/xep_0013/stanza.py
Normal file
53
sleekxmpp/plugins/xep_0013/stanza.py
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
"""
|
||||||
|
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 permissio
|
||||||
|
"""
|
||||||
|
|
||||||
|
from sleekxmpp.jid import JID
|
||||||
|
from sleekxmpp.xmlstream import ElementBase, register_stanza_plugin
|
||||||
|
|
||||||
|
|
||||||
|
class Offline(ElementBase):
|
||||||
|
name = 'offline'
|
||||||
|
namespace = 'http://jabber.org/protocol/offline'
|
||||||
|
plugin_attrib = 'offline'
|
||||||
|
interfaces = set(['fetch', 'purge', 'results'])
|
||||||
|
bool_interfaces = interfaces
|
||||||
|
|
||||||
|
def setup(self, xml=None):
|
||||||
|
ElementBase.setup(self, xml)
|
||||||
|
self._results = []
|
||||||
|
|
||||||
|
# The results interface is meant only as an easy
|
||||||
|
# way to access the set of collected message responses
|
||||||
|
# from the query.
|
||||||
|
|
||||||
|
def get_results(self):
|
||||||
|
return self._results
|
||||||
|
|
||||||
|
def set_results(self, values):
|
||||||
|
self._results = values
|
||||||
|
|
||||||
|
def del_results(self):
|
||||||
|
self._results = []
|
||||||
|
|
||||||
|
|
||||||
|
class Item(ElementBase):
|
||||||
|
name = 'item'
|
||||||
|
namespace = 'http://jabber.org/protocol/offline'
|
||||||
|
plugin_attrib = 'item'
|
||||||
|
interfaces = set(['action', 'node', 'jid'])
|
||||||
|
|
||||||
|
actions = set(['view', 'remove'])
|
||||||
|
|
||||||
|
def get_jid(self):
|
||||||
|
return JID(self._get_attr('jid'))
|
||||||
|
|
||||||
|
def set_jid(self, value):
|
||||||
|
self._set_attr('jid', str(value))
|
||||||
|
|
||||||
|
|
||||||
|
register_stanza_plugin(Offline, Item, iterable=True)
|
@ -22,7 +22,6 @@ class MAM(ElementBase):
|
|||||||
|
|
||||||
def setup(self, xml=None):
|
def setup(self, xml=None):
|
||||||
ElementBase.setup(self, xml)
|
ElementBase.setup(self, xml)
|
||||||
|
|
||||||
self._results = []
|
self._results = []
|
||||||
|
|
||||||
def get_start(self):
|
def get_start(self):
|
||||||
@ -49,7 +48,6 @@ class MAM(ElementBase):
|
|||||||
def set_with(self, value):
|
def set_with(self, value):
|
||||||
self._set_sub_text('with', str(value))
|
self._set_sub_text('with', str(value))
|
||||||
|
|
||||||
|
|
||||||
# The results interface is meant only as an easy
|
# The results interface is meant only as an easy
|
||||||
# way to access the set of collected message responses
|
# way to access the set of collected message responses
|
||||||
# from the query.
|
# from the query.
|
||||||
|
Loading…
Reference in New Issue
Block a user