Compare commits
1 Commits
master
...
xep-446-co
Author | SHA1 | Date | |
---|---|---|---|
![]() |
04244ecf82 |
8
doap.xml
8
doap.xml
@ -941,14 +941,6 @@
|
|||||||
<xmpp:since>1.8.6</xmpp:since>
|
<xmpp:since>1.8.6</xmpp:since>
|
||||||
</xmpp:SupportedXep>
|
</xmpp:SupportedXep>
|
||||||
</implements>
|
</implements>
|
||||||
<implements>
|
|
||||||
<xmpp:SupportedXep>
|
|
||||||
<xmpp:xep rdf:resource="https://xmpp.org/extensions/xep-0482.html"/>
|
|
||||||
<xmpp:status>complete</xmpp:status>
|
|
||||||
<xmpp:version>0.1.0</xmpp:version>
|
|
||||||
<xmpp:since>1.8.7</xmpp:since>
|
|
||||||
</xmpp:SupportedXep>
|
|
||||||
</implements>
|
|
||||||
<implements>
|
<implements>
|
||||||
<xmpp:SupportedXep>
|
<xmpp:SupportedXep>
|
||||||
<xmpp:xep rdf:resource="https://xmpp.org/extensions/xep-0490.html"/>
|
<xmpp:xep rdf:resource="https://xmpp.org/extensions/xep-0490.html"/>
|
||||||
|
@ -17,7 +17,6 @@ Plugin index
|
|||||||
xep_0049
|
xep_0049
|
||||||
xep_0050
|
xep_0050
|
||||||
xep_0054
|
xep_0054
|
||||||
xep_0055
|
|
||||||
xep_0059
|
xep_0059
|
||||||
xep_0060
|
xep_0060
|
||||||
xep_0065
|
xep_0065
|
||||||
@ -32,7 +31,6 @@ Plugin index
|
|||||||
xep_0085
|
xep_0085
|
||||||
xep_0086
|
xep_0086
|
||||||
xep_0092
|
xep_0092
|
||||||
xep_0100
|
|
||||||
xep_0106
|
xep_0106
|
||||||
xep_0107
|
xep_0107
|
||||||
xep_0108
|
xep_0108
|
||||||
@ -64,15 +62,12 @@ Plugin index
|
|||||||
xep_0256
|
xep_0256
|
||||||
xep_0257
|
xep_0257
|
||||||
xep_0258
|
xep_0258
|
||||||
xep_0264
|
|
||||||
xep_0279
|
xep_0279
|
||||||
xep_0280
|
xep_0280
|
||||||
xep_0292
|
|
||||||
xep_0297
|
xep_0297
|
||||||
xep_0300
|
xep_0300
|
||||||
xep_0308
|
xep_0308
|
||||||
xep_0313
|
xep_0313
|
||||||
xep_0317
|
|
||||||
xep_0319
|
xep_0319
|
||||||
xep_0332
|
xep_0332
|
||||||
xep_0333
|
xep_0333
|
||||||
@ -84,13 +79,9 @@ Plugin index
|
|||||||
xep_0359
|
xep_0359
|
||||||
xep_0363
|
xep_0363
|
||||||
xep_0369
|
xep_0369
|
||||||
xep_0372
|
|
||||||
xep_0377
|
xep_0377
|
||||||
xep_0380
|
xep_0380
|
||||||
xep_0382
|
|
||||||
xep_0385
|
|
||||||
xep_0394
|
xep_0394
|
||||||
xep_0402
|
|
||||||
xep_0403
|
xep_0403
|
||||||
xep_0404
|
xep_0404
|
||||||
xep_0405
|
xep_0405
|
||||||
@ -103,9 +94,4 @@ Plugin index
|
|||||||
xep_0439
|
xep_0439
|
||||||
xep_0441
|
xep_0441
|
||||||
xep_0444
|
xep_0444
|
||||||
xep_0446
|
|
||||||
xep_0447
|
|
||||||
xep_0461
|
|
||||||
xep_0469
|
|
||||||
xep_0490
|
|
||||||
xep_0492
|
xep_0492
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
|
|
||||||
XEP-0100: Gateway interaction
|
XEP-0106: Gateway interaction
|
||||||
=============================
|
=============================
|
||||||
|
|
||||||
.. module:: slixmpp.plugins.xep_0100
|
.. module:: slixmpp.plugins.xep_0100
|
||||||
|
@ -1,18 +0,0 @@
|
|||||||
|
|
||||||
XEP-0264: Jingle Content Thumbnails
|
|
||||||
===================================
|
|
||||||
|
|
||||||
.. module:: slixmpp.plugins.xep_0264
|
|
||||||
|
|
||||||
.. autoclass:: XEP_0264
|
|
||||||
:members:
|
|
||||||
:exclude-members: session_bind, plugin_init, plugin_end
|
|
||||||
|
|
||||||
|
|
||||||
Stanza elements
|
|
||||||
---------------
|
|
||||||
|
|
||||||
.. automodule:: slixmpp.plugins.xep_0264.stanza
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
|
|
@ -1,18 +0,0 @@
|
|||||||
|
|
||||||
XEP-0317: Hats
|
|
||||||
==============
|
|
||||||
|
|
||||||
.. module:: slixmpp.plugins.xep_0317
|
|
||||||
|
|
||||||
.. autoclass:: XEP_0317
|
|
||||||
:members:
|
|
||||||
:exclude-members: session_bind, plugin_init, plugin_end
|
|
||||||
|
|
||||||
|
|
||||||
Stanza elements
|
|
||||||
---------------
|
|
||||||
|
|
||||||
.. automodule:: slixmpp.plugins.xep_0317.stanza
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
|
|
@ -1,18 +0,0 @@
|
|||||||
|
|
||||||
XEP-0372: References
|
|
||||||
====================
|
|
||||||
|
|
||||||
.. module:: slixmpp.plugins.xep_0372
|
|
||||||
|
|
||||||
.. autoclass:: XEP_0372
|
|
||||||
:members:
|
|
||||||
:exclude-members: session_bind, plugin_init, plugin_end
|
|
||||||
|
|
||||||
|
|
||||||
Stanza elements
|
|
||||||
---------------
|
|
||||||
|
|
||||||
.. automodule:: slixmpp.plugins.xep_0372.stanza
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
|
|
@ -1,18 +0,0 @@
|
|||||||
|
|
||||||
XEP-0382: Spoiler Messages
|
|
||||||
==========================
|
|
||||||
|
|
||||||
.. module:: slixmpp.plugins.xep_0382
|
|
||||||
|
|
||||||
.. autoclass:: XEP_0382
|
|
||||||
:members:
|
|
||||||
:exclude-members: session_bind, plugin_init, plugin_end
|
|
||||||
|
|
||||||
|
|
||||||
Stanza elements
|
|
||||||
---------------
|
|
||||||
|
|
||||||
.. automodule:: slixmpp.plugins.xep_0382.stanza
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
|
|
@ -1,18 +0,0 @@
|
|||||||
|
|
||||||
XEP-0385: Stateless Inline Media Sharing (SIMS)
|
|
||||||
===============================================
|
|
||||||
|
|
||||||
.. module:: slixmpp.plugins.xep_0385
|
|
||||||
|
|
||||||
.. autoclass:: XEP_0385
|
|
||||||
:members:
|
|
||||||
:exclude-members: session_bind, plugin_init, plugin_end
|
|
||||||
|
|
||||||
|
|
||||||
Stanza elements
|
|
||||||
---------------
|
|
||||||
|
|
||||||
.. automodule:: slixmpp.plugins.xep_0385.stanza
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
|
|
@ -1,18 +0,0 @@
|
|||||||
|
|
||||||
XEP-0402: PEP Native Bookmarks
|
|
||||||
==============================
|
|
||||||
|
|
||||||
.. module:: slixmpp.plugins.xep_0402
|
|
||||||
|
|
||||||
.. autoclass:: XEP_0402
|
|
||||||
:members:
|
|
||||||
:exclude-members: session_bind, plugin_init, plugin_end
|
|
||||||
|
|
||||||
|
|
||||||
Stanza elements
|
|
||||||
---------------
|
|
||||||
|
|
||||||
.. automodule:: slixmpp.plugins.xep_0402.stanza
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
|
|
@ -1,18 +0,0 @@
|
|||||||
|
|
||||||
XEP-0446: File metadata element
|
|
||||||
===============================
|
|
||||||
|
|
||||||
.. module:: slixmpp.plugins.xep_0446
|
|
||||||
|
|
||||||
.. autoclass:: XEP_0446
|
|
||||||
:members:
|
|
||||||
:exclude-members: session_bind, plugin_init, plugin_end
|
|
||||||
|
|
||||||
|
|
||||||
Stanza elements
|
|
||||||
---------------
|
|
||||||
|
|
||||||
.. automodule:: slixmpp.plugins.xep_0446.stanza
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
|
|
@ -1,18 +0,0 @@
|
|||||||
|
|
||||||
XEP-0447: Stateless File Sharing
|
|
||||||
================================
|
|
||||||
|
|
||||||
.. module:: slixmpp.plugins.xep_0447
|
|
||||||
|
|
||||||
.. autoclass:: XEP_0447
|
|
||||||
:members:
|
|
||||||
:exclude-members: session_bind, plugin_init, plugin_end
|
|
||||||
|
|
||||||
|
|
||||||
Stanza elements
|
|
||||||
---------------
|
|
||||||
|
|
||||||
.. automodule:: slixmpp.plugins.xep_0447.stanza
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
|
|
@ -1,18 +0,0 @@
|
|||||||
|
|
||||||
XEP-0461: Message Replies
|
|
||||||
=========================
|
|
||||||
|
|
||||||
.. module:: slixmpp.plugins.xep_0461
|
|
||||||
|
|
||||||
.. autoclass:: XEP_0461
|
|
||||||
:members:
|
|
||||||
:exclude-members: session_bind, plugin_init, plugin_end
|
|
||||||
|
|
||||||
|
|
||||||
Stanza elements
|
|
||||||
---------------
|
|
||||||
|
|
||||||
.. automodule:: slixmpp.plugins.xep_0461.stanza
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
|
|
@ -1,17 +0,0 @@
|
|||||||
XEP-0469: Bookmark Pinning
|
|
||||||
==========================
|
|
||||||
|
|
||||||
.. module:: slixmpp.plugins.xep_0469
|
|
||||||
|
|
||||||
.. autoclass:: XEP_0469
|
|
||||||
:members:
|
|
||||||
:exclude-members: session_bind, plugin_init, plugin_end
|
|
||||||
|
|
||||||
|
|
||||||
Stanza elements
|
|
||||||
---------------
|
|
||||||
|
|
||||||
.. automodule:: slixmpp.plugins.xep_0469.stanza
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
|
|
@ -1,18 +0,0 @@
|
|||||||
|
|
||||||
XEP-0490: Message Displayed Synchronization
|
|
||||||
===========================================
|
|
||||||
|
|
||||||
.. module:: slixmpp.plugins.xep_0490
|
|
||||||
|
|
||||||
.. autoclass:: XEP_0490
|
|
||||||
:members:
|
|
||||||
:exclude-members: session_bind, plugin_init, plugin_end
|
|
||||||
|
|
||||||
|
|
||||||
Stanza elements
|
|
||||||
---------------
|
|
||||||
|
|
||||||
.. automodule:: slixmpp.plugins.xep_0490.stanza
|
|
||||||
:members:
|
|
||||||
:undoc-members:
|
|
||||||
|
|
@ -1,6 +1,6 @@
|
|||||||
|
|
||||||
XEP-0492: Chat Notification Settings
|
XEP-0492: Chat Notification Settings
|
||||||
====================================
|
===========================
|
||||||
|
|
||||||
.. module:: slixmpp.plugins.xep_0492
|
.. module:: slixmpp.plugins.xep_0492
|
||||||
|
|
||||||
|
@ -10,8 +10,8 @@ sendxmpp-py
|
|||||||
~~~~~~~~~~~
|
~~~~~~~~~~~
|
||||||
sendxmpp is a command line program and is the XMPP equivalent of sendmail. It is a Python version of the original sendxmpp which is written in Perl.
|
sendxmpp is a command line program and is the XMPP equivalent of sendmail. It is a Python version of the original sendxmpp which is written in Perl.
|
||||||
|
|
||||||
- `Source <https://code.moparisthebest.com/moparisthebest/sendxmpp-py>`__
|
- `Source <https://code.moparisthebest.com/moparisthebest/sendxmpp-py>`_
|
||||||
- `Groupchat <xmpp:xmpp-ircd@chatrooms.hackerposse.com?join>`__
|
- `Groupchat <xmpp:xmpp-ircd@chatrooms.hackerposse.com?join>`_
|
||||||
|
|
||||||
Bots
|
Bots
|
||||||
----
|
----
|
||||||
@ -20,65 +20,69 @@ BotLogMauve
|
|||||||
~~~~~~~~~~~
|
~~~~~~~~~~~
|
||||||
XMPP bot which logs groupchat messages. Logs are in text format, with one file per day and per groupchat.
|
XMPP bot which logs groupchat messages. Logs are in text format, with one file per day and per groupchat.
|
||||||
|
|
||||||
- `Source <https://git.khaganat.net/khaganat/BotLogMauve>`__
|
- `Source <https://git.khaganat.net/khaganat/BotLogMauve>`_
|
||||||
|
|
||||||
BukuBot
|
BukuBot
|
||||||
~~~~~~~
|
~~~~~~~
|
||||||
BukuBot makes it possible to manage and search your bookmarks from your chat.
|
BukuBot makes it possible to manage and search your bookmarks from your chat.
|
||||||
|
|
||||||
- `Source <https://codeberg.org/sch/BukuBot>`__
|
- `Source <https://codeberg.org/sch/BukuBot>`_
|
||||||
|
|
||||||
LinkBot
|
LinkBot
|
||||||
~~~~~~~
|
~~~~~~~
|
||||||
This bot reveals the title of any shared link in a groupchat for quick content insight.
|
This bot reveals the title of any shared link in a groupchat for quick content insight.
|
||||||
|
|
||||||
- `Source <https://git.xmpp-it.net/mario/XMPPBot>`__
|
- `Source <https://git.xmpp-it.net/mario/XMPPBot>`_
|
||||||
|
|
||||||
llama-bot
|
llama-bot
|
||||||
~~~~~~~~~
|
~~~~~~~~~
|
||||||
Llama-bot enables engaging communication with the LLM (large language model) of llama.cpp, providing seamless and dynamic conversation with it.
|
Llama-bot enables engaging communication with the LLM (large language model) of llama.cpp, providing seamless and dynamic conversation with it.
|
||||||
|
|
||||||
- `Source <https://github.com/decent-im/llama-bot>`__
|
- `Groupchat <xmpp:slixmpp@muc.poez.io?join>`_
|
||||||
- `Demo <xmpp:llama@decent.im?message>`__
|
- `Source <https://github.com/decent-im/llama-bot>`_
|
||||||
|
- `Demo <xmpp:llama@decent.im?message>`_
|
||||||
|
|
||||||
Morbot
|
Morbot
|
||||||
~~~~~~
|
~~~~~~
|
||||||
Morbot is a simple Slixmpp bot that will take new articles from listed RSS feeds and send them to assigned XMPP MUCs.
|
Morbot is a simple Slixmpp bot that will take new articles from listed RSS feeds and send them to assigned XMPP MUCs.
|
||||||
|
|
||||||
- `Source <https://codeberg.org/TheCoffeMaker/Morbot>`__
|
- `Groupchat <xmpp:slixmpp@muc.poez.io?join>`_
|
||||||
|
- `Source <https://codeberg.org/TheCoffeMaker/Morbot>`_
|
||||||
|
|
||||||
Slixfeed
|
Slixfeed
|
||||||
~~~~~~~~
|
~~~~~~~~
|
||||||
Slixfeed aims to be an easy to use and fully-featured news aggregator bot for XMPP. It provides a convenient access to Blogs, Fediverse and News websites along with filtering functionality.
|
Slixfeed aims to be an easy to use and fully-featured news aggregator bot for XMPP. It provides a convenient access to Blogs, Fediverse and News websites along with filtering functionality.
|
||||||
|
|
||||||
- `Groupchat <xmpp:slixfeed@chat.woodpeckersnest.space?join>`__
|
- `Groupchat <xmpp:slixfeed@chat.woodpeckersnest.space?join>`_
|
||||||
- `Source <https://gitgud.io/sjehuda/slixfeed>`__
|
- `Source <https://gitgud.io/sjehuda/slixfeed>`_
|
||||||
|
|
||||||
sms4you
|
sms4you
|
||||||
~~~~~~~
|
~~~~~~~
|
||||||
sms4you forwards messages from and to SMS and connects either with sms4you-xmpp or sms4you-email to choose the other mean of communication. Nice for receiving or sending SMS, independently from carrying a SIM card.
|
sms4you forwards messages from and to SMS and connects either with sms4you-xmpp or sms4you-email to choose the other mean of communication. Nice for receiving or sending SMS, independently from carrying a SIM card.
|
||||||
|
|
||||||
- `Homepage <https://sms4you-team.pages.debian.net/sms4you/>`__
|
- `Groupchat <xmpp:slixmpp@muc.poez.io?join>`_
|
||||||
- `Source <https://salsa.debian.org/sms4you-team/sms4you>`__
|
- `Homepage <https://sms4you-team.pages.debian.net/sms4you/>`_
|
||||||
|
- `Source <https://salsa.debian.org/sms4you-team/sms4you>`_
|
||||||
|
|
||||||
Stable Diffusion
|
Stable Diffusion
|
||||||
~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~
|
||||||
XMPP bot that generates digital images from textual descriptions.
|
XMPP bot that generates digital images from textual descriptions.
|
||||||
|
|
||||||
- `Groupchat <xmpp:slidge@conference.nicoco.fr?join>`__
|
- `Groupchat <xmpp:slidge@conference.nicoco.fr?join>`_
|
||||||
- `Source <https://www.nicoco.fr/blog/2022/08/31/xmpp-bot-stable-diffusion/>`__
|
- `Source <https://www.nicoco.fr/blog/2022/08/31/xmpp-bot-stable-diffusion/>`_
|
||||||
|
|
||||||
WhisperBot
|
WhisperBot
|
||||||
~~~~~~~~~~
|
~~~~~~~~~~
|
||||||
XMPP bot that transliterates audio messages using OpenAI's Whisper libraries.
|
XMPP bot that transliterates audio messages using OpenAI's Whisper libraries.
|
||||||
|
|
||||||
- `Source <https://codeberg.org/TheCoffeMaker/WhisperBot>`__
|
- `Groupchat <xmpp:slixmpp@muc.poez.io?join>`_
|
||||||
|
- `Source <https://codeberg.org/TheCoffeMaker/WhisperBot>`_
|
||||||
|
|
||||||
XMPP MUC Message Gateway
|
XMPP MUC Message Gateway
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
A multipurpose JSON forwarder microservice from HTTP POST to XMPP MUC room over TLSv1.2 with SliXMPP.
|
A multipurpose JSON forwarder microservice from HTTP POST to XMPP MUC room over TLSv1.2 with SliXMPP.
|
||||||
|
|
||||||
- `Source <https://github.com/immanuelfodor/xmpp-muc-message-gateway>`__
|
- `Source <https://github.com/immanuelfodor/xmpp-muc-message-gateway>`_
|
||||||
|
|
||||||
Services
|
Services
|
||||||
--------
|
--------
|
||||||
@ -87,14 +91,14 @@ AtomToPubsub
|
|||||||
~~~~~~~~~~~~
|
~~~~~~~~~~~~
|
||||||
AtomToPubsub is a simple Python script that parses Atom + RSS feeds and pushes the entries to a designated XMPP Pubsub Node.
|
AtomToPubsub is a simple Python script that parses Atom + RSS feeds and pushes the entries to a designated XMPP Pubsub Node.
|
||||||
|
|
||||||
- `Groupchat <xmpp:movim@conference.movim.eu?join>`__
|
- `Groupchat <xmpp:movim@conference.movim.eu?join>`_
|
||||||
- `Source <https://github.com/imattau/atomtopubsub>`__
|
- `Source <https://github.com/imattau/atomtopubsub>`_
|
||||||
|
|
||||||
Slidge
|
Slidge
|
||||||
~~~~~~
|
~~~~~~
|
||||||
|
|
||||||
Slidge is a general purpose XMPP gateway framework in Python.
|
Slidge is a general purpose XMPP gateway framework in Python.
|
||||||
|
|
||||||
- `Groupchat <xmpp:slidge@conference.nicoco.fr?join>`__
|
- `Groupchat <xmpp:slidge@conference.nicoco.fr?join>`_
|
||||||
- `Homepage <https://slidge.im/core/>`__
|
- `Homepage <https://slidge.im/core/>`_
|
||||||
- `Source <https://sr.ht/~nicoco/slidge>`__
|
- `Source <https://sr.ht/~nicoco/slidge>`_
|
||||||
|
@ -50,39 +50,10 @@ Running the event loop
|
|||||||
only run for this amount of time, and if ``forever`` is False it will
|
only run for this amount of time, and if ``forever`` is False it will
|
||||||
run until disconnection).
|
run until disconnection).
|
||||||
|
|
||||||
This wrapper should be removed in slixmpp 1.9.0.
|
|
||||||
|
|
||||||
Therefore you can handle the event loop in any way you like
|
Therefore you can handle the event loop in any way you like
|
||||||
instead of using ``process()``.
|
instead of using ``process()``.
|
||||||
|
|
||||||
|
|
||||||
Using connect()
|
|
||||||
~~~~~~~~~~~~~~~
|
|
||||||
|
|
||||||
:meth:`.XMLStream.connect` schedules a lot of things in the background, but that
|
|
||||||
only holds true if the event loop is running!
|
|
||||||
|
|
||||||
That is why in all examples we usually call connect() right before calling
|
|
||||||
a `loop.run_…` function, or the deprecated `process()` function.
|
|
||||||
|
|
||||||
Using a different event loop
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
||||||
|
|
||||||
Immediately upon XMPP object creation (`ClientXMPP` / `ComponentXMPP`) you
|
|
||||||
should sets its `loop` attribute to whatever you want, and ideally this
|
|
||||||
should work. This path is less tested, so it may break, if that is the case
|
|
||||||
please report a bug.
|
|
||||||
|
|
||||||
Any access to the `loop` attribute if not user-initialized will set it
|
|
||||||
to the default asyncio event loop by default.
|
|
||||||
|
|
||||||
.. warning::
|
|
||||||
|
|
||||||
If the loop attribute is modified at runtime, the application will probably
|
|
||||||
end up in an hybrid state and asyncio may complain loudly that things bound
|
|
||||||
to an event loop are being ran in another. Try to avoid that situation.
|
|
||||||
|
|
||||||
|
|
||||||
Examples
|
Examples
|
||||||
~~~~~~~~
|
~~~~~~~~
|
||||||
|
|
||||||
@ -102,11 +73,10 @@ callbacks while everything is not ready.
|
|||||||
callback = lambda _: client.connected_event.set()
|
callback = lambda _: client.connected_event.set()
|
||||||
client.add_event_handler('session_start', callback)
|
client.add_event_handler('session_start', callback)
|
||||||
client.connect()
|
client.connect()
|
||||||
loop = asyncio.get_event_loop()
|
|
||||||
loop.run_until_complete(event.wait())
|
loop.run_until_complete(event.wait())
|
||||||
# do some other stuff before running the event loop, e.g.
|
# do some other stuff before running the event loop, e.g.
|
||||||
# loop.run_until_complete(httpserver.init())
|
# loop.run_until_complete(httpserver.init())
|
||||||
loop.run_forever()
|
client.process()
|
||||||
|
|
||||||
|
|
||||||
Use with other asyncio-based libraries
|
Use with other asyncio-based libraries
|
||||||
@ -136,7 +106,7 @@ a simple <message>.
|
|||||||
client.add_event_handler('session_start', get_pythonorg)
|
client.add_event_handler('session_start', get_pythonorg)
|
||||||
client.add_event_handler('session_start', get_asyncioorg)
|
client.add_event_handler('session_start', get_asyncioorg)
|
||||||
client.connect()
|
client.connect()
|
||||||
client.loop.run_until_complete(client.disconnected)
|
client.process()
|
||||||
|
|
||||||
|
|
||||||
Blocking Iq
|
Blocking Iq
|
||||||
@ -166,6 +136,6 @@ JID indicating its findings.
|
|||||||
|
|
||||||
client = ExampleClient('jid@example', 'password')
|
client = ExampleClient('jid@example', 'password')
|
||||||
client.connect()
|
client.connect()
|
||||||
client.loop.run_until_complete(client.disconnected)
|
client.process()
|
||||||
|
|
||||||
|
|
||||||
|
@ -139,7 +139,7 @@ class ClientXMPP(BaseXMPP):
|
|||||||
|
|
||||||
def connect(self, address: Optional[Tuple[str, int]] = None, # type: ignore
|
def connect(self, address: Optional[Tuple[str, int]] = None, # type: ignore
|
||||||
use_ssl: Optional[bool] = None, force_starttls: Optional[bool] = None,
|
use_ssl: Optional[bool] = None, force_starttls: Optional[bool] = None,
|
||||||
disable_starttls: Optional[bool] = None) -> asyncio.Future:
|
disable_starttls: Optional[bool] = None) -> None:
|
||||||
"""Connect to the XMPP server.
|
"""Connect to the XMPP server.
|
||||||
|
|
||||||
When no address is given, a SRV lookup for the server will
|
When no address is given, a SRV lookup for the server will
|
||||||
@ -166,9 +166,8 @@ class ClientXMPP(BaseXMPP):
|
|||||||
host, port = (self.boundjid.host, 5222)
|
host, port = (self.boundjid.host, 5222)
|
||||||
self.dns_service = 'xmpp-client'
|
self.dns_service = 'xmpp-client'
|
||||||
|
|
||||||
return XMLStream.connect(self, host, port, use_ssl=use_ssl,
|
XMLStream.connect(self, host, port, use_ssl=use_ssl,
|
||||||
force_starttls=force_starttls,
|
force_starttls=force_starttls, disable_starttls=disable_starttls)
|
||||||
disable_starttls=disable_starttls)
|
|
||||||
|
|
||||||
def register_feature(self, name: str, handler: Callable, restart: bool = False, order: int = 5000) -> None:
|
def register_feature(self, name: str, handler: Callable, restart: bool = False, order: int = 5000) -> None:
|
||||||
"""Register a stream feature handler.
|
"""Register a stream feature handler.
|
||||||
|
@ -9,7 +9,6 @@
|
|||||||
import logging
|
import logging
|
||||||
import hashlib
|
import hashlib
|
||||||
|
|
||||||
from asyncio import Future
|
|
||||||
from typing import Optional
|
from typing import Optional
|
||||||
|
|
||||||
from slixmpp import Message, Iq, Presence
|
from slixmpp import Message, Iq, Presence
|
||||||
@ -98,7 +97,7 @@ class ComponentXMPP(BaseXMPP):
|
|||||||
|
|
||||||
def connect(self, host: Optional[str] = None, port: int = 0, use_ssl: Optional[bool] = None,
|
def connect(self, host: Optional[str] = None, port: int = 0, use_ssl: Optional[bool] = None,
|
||||||
force_starttls: Optional[bool] = None,
|
force_starttls: Optional[bool] = None,
|
||||||
disable_starttls: Optional[bool] = None) -> Future:
|
disable_starttls: Optional[bool] = None) -> None:
|
||||||
"""Connect to the server.
|
"""Connect to the server.
|
||||||
|
|
||||||
|
|
||||||
@ -119,7 +118,7 @@ class ComponentXMPP(BaseXMPP):
|
|||||||
self.server_name = self.boundjid.host
|
self.server_name = self.boundjid.host
|
||||||
|
|
||||||
log.debug("Connecting to %s:%s", host, port)
|
log.debug("Connecting to %s:%s", host, port)
|
||||||
return XMLStream.connect(self, host=self.server_host, port=self.server_port, use_ssl=use_ssl)
|
XMLStream.connect(self, host=self.server_host, port=self.server_port, use_ssl=use_ssl)
|
||||||
|
|
||||||
def incoming_filter(self, xml):
|
def incoming_filter(self, xml):
|
||||||
"""
|
"""
|
||||||
|
@ -122,7 +122,6 @@ PLUGINS = [
|
|||||||
'xep_0447', # Stateless file sharing
|
'xep_0447', # Stateless file sharing
|
||||||
'xep_0461', # Message Replies
|
'xep_0461', # Message Replies
|
||||||
'xep_0469', # Bookmarks Pinning
|
'xep_0469', # Bookmarks Pinning
|
||||||
'xep_0482', # Call Invites
|
|
||||||
'xep_0490', # Message Displayed Synchronization
|
'xep_0490', # Message Displayed Synchronization
|
||||||
'xep_0492', # Chat Notification Settings
|
'xep_0492', # Chat Notification Settings
|
||||||
# Meant to be imported by plugins
|
# Meant to be imported by plugins
|
||||||
|
@ -162,7 +162,7 @@ class XEP_0009(BasePlugin):
|
|||||||
if not forwarded and self.xmpp.event_handled('jabber_rpc_method_call') > 1:
|
if not forwarded and self.xmpp.event_handled('jabber_rpc_method_call') > 1:
|
||||||
return
|
return
|
||||||
# Reply with error by default
|
# Reply with error by default
|
||||||
error = self.xmpp.plugin['xep_0009']._item_not_found(iq)
|
error = self.client.plugin['xep_0009']._item_not_found(iq)
|
||||||
error.send()
|
error.send()
|
||||||
|
|
||||||
def _on_jabber_rpc_method_response(self, iq, forwarded=False):
|
def _on_jabber_rpc_method_response(self, iq, forwarded=False):
|
||||||
@ -175,7 +175,7 @@ class XEP_0009(BasePlugin):
|
|||||||
"""
|
"""
|
||||||
if not forwarded and self.xmpp.event_handled('jabber_rpc_method_response') > 1:
|
if not forwarded and self.xmpp.event_handled('jabber_rpc_method_response') > 1:
|
||||||
return
|
return
|
||||||
error = self.xmpp.plugin['xep_0009']._recipient_unvailable(iq)
|
error = self.client.plugin['xep_0009']._recpient_unavailable(iq)
|
||||||
error.send()
|
error.send()
|
||||||
|
|
||||||
def _on_jabber_rpc_method_fault(self, iq, forwarded=False):
|
def _on_jabber_rpc_method_fault(self, iq, forwarded=False):
|
||||||
@ -188,7 +188,7 @@ class XEP_0009(BasePlugin):
|
|||||||
"""
|
"""
|
||||||
if not forwarded and self.xmpp.event_handled('jabber_rpc_method_fault') > 1:
|
if not forwarded and self.xmpp.event_handled('jabber_rpc_method_fault') > 1:
|
||||||
return
|
return
|
||||||
error = self.xmpp.plugin['xep_0009']._recipient_unvailable(iq)
|
error = self.client.plugin['xep_0009']._recpient_unavailable(iq)
|
||||||
error.send()
|
error.send()
|
||||||
|
|
||||||
def _on_jabber_rpc_error(self, iq, forwarded=False):
|
def _on_jabber_rpc_error(self, iq, forwarded=False):
|
||||||
@ -201,7 +201,7 @@ class XEP_0009(BasePlugin):
|
|||||||
"""
|
"""
|
||||||
if not forwarded and self.xmpp.event_handled('jabber_rpc_error') > 1:
|
if not forwarded and self.xmpp.event_handled('jabber_rpc_error') > 1:
|
||||||
return
|
return
|
||||||
error = self.xmpp.plugin['xep_0009']._recipient_unvailable(iq)
|
error = self.client.plugin['xep_0009']._recpient_unavailable(iq, iq.get_payload())
|
||||||
error.send()
|
error.send()
|
||||||
|
|
||||||
def _send_fault(self, iq, fault_xml): #
|
def _send_fault(self, iq, fault_xml): #
|
||||||
|
@ -57,9 +57,6 @@ class XEP_0030(BasePlugin):
|
|||||||
Given Given A single node
|
Given Given A single node
|
||||||
====== ======= ============================
|
====== ======= ============================
|
||||||
|
|
||||||
Adding information for a given node without specifying the JID will
|
|
||||||
use the bound JID and therefore must be done after the bind happens.
|
|
||||||
|
|
||||||
Stream Handlers:
|
Stream Handlers:
|
||||||
|
|
||||||
::
|
::
|
||||||
|
@ -9,7 +9,6 @@ from typing import (
|
|||||||
Set,
|
Set,
|
||||||
Tuple,
|
Tuple,
|
||||||
Union,
|
Union,
|
||||||
Dict,
|
|
||||||
)
|
)
|
||||||
from slixmpp.xmlstream import ElementBase, ET
|
from slixmpp.xmlstream import ElementBase, ET
|
||||||
|
|
||||||
@ -145,25 +144,6 @@ class DiscoInfo(ElementBase):
|
|||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def dict_identities(self, lang: Optional[str] = None) -> Set[Dict[str, str]]:
|
|
||||||
"""
|
|
||||||
Return the set of all identities, each one as a dict with
|
|
||||||
category, type, xml_lang, and name keys.
|
|
||||||
|
|
||||||
:param lang: If there is a need to filter identities by lang.
|
|
||||||
"""
|
|
||||||
ids = self.get_identities(lang=lang, dedupe=True)
|
|
||||||
dict_ids = set()
|
|
||||||
for identity in ids:
|
|
||||||
dict_ids.add({
|
|
||||||
'category': identity[0],
|
|
||||||
'type': identity[1],
|
|
||||||
'xml_lang': identity[2],
|
|
||||||
'name': identity[3],
|
|
||||||
})
|
|
||||||
return dict_ids
|
|
||||||
|
|
||||||
|
|
||||||
def get_identities(self, lang: Optional[str] = None, dedupe: bool = True
|
def get_identities(self, lang: Optional[str] = None, dedupe: bool = True
|
||||||
) -> Iterable[IdentityType]:
|
) -> Iterable[IdentityType]:
|
||||||
"""
|
"""
|
||||||
|
@ -326,10 +326,7 @@ class XEP_0050(BasePlugin):
|
|||||||
iq['command']['actions'] = actions
|
iq['command']['actions'] = actions
|
||||||
iq['command']['status'] = 'executing'
|
iq['command']['status'] = 'executing'
|
||||||
else:
|
else:
|
||||||
actions = ['complete']
|
iq['command']['actions'] = ['complete']
|
||||||
if session['allow_prev']:
|
|
||||||
actions.append('prev')
|
|
||||||
iq['command']['actions'] = actions
|
|
||||||
iq['command']['status'] = 'executing'
|
iq['command']['status'] = 'executing'
|
||||||
|
|
||||||
iq['command']['notes'] = session['notes']
|
iq['command']['notes'] = session['notes']
|
||||||
|
@ -464,10 +464,10 @@ class XEP_0060(BasePlugin):
|
|||||||
"""
|
"""
|
||||||
Retrieve the ItemIDs hosted by a given node, using disco.
|
Retrieve the ItemIDs hosted by a given node, using disco.
|
||||||
"""
|
"""
|
||||||
return self.xmpp['xep_0030'].get_items(jid, node, ifrom=ifrom,
|
self.xmpp['xep_0030'].get_items(jid, node, ifrom=ifrom,
|
||||||
callback=callback, timeout=timeout,
|
callback=callback, timeout=timeout,
|
||||||
iterator=iterator,
|
iterator=iterator,
|
||||||
timeout_callback=timeout_callback)
|
timeout_callback=timeout_callback)
|
||||||
|
|
||||||
def modify_affiliations(self, jid, node, affiliations=None, ifrom=None,
|
def modify_affiliations(self, jid, node, affiliations=None, ifrom=None,
|
||||||
timeout_callback=None, callback=None,
|
timeout_callback=None, callback=None,
|
||||||
|
@ -48,12 +48,8 @@ class XEP_0223(BasePlugin):
|
|||||||
|
|
||||||
:param node: Node to set the configuration at.
|
:param node: Node to set the configuration at.
|
||||||
"""
|
"""
|
||||||
config = self.xmpp['xep_0004'].stanza.Form()
|
config = self.xmpp['xep_0004'].Form()
|
||||||
config['type'] = 'submit'
|
config['type'] = 'submit'
|
||||||
config.add_field(
|
|
||||||
var='FORM_TYPE',
|
|
||||||
ftype='hidden',
|
|
||||||
value='http://jabber.org/protocol/pubsub#node_config')
|
|
||||||
|
|
||||||
for field, value in self.profile.items():
|
for field, value in self.profile.items():
|
||||||
config.add_field(var=field, value=value)
|
config.add_field(var=field, value=value)
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
from slixmpp.plugins.base import register_plugin
|
from slixmpp.plugins.base import register_plugin
|
||||||
|
|
||||||
from . import stanza, vcard4
|
from . import stanza, vcard4
|
||||||
from .vcard4 import XEP_0292
|
|
||||||
|
|
||||||
register_plugin(vcard4.XEP_0292)
|
register_plugin(vcard4.XEP_0292)
|
||||||
|
@ -1,12 +1,11 @@
|
|||||||
|
|
||||||
# Slixmpp: The Slick XMPP Library
|
# Slixmpp: The Slick XMPP Library
|
||||||
# Copyright (C) 2012 Nathanael C. Fritz, Lance J.T. Stout
|
# Copyright (C) 2012 Nathanael C. Fritz, Lance J.T. Stout
|
||||||
# This file is part of Slixmpp.
|
# This file is part of Slixmpp.
|
||||||
# See the file LICENSE for copying permissio
|
# See the file LICENSE for copying permissio
|
||||||
import logging
|
import logging
|
||||||
from typing import Optional
|
|
||||||
|
|
||||||
from slixmpp.stanza import Message
|
from slixmpp.stanza import Message
|
||||||
from slixmpp.jid import JID
|
|
||||||
from slixmpp.xmlstream.handler import Callback
|
from slixmpp.xmlstream.handler import Callback
|
||||||
from slixmpp.xmlstream.matcher import StanzaPath
|
from slixmpp.xmlstream.matcher import StanzaPath
|
||||||
from slixmpp.xmlstream import register_stanza_plugin
|
from slixmpp.xmlstream import register_stanza_plugin
|
||||||
@ -50,54 +49,3 @@ class XEP_0308(BasePlugin):
|
|||||||
|
|
||||||
def _handle_correction(self, msg: Message):
|
def _handle_correction(self, msg: Message):
|
||||||
self.xmpp.event('message_correction', msg)
|
self.xmpp.event('message_correction', msg)
|
||||||
|
|
||||||
def build_correction(self, id_to_replace: str, mto: JID,
|
|
||||||
mfrom: Optional[JID] = None, mtype: str = 'chat',
|
|
||||||
mbody: str = '') -> Message:
|
|
||||||
"""
|
|
||||||
Build a corrected message.
|
|
||||||
|
|
||||||
:param id_to_replace: The id of the original message.
|
|
||||||
:param mto: Recipient of the message, must be the same as the original
|
|
||||||
message.
|
|
||||||
:param mfrom: Sender of the message, must be the same as the original
|
|
||||||
message.
|
|
||||||
:param mtype: Type of the message, must be the send as the original
|
|
||||||
message.
|
|
||||||
:param mbody: The corrected message body.
|
|
||||||
"""
|
|
||||||
msg = self.xmpp.make_message(
|
|
||||||
mto=mto,
|
|
||||||
mfrom=mfrom,
|
|
||||||
mbody=mbody,
|
|
||||||
mtype=mtype
|
|
||||||
)
|
|
||||||
msg['replace']['id'] = id_to_replace
|
|
||||||
return msg
|
|
||||||
|
|
||||||
def correct_message(self, msg: Message, body: str) -> Message:
|
|
||||||
"""
|
|
||||||
Send a correction to an existing message.
|
|
||||||
|
|
||||||
:param msg: The message that must be replaced.
|
|
||||||
:param body: The body to set in the correcting message.
|
|
||||||
:returns: The message that was sent.
|
|
||||||
"""
|
|
||||||
to_replace = msg['id']
|
|
||||||
mto = msg['to']
|
|
||||||
mfrom = msg['from']
|
|
||||||
mtype = msg['type']
|
|
||||||
if not to_replace:
|
|
||||||
raise ValueError('No available ID for replacing the message')
|
|
||||||
if not mto:
|
|
||||||
raise ValueError('No available recipient JID')
|
|
||||||
|
|
||||||
new = self.build_correction(
|
|
||||||
id_to_replace=to_replace,
|
|
||||||
mto=mto,
|
|
||||||
mfrom=mfrom,
|
|
||||||
mtype=mtype,
|
|
||||||
mbody=body,
|
|
||||||
)
|
|
||||||
new.send()
|
|
||||||
return new
|
|
||||||
|
@ -138,10 +138,10 @@ class XEP_0356(BasePlugin):
|
|||||||
|
|
||||||
Raises ValueError if the server did not advertise the corresponding privileges
|
Raises ValueError if the server did not advertise the corresponding privileges
|
||||||
|
|
||||||
Here is an example of a roster_items value:
|
:param jid: user we want to add or modify roster items
|
||||||
|
:param roster_items: a dict containing the roster items' JIDs as keys and
|
||||||
.. code-block:: json
|
nested dicts containing names, subscriptions and groups.
|
||||||
|
Example:
|
||||||
{
|
{
|
||||||
"friend1@example.com": {
|
"friend1@example.com": {
|
||||||
"name": "Friend 1",
|
"name": "Friend 1",
|
||||||
@ -152,13 +152,8 @@ class XEP_0356(BasePlugin):
|
|||||||
"name": "Friend 2",
|
"name": "Friend 2",
|
||||||
"subscription": "from",
|
"subscription": "from",
|
||||||
"groups": ["group3"],
|
"groups": ["group3"],
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
:param jid: user we want to add or modify roster items
|
|
||||||
:param roster_items: a dict containing the roster items' JIDs as keys and
|
|
||||||
nested dicts containing names, subscriptions and groups.
|
|
||||||
|
|
||||||
"""
|
"""
|
||||||
if isinstance(jid, str):
|
if isinstance(jid, str):
|
||||||
jid = JID(jid)
|
jid = JID(jid)
|
||||||
|
@ -14,12 +14,11 @@ class Reply(ElementBase):
|
|||||||
interfaces = {"id", "to"}
|
interfaces = {"id", "to"}
|
||||||
|
|
||||||
def add_quoted_fallback(self, fallback: str, nickname: Optional[str] = None):
|
def add_quoted_fallback(self, fallback: str, nickname: Optional[str] = None):
|
||||||
r"""
|
"""
|
||||||
Add plain text fallback for clients not implementing XEP-0461.
|
Add plain text fallback for clients not implementing XEP-0461.
|
||||||
|
|
||||||
|
|
||||||
``msg["reply"].add_quoted_fallback("Some text", "Bob")`` will
|
``msg["reply"].add_quoted_fallback("Some text", "Bob")`` will
|
||||||
prepend ``> Bob:\n> Some text\n`` to the body of the message, and set the
|
prepend "> Bob:\n> Some text\n" to the body of the message, and set the
|
||||||
fallback_body attributes accordingly, so that clients implementing
|
fallback_body attributes accordingly, so that clients implementing
|
||||||
XEP-0461 can hide the fallback text.
|
XEP-0461 can hide the fallback text.
|
||||||
|
|
||||||
|
@ -1,11 +0,0 @@
|
|||||||
# Slixmpp: The Slick XMPP Library
|
|
||||||
# Copyright (C) 2025 Mathieu Pasquet
|
|
||||||
# This file is part of Slixmpp.
|
|
||||||
# See the file LICENSE for copying permissio
|
|
||||||
from slixmpp.plugins.base import register_plugin
|
|
||||||
|
|
||||||
from slixmpp.plugins.xep_0482 import stanza
|
|
||||||
from slixmpp.plugins.xep_0482.call_invites import XEP_0482
|
|
||||||
|
|
||||||
|
|
||||||
register_plugin(XEP_0482)
|
|
@ -1,55 +0,0 @@
|
|||||||
# Slixmpp: The Slick XMPP Library
|
|
||||||
# Copyright (C) 2025 Mathieu Pasquet
|
|
||||||
# This file is part of Slixmpp.
|
|
||||||
# See the file LICENSE for copying permissio
|
|
||||||
import logging
|
|
||||||
from typing import Optional
|
|
||||||
|
|
||||||
from slixmpp.stanza import Message
|
|
||||||
from slixmpp.jid import JID
|
|
||||||
from slixmpp.xmlstream.handler import Callback
|
|
||||||
from slixmpp.xmlstream.matcher import StanzaPath
|
|
||||||
from slixmpp.xmlstream import register_stanza_plugin
|
|
||||||
from slixmpp.plugins import BasePlugin
|
|
||||||
from slixmpp.plugins.xep_0482 import stanza
|
|
||||||
|
|
||||||
|
|
||||||
log = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class XEP_0482(BasePlugin):
|
|
||||||
|
|
||||||
"""
|
|
||||||
XEP-0482: Call Invites
|
|
||||||
|
|
||||||
This plugin defines the stanza elements for Call Invites, as well as new
|
|
||||||
events:
|
|
||||||
|
|
||||||
- `call-invite`
|
|
||||||
- `call-reject`
|
|
||||||
- `call-retract`
|
|
||||||
- `call-leave`
|
|
||||||
- `call-left`
|
|
||||||
"""
|
|
||||||
|
|
||||||
name = 'xep_0482'
|
|
||||||
description = 'XEP-0482: Call Invites'
|
|
||||||
dependencies = set()
|
|
||||||
stanza = stanza
|
|
||||||
|
|
||||||
def plugin_init(self):
|
|
||||||
stanza.register_plugins()
|
|
||||||
|
|
||||||
for event in ('invite', 'reject', 'retract', 'leave', 'left'):
|
|
||||||
self.xmpp.register_handler(
|
|
||||||
Callback(f'Call {event}',
|
|
||||||
StanzaPath(f'message/call-{event}'),
|
|
||||||
self._handle_event))
|
|
||||||
def _handle_event(self, message):
|
|
||||||
for event in ('invite', 'reject', 'retract', 'leave', 'left'):
|
|
||||||
if message.get_plugin(f'call-{event}', check=True):
|
|
||||||
self.xmpp.event(f'call-{event}')
|
|
||||||
|
|
||||||
def plugin_end(self):
|
|
||||||
for event in ('invite', 'reject', 'retract', 'leave', 'left'):
|
|
||||||
self.xmpp.remove_handler(f'Call {event}')
|
|
@ -1,102 +0,0 @@
|
|||||||
# Slixmpp: The Slick XMPP Library
|
|
||||||
# Copyright (C) 2025 Mathieu Pasquet
|
|
||||||
# This file is part of Slixmpp.
|
|
||||||
# See the file LICENSE for copying permission
|
|
||||||
|
|
||||||
from typing import Tuple, List, Optional
|
|
||||||
from slixmpp import Message
|
|
||||||
from slixmpp.jid import JID
|
|
||||||
from slixmpp.xmlstream import ElementBase, register_stanza_plugin
|
|
||||||
|
|
||||||
NS = 'urn:xmpp:call-invites:0'
|
|
||||||
|
|
||||||
|
|
||||||
class Jingle(ElementBase):
|
|
||||||
name = 'jingle'
|
|
||||||
namespace = NS
|
|
||||||
plugin_attrib = 'jingle'
|
|
||||||
plugin_multi_attrib = 'jingles'
|
|
||||||
interfaces = {'sid', 'jid'}
|
|
||||||
|
|
||||||
def set_jid(self, value: JID) -> None:
|
|
||||||
if not isinstance(value, JID):
|
|
||||||
try:
|
|
||||||
value = JID(value)
|
|
||||||
except ValueError:
|
|
||||||
raise ValueError(f'"jid" must be a valid JID object')
|
|
||||||
self.xml.attrib['jid'] = value.full
|
|
||||||
|
|
||||||
def get_jid(self) -> Optional[JID]:
|
|
||||||
try:
|
|
||||||
return JID(self.xml.attrib.get('jid', ''))
|
|
||||||
except ValueError:
|
|
||||||
return None
|
|
||||||
|
|
||||||
|
|
||||||
class External(ElementBase):
|
|
||||||
name = 'external'
|
|
||||||
namespace = NS
|
|
||||||
plugin_attrib = 'external'
|
|
||||||
plugin_multi_attrib = 'externals'
|
|
||||||
interfaces = {'uri'}
|
|
||||||
|
|
||||||
|
|
||||||
class Invite(ElementBase):
|
|
||||||
name = 'invite'
|
|
||||||
namespace = NS
|
|
||||||
plugin_attrib = 'call-invite'
|
|
||||||
interfaces = {'video'}
|
|
||||||
|
|
||||||
def get_methods(self) -> Tuple[List[Jingle], List[External]]:
|
|
||||||
return (self['jingles'], self['externals'])
|
|
||||||
|
|
||||||
def set_video(self, value: bool) -> None:
|
|
||||||
if not isinstance(value, bool):
|
|
||||||
raise ValueError(f'Invalid value for the video attribute: {value}')
|
|
||||||
self.xml.attrib['video'] = str(value).lower()
|
|
||||||
|
|
||||||
def get_video(self) -> bool:
|
|
||||||
vid = self.xml.attrib.get('video', 'false').lower()
|
|
||||||
return vid == 'true'
|
|
||||||
|
|
||||||
|
|
||||||
class Retract(ElementBase):
|
|
||||||
name = 'retract'
|
|
||||||
namespace = NS
|
|
||||||
plugin_attrib = 'call-retract'
|
|
||||||
interfaces = {'id'}
|
|
||||||
|
|
||||||
|
|
||||||
class Accept(ElementBase):
|
|
||||||
name = 'accept'
|
|
||||||
namespace = NS
|
|
||||||
plugin_attrib = 'call-accept'
|
|
||||||
interfaces = {'id'}
|
|
||||||
|
|
||||||
|
|
||||||
class Reject(ElementBase):
|
|
||||||
name = 'reject'
|
|
||||||
namespace = NS
|
|
||||||
plugin_attrib = 'call-reject'
|
|
||||||
interfaces = {'id'}
|
|
||||||
|
|
||||||
|
|
||||||
class Left(ElementBase):
|
|
||||||
name = 'left'
|
|
||||||
namespace = NS
|
|
||||||
plugin_attrib = 'call-left'
|
|
||||||
interfaces = {'id'}
|
|
||||||
|
|
||||||
|
|
||||||
def register_plugins() -> None:
|
|
||||||
register_stanza_plugin(Message, Invite)
|
|
||||||
register_stanza_plugin(Message, Retract)
|
|
||||||
register_stanza_plugin(Message, Accept)
|
|
||||||
register_stanza_plugin(Message, Reject)
|
|
||||||
register_stanza_plugin(Message, Left)
|
|
||||||
|
|
||||||
register_stanza_plugin(Invite, Jingle, iterable=True)
|
|
||||||
register_stanza_plugin(Invite, External, iterable=True)
|
|
||||||
|
|
||||||
register_stanza_plugin(Accept, Jingle)
|
|
||||||
register_stanza_plugin(Accept, External)
|
|
@ -375,23 +375,7 @@ class XMLStream(asyncio.BaseProtocol):
|
|||||||
@property
|
@property
|
||||||
def loop(self) -> AbstractEventLoop:
|
def loop(self) -> AbstractEventLoop:
|
||||||
if self._loop is None:
|
if self._loop is None:
|
||||||
try:
|
self._loop = asyncio.get_event_loop()
|
||||||
with warnings.catch_warnings():
|
|
||||||
warnings.simplefilter("ignore")
|
|
||||||
self._loop = asyncio.get_event_loop()
|
|
||||||
# We do not know what exception will be raised in the future
|
|
||||||
# instead of the warning
|
|
||||||
except Exception:
|
|
||||||
try:
|
|
||||||
current = asyncio.get_running_loop()
|
|
||||||
except RuntimeError:
|
|
||||||
current = None
|
|
||||||
if current is not None:
|
|
||||||
self._loop = current
|
|
||||||
else:
|
|
||||||
loop = asyncio.new_event_loop()
|
|
||||||
asyncio.set_event_loop(loop)
|
|
||||||
self._loop = loop
|
|
||||||
return self._loop
|
return self._loop
|
||||||
|
|
||||||
@loop.setter
|
@loop.setter
|
||||||
@ -425,10 +409,9 @@ class XMLStream(asyncio.BaseProtocol):
|
|||||||
self.disconnected.set_result(True)
|
self.disconnected.set_result(True)
|
||||||
self.disconnected = asyncio.Future()
|
self.disconnected = asyncio.Future()
|
||||||
|
|
||||||
def connect(self, host: str = '', port: int = 0,
|
def connect(self, host: str = '', port: int = 0, use_ssl: Optional[bool] = None,
|
||||||
use_ssl: Optional[bool] = None,
|
|
||||||
force_starttls: Optional[bool] = None,
|
force_starttls: Optional[bool] = None,
|
||||||
disable_starttls: Optional[bool] = None) -> asyncio.Future:
|
disable_starttls: Optional[bool] = None) -> None:
|
||||||
"""Create a new socket and connect to the server.
|
"""Create a new socket and connect to the server.
|
||||||
|
|
||||||
:param host: The name of the desired server for the connection.
|
:param host: The name of the desired server for the connection.
|
||||||
@ -447,7 +430,6 @@ class XMLStream(asyncio.BaseProtocol):
|
|||||||
upgrade to TLS, even if the server provides
|
upgrade to TLS, even if the server provides
|
||||||
it. Use this for example if you’re on
|
it. Use this for example if you’re on
|
||||||
localhost
|
localhost
|
||||||
:returns: A future on the current connection attempt
|
|
||||||
|
|
||||||
"""
|
"""
|
||||||
if self._run_out_filters is None or self._run_out_filters.done():
|
if self._run_out_filters is None or self._run_out_filters.done():
|
||||||
@ -479,14 +461,8 @@ class XMLStream(asyncio.BaseProtocol):
|
|||||||
self._connect_routine(),
|
self._connect_routine(),
|
||||||
loop=self.loop,
|
loop=self.loop,
|
||||||
)
|
)
|
||||||
return self._current_connection_attempt
|
|
||||||
|
|
||||||
async def _connect_routine(self) -> Optional[asyncio.Future]:
|
async def _connect_routine(self) -> None:
|
||||||
"""
|
|
||||||
Returns None if the attempt was canceled or if the connection succeeded
|
|
||||||
(cancelling done manually by the library user, so that should be known)
|
|
||||||
or the next connection attempt future if a new try has been scheduled.
|
|
||||||
"""
|
|
||||||
self.event_when_connected = "connected"
|
self.event_when_connected = "connected"
|
||||||
|
|
||||||
if self._connect_loop_wait > 0:
|
if self._connect_loop_wait > 0:
|
||||||
@ -511,7 +487,7 @@ class XMLStream(asyncio.BaseProtocol):
|
|||||||
ssl_context = None
|
ssl_context = None
|
||||||
|
|
||||||
if self._current_connection_attempt is None:
|
if self._current_connection_attempt is None:
|
||||||
return None
|
return
|
||||||
try:
|
try:
|
||||||
server_hostname = self.default_domain if self.use_ssl else None
|
server_hostname = self.default_domain if self.use_ssl else None
|
||||||
await self.loop.create_connection(lambda: self,
|
await self.loop.create_connection(lambda: self,
|
||||||
@ -523,12 +499,11 @@ class XMLStream(asyncio.BaseProtocol):
|
|||||||
except Socket.gaierror as e:
|
except Socket.gaierror as e:
|
||||||
self.event('connection_failed',
|
self.event('connection_failed',
|
||||||
'No DNS record available for %s' % self.default_domain)
|
'No DNS record available for %s' % self.default_domain)
|
||||||
return self.reschedule_connection_attempt()
|
self.reschedule_connection_attempt()
|
||||||
except OSError as e:
|
except OSError as e:
|
||||||
log.debug('Connection failed: %s', e)
|
log.debug('Connection failed: %s', e)
|
||||||
self.event("connection_failed", e)
|
self.event("connection_failed", e)
|
||||||
return self.reschedule_connection_attempt()
|
self.reschedule_connection_attempt()
|
||||||
return None
|
|
||||||
|
|
||||||
def process(self, *, forever: bool = True, timeout: Optional[int] = None) -> None:
|
def process(self, *, forever: bool = True, timeout: Optional[int] = None) -> None:
|
||||||
"""Process all the available XMPP events (receiving or sending data on the
|
"""Process all the available XMPP events (receiving or sending data on the
|
||||||
@ -664,22 +639,19 @@ class XMLStream(asyncio.BaseProtocol):
|
|||||||
self._set_disconnected_future()
|
self._set_disconnected_future()
|
||||||
self.event("disconnected", self.disconnect_reason or exception)
|
self.event("disconnected", self.disconnect_reason or exception)
|
||||||
|
|
||||||
def reschedule_connection_attempt(self) -> Optional[asyncio.Future]:
|
def reschedule_connection_attempt(self) -> None:
|
||||||
"""
|
"""
|
||||||
Increase the exponential back-off and initate another background
|
Increase the exponential back-off and initate another background
|
||||||
_connect_routine call to connect to the server.
|
_connect_routine call to connect to the server.
|
||||||
|
|
||||||
:returns: A future on the next scheduled connection attempt.
|
|
||||||
"""
|
"""
|
||||||
# abort if there is no ongoing connection attempt
|
# abort if there is no ongoing connection attempt
|
||||||
if self._current_connection_attempt is None:
|
if self._current_connection_attempt is None:
|
||||||
return None
|
return
|
||||||
self._connect_loop_wait = min(300, self._connect_loop_wait * 2 + 1)
|
self._connect_loop_wait = min(300, self._connect_loop_wait * 2 + 1)
|
||||||
self._current_connection_attempt = asyncio.ensure_future(
|
self._current_connection_attempt = asyncio.ensure_future(
|
||||||
self._connect_routine(),
|
self._connect_routine(),
|
||||||
loop=self.loop,
|
loop=self.loop,
|
||||||
)
|
)
|
||||||
return self._current_connection_attempt
|
|
||||||
|
|
||||||
def cancel_connection_attempt(self) -> None:
|
def cancel_connection_attempt(self) -> None:
|
||||||
"""
|
"""
|
||||||
|
@ -1,26 +0,0 @@
|
|||||||
import unittest
|
|
||||||
from slixmpp import Message
|
|
||||||
from slixmpp.test import SlixTest
|
|
||||||
from slixmpp.plugins.xep_0308 import Replace
|
|
||||||
from slixmpp.xmlstream import register_stanza_plugin
|
|
||||||
|
|
||||||
|
|
||||||
class TestCorrectStanza(SlixTest):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
register_stanza_plugin(Message, Replace)
|
|
||||||
|
|
||||||
def testBuild(self):
|
|
||||||
"""Test that the element is created correctly."""
|
|
||||||
msg = Message()
|
|
||||||
msg['type'] = 'chat'
|
|
||||||
msg['replace']['id'] = 'toto123'
|
|
||||||
|
|
||||||
self.check(msg, """
|
|
||||||
<message type="chat">
|
|
||||||
<replace xmlns="urn:xmpp:message-correct:0" id="toto123"/>
|
|
||||||
</message>
|
|
||||||
""")
|
|
||||||
|
|
||||||
|
|
||||||
suite = unittest.TestLoader().loadTestsFromTestCase(TestCorrectStanza)
|
|
@ -1,42 +0,0 @@
|
|||||||
import unittest
|
|
||||||
from slixmpp import Message
|
|
||||||
from slixmpp.jid import JID
|
|
||||||
from slixmpp.test import SlixTest
|
|
||||||
from slixmpp.plugins.xep_0482 import stanza
|
|
||||||
from slixmpp.plugins.xep_0482.stanza import External, Jingle
|
|
||||||
from slixmpp.xmlstream import register_stanza_plugin
|
|
||||||
|
|
||||||
|
|
||||||
class TestCallInviteStanza(SlixTest):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
stanza.register_plugins()
|
|
||||||
|
|
||||||
def test_invite(self):
|
|
||||||
"""Test that the element is created correctly."""
|
|
||||||
msg = Message()
|
|
||||||
msg['call-invite']['video'] = True
|
|
||||||
jingle = Jingle()
|
|
||||||
jingle['sid'] = 'toto'
|
|
||||||
jingle['jid'] = JID('toto@example.com/m')
|
|
||||||
external = External()
|
|
||||||
external['uri'] = "https://example.com/call"
|
|
||||||
msg['call-invite'].append(jingle)
|
|
||||||
msg['call-invite'].append(external)
|
|
||||||
|
|
||||||
self.check(msg, """
|
|
||||||
<message>
|
|
||||||
<invite xmlns="urn:xmpp:call-invites:0" video="true">
|
|
||||||
<jingle sid="toto" jid="toto@example.com/m" />
|
|
||||||
<external uri="https://example.com/call" />
|
|
||||||
</invite>
|
|
||||||
</message>
|
|
||||||
""")
|
|
||||||
|
|
||||||
self.assertEqual(
|
|
||||||
msg['call-invite'].get_methods(),
|
|
||||||
([jingle], [external]),
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
suite = unittest.TestLoader().loadTestsFromTestCase(TestCallInviteStanza)
|
|
@ -1,28 +0,0 @@
|
|||||||
import unittest
|
|
||||||
from slixmpp.test import SlixTest
|
|
||||||
|
|
||||||
|
|
||||||
class TestPrivatePEP(SlixTest):
|
|
||||||
|
|
||||||
def testConfigureNode(self):
|
|
||||||
self.stream_start(mode='client', plugins=['xep_0223'])
|
|
||||||
|
|
||||||
self.xmpp.plugin['xep_0223'].configure(node="toto")
|
|
||||||
self.send("""
|
|
||||||
<iq type="set" id="1">
|
|
||||||
<pubsub xmlns='http://jabber.org/protocol/pubsub#owner'>
|
|
||||||
<configure node='toto'>
|
|
||||||
<x xmlns='jabber:x:data' type='submit'>
|
|
||||||
<field var='FORM_TYPE' type='hidden'>
|
|
||||||
<value>http://jabber.org/protocol/pubsub#node_config</value>
|
|
||||||
</field>
|
|
||||||
<field var='pubsub#persist_items'><value>1</value></field>
|
|
||||||
<field var='pubsub#access_model'><value>whitelist</value></field>
|
|
||||||
</x>
|
|
||||||
</configure>
|
|
||||||
</pubsub>
|
|
||||||
</iq>
|
|
||||||
""")
|
|
||||||
|
|
||||||
|
|
||||||
suite = unittest.TestLoader().loadTestsFromTestCase(TestPrivatePEP)
|
|
@ -1,53 +0,0 @@
|
|||||||
import unittest
|
|
||||||
from slixmpp.jid import JID
|
|
||||||
from slixmpp.test import SlixTest
|
|
||||||
|
|
||||||
|
|
||||||
class TestStreamCorrect(SlixTest):
|
|
||||||
|
|
||||||
def test_recv_correct(self):
|
|
||||||
self.stream_start(mode='client', plugins=['xep_0308'])
|
|
||||||
|
|
||||||
recv = []
|
|
||||||
|
|
||||||
def recv_correct(msg):
|
|
||||||
recv.append(msg)
|
|
||||||
|
|
||||||
self.xmpp.add_event_handler('message_correction', recv_correct)
|
|
||||||
|
|
||||||
self.recv("""
|
|
||||||
<message from="example.com" to="toto@example">
|
|
||||||
<replace xmlns="urn:xmpp:message-correct:0" id="tototo"/>
|
|
||||||
<body>oucou</body>
|
|
||||||
</message>
|
|
||||||
""")
|
|
||||||
received = recv[0]
|
|
||||||
self.assertEqual(received['replace']['id'], "tototo")
|
|
||||||
|
|
||||||
|
|
||||||
def test_send_correct(self):
|
|
||||||
self.stream_start(mode='client', plugins=['xep_0308'])
|
|
||||||
|
|
||||||
corrected = self.xmpp.plugin['xep_0308'].build_correction(
|
|
||||||
id_to_replace="12345",
|
|
||||||
mto=JID('toto@example.com'),
|
|
||||||
mbody="I am replacing",
|
|
||||||
)
|
|
||||||
self.assertEqual(corrected['replace']['id'], '12345')
|
|
||||||
self.assertEqual(corrected['to'], JID('toto@example.com'))
|
|
||||||
self.assertEqual(corrected['body'], 'I am replacing')
|
|
||||||
corrected['id'] = 'my id'
|
|
||||||
|
|
||||||
corrected = self.xmpp.plugin['xep_0308'].correct_message(
|
|
||||||
corrected,
|
|
||||||
'This is new',
|
|
||||||
)
|
|
||||||
self.send("""
|
|
||||||
<message type="chat" to="toto@example.com">
|
|
||||||
<body>This is new</body>
|
|
||||||
<replace xmlns="urn:xmpp:message-correct:0" id="my id" />
|
|
||||||
</message>
|
|
||||||
""")
|
|
||||||
|
|
||||||
|
|
||||||
suite = unittest.TestLoader().loadTestsFromTestCase(TestStreamCorrect)
|
|
Loading…
Reference in New Issue
Block a user