55 lines
2 KiB
Python
55 lines
2 KiB
Python
|
#!/usr/bin/env python3
|
||
|
"""
|
||
|
A simple UDP echo server.
|
||
|
"""
|
||
|
import argparse
|
||
|
import itertools
|
||
|
import logging
|
||
|
import socket
|
||
|
|
||
|
logger = logging.getLogger(__name__)
|
||
|
|
||
|
# the buffer for receiving incoming messages
|
||
|
BUFFER_SIZE = 4096
|
||
|
|
||
|
def receive_next(sock):
|
||
|
"Repeatedly tries receiving on the given socket until some data comes in."
|
||
|
logger.debug("Waiting to receive data...")
|
||
|
while True:
|
||
|
try:
|
||
|
return sock.recvfrom(BUFFER_SIZE)
|
||
|
except socket.timeout:
|
||
|
logger.debug("No data received yet: retrying.")
|
||
|
pass
|
||
|
|
||
|
def receive_and_send_one(sock):
|
||
|
"Waits for a single datagram over the socket and echoes it back."
|
||
|
input_data, addr = receive_next(sock)
|
||
|
message = input_data.decode()
|
||
|
logger.info("Received message from %s: %s (%s bytes).", addr, message, len(input_data))
|
||
|
output_len = sock.sendto(input_data, addr)
|
||
|
logger.info("Echoed message back to %s: %s (%s bytes).", addr, message, output_len)
|
||
|
|
||
|
def start(args):
|
||
|
"Runs the server."
|
||
|
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
|
||
|
sock.settimeout(5) # seconds
|
||
|
sock.bind((args.host, args.port))
|
||
|
logger.info("Listening on %s:%s.", args.host, args.port)
|
||
|
try:
|
||
|
for i in itertools.count(1):
|
||
|
receive_and_send_one(sock)
|
||
|
finally:
|
||
|
logger.info("Shutting down.")
|
||
|
sock.close()
|
||
|
|
||
|
if __name__ == "__main__":
|
||
|
parser = argparse.ArgumentParser(__doc__, formatter_class=argparse.RawDescriptionHelpFormatter)
|
||
|
parser.add_argument('--host', help='The host that the server socket should bind to.', default="0.0.0.0")
|
||
|
parser.add_argument('--port', help='The port that the server socket should bind to.', type=int, default=123)
|
||
|
parser.add_argument('--verbose', '-v', help="Increases the logging verbosity level.", action='count')
|
||
|
args = parser.parse_args()
|
||
|
logging.basicConfig(level=logging.DEBUG if args.verbose else logging.INFO,
|
||
|
format='%(asctime)s %(levelname)s %(message)s')
|
||
|
start(args)
|