Mini Shell
# Copyright (c) Twisted Matrix Laboratories.
# See LICENSE for details.
"""
Tests for L{twisted.words.service}.
"""
import time
from twisted.cred import checkers, credentials, portal
from twisted.internet import address, defer, reactor
from twisted.internet.defer import Deferred, DeferredList, maybeDeferred, succeed
from twisted.spread import pb
from twisted.test import proto_helpers
from twisted.trial import unittest
from twisted.words import ewords, service
from twisted.words.protocols import irc
class RealmTests(unittest.TestCase):
def _entityCreationTest(self, kind):
# Kind is "user" or "group"
realm = service.InMemoryWordsRealm("realmname")
name = "test" + kind.lower()
create = getattr(realm, "create" + kind.title())
get = getattr(realm, "get" + kind.title())
flag = "create" + kind.title() + "OnRequest"
dupExc = getattr(ewords, "Duplicate" + kind.title())
noSuchExc = getattr(ewords, "NoSuch" + kind.title())
# Creating should succeed
p = self.successResultOf(create(name))
self.assertEqual(name, p.name)
# Creating the same user again should not
self.failureResultOf(create(name)).trap(dupExc)
# Getting a non-existent user should succeed if createUserOnRequest is True
setattr(realm, flag, True)
p = self.successResultOf(get("new" + kind.lower()))
self.assertEqual("new" + kind.lower(), p.name)
# Getting that user again should return the same object
newp = self.successResultOf(get("new" + kind.lower()))
self.assertIdentical(p, newp)
# Getting a non-existent user should fail if createUserOnRequest is False
setattr(realm, flag, False)
self.failureResultOf(get("another" + kind.lower())).trap(noSuchExc)
def testUserCreation(self):
return self._entityCreationTest("User")
def testGroupCreation(self):
return self._entityCreationTest("Group")
def testUserRetrieval(self):
realm = service.InMemoryWordsRealm("realmname")
# Make a user to play around with
user = self.successResultOf(realm.createUser("testuser"))
# Make sure getting the user returns the same object
retrieved = self.successResultOf(realm.getUser("testuser"))
self.assertIdentical(user, retrieved)
# Make sure looking up the user also returns the same object
lookedUp = self.successResultOf(realm.lookupUser("testuser"))
self.assertIdentical(retrieved, lookedUp)
# Make sure looking up a user who does not exist fails
(self.failureResultOf(realm.lookupUser("nosuchuser")).trap(ewords.NoSuchUser))
def testUserAddition(self):
realm = service.InMemoryWordsRealm("realmname")
# Create and manually add a user to the realm
p = service.User("testuser")
user = self.successResultOf(realm.addUser(p))
self.assertIdentical(p, user)
# Make sure getting that user returns the same object
retrieved = self.successResultOf(realm.getUser("testuser"))
self.assertIdentical(user, retrieved)
# Make sure looking up that user returns the same object
lookedUp = self.successResultOf(realm.lookupUser("testuser"))
self.assertIdentical(retrieved, lookedUp)
def testGroupRetrieval(self):
realm = service.InMemoryWordsRealm("realmname")
group = self.successResultOf(realm.createGroup("testgroup"))
retrieved = self.successResultOf(realm.getGroup("testgroup"))
self.assertIdentical(group, retrieved)
(self.failureResultOf(realm.getGroup("nosuchgroup")).trap(ewords.NoSuchGroup))
def testGroupAddition(self):
realm = service.InMemoryWordsRealm("realmname")
p = service.Group("testgroup")
self.successResultOf(realm.addGroup(p))
group = self.successResultOf(realm.getGroup("testGroup"))
self.assertIdentical(p, group)
def testGroupUsernameCollision(self):
"""
Try creating a group with the same name as an existing user and
assert that it succeeds, since users and groups should not be in the
same namespace and collisions should be impossible.
"""
realm = service.InMemoryWordsRealm("realmname")
self.successResultOf(realm.createUser("test"))
self.successResultOf(realm.createGroup("test"))
def testEnumeration(self):
realm = service.InMemoryWordsRealm("realmname")
self.successResultOf(realm.createGroup("groupone"))
self.successResultOf(realm.createGroup("grouptwo"))
groups = self.successResultOf(realm.itergroups())
n = [g.name for g in groups]
n.sort()
self.assertEqual(n, ["groupone", "grouptwo"])
class TestCaseUserAgg:
def __init__(
self,
user,
realm,
factory,
address=address.IPv4Address("TCP", "127.0.0.1", 54321),
):
self.user = user
self.transport = proto_helpers.StringTransportWithDisconnection()
self.protocol = factory.buildProtocol(address)
self.transport.protocol = self.protocol
self.user.mind = self.protocol
self.protocol.makeConnection(self.transport)
def write(self, stuff):
self.protocol.dataReceived(stuff)
class IRCProtocolTests(unittest.TestCase):
STATIC_USERS = [
"useruser",
"otheruser",
"someguy",
"firstuser",
"username",
"userone",
"usertwo",
"userthree",
"userfour",
b"userfive",
"someuser",
]
def setUp(self):
self.realm = service.InMemoryWordsRealm("realmname")
self.checker = checkers.InMemoryUsernamePasswordDatabaseDontUse()
self.portal = portal.Portal(self.realm, [self.checker])
self.factory = service.IRCFactory(self.realm, self.portal)
c = []
for nick in self.STATIC_USERS:
if isinstance(nick, bytes):
nick = nick.decode("utf-8")
c.append(self.realm.createUser(nick))
self.checker.addUser(nick, nick + "_password")
return DeferredList(c)
def _assertGreeting(self, user):
"""
The user has been greeted with the four messages that are (usually)
considered to start an IRC session.
Asserts that the required responses were received.
"""
# Make sure we get 1-4 at least
response = self._response(user)
expected = [irc.RPL_WELCOME, irc.RPL_YOURHOST, irc.RPL_CREATED, irc.RPL_MYINFO]
for (prefix, command, args) in response:
if command in expected:
expected.remove(command)
self.assertFalse(expected, f"Missing responses for {expected!r}")
def _login(self, user, nick, password=None):
if password is None:
password = nick + "_password"
user.write(f"PASS {password}\r\n")
user.write(f"NICK {nick} extrainfo\r\n")
def _loggedInUser(self, name):
user = self.successResultOf(self.realm.lookupUser(name))
agg = TestCaseUserAgg(user, self.realm, self.factory)
self._login(agg, name)
return agg
def _response(self, user, messageType=None):
"""
Extracts the user's response, and returns a list of parsed lines.
If messageType is defined, only messages of that type will be returned.
"""
response = user.transport.value()
if bytes != str and isinstance(response, bytes):
response = response.decode("utf-8")
response = response.splitlines()
user.transport.clear()
result = []
for message in map(irc.parsemsg, response):
if messageType is None or message[1] == messageType:
result.append(message)
return result
def testPASSLogin(self):
user = self._loggedInUser("firstuser")
self._assertGreeting(user)
def test_nickServLogin(self):
"""
Sending NICK without PASS will prompt the user for their password.
When the user sends their password to NickServ, it will respond with a
Greeting.
"""
firstuser = self.successResultOf(self.realm.lookupUser("firstuser"))
user = TestCaseUserAgg(firstuser, self.realm, self.factory)
user.write("NICK firstuser extrainfo\r\n")
response = self._response(user, "PRIVMSG")
self.assertEqual(len(response), 1)
self.assertEqual(response[0][0], service.NICKSERV)
self.assertEqual(response[0][1], "PRIVMSG")
self.assertEqual(response[0][2], ["firstuser", "Password?"])
user.transport.clear()
user.write("PRIVMSG nickserv firstuser_password\r\n")
self._assertGreeting(user)
def testFailedLogin(self):
firstuser = self.successResultOf(self.realm.lookupUser("firstuser"))
user = TestCaseUserAgg(firstuser, self.realm, self.factory)
self._login(user, "firstuser", "wrongpass")
response = self._response(user, "PRIVMSG")
self.assertEqual(len(response), 1)
self.assertEqual(response[0][2], ["firstuser", "Login failed. Goodbye."])
def testLogout(self):
logout = []
firstuser = self.successResultOf(self.realm.lookupUser("firstuser"))
user = TestCaseUserAgg(firstuser, self.realm, self.factory)
self._login(user, "firstuser")
user.protocol.logout = lambda: logout.append(True)
user.write("QUIT\r\n")
self.assertEqual(logout, [True])
def testJoin(self):
firstuser = self.successResultOf(self.realm.lookupUser("firstuser"))
somechannel = self.successResultOf(self.realm.createGroup("somechannel"))
somechannel.meta["topic"] = "some random topic"
# Bring in one user, make sure he gets into the channel sanely
user = TestCaseUserAgg(firstuser, self.realm, self.factory)
self._login(user, "firstuser")
user.transport.clear()
user.write("JOIN #somechannel\r\n")
response = self._response(user)
self.assertEqual(len(response), 5)
# Join message
self.assertEqual(response[0][0], "firstuser!firstuser@realmname")
self.assertEqual(response[0][1], "JOIN")
self.assertEqual(response[0][2], ["#somechannel"])
# User list
self.assertEqual(response[1][1], "353")
self.assertEqual(response[2][1], "366")
# Topic (or lack thereof, as the case may be)
self.assertEqual(response[3][1], "332")
self.assertEqual(response[4][1], "333")
# Hook up another client! It is a CHAT SYSTEM!!!!!!!
other = self._loggedInUser("otheruser")
other.transport.clear()
user.transport.clear()
other.write("JOIN #somechannel\r\n")
# At this point, both users should be in the channel
response = self._response(other)
event = self._response(user)
self.assertEqual(len(event), 1)
self.assertEqual(event[0][0], "otheruser!otheruser@realmname")
self.assertEqual(event[0][1], "JOIN")
self.assertEqual(event[0][2], ["#somechannel"])
self.assertEqual(response[1][0], "realmname")
self.assertEqual(response[1][1], "353")
self.assertIn(
response[1][2],
[
["otheruser", "=", "#somechannel", "firstuser otheruser"],
["otheruser", "=", "#somechannel", "otheruser firstuser"],
],
)
def test_joinTopicless(self):
"""
When a user joins a group without a topic, no topic information is
sent to that user.
"""
firstuser = self.successResultOf(self.realm.lookupUser("firstuser"))
self.successResultOf(self.realm.createGroup("somechannel"))
# Bring in one user, make sure he gets into the channel sanely
user = TestCaseUserAgg(firstuser, self.realm, self.factory)
self._login(user, "firstuser")
user.transport.clear()
user.write("JOIN #somechannel\r\n")
response = self._response(user)
responseCodes = [r[1] for r in response]
self.assertNotIn("332", responseCodes)
self.assertNotIn("333", responseCodes)
def testLeave(self):
user = self._loggedInUser("useruser")
self.successResultOf(self.realm.createGroup("somechannel"))
user.write("JOIN #somechannel\r\n")
user.transport.clear()
other = self._loggedInUser("otheruser")
other.write("JOIN #somechannel\r\n")
user.transport.clear()
other.transport.clear()
user.write("PART #somechannel\r\n")
response = self._response(user)
event = self._response(other)
self.assertEqual(len(response), 1)
self.assertEqual(response[0][0], "useruser!useruser@realmname")
self.assertEqual(response[0][1], "PART")
self.assertEqual(response[0][2], ["#somechannel", "leaving"])
self.assertEqual(response, event)
# Now again, with a part message
user.write("JOIN #somechannel\r\n")
user.transport.clear()
other.transport.clear()
user.write("PART #somechannel :goodbye stupidheads\r\n")
response = self._response(user)
event = self._response(other)
self.assertEqual(len(response), 1)
self.assertEqual(response[0][0], "useruser!useruser@realmname")
self.assertEqual(response[0][1], "PART")
self.assertEqual(response[0][2], ["#somechannel", "goodbye stupidheads"])
self.assertEqual(response, event)
user.write(b"JOIN #somechannel\r\n")
user.transport.clear()
other.transport.clear()
user.write(b"PART #somechannel :goodbye stupidheads1\r\n")
response = self._response(user)
event = self._response(other)
self.assertEqual(len(response), 1)
self.assertEqual(response[0][0], "useruser!useruser@realmname")
self.assertEqual(response[0][1], "PART")
self.assertEqual(response[0][2], ["#somechannel", "goodbye stupidheads1"])
self.assertEqual(response, event)
def testGetTopic(self):
user = self._loggedInUser("useruser")
group = service.Group("somechannel")
group.meta["topic"] = "This is a test topic."
group.meta["topic_author"] = "some_fellow"
group.meta["topic_date"] = 77777777
self.successResultOf(self.realm.addGroup(group))
user.transport.clear()
user.write("JOIN #somechannel\r\n")
response = self._response(user)
self.assertEqual(response[3][0], "realmname")
self.assertEqual(response[3][1], "332")
# XXX Sigh. irc.parsemsg() is not as correct as one might hope.
self.assertEqual(
response[3][2], ["useruser", "#somechannel", "This is a test topic."]
)
self.assertEqual(response[4][1], "333")
self.assertEqual(
response[4][2], ["useruser", "#somechannel", "some_fellow", "77777777"]
)
user.transport.clear()
user.write("TOPIC #somechannel\r\n")
response = self._response(user)
self.assertEqual(response[0][1], "332")
self.assertEqual(
response[0][2], ["useruser", "#somechannel", "This is a test topic."]
)
self.assertEqual(response[1][1], "333")
self.assertEqual(
response[1][2], ["useruser", "#somechannel", "some_fellow", "77777777"]
)
def testSetTopic(self):
user = self._loggedInUser("useruser")
somechannel = self.successResultOf(self.realm.createGroup("somechannel"))
user.write("JOIN #somechannel\r\n")
other = self._loggedInUser("otheruser")
other.write("JOIN #somechannel\r\n")
user.transport.clear()
other.transport.clear()
other.write("TOPIC #somechannel :This is the new topic.\r\n")
response = self._response(other)
event = self._response(user)
self.assertEqual(response, event)
self.assertEqual(response[0][0], "otheruser!otheruser@realmname")
self.assertEqual(response[0][1], "TOPIC")
self.assertEqual(response[0][2], ["#somechannel", "This is the new topic."])
other.transport.clear()
somechannel.meta["topic_date"] = 12345
other.write("TOPIC #somechannel\r\n")
response = self._response(other)
self.assertEqual(response[0][1], "332")
self.assertEqual(
response[0][2], ["otheruser", "#somechannel", "This is the new topic."]
)
self.assertEqual(response[1][1], "333")
self.assertEqual(
response[1][2], ["otheruser", "#somechannel", "otheruser", "12345"]
)
other.transport.clear()
other.write("TOPIC #asdlkjasd\r\n")
response = self._response(other)
self.assertEqual(response[0][1], "403")
def testGroupMessage(self):
user = self._loggedInUser("useruser")
self.successResultOf(self.realm.createGroup("somechannel"))
user.write("JOIN #somechannel\r\n")
other = self._loggedInUser("otheruser")
other.write("JOIN #somechannel\r\n")
user.transport.clear()
other.transport.clear()
user.write("PRIVMSG #somechannel :Hello, world.\r\n")
response = self._response(user)
event = self._response(other)
self.assertFalse(response)
self.assertEqual(len(event), 1)
self.assertEqual(event[0][0], "useruser!useruser@realmname")
self.assertEqual(event[0][1], "PRIVMSG", -1)
self.assertEqual(event[0][2], ["#somechannel", "Hello, world."])
def testPrivateMessage(self):
user = self._loggedInUser("useruser")
other = self._loggedInUser("otheruser")
user.transport.clear()
other.transport.clear()
user.write("PRIVMSG otheruser :Hello, monkey.\r\n")
response = self._response(user)
event = self._response(other)
self.assertFalse(response)
self.assertEqual(len(event), 1)
self.assertEqual(event[0][0], "useruser!useruser@realmname")
self.assertEqual(event[0][1], "PRIVMSG")
self.assertEqual(event[0][2], ["otheruser", "Hello, monkey."])
user.write("PRIVMSG nousernamedthis :Hello, monkey.\r\n")
response = self._response(user)
self.assertEqual(len(response), 1)
self.assertEqual(response[0][0], "realmname")
self.assertEqual(response[0][1], "401")
self.assertEqual(
response[0][2], ["useruser", "nousernamedthis", "No such nick/channel."]
)
def testOper(self):
user = self._loggedInUser("useruser")
user.transport.clear()
user.write("OPER user pass\r\n")
response = self._response(user)
self.assertEqual(len(response), 1)
self.assertEqual(response[0][1], "491")
def testGetUserMode(self):
user = self._loggedInUser("useruser")
user.transport.clear()
user.write("MODE useruser\r\n")
response = self._response(user)
self.assertEqual(len(response), 1)
self.assertEqual(response[0][0], "realmname")
self.assertEqual(response[0][1], "221")
self.assertEqual(response[0][2], ["useruser", "+"])
def testSetUserMode(self):
user = self._loggedInUser("useruser")
user.transport.clear()
user.write("MODE useruser +abcd\r\n")
response = self._response(user)
self.assertEqual(len(response), 1)
self.assertEqual(response[0][1], "472")
def testGetGroupMode(self):
user = self._loggedInUser("useruser")
self.successResultOf(self.realm.createGroup("somechannel"))
user.write("JOIN #somechannel\r\n")
user.transport.clear()
user.write("MODE #somechannel\r\n")
response = self._response(user)
self.assertEqual(len(response), 1)
self.assertEqual(response[0][1], "324")
def testSetGroupMode(self):
user = self._loggedInUser("useruser")
self.successResultOf(self.realm.createGroup("groupname"))
user.write("JOIN #groupname\r\n")
user.transport.clear()
user.write("MODE #groupname +abcd\r\n")
response = self._response(user)
self.assertEqual(len(response), 1)
self.assertEqual(response[0][1], "472")
def testWho(self):
group = service.Group("groupname")
self.successResultOf(self.realm.addGroup(group))
users = []
for nick in "userone", "usertwo", "userthree":
u = self._loggedInUser(nick)
users.append(u)
users[-1].write("JOIN #groupname\r\n")
for user in users:
user.transport.clear()
users[0].write("WHO #groupname\r\n")
r = self._response(users[0])
self.assertFalse(self._response(users[1]))
self.assertFalse(self._response(users[2]))
wantusers = ["userone", "usertwo", "userthree"]
for (prefix, code, stuff) in r[:-1]:
self.assertEqual(prefix, "realmname")
self.assertEqual(code, "352")
(
myname,
group,
theirname,
theirhost,
theirserver,
theirnick,
flag,
extra,
) = stuff
self.assertEqual(myname, "userone")
self.assertEqual(group, "#groupname")
self.assertTrue(theirname in wantusers)
self.assertEqual(theirhost, "realmname")
self.assertEqual(theirserver, "realmname")
wantusers.remove(theirnick)
self.assertEqual(flag, "H")
self.assertEqual(extra, "0 " + theirnick)
self.assertFalse(wantusers)
prefix, code, stuff = r[-1]
self.assertEqual(prefix, "realmname")
self.assertEqual(code, "315")
myname, channel, extra = stuff
self.assertEqual(myname, "userone")
self.assertEqual(channel, "#groupname")
self.assertEqual(extra, "End of /WHO list.")
def testList(self):
user = self._loggedInUser("someuser")
user.transport.clear()
somegroup = self.successResultOf(self.realm.createGroup("somegroup"))
somegroup.size = lambda: succeed(17)
somegroup.meta["topic"] = "this is the topic woo"
# Test one group
user.write("LIST #somegroup\r\n")
r = self._response(user)
self.assertEqual(len(r), 2)
resp, end = r
self.assertEqual(resp[0], "realmname")
self.assertEqual(resp[1], "322")
self.assertEqual(resp[2][0], "someuser")
self.assertEqual(resp[2][1], "somegroup")
self.assertEqual(resp[2][2], "17")
self.assertEqual(resp[2][3], "this is the topic woo")
self.assertEqual(end[0], "realmname")
self.assertEqual(end[1], "323")
self.assertEqual(end[2][0], "someuser")
self.assertEqual(end[2][1], "End of /LIST")
user.transport.clear()
# Test all groups
user.write("LIST\r\n")
r = self._response(user)
self.assertEqual(len(r), 2)
fg1, end = r
self.assertEqual(fg1[1], "322")
self.assertEqual(fg1[2][1], "somegroup")
self.assertEqual(fg1[2][2], "17")
self.assertEqual(fg1[2][3], "this is the topic woo")
self.assertEqual(end[1], "323")
def testWhois(self):
user = self._loggedInUser("someguy")
otherguy = service.User("otherguy")
otherguy.itergroups = lambda: iter(
[service.Group("groupA"), service.Group("groupB")]
)
otherguy.signOn = 10
otherguy.lastMessage = time.time() - 15
self.successResultOf(self.realm.addUser(otherguy))
user.transport.clear()
user.write("WHOIS otherguy\r\n")
r = self._response(user)
self.assertEqual(len(r), 5)
wuser, wserver, idle, channels, end = r
self.assertEqual(wuser[0], "realmname")
self.assertEqual(wuser[1], "311")
self.assertEqual(wuser[2][0], "someguy")
self.assertEqual(wuser[2][1], "otherguy")
self.assertEqual(wuser[2][2], "otherguy")
self.assertEqual(wuser[2][3], "realmname")
self.assertEqual(wuser[2][4], "*")
self.assertEqual(wuser[2][5], "otherguy")
self.assertEqual(wserver[0], "realmname")
self.assertEqual(wserver[1], "312")
self.assertEqual(wserver[2][0], "someguy")
self.assertEqual(wserver[2][1], "otherguy")
self.assertEqual(wserver[2][2], "realmname")
self.assertEqual(wserver[2][3], "Hi mom!")
self.assertEqual(idle[0], "realmname")
self.assertEqual(idle[1], "317")
self.assertEqual(idle[2][0], "someguy")
self.assertEqual(idle[2][1], "otherguy")
self.assertEqual(idle[2][2], "15")
self.assertEqual(idle[2][3], "10")
self.assertEqual(idle[2][4], "seconds idle, signon time")
self.assertEqual(channels[0], "realmname")
self.assertEqual(channels[1], "319")
self.assertEqual(channels[2][0], "someguy")
self.assertEqual(channels[2][1], "otherguy")
self.assertEqual(channels[2][2], "#groupA #groupB")
self.assertEqual(end[0], "realmname")
self.assertEqual(end[1], "318")
self.assertEqual(end[2][0], "someguy")
self.assertEqual(end[2][1], "otherguy")
self.assertEqual(end[2][2], "End of WHOIS list.")
class TestMind(service.PBMind):
def __init__(self, *a, **kw):
self.joins = []
self.parts = []
self.messages = []
self.meta = []
def remote_userJoined(self, user, group):
self.joins.append((user, group))
def remote_userLeft(self, user, group, reason):
self.parts.append((user, group, reason))
def remote_receive(self, sender, recipient, message):
self.messages.append((sender, recipient, message))
def remote_groupMetaUpdate(self, group, meta):
self.meta.append((group, meta))
pb.setUnjellyableForClass(TestMind, service.PBMindReference)
class PBProtocolTests(unittest.TestCase):
def setUp(self):
self.realm = service.InMemoryWordsRealm("realmname")
self.checker = checkers.InMemoryUsernamePasswordDatabaseDontUse()
self.portal = portal.Portal(self.realm, [self.checker])
self.serverFactory = pb.PBServerFactory(self.portal)
self.serverFactory.protocol = self._protocolFactory
self.serverFactory.unsafeTracebacks = True
self.clientFactory = pb.PBClientFactory()
self.clientFactory.unsafeTracebacks = True
self.serverPort = reactor.listenTCP(0, self.serverFactory)
self.clientConn = reactor.connectTCP(
"127.0.0.1", self.serverPort.getHost().port, self.clientFactory
)
def _protocolFactory(self, *args, **kw):
self._serverProtocol = pb.Broker(0)
return self._serverProtocol
def tearDown(self):
d3 = Deferred()
self._serverProtocol.notifyOnDisconnect(lambda: d3.callback(None))
return DeferredList(
[
maybeDeferred(self.serverPort.stopListening),
maybeDeferred(self.clientConn.disconnect),
d3,
]
)
def _loggedInAvatar(self, name, password, mind):
nameBytes = name
if isinstance(name, str):
nameBytes = name.encode("ascii")
creds = credentials.UsernamePassword(nameBytes, password)
self.checker.addUser(nameBytes, password)
d = self.realm.createUser(name)
d.addCallback(lambda ign: self.clientFactory.login(creds, mind))
return d
@defer.inlineCallbacks
def testGroups(self):
mindone = TestMind()
one = yield self._loggedInAvatar("one", b"p1", mindone)
mindtwo = TestMind()
two = yield self._loggedInAvatar("two", b"p2", mindtwo)
mindThree = TestMind()
three = yield self._loggedInAvatar(b"three", b"p3", mindThree)
yield self.realm.createGroup("foobar")
yield self.realm.createGroup(b"barfoo")
groupone = yield one.join("foobar")
grouptwo = yield two.join(b"barfoo")
yield two.join("foobar")
yield two.join(b"barfoo")
yield three.join("foobar")
yield groupone.send({b"text": b"hello, monkeys"})
yield groupone.leave()
yield grouptwo.leave()
Zerion Mini Shell 1.0