Skip to content

Instantly share code, notes, and snippets.

@sneils
Created July 24, 2015 10:35
Show Gist options
  • Save sneils/7e87e27d8bb42f19d3f0 to your computer and use it in GitHub Desktop.
Save sneils/7e87e27d8bb42f19d3f0 to your computer and use it in GitHub Desktop.
import asyncio
clients = []
class SimpleChatClientProtocol(asyncio.Protocol):
def connection_made(self, transport):
self.transport = transport
peername = transport.get_extra_info("peername")
self.username = "{:s}:{:d}".format(*peername)
print("connection_made: {}".format(self.username))
for client in clients:
client.send("{:s} connected".format(self.username))
clients.append(self)
def send(self, text):
self.transport.write("{:s}\n".format(text).encode())
def data_received(self, data):
print("data_received: {}".format(data.decode()))
incoming = data.decode()
if len(incoming) == 0:
return
if incoming.find("/") == 0:
parts = incoming.split()
command = parts[0]
print("found command block: {:s}".format(command))
if command == "/username":
if len(parts) != 2:
self.send("< command: invalid payload! Usage: /username <username>")
return
print("{:s} changing username to {:s}".format(self.username, parts[1]))
for client in clients:
if client != self:
client.send("{:s} changed username to {:s}".format(self.username, parts[1]))
self.username = parts[1]
self.send("< username changed")
return
elif command == "/whisper":
if len(parts) < 3:
self.send("< command: invalid payload! Usage: /w <username> <message>")
return
for client in clients:
if client.username == parts[1]:
client.send("{:s} whispered: {:s}".format(self.username, " ".join(parts[2:])))
return
self.send("< command: whisper target username not found!")
return
elif command == "/exit":
self.send("< command: disconnecting, bye!")
print("client_disconnect: {:s}".format(self.username))
self.transport.close()
return
for client in clients:
if client is not self:
client.send("{:s} wrote: {:s}".format(self.username, incoming))
def connection_lost(self, ex):
print("connection_lost: {}".format(self.username))
clients.remove(self)
for client in clients:
client.send("{:s} disconnected".format(self.username))
if __name__ == '__main__':
print("starting up..")
loop = asyncio.get_event_loop()
coro = loop.create_server(SimpleChatClientProtocol, port=1234)
server = loop.run_until_complete(coro)
for socket in server.sockets:
print("serving on {}".format(socket.getsockname()))
loop.run_forever()
@sneils
Copy link
Author

sneils commented Jul 28, 2015

import asyncio
import sys

class SimpleChatClientProtocol(asyncio.Protocol):
    def __init__(self):
        # add console listener to event loop to listen for input
        self.loop = asyncio.get_event_loop()
        self.loop.add_reader(sys.stdin, self.process_input)

    def connection_made(self, transport):
        print("> connection_made")
        self.transport = transport

    def data_received(self, data):
        print("> data_received")
        print(data.decode())

    def connection_lost(self, ex):
        print("> connection_lost")

        # remove console listener and shut down event loop
        self.loop.remove_reader(sys.stdin)
        self.loop.stop()

    def process_input(self):
        text = sys.stdin.readline()
        self.transport.write(text.encode())

        if text == "/exit":
            # close transport to invoke connection_lost
            self.transport.close()

if __name__ == '__main__':
    loop = asyncio.get_event_loop()
    coro = loop.create_connection(SimpleChatClientProtocol, host='localhost', port=1234)
    loop.run_until_complete(coro)
    loop.run_forever()
    loop.close()

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment