Merge branch 'xep-0444-and-not-protoxep' into 'master'
Promote protoxep_reactions to XEP-0444 See merge request poezio/slixmpp!64
This commit is contained in:
commit
c86a6ad299
8 changed files with 161 additions and 52 deletions
|
@ -9,7 +9,8 @@ test:
|
||||||
image: ubuntu:latest
|
image: ubuntu:latest
|
||||||
script:
|
script:
|
||||||
- apt update
|
- apt update
|
||||||
- apt install -y python3 cython3 gpg
|
- apt install -y python3 python3-pip cython3 gpg
|
||||||
|
- pip3 install emoji
|
||||||
- ./run_tests.py
|
- ./run_tests.py
|
||||||
|
|
||||||
trigger_poezio:
|
trigger_poezio:
|
||||||
|
|
3
setup.py
3
setup.py
|
@ -30,6 +30,7 @@ CLASSIFIERS = [
|
||||||
'Programming Language :: Python',
|
'Programming Language :: Python',
|
||||||
'Programming Language :: Python :: 3.7',
|
'Programming Language :: Python :: 3.7',
|
||||||
'Programming Language :: Python :: 3.8',
|
'Programming Language :: Python :: 3.8',
|
||||||
|
'Programming Language :: Python :: 3.9',
|
||||||
'Topic :: Internet :: XMPP',
|
'Topic :: Internet :: XMPP',
|
||||||
'Topic :: Software Development :: Libraries :: Python Modules',
|
'Topic :: Software Development :: Libraries :: Python Modules',
|
||||||
]
|
]
|
||||||
|
@ -82,7 +83,7 @@ setup(
|
||||||
platforms=['any'],
|
platforms=['any'],
|
||||||
packages=packages,
|
packages=packages,
|
||||||
ext_modules=ext_modules,
|
ext_modules=ext_modules,
|
||||||
install_requires=['aiodns>=1.0', 'pyasn1', 'pyasn1_modules', 'aiohttp'],
|
install_requires=['aiodns>=1.0', 'pyasn1', 'pyasn1_modules', 'aiohttp', 'emoji'],
|
||||||
classifiers=CLASSIFIERS,
|
classifiers=CLASSIFIERS,
|
||||||
cmdclass={'test': TestCommand}
|
cmdclass={'test': TestCommand}
|
||||||
)
|
)
|
||||||
|
|
|
@ -86,6 +86,6 @@ __all__ = [
|
||||||
'xep_0325', # IoT Systems Control
|
'xep_0325', # IoT Systems Control
|
||||||
'xep_0332', # HTTP Over XMPP Transport
|
'xep_0332', # HTTP Over XMPP Transport
|
||||||
'xep_0377', # Spam reporting
|
'xep_0377', # Spam reporting
|
||||||
'protoxep_reactions', # https://dino.im/xeps/reactions.html
|
'xep_0444', # Message Reactions
|
||||||
'protoxep_occupantid', # https://dino.im/xeps/occupant-id.html
|
'protoxep_occupantid', # https://dino.im/xeps/occupant-id.html
|
||||||
]
|
]
|
||||||
|
|
|
@ -1,31 +0,0 @@
|
||||||
"""
|
|
||||||
Slixmpp: The Slick XMPP Library
|
|
||||||
Copyright (C) 2019 Mathieu Pasquet
|
|
||||||
This file is part of Slixmpp.
|
|
||||||
|
|
||||||
See the file LICENSE for copying permission.
|
|
||||||
"""
|
|
||||||
|
|
||||||
from slixmpp.xmlstream import ElementBase, register_stanza_plugin
|
|
||||||
|
|
||||||
|
|
||||||
class Reactions(ElementBase):
|
|
||||||
name = 'reactions'
|
|
||||||
plugin_attrib = 'reactions'
|
|
||||||
namespace = 'urn:xmpp:reactions:0'
|
|
||||||
interfaces = {'to'}
|
|
||||||
|
|
||||||
|
|
||||||
class Reaction(ElementBase):
|
|
||||||
name = 'reaction'
|
|
||||||
namespace = 'urn:xmpp:reactions:0'
|
|
||||||
interfaces = {'value'}
|
|
||||||
|
|
||||||
def get_value(self) -> str:
|
|
||||||
return self.xml.text
|
|
||||||
|
|
||||||
def set_value(self, value: str):
|
|
||||||
self.xml.text = value
|
|
||||||
|
|
||||||
|
|
||||||
register_stanza_plugin(Reactions, Reaction, iterable=True)
|
|
|
@ -1,11 +1,11 @@
|
||||||
"""
|
"""
|
||||||
Slixmpp: The Slick XMPP Library
|
Slixmpp: The Slick XMPP Library
|
||||||
Copyright (C) 2019 Mathieu Pasquet
|
Copyright (C) 2020 Mathieu Pasquet
|
||||||
This file is part of Slixmpp.
|
This file is part of Slixmpp.
|
||||||
|
|
||||||
See the file LICENSE for copying permission.
|
See the file LICENSE for copying permission.
|
||||||
"""
|
"""
|
||||||
from slixmpp.plugins.base import register_plugin
|
from slixmpp.plugins.base import register_plugin
|
||||||
from slixmpp.plugins.protoxep_reactions.reactions import XEP_Reactions
|
from slixmpp.plugins.xep_0444.reactions import XEP_0444
|
||||||
|
|
||||||
register_plugin(XEP_Reactions)
|
register_plugin(XEP_0444)
|
|
@ -1,26 +1,28 @@
|
||||||
"""
|
"""
|
||||||
Slixmpp: The Slick XMPP Library
|
Slixmpp: The Slick XMPP Library
|
||||||
Copyright (C) 2019 Mathieu Pasquet
|
Copyright (C) 2020 Mathieu Pasquet
|
||||||
This file is part of Slixmpp.
|
This file is part of Slixmpp.
|
||||||
|
|
||||||
See the file LICENSE for copying permission.
|
See the file LICENSE for copying permission.
|
||||||
"""
|
"""
|
||||||
from typing import Iterable
|
from typing import Iterable
|
||||||
|
|
||||||
|
from slixmpp import JID
|
||||||
from slixmpp.plugins import BasePlugin
|
from slixmpp.plugins import BasePlugin
|
||||||
from slixmpp.stanza import Message
|
from slixmpp.stanza import Message
|
||||||
from slixmpp.xmlstream import register_stanza_plugin
|
from slixmpp.xmlstream import register_stanza_plugin
|
||||||
from slixmpp.xmlstream.matcher import MatchXMLMask
|
from slixmpp.xmlstream.matcher import MatchXMLMask
|
||||||
from slixmpp.xmlstream.handler import Callback
|
from slixmpp.xmlstream.handler import Callback
|
||||||
|
|
||||||
from slixmpp.plugins.protoxep_reactions import stanza
|
from slixmpp.plugins.xep_0444 import stanza
|
||||||
|
|
||||||
|
|
||||||
class XEP_Reactions(BasePlugin):
|
class XEP_0444(BasePlugin):
|
||||||
name = 'protoxep_reactions'
|
name = 'xep_0444'
|
||||||
description = 'XEP-XXXX: Message Reactions'
|
description = 'XEP-0444: Message Reactions'
|
||||||
dependencies = {'xep_0030'}
|
dependencies = {'xep_0030', 'xep_0334'}
|
||||||
stanza = stanza
|
stanza = stanza
|
||||||
|
namespace = stanza.NS
|
||||||
|
|
||||||
def plugin_init(self):
|
def plugin_init(self):
|
||||||
self.xmpp.register_handler(
|
self.xmpp.register_handler(
|
||||||
|
@ -30,25 +32,32 @@ class XEP_Reactions(BasePlugin):
|
||||||
self._handle_reactions,
|
self._handle_reactions,
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
self.xmpp['xep_0030'].add_feature('urn:xmpp:reactions:0')
|
|
||||||
register_stanza_plugin(Message, stanza.Reactions)
|
register_stanza_plugin(Message, stanza.Reactions)
|
||||||
|
register_stanza_plugin(stanza.Reactions, stanza.Reaction, iterable=True)
|
||||||
|
|
||||||
|
def session_bind(self, event):
|
||||||
|
self.xmpp['xep_0030'].add_feature(stanza.NS)
|
||||||
|
|
||||||
def plugin_end(self):
|
def plugin_end(self):
|
||||||
self.xmpp.remove_handler('Reaction received')
|
self.xmpp.remove_handler('Reaction received')
|
||||||
self.xmpp['xep_0030'].remove_feature('urn:xmpp:reactions:0')
|
self.xmpp['xep_0030'].remove_feature(stanza.NS)
|
||||||
|
|
||||||
def _handle_reactions(self, message: Message):
|
def _handle_reactions(self, message: Message):
|
||||||
self.xmpp.event('reactions', message)
|
self.xmpp.event('reactions', message)
|
||||||
|
|
||||||
|
def send_reactions(self, to: JID, to_id: str, reactions: Iterable[str], *, store=True):
|
||||||
|
"""Send reactions related to a message"""
|
||||||
|
msg = self.xmpp.make_message(mto=to)
|
||||||
|
self.set_reactions(msg, to_id, reactions)
|
||||||
|
if store:
|
||||||
|
msg.enable('store')
|
||||||
|
msg.send()
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def set_reactions(message: Message, to_id: str, reactions: Iterable[str]):
|
def set_reactions(message: Message, to_id: str, reactions: Iterable[str]):
|
||||||
"""
|
"""Add reactions to a Message object."""
|
||||||
Add reactions to a Message object.
|
message['reactions']['id'] = to_id
|
||||||
"""
|
|
||||||
reactions_stanza = stanza.Reactions()
|
|
||||||
reactions_stanza['to'] = to_id
|
|
||||||
for reaction in reactions:
|
for reaction in reactions:
|
||||||
reaction_stanza = stanza.Reaction()
|
reaction_stanza = stanza.Reaction()
|
||||||
reaction_stanza['value'] = reaction
|
reaction_stanza['value'] = reaction
|
||||||
reactions_stanza.append(reaction_stanza)
|
message['reactions'].append(reaction_stanza)
|
||||||
message.append(reactions_stanza)
|
|
60
slixmpp/plugins/xep_0444/stanza.py
Normal file
60
slixmpp/plugins/xep_0444/stanza.py
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
"""
|
||||||
|
Slixmpp: The Slick XMPP Library
|
||||||
|
Copyright (C) 2020 Mathieu Pasquet
|
||||||
|
This file is part of Slixmpp.
|
||||||
|
|
||||||
|
See the file LICENSE for copying permission.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from typing import Set, Iterable
|
||||||
|
from slixmpp.xmlstream import ElementBase
|
||||||
|
try:
|
||||||
|
from emoji import UNICODE_EMOJI
|
||||||
|
except ImportError:
|
||||||
|
UNICODE_EMOJI = None
|
||||||
|
|
||||||
|
|
||||||
|
NS = 'urn:xmpp:reactions:0'
|
||||||
|
|
||||||
|
class Reactions(ElementBase):
|
||||||
|
name = 'reactions'
|
||||||
|
plugin_attrib = 'reactions'
|
||||||
|
namespace = NS
|
||||||
|
interfaces = {'id', 'values'}
|
||||||
|
|
||||||
|
def get_values(self, *, all_chars=False) -> Set[str]:
|
||||||
|
""""Get all reactions as str"""
|
||||||
|
reactions = set()
|
||||||
|
for reaction in self:
|
||||||
|
value = reaction['value']
|
||||||
|
if UNICODE_EMOJI and not all_chars:
|
||||||
|
if value in UNICODE_EMOJI:
|
||||||
|
reactions.add(reaction['value'])
|
||||||
|
else:
|
||||||
|
reactions.add(reaction['value'])
|
||||||
|
return reactions
|
||||||
|
|
||||||
|
def set_values(self, values: Iterable[str], *, all_chars=False):
|
||||||
|
""""Set all reactions as str"""
|
||||||
|
for element in self.xml.findall('reaction'):
|
||||||
|
self.xml.remove(element)
|
||||||
|
for reaction_txt in values:
|
||||||
|
reaction = Reaction()
|
||||||
|
reaction.set_value(reaction_txt, all_chars=all_chars)
|
||||||
|
self.append(reaction)
|
||||||
|
|
||||||
|
|
||||||
|
class Reaction(ElementBase):
|
||||||
|
name = 'reaction'
|
||||||
|
namespace = NS
|
||||||
|
interfaces = {'value'}
|
||||||
|
|
||||||
|
def get_value(self) -> str:
|
||||||
|
return self.xml.text
|
||||||
|
|
||||||
|
def set_value(self, value: str, *, all_chars=False):
|
||||||
|
if UNICODE_EMOJI and not all_chars:
|
||||||
|
if not value in UNICODE_EMOJI:
|
||||||
|
raise ValueError("%s is not a valid emoji" % value)
|
||||||
|
self.xml.text = value
|
||||||
|
|
69
tests/test_stanza_xep_0444.py
Normal file
69
tests/test_stanza_xep_0444.py
Normal file
|
@ -0,0 +1,69 @@
|
||||||
|
"""
|
||||||
|
Slixmpp: The Slick XMPP Library
|
||||||
|
Copyright (C) 2020 Mathieu Pasquet
|
||||||
|
This file is part of Slixmpp.
|
||||||
|
|
||||||
|
See the file LICENSE for copying permission.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import unittest
|
||||||
|
from slixmpp import Message
|
||||||
|
from slixmpp.test import SlixTest
|
||||||
|
from slixmpp.plugins.xep_0444 import XEP_0444
|
||||||
|
import slixmpp.plugins.xep_0444.stanza as stanza
|
||||||
|
from slixmpp.xmlstream import register_stanza_plugin
|
||||||
|
|
||||||
|
|
||||||
|
class TestReactions(SlixTest):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
register_stanza_plugin(Message, stanza.Reactions)
|
||||||
|
register_stanza_plugin(stanza.Reactions, stanza.Reaction)
|
||||||
|
|
||||||
|
def testCreateReactions(self):
|
||||||
|
"""Testing creating Reactions."""
|
||||||
|
|
||||||
|
xmlstring = """
|
||||||
|
<message>
|
||||||
|
<reactions xmlns="urn:xmpp:reactions:0" id="abcd">
|
||||||
|
<reaction>😃</reaction>
|
||||||
|
<reaction>🤗</reaction>
|
||||||
|
</reactions>
|
||||||
|
</message>
|
||||||
|
"""
|
||||||
|
|
||||||
|
msg = self.Message()
|
||||||
|
msg['reactions']['id'] = 'abcd'
|
||||||
|
msg['reactions']['values'] = ['😃', '🤗']
|
||||||
|
|
||||||
|
self.check(msg, xmlstring, use_values=False)
|
||||||
|
|
||||||
|
self.assertEqual({'😃', '🤗'}, msg['reactions']['values'])
|
||||||
|
|
||||||
|
|
||||||
|
def testCreateReactionsUnrestricted(self):
|
||||||
|
"""Testing creating Reactions with the extra all_chars arg."""
|
||||||
|
|
||||||
|
xmlstring = """
|
||||||
|
<message>
|
||||||
|
<reactions xmlns="urn:xmpp:reactions:0" id="abcd">
|
||||||
|
<reaction>😃</reaction>
|
||||||
|
<reaction>🤗</reaction>
|
||||||
|
<reaction>toto</reaction>
|
||||||
|
</reactions>
|
||||||
|
</message>
|
||||||
|
"""
|
||||||
|
|
||||||
|
msg = self.Message()
|
||||||
|
msg['reactions']['id'] = 'abcd'
|
||||||
|
msg['reactions'].set_values(['😃', '🤗', 'toto'], all_chars=True)
|
||||||
|
|
||||||
|
self.check(msg, xmlstring, use_values=False)
|
||||||
|
|
||||||
|
self.assertEqual({'😃', '🤗'}, msg['reactions']['values'])
|
||||||
|
self.assertEqual({'😃', '🤗', 'toto'}, msg['reactions'].get_values(all_chars=True))
|
||||||
|
with self.assertRaises(ValueError):
|
||||||
|
msg['reactions'].set_values(['😃', '🤗', 'toto'], all_chars=False)
|
||||||
|
|
||||||
|
|
||||||
|
suite = unittest.TestLoader().loadTestsFromTestCase(TestReactions)
|
Loading…
Reference in a new issue