Merge branch 'component-registration' into 'master'
Add registration to components See merge request poezio/slixmpp!130
This commit is contained in:
commit
626bf5ff8a
2 changed files with 264 additions and 4 deletions
|
@ -8,6 +8,8 @@ import ssl
|
||||||
|
|
||||||
from slixmpp.stanza import StreamFeatures, Iq
|
from slixmpp.stanza import StreamFeatures, Iq
|
||||||
from slixmpp.xmlstream import register_stanza_plugin, JID
|
from slixmpp.xmlstream import register_stanza_plugin, JID
|
||||||
|
from slixmpp.xmlstream.handler import CoroutineCallback
|
||||||
|
from slixmpp.xmlstream.matcher import StanzaPath
|
||||||
from slixmpp.plugins import BasePlugin
|
from slixmpp.plugins import BasePlugin
|
||||||
from slixmpp.plugins.xep_0077 import stanza, Register, RegisterFeature
|
from slixmpp.plugins.xep_0077 import stanza, Register, RegisterFeature
|
||||||
|
|
||||||
|
@ -19,6 +21,36 @@ class XEP_0077(BasePlugin):
|
||||||
|
|
||||||
"""
|
"""
|
||||||
XEP-0077: In-Band Registration
|
XEP-0077: In-Band Registration
|
||||||
|
|
||||||
|
Events:
|
||||||
|
|
||||||
|
::
|
||||||
|
|
||||||
|
user_register -- After succesful validation and add to the user store
|
||||||
|
in api["user_validate"]
|
||||||
|
user_unregister -- After succesful user removal in api["user_remove"]
|
||||||
|
|
||||||
|
Config:
|
||||||
|
|
||||||
|
::
|
||||||
|
|
||||||
|
form_fields are only form_instructions are only used for component registration
|
||||||
|
in case api["make_registration_form"] is not overriden.
|
||||||
|
|
||||||
|
API:
|
||||||
|
|
||||||
|
::
|
||||||
|
|
||||||
|
user_get(jid, node, ifrom, iq)
|
||||||
|
Returns a dict-like object containing `form_fields` for this user or None
|
||||||
|
user_remove(jid, node, ifrom, iq)
|
||||||
|
Removes a user or raise KeyError in case the user is not found in the user store
|
||||||
|
make_registration_form(self, jid, node, ifrom, iq)
|
||||||
|
Returns an iq reply to a registration form request, pre-filled and with
|
||||||
|
<registered/> in case the requesting entity is already registered to us
|
||||||
|
user_validate((self, jid, node, ifrom, registration)
|
||||||
|
Add the user to the user store or raise ValueError(msg) if any problem is encountered
|
||||||
|
msg is sent back to the XMPP client as an error message.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
name = 'xep_0077'
|
name = 'xep_0077'
|
||||||
|
@ -28,18 +60,36 @@ class XEP_0077(BasePlugin):
|
||||||
default_config = {
|
default_config = {
|
||||||
'create_account': True,
|
'create_account': True,
|
||||||
'force_registration': False,
|
'force_registration': False,
|
||||||
'order': 50
|
'order': 50,
|
||||||
|
"form_fields": {"username", "password"},
|
||||||
|
"form_instructions": "Enter your credentials",
|
||||||
}
|
}
|
||||||
|
|
||||||
def plugin_init(self):
|
def plugin_init(self):
|
||||||
register_stanza_plugin(StreamFeatures, RegisterFeature)
|
register_stanza_plugin(StreamFeatures, RegisterFeature)
|
||||||
register_stanza_plugin(Iq, Register)
|
register_stanza_plugin(Iq, Register)
|
||||||
|
|
||||||
if not self.xmpp.is_component:
|
if self.xmpp.is_component:
|
||||||
self.xmpp.register_feature('register',
|
self.xmpp["xep_0030"].add_feature("jabber:iq:register")
|
||||||
|
self.xmpp.register_handler(
|
||||||
|
CoroutineCallback(
|
||||||
|
"registration",
|
||||||
|
StanzaPath("/iq/register"),
|
||||||
|
self._handle_registration,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
self._user_store = {}
|
||||||
|
self.api.register(self._user_get, "user_get")
|
||||||
|
self.api.register(self._user_remove, "user_remove")
|
||||||
|
self.api.register(self._make_registration_form, "make_registration_form")
|
||||||
|
self.api.register(self._user_validate, "user_validate")
|
||||||
|
else:
|
||||||
|
self.xmpp.register_feature(
|
||||||
|
"register",
|
||||||
self._handle_register_feature,
|
self._handle_register_feature,
|
||||||
restart=False,
|
restart=False,
|
||||||
order=self.order)
|
order=self.order,
|
||||||
|
)
|
||||||
|
|
||||||
register_stanza_plugin(Register, self.xmpp['xep_0004'].stanza.Form)
|
register_stanza_plugin(Register, self.xmpp['xep_0004'].stanza.Form)
|
||||||
register_stanza_plugin(Register, self.xmpp['xep_0066'].stanza.OOB)
|
register_stanza_plugin(Register, self.xmpp['xep_0066'].stanza.OOB)
|
||||||
|
@ -50,6 +100,92 @@ class XEP_0077(BasePlugin):
|
||||||
if not self.xmpp.is_component:
|
if not self.xmpp.is_component:
|
||||||
self.xmpp.unregister_feature('register', self.order)
|
self.xmpp.unregister_feature('register', self.order)
|
||||||
|
|
||||||
|
def _user_get(self, jid, node, ifrom, iq):
|
||||||
|
return self._user_store.get(iq["from"].bare)
|
||||||
|
|
||||||
|
def _user_remove(self, jid, node, ifrom, iq):
|
||||||
|
return self._user_store.pop(iq["from"].bare)
|
||||||
|
|
||||||
|
def _make_registration_form(self, jid, node, ifrom, iq: Iq):
|
||||||
|
reg = iq["register"]
|
||||||
|
user = self.api["user_get"](None, None, None, iq)
|
||||||
|
|
||||||
|
|
||||||
|
if user is None:
|
||||||
|
user = {}
|
||||||
|
else:
|
||||||
|
reg["registered"] = True
|
||||||
|
|
||||||
|
reg["instructions"] = self.form_instructions
|
||||||
|
|
||||||
|
for field in self.form_fields:
|
||||||
|
data = user.get(field, "")
|
||||||
|
if data:
|
||||||
|
reg[field] = data
|
||||||
|
else:
|
||||||
|
# Add a blank field
|
||||||
|
reg.add_field(field)
|
||||||
|
|
||||||
|
reply = iq.reply()
|
||||||
|
reply.set_payload(reg.xml)
|
||||||
|
return reply
|
||||||
|
|
||||||
|
def _user_validate(self, jid, node, ifrom, registration):
|
||||||
|
self._user_store[ifrom.bare] = {key: registration[key] for key in self.form_fields}
|
||||||
|
|
||||||
|
async def _handle_registration(self, iq: Iq):
|
||||||
|
if iq["type"] == "get":
|
||||||
|
self._send_form(iq)
|
||||||
|
elif iq["type"] == "set":
|
||||||
|
if iq["register"]["remove"]:
|
||||||
|
try:
|
||||||
|
self.api["user_remove"](None, None, iq["from"], iq)
|
||||||
|
except KeyError:
|
||||||
|
_send_error(
|
||||||
|
iq,
|
||||||
|
"404",
|
||||||
|
"cancel",
|
||||||
|
"item-not-found",
|
||||||
|
"User not found",
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
reply = iq.reply()
|
||||||
|
reply.send()
|
||||||
|
self.xmpp.event("user_unregister", iq)
|
||||||
|
return
|
||||||
|
|
||||||
|
|
||||||
|
for field in self.form_fields:
|
||||||
|
if not iq["register"][field]:
|
||||||
|
# Incomplete Registration
|
||||||
|
_send_error(
|
||||||
|
iq,
|
||||||
|
"406",
|
||||||
|
"modify",
|
||||||
|
"not-acceptable",
|
||||||
|
"Please fill in all fields.",
|
||||||
|
)
|
||||||
|
return
|
||||||
|
|
||||||
|
try:
|
||||||
|
self.api["user_validate"](None, None, iq["from"], iq["register"])
|
||||||
|
except ValueError as e:
|
||||||
|
_send_error(
|
||||||
|
iq,
|
||||||
|
"406",
|
||||||
|
"modify",
|
||||||
|
"not-acceptable",
|
||||||
|
e.args,
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
reply = iq.reply()
|
||||||
|
reply.send()
|
||||||
|
self.xmpp.event("user_register", iq)
|
||||||
|
|
||||||
|
def _send_form(self, iq):
|
||||||
|
reply = self.api["make_registration_form"](None, None, iq["from"], iq)
|
||||||
|
reply.send()
|
||||||
|
|
||||||
def _force_registration(self, event):
|
def _force_registration(self, event):
|
||||||
if self.force_registration:
|
if self.force_registration:
|
||||||
self.xmpp.add_filter('in', self._force_stream_feature)
|
self.xmpp.add_filter('in', self._force_stream_feature)
|
||||||
|
@ -109,3 +245,15 @@ class XEP_0077(BasePlugin):
|
||||||
iq['register']['username'] = self.xmpp.boundjid.user
|
iq['register']['username'] = self.xmpp.boundjid.user
|
||||||
iq['register']['password'] = password
|
iq['register']['password'] = password
|
||||||
return iq.send(timeout=timeout, callback=callback)
|
return iq.send(timeout=timeout, callback=callback)
|
||||||
|
|
||||||
|
def _send_error(iq, code, error_type, name, text=""):
|
||||||
|
# It would be nice to raise XMPPError but the iq payload
|
||||||
|
# should include the register info
|
||||||
|
reply = iq.reply()
|
||||||
|
reply.set_payload(iq["register"].xml)
|
||||||
|
reply.error()
|
||||||
|
reply["error"]["code"] = code
|
||||||
|
reply["error"]["type"] = error_type
|
||||||
|
reply["error"]["condition"] = name
|
||||||
|
reply["error"]["text"] = text
|
||||||
|
reply.send()
|
||||||
|
|
112
tests/test_stream_xep_0077.py
Normal file
112
tests/test_stream_xep_0077.py
Normal file
|
@ -0,0 +1,112 @@
|
||||||
|
"""
|
||||||
|
This only covers the component registration side of the XEP-0077 plugin
|
||||||
|
"""
|
||||||
|
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
from slixmpp import ComponentXMPP, Iq
|
||||||
|
from slixmpp.xmlstream import register_stanza_plugin
|
||||||
|
from slixmpp.test import SlixTest
|
||||||
|
from slixmpp.plugins.xep_0077 import Register
|
||||||
|
|
||||||
|
|
||||||
|
class TestRegistration(SlixTest):
|
||||||
|
def setUp(self):
|
||||||
|
self.stream_start(
|
||||||
|
mode="component", plugins=["xep_0077"], jid="shakespeare.lit", server="lit"
|
||||||
|
)
|
||||||
|
|
||||||
|
def testRegistrationForm(self):
|
||||||
|
self.stream_start(
|
||||||
|
mode="component", plugins=["xep_0077"], jid="shakespeare.lit", server="lit"
|
||||||
|
)
|
||||||
|
self.recv(
|
||||||
|
"""
|
||||||
|
<iq type='get' id='reg1' to='shakespeare.lit' from='bill@server/resource'>
|
||||||
|
<query xmlns='jabber:iq:register'/>
|
||||||
|
</iq>
|
||||||
|
""",
|
||||||
|
)
|
||||||
|
self.send(
|
||||||
|
f"""
|
||||||
|
<iq type='result' id='reg1' from='shakespeare.lit' to='bill@server/resource'>
|
||||||
|
<query xmlns='jabber:iq:register'>
|
||||||
|
<instructions>{self.xmpp["xep_0077"].form_instructions}</instructions>
|
||||||
|
<username/>
|
||||||
|
<password/>
|
||||||
|
</query>
|
||||||
|
</iq>
|
||||||
|
""",
|
||||||
|
use_values=False # Fails inconsistently without this
|
||||||
|
)
|
||||||
|
|
||||||
|
def testRegistrationSuccessAndModif(self):
|
||||||
|
self.recv(
|
||||||
|
"""
|
||||||
|
<iq type='set' id='reg2' to='shakespeare.lit' from="bill@server/resource">
|
||||||
|
<query xmlns='jabber:iq:register'>
|
||||||
|
<username>bill</username>
|
||||||
|
<password>Calliope</password>
|
||||||
|
</query>
|
||||||
|
</iq>
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
self.send("<iq type='result' id='reg2' from='shakespeare.lit' to='bill@server/resource'/>")
|
||||||
|
user_store = self.xmpp["xep_0077"]._user_store
|
||||||
|
self.assertEqual(user_store["bill@server"]["username"], "bill")
|
||||||
|
self.assertEqual(user_store["bill@server"]["password"], "Calliope")
|
||||||
|
|
||||||
|
self.recv(
|
||||||
|
"""
|
||||||
|
<iq type='get' id='reg1' to='shakespeare.lit' from="bill@server/resource">
|
||||||
|
<query xmlns='jabber:iq:register'/>
|
||||||
|
</iq>
|
||||||
|
""",
|
||||||
|
)
|
||||||
|
self.send(
|
||||||
|
f"""
|
||||||
|
<iq type='result' id='reg1' to="bill@server/resource" from='shakespeare.lit'>
|
||||||
|
<query xmlns='jabber:iq:register'>
|
||||||
|
<instructions>{self.xmpp["xep_0077"].form_instructions}</instructions>
|
||||||
|
<username>bill</username>
|
||||||
|
<password>Calliope</password>
|
||||||
|
<registered />
|
||||||
|
</query>
|
||||||
|
</iq>
|
||||||
|
""",
|
||||||
|
use_values=False # Fails inconsistently without this
|
||||||
|
)
|
||||||
|
|
||||||
|
def testRegistrationAndRemove(self):
|
||||||
|
self.recv(
|
||||||
|
"""
|
||||||
|
<iq type='set' id='reg2' to='shakespeare.lit' from="bill@shakespeare.lit/globe">
|
||||||
|
<query xmlns='jabber:iq:register'>
|
||||||
|
<username>bill</username>
|
||||||
|
<password>Calliope</password>
|
||||||
|
</query>
|
||||||
|
</iq>
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
self.send("<iq type='result' id='reg2' from='shakespeare.lit' to='bill@shakespeare.lit/globe'/>")
|
||||||
|
pseudo_iq = self.xmpp.Iq()
|
||||||
|
pseudo_iq["from"] = "bill@shakespeare.lit/globe"
|
||||||
|
user = self.xmpp["xep_0077"].api["user_get"](None, None, None, pseudo_iq)
|
||||||
|
self.assertEqual(user["username"], "bill")
|
||||||
|
self.assertEqual(user["password"], "Calliope")
|
||||||
|
self.recv(
|
||||||
|
"""
|
||||||
|
<iq type='set' from='bill@shakespeare.lit/globe' id='unreg1'>
|
||||||
|
<query xmlns='jabber:iq:register'>
|
||||||
|
<remove/>
|
||||||
|
</query>
|
||||||
|
</iq>
|
||||||
|
"""
|
||||||
|
)
|
||||||
|
self.send("<iq type='result' to='bill@shakespeare.lit/globe' id='unreg1'/>")
|
||||||
|
user_store = self.xmpp["xep_0077"]._user_store
|
||||||
|
self.assertIs(user_store.get("bill@shakespeare.lit"), None)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
suite = unittest.TestLoader().loadTestsFromTestCase(TestRegistration)
|
Loading…
Reference in a new issue