-
Notifications
You must be signed in to change notification settings - Fork 22
/
echo.py
executable file
·99 lines (82 loc) · 3.26 KB
/
echo.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
#!/usr/bin/env python3
import datetime
import socket
import sys
# Block size is set to 8192 because thats usually the max header size
BLOCK_SIZE = 8192
def serve(host='0.0.0.0', port=3246, verbosity=1):
try:
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sock.bind((host, port))
sock.listen(1)
if verbosity > 0:
print('Echoing from http://{}:{}'.format(host, port))
while True:
connection, client_address = sock.accept()
request = {}
bytes_left = BLOCK_SIZE
while bytes_left > 0:
if bytes_left > BLOCK_SIZE:
data = connection.recv(BLOCK_SIZE)
else:
data = connection.recv(max(0, bytes_left))
if not 'header' in request:
request = build_request(data)
header_length = len(request['raw']) - len(request['body'])
body_length_read = BLOCK_SIZE - header_length
if 'content-length' in request['header']:
bytes_left = int(request['header']['content-length']) - body_length_read
else:
bytes_left = 0
else:
request['raw'] += data
request['body'] += data.decode('utf-8', 'ignore')
bytes_left -= BLOCK_SIZE
request_time = datetime.datetime.now().ctime()
if verbosity > 0:
print(' - '.join([client_address[0], request_time, request['header']['request-line']]))
raw_decoded = request['raw'].decode('utf-8', 'ignore')
response = "HTTP/1.1 200 OK\nAccess-Control-Allow-Origin: *\n\n{}".format(raw_decoded)
if verbosity == 2:
print("-"*10)
print(response)
print("-"*40)
connection.sendall(response.encode())
connection.close()
except KeyboardInterrupt:
print("\nExiting...")
finally:
sock.close()
def build_request(first_chunk):
lines = first_chunk.decode('utf-8', 'ignore').split('\r\n')
h = {'request-line': lines[0]}
i = 1
while i < len(lines[1:]) and lines[i] != '':
k, v = lines[i].split(': ')
h.update({k.lower(): v})
i += 1
r = {
"header": h,
"raw": first_chunk,
"body": lines[-1]
}
return r
if __name__ == '__main__':
from argparse import ArgumentParser
parser = ArgumentParser(description="Server that returns any http request made to it")
parser.add_argument('-b', '--bind', default='localhost', help='host to bind to')
parser.add_argument('-p', '--port', default=3246, type=int, help='port to listen on')
parser.add_argument('-v', '--verbose', action='store_true', help='print all requests to terminal')
parser.add_argument('-q', '--quiet', action='store_true', help='silence all output (overrides --verbose)')
args = parser.parse_args()
host = args.bind
port = args.port
verbose = args.verbose
quiet = args.quiet
verbosity = 1
if verbose:
verbosity = 2
if quiet:
verbosity = 0
serve(host, port, verbosity)