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:
Link Mauve 2020-11-27 19:54:13 +01:00
commit c86a6ad299
8 changed files with 161 additions and 52 deletions

View file

@ -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:

View file

@ -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}
) )

View file

@ -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
] ]

View file

@ -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)

View file

@ -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)

View file

@ -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)

View 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

View 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)