mirror of
https://github.com/correl/SleekXMPP.git
synced 2024-12-25 03:00:17 +00:00
Merge branch 'develop' into exceptions
This commit is contained in:
commit
20df6348a4
10 changed files with 317 additions and 39 deletions
11
sleekxmpp/plugins/xep_0066/__init__.py
Normal file
11
sleekxmpp/plugins/xep_0066/__init__.py
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
"""
|
||||||
|
SleekXMPP: The Sleek XMPP Library
|
||||||
|
Copyright (C) 2011 Nathanael C. Fritz, Lance J.T. Stout
|
||||||
|
This file is part of SleekXMPP.
|
||||||
|
|
||||||
|
See the file LICENSE for copying permission.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from sleekxmpp.plugins.xep_0066 import stanza
|
||||||
|
from sleekxmpp.plugins.xep_0066.stanza import OOB, OOBTransfer
|
||||||
|
from sleekxmpp.plugins.xep_0066.oob import xep_0066
|
89
sleekxmpp/plugins/xep_0066/oob.py
Normal file
89
sleekxmpp/plugins/xep_0066/oob.py
Normal file
|
@ -0,0 +1,89 @@
|
||||||
|
"""
|
||||||
|
SleekXMPP: The Sleek XMPP Library
|
||||||
|
Copyright (C) 2011 Nathanael C. Fritz, Lance J.T. Stout
|
||||||
|
This file is part of SleekXMPP.
|
||||||
|
|
||||||
|
See the file LICENSE for copying permission.
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
from sleekxmpp.stanza import Message, Presence, Iq
|
||||||
|
from sleekxmpp.xmlstream import register_stanza_plugin
|
||||||
|
from sleekxmpp.xmlstream.handler import Callback
|
||||||
|
from sleekxmpp.xmlstream.matcher import StanzaPath
|
||||||
|
from sleekxmpp.plugins.base import base_plugin
|
||||||
|
from sleekxmpp.plugins.xep_0066 import stanza
|
||||||
|
|
||||||
|
|
||||||
|
class xep_0066(base_plugin):
|
||||||
|
|
||||||
|
"""
|
||||||
|
XEP-0066: Out-of-Band Data
|
||||||
|
|
||||||
|
Out-of-Band Data is a basic method for transferring files between
|
||||||
|
XMPP agents. The URL of the resource in question is sent to the receiving
|
||||||
|
entity, which then downloads the resource before responding to the OOB
|
||||||
|
request. OOB is also used as a generic means to transmit URLs in other
|
||||||
|
stanzas to indicate where to find additional information.
|
||||||
|
|
||||||
|
Also see <http://www.xmpp.org/extensions/xep-0066.html>.
|
||||||
|
|
||||||
|
Events:
|
||||||
|
oob_transfer -- Raised when a request to download a resource
|
||||||
|
has been received.
|
||||||
|
|
||||||
|
Methods:
|
||||||
|
send_oob -- Send a request to another entity to download a file
|
||||||
|
or other addressable resource.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def plugin_init(self):
|
||||||
|
"""Start the XEP-0066 plugin."""
|
||||||
|
self.xep = '0066'
|
||||||
|
self.description = 'Out-of-Band Transfer'
|
||||||
|
self.stanza = stanza
|
||||||
|
|
||||||
|
register_stanza_plugin(Iq, stanza.OOBTransfer)
|
||||||
|
register_stanza_plugin(Message, stanza.OOB)
|
||||||
|
register_stanza_plugin(Presence, stanza.OOB)
|
||||||
|
|
||||||
|
self.xmpp.register_handler(
|
||||||
|
Callback('OOB Transfer',
|
||||||
|
StanzaPath('iq@type=set/oob_transfer'),
|
||||||
|
self._handle_transfer))
|
||||||
|
|
||||||
|
def post_init(self):
|
||||||
|
"""Handle cross-plugin dependencies."""
|
||||||
|
base_plugin.post_init(self)
|
||||||
|
self.xmpp['xep_0030'].add_feature(stanza.OOBTransfer.namespace)
|
||||||
|
self.xmpp['xep_0030'].add_feature(stanza.OOB.namespace)
|
||||||
|
|
||||||
|
def send_oob(self, to, url, desc=None, ifrom=None, **iqargs):
|
||||||
|
"""
|
||||||
|
Initiate a basic file transfer by sending the URL of
|
||||||
|
a file or other resource.
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
url -- The URL of the resource to transfer.
|
||||||
|
desc -- An optional human readable description of the item
|
||||||
|
that is to be transferred.
|
||||||
|
ifrom -- Specifiy the sender's JID.
|
||||||
|
block -- If true, block and wait for the stanzas' reply.
|
||||||
|
timeout -- The time in seconds to block while waiting for
|
||||||
|
a reply. If None, then wait indefinitely.
|
||||||
|
callback -- Optional callback to execute when a reply is
|
||||||
|
received instead of blocking and waiting for
|
||||||
|
the reply.
|
||||||
|
"""
|
||||||
|
iq = self.xmpp.Iq()
|
||||||
|
iq['type'] = 'set'
|
||||||
|
iq['to'] = to
|
||||||
|
if ifrom:
|
||||||
|
iq['from'] = ifrom
|
||||||
|
iq['oob_transfer']['url'] = url
|
||||||
|
iq['oob_transfer']['desc'] = desc
|
||||||
|
return iq.send(**iqargs)
|
||||||
|
|
||||||
|
def _handle_transfer(self, iq):
|
||||||
|
"""Handle receiving an out-of-band transfer request."""
|
||||||
|
self.xmpp.event('oob_transfer', iq)
|
33
sleekxmpp/plugins/xep_0066/stanza.py
Normal file
33
sleekxmpp/plugins/xep_0066/stanza.py
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
"""
|
||||||
|
SleekXMPP: The Sleek XMPP Library
|
||||||
|
Copyright (C) 2011 Nathanael C. Fritz, Lance J.T. Stout
|
||||||
|
This file is part of SleekXMPP.
|
||||||
|
|
||||||
|
See the file LICENSE for copying permission.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from sleekxmpp.xmlstream import ElementBase
|
||||||
|
|
||||||
|
|
||||||
|
class OOBTransfer(ElementBase):
|
||||||
|
|
||||||
|
"""
|
||||||
|
"""
|
||||||
|
|
||||||
|
name = 'query'
|
||||||
|
namespace = 'jabber:iq:oob'
|
||||||
|
plugin_attrib = 'oob_transfer'
|
||||||
|
interfaces = set(('url', 'desc', 'sid'))
|
||||||
|
sub_interfaces = set(('url', 'desc'))
|
||||||
|
|
||||||
|
|
||||||
|
class OOB(ElementBase):
|
||||||
|
|
||||||
|
"""
|
||||||
|
"""
|
||||||
|
|
||||||
|
name = 'x'
|
||||||
|
namespace = 'jabber:x:oob'
|
||||||
|
plugin_attrib = 'oob'
|
||||||
|
interfaces = set(('url', 'desc'))
|
||||||
|
sub_interfaces = interfaces
|
|
@ -64,8 +64,7 @@ class RootStanza(StanzaBase):
|
||||||
# log the error
|
# log the error
|
||||||
log.exception('Error handling {%s}%s stanza' %
|
log.exception('Error handling {%s}%s stanza' %
|
||||||
(self.namespace, self.name))
|
(self.namespace, self.name))
|
||||||
# Finally raise the exception, so it can be handled (or not)
|
# Finally raise the exception to a global exception handler
|
||||||
# at a higher level by using sys.excepthook.
|
self.stream.exception(e)
|
||||||
raise e
|
|
||||||
|
|
||||||
register_stanza_plugin(RootStanza, Error)
|
register_stanza_plugin(RootStanza, Error)
|
||||||
|
|
|
@ -764,7 +764,6 @@ class XMLStream(object):
|
||||||
Event handlers and the send queue will be threaded
|
Event handlers and the send queue will be threaded
|
||||||
regardless of this parameter's value.
|
regardless of this parameter's value.
|
||||||
"""
|
"""
|
||||||
self._thread_excepthook()
|
|
||||||
self.scheduler.process(threaded=True)
|
self.scheduler.process(threaded=True)
|
||||||
|
|
||||||
def start_thread(name, target):
|
def start_thread(name, target):
|
||||||
|
@ -1052,30 +1051,16 @@ class XMLStream(object):
|
||||||
self.event_queue.put(('quit', None, None))
|
self.event_queue.put(('quit', None, None))
|
||||||
return
|
return
|
||||||
|
|
||||||
def _thread_excepthook(self):
|
def exception(self, exception):
|
||||||
"""
|
"""
|
||||||
If a threaded event handler raises an exception, there is no way to
|
Process an unknown exception.
|
||||||
catch it except with an excepthook. Currently, each thread has its own
|
|
||||||
excepthook, but ideally we could use the main sys.excepthook.
|
|
||||||
|
|
||||||
Modifies threading.Thread to use sys.excepthook when an exception
|
Meant to be overridden.
|
||||||
is not caught.
|
|
||||||
|
Arguments:
|
||||||
|
exception -- An unhandled exception object.
|
||||||
"""
|
"""
|
||||||
init_old = threading.Thread.__init__
|
pass
|
||||||
|
|
||||||
def init(self, *args, **kwargs):
|
|
||||||
init_old(self, *args, **kwargs)
|
|
||||||
run_old = self.run
|
|
||||||
|
|
||||||
def run_with_except_hook(*args, **kw):
|
|
||||||
try:
|
|
||||||
run_old(*args, **kw)
|
|
||||||
except (KeyboardInterrupt, SystemExit):
|
|
||||||
raise
|
|
||||||
except:
|
|
||||||
sys.excepthook(*sys.exc_info())
|
|
||||||
self.run = run_with_except_hook
|
|
||||||
threading.Thread.__init__ = init
|
|
||||||
|
|
||||||
|
|
||||||
# To comply with PEP8, method names now use underscores.
|
# To comply with PEP8, method names now use underscores.
|
||||||
|
|
|
@ -12,7 +12,6 @@ class TestStreamExceptions(SleekTest):
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def tearDown(self):
|
def tearDown(self):
|
||||||
sys.excepthook = sys.__excepthook__
|
|
||||||
self.stream_close()
|
self.stream_close()
|
||||||
|
|
||||||
def testExceptionReply(self):
|
def testExceptionReply(self):
|
||||||
|
@ -23,8 +22,6 @@ class TestStreamExceptions(SleekTest):
|
||||||
msg['body'] = 'Body changed'
|
msg['body'] = 'Body changed'
|
||||||
raise XMPPError(clear=False)
|
raise XMPPError(clear=False)
|
||||||
|
|
||||||
|
|
||||||
sys.excepthook = lambda *args, **kwargs: None
|
|
||||||
self.stream_start()
|
self.stream_start()
|
||||||
self.xmpp.add_event_handler('message', message)
|
self.xmpp.add_event_handler('message', message)
|
||||||
|
|
||||||
|
@ -44,6 +41,49 @@ class TestStreamExceptions(SleekTest):
|
||||||
</message>
|
</message>
|
||||||
""")
|
""")
|
||||||
|
|
||||||
|
def testExceptionContinueWorking(self):
|
||||||
|
"""Test that Sleek continues to respond after an XMPPError is raised."""
|
||||||
|
|
||||||
|
def message(msg):
|
||||||
|
msg.reply()
|
||||||
|
msg['body'] = 'Body changed'
|
||||||
|
raise XMPPError(clear=False)
|
||||||
|
|
||||||
|
self.stream_start()
|
||||||
|
self.xmpp.add_event_handler('message', message)
|
||||||
|
|
||||||
|
self.recv("""
|
||||||
|
<message>
|
||||||
|
<body>This is going to cause an error.</body>
|
||||||
|
</message>
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.send("""
|
||||||
|
<message type="error">
|
||||||
|
<body>This is going to cause an error.</body>
|
||||||
|
<error type="cancel" code="500">
|
||||||
|
<undefined-condition
|
||||||
|
xmlns="urn:ietf:params:xml:ns:xmpp-stanzas" />
|
||||||
|
</error>
|
||||||
|
</message>
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.recv("""
|
||||||
|
<message>
|
||||||
|
<body>This is going to cause an error.</body>
|
||||||
|
</message>
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.send("""
|
||||||
|
<message type="error">
|
||||||
|
<body>This is going to cause an error.</body>
|
||||||
|
<error type="cancel" code="500">
|
||||||
|
<undefined-condition
|
||||||
|
xmlns="urn:ietf:params:xml:ns:xmpp-stanzas" />
|
||||||
|
</error>
|
||||||
|
</message>
|
||||||
|
""")
|
||||||
|
|
||||||
def testXMPPErrorException(self):
|
def testXMPPErrorException(self):
|
||||||
"""Test raising an XMPPError exception."""
|
"""Test raising an XMPPError exception."""
|
||||||
|
|
||||||
|
@ -153,9 +193,8 @@ class TestStreamExceptions(SleekTest):
|
||||||
def catch_error(*args, **kwargs):
|
def catch_error(*args, **kwargs):
|
||||||
raised_errors.append(True)
|
raised_errors.append(True)
|
||||||
|
|
||||||
sys.excepthook = catch_error
|
|
||||||
|
|
||||||
self.stream_start()
|
self.stream_start()
|
||||||
|
self.xmpp.exception = catch_error
|
||||||
self.xmpp.add_event_handler('message', message)
|
self.xmpp.add_event_handler('message', message)
|
||||||
|
|
||||||
self.recv("""
|
self.recv("""
|
||||||
|
@ -178,6 +217,58 @@ class TestStreamExceptions(SleekTest):
|
||||||
|
|
||||||
self.assertEqual(raised_errors, [True], "Exception was not raised: %s" % raised_errors)
|
self.assertEqual(raised_errors, [True], "Exception was not raised: %s" % raised_errors)
|
||||||
|
|
||||||
|
def testUnknownException(self):
|
||||||
|
"""Test Sleek continues to respond after an unknown exception."""
|
||||||
|
|
||||||
|
raised_errors = []
|
||||||
|
|
||||||
|
def message(msg):
|
||||||
|
raise ValueError("Did something wrong")
|
||||||
|
|
||||||
|
def catch_error(*args, **kwargs):
|
||||||
|
raised_errors.append(True)
|
||||||
|
|
||||||
|
self.stream_start()
|
||||||
|
self.xmpp.exception = catch_error
|
||||||
|
self.xmpp.add_event_handler('message', message)
|
||||||
|
|
||||||
|
self.recv("""
|
||||||
|
<message>
|
||||||
|
<body>This is going to cause an error.</body>
|
||||||
|
</message>
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.send("""
|
||||||
|
<message type="error">
|
||||||
|
<error type="cancel" code="500">
|
||||||
|
<undefined-condition
|
||||||
|
xmlns="urn:ietf:params:xml:ns:xmpp-stanzas" />
|
||||||
|
<text xmlns="urn:ietf:params:xml:ns:xmpp-stanzas">
|
||||||
|
SleekXMPP got into trouble.
|
||||||
|
</text>
|
||||||
|
</error>
|
||||||
|
</message>
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.recv("""
|
||||||
|
<message>
|
||||||
|
<body>This is going to cause an error.</body>
|
||||||
|
</message>
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.send("""
|
||||||
|
<message type="error">
|
||||||
|
<error type="cancel" code="500">
|
||||||
|
<undefined-condition
|
||||||
|
xmlns="urn:ietf:params:xml:ns:xmpp-stanzas" />
|
||||||
|
<text xmlns="urn:ietf:params:xml:ns:xmpp-stanzas">
|
||||||
|
SleekXMPP got into trouble.
|
||||||
|
</text>
|
||||||
|
</error>
|
||||||
|
</message>
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.assertEqual(raised_errors, [True, True], "Exceptions were not raised: %s" % raised_errors)
|
||||||
|
|
||||||
|
|
||||||
suite = unittest.TestLoader().loadTestsFromTestCase(TestStreamExceptions)
|
suite = unittest.TestLoader().loadTestsFromTestCase(TestStreamExceptions)
|
||||||
|
|
|
@ -12,7 +12,6 @@ class TestStreamDisco(SleekTest):
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def tearDown(self):
|
def tearDown(self):
|
||||||
sys.excepthook = sys.__excepthook__
|
|
||||||
self.stream_close()
|
self.stream_close()
|
||||||
|
|
||||||
def testInfoEmptyDefaultNode(self):
|
def testInfoEmptyDefaultNode(self):
|
||||||
|
@ -531,11 +530,6 @@ class TestStreamDisco(SleekTest):
|
||||||
|
|
||||||
raised_exceptions = []
|
raised_exceptions = []
|
||||||
|
|
||||||
def catch_exception(*args, **kwargs):
|
|
||||||
raised_exceptions.append(True)
|
|
||||||
|
|
||||||
sys.excepthook = catch_exception
|
|
||||||
|
|
||||||
self.stream_start(mode='client',
|
self.stream_start(mode='client',
|
||||||
plugins=['xep_0030', 'xep_0059'])
|
plugins=['xep_0030', 'xep_0059'])
|
||||||
|
|
||||||
|
@ -544,8 +538,14 @@ class TestStreamDisco(SleekTest):
|
||||||
iterator=True)
|
iterator=True)
|
||||||
results.amount = 10
|
results.amount = 10
|
||||||
|
|
||||||
|
def run_test():
|
||||||
|
try:
|
||||||
|
results.next()
|
||||||
|
except StopIteration:
|
||||||
|
raised_exceptions.append(True)
|
||||||
|
|
||||||
t = threading.Thread(name="get_items_iterator",
|
t = threading.Thread(name="get_items_iterator",
|
||||||
target=results.next)
|
target=run_test)
|
||||||
t.start()
|
t.start()
|
||||||
|
|
||||||
self.send("""
|
self.send("""
|
||||||
|
|
72
tests/test_stream_xep_0066.py
Normal file
72
tests/test_stream_xep_0066.py
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
import time
|
||||||
|
import threading
|
||||||
|
|
||||||
|
from sleekxmpp.test import *
|
||||||
|
|
||||||
|
|
||||||
|
class TestOOB(SleekTest):
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
self.stream_close()
|
||||||
|
|
||||||
|
def testSendOOB(self):
|
||||||
|
"""Test sending an OOB transfer request."""
|
||||||
|
self.stream_start(plugins=['xep_0066', 'xep_0030'])
|
||||||
|
|
||||||
|
url = 'http://github.com/fritzy/SleekXMPP/blob/master/README'
|
||||||
|
|
||||||
|
t = threading.Thread(
|
||||||
|
name='send_oob',
|
||||||
|
target=self.xmpp['xep_0066'].send_oob,
|
||||||
|
args=('user@example.com', url),
|
||||||
|
kwargs={'desc': 'SleekXMPP README'})
|
||||||
|
|
||||||
|
t.start()
|
||||||
|
|
||||||
|
self.send("""
|
||||||
|
<iq to="user@example.com" type="set" id="1">
|
||||||
|
<query xmlns="jabber:iq:oob">
|
||||||
|
<url>http://github.com/fritzy/SleekXMPP/blob/master/README</url>
|
||||||
|
<desc>SleekXMPP README</desc>
|
||||||
|
</query>
|
||||||
|
</iq>
|
||||||
|
""")
|
||||||
|
|
||||||
|
self.recv("""
|
||||||
|
<iq id="1" type="result"
|
||||||
|
to="tester@localhost"
|
||||||
|
from="user@example.com" />
|
||||||
|
""")
|
||||||
|
|
||||||
|
t.join()
|
||||||
|
|
||||||
|
def testReceiveOOB(self):
|
||||||
|
"""Test receiving an OOB request."""
|
||||||
|
self.stream_start(plugins=['xep_0066', 'xep_0030'])
|
||||||
|
|
||||||
|
events = []
|
||||||
|
|
||||||
|
def receive_oob(iq):
|
||||||
|
events.append(iq['oob_transfer']['url'])
|
||||||
|
|
||||||
|
self.xmpp.add_event_handler('oob_transfer', receive_oob)
|
||||||
|
|
||||||
|
self.recv("""
|
||||||
|
<iq to="tester@localhost"
|
||||||
|
from="user@example.com"
|
||||||
|
type="set" id="1">
|
||||||
|
<query xmlns="jabber:iq:oob">
|
||||||
|
<url>http://github.com/fritzy/SleekXMPP/blob/master/README</url>
|
||||||
|
<desc>SleekXMPP README</desc>
|
||||||
|
</query>
|
||||||
|
</iq>
|
||||||
|
""")
|
||||||
|
|
||||||
|
time.sleep(0.1)
|
||||||
|
|
||||||
|
self.assertEqual(events,
|
||||||
|
['http://github.com/fritzy/SleekXMPP/blob/master/README'],
|
||||||
|
'URL was not received: %s' % events)
|
||||||
|
|
||||||
|
|
||||||
|
suite = unittest.TestLoader().loadTestsFromTestCase(TestOOB)
|
|
@ -13,7 +13,6 @@ class TestStreamExtendedDisco(SleekTest):
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def tearDown(self):
|
def tearDown(self):
|
||||||
sys.excepthook = sys.__excepthook__
|
|
||||||
self.stream_close()
|
self.stream_close()
|
||||||
|
|
||||||
def testUsingExtendedInfo(self):
|
def testUsingExtendedInfo(self):
|
||||||
|
|
|
@ -13,7 +13,6 @@ class TestStreamDirectInvite(SleekTest):
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def tearDown(self):
|
def tearDown(self):
|
||||||
sys.excepthook = sys.__excepthook__
|
|
||||||
self.stream_close()
|
self.stream_close()
|
||||||
|
|
||||||
def testReceiveInvite(self):
|
def testReceiveInvite(self):
|
||||||
|
|
Loading…
Reference in a new issue