forked from ea8606/socket-http-server-homework
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhttp_server.py
165 lines (128 loc) · 4.86 KB
/
http_server.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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
import socket
import sys
import os
import mimetypes
workd = '/'.join([os.getcwd(), 'webroot'])
def response_ok(body=b"This is a minimal response", mimetype=b"text/plain"):
"""
returns a basic HTTP response
Ex:
response_ok(
b"<html><h1>Welcome:</h1></html>",
b"text/html"
) ->
b'''
HTTP/1.1 200 OK\r\n
Content-Type: text/html\r\n
\r\n
<html><h1>Welcome:</h1></html>\r\n
'''
"""
return b"\r\n".join([
b"HTTP/1.1 200 OK",
b"Content-Type: " + mimetype,
b"",
body,
])
def parse_request(request):
method, uri, version = request.split("\r\n")[0].split(" ")
if method != "GET":
raise NotImplementedError
return uri
def response_method_not_allowed():
"""Returns a 405 Method Not Allowed response"""
return b"\r\n".join([
b"HTTP/1.1 405 Method Not Allowed",
b"",
b"You can't do that on this server!",
])
def response_not_found():
"""Returns a 404 Not Found response"""
# TODO: Construct and return a 404 "not found" response
# You can re-use most of the code from the 405 Method Not
# Allowed response.
return b"\r\n".join([
b"HTTP/1.1 404 Page Not Found",
b"",
b"You can't do that on this server!",
])
def resolve_uri(uri):
"""
This method should return appropriate content and a mime type.
If the requested URI is a directory, then the content should be a
plain-text listing of the contents with mimetype `text/plain`.
If the URI is a file, it should return the contents of that file
and its correct mimetype.
If the URI does not map to a real location, it should raise a
NameError that the server can catch to return a 404 response.
Ex:
resolve_uri('/a_web_page.html') -> (b"<html><h1>North Carolina...",
b"text/html")
resolve_uri('/images/sample_1.png')
-> (b"A12BCF...", # contents of sample_1.png
b"image/png")
resolve_uri('/') -> (b"images/, a_web_page.html, make_type.py,...",
b"text/plain")
resolve_uri('/a_page_that_doesnt_exist.html') -> Raises a NameError
"""
# TODO: Fill content and mime_type according to the function description
# above. If the provided URI does not correspdond to a real file or
# directory, then raise a NameError.
# Hint: When opening a file, use open(filename, "rb") to open and read the
# file as a stream of bytes.
if uri == '/':
content = '\n'.join([x for x in os.listdir(workd)]).encode()
mime_type = 'text/plain'
else:
try:
with open(''.join([workd, uri]), 'rb') as f:
content = f.read()
except FileNotFoundError:
raise NameError
mime_type = mimetypes.guess_type(uri)[0]
return content, mime_type.encode()
def server(log_buffer=sys.stderr):
address = ('0.0.0.0', int(os.environ.get('PORT', 10000)))
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
print("making a server on {0}:{1}".format(*address), file=log_buffer)
sock.bind(address)
sock.listen(1)
try:
while True:
print('waiting for a connection', file=log_buffer)
conn, addr = sock.accept() # blocking
try:
print('connection - {0}:{1}'.format(*addr), file=log_buffer)
request = ''
while True:
data = conn.recv(1024)
request += data.decode('utf8')
if len(data) < 1024:
break
if b'\r\n\r\n' in data:
break
try:
uri = parse_request(request)
except NotImplementedError:
response = response_method_not_allowed()
else:
# TODO: resolve_uri will raise a NameError if the file
# specified by uri can't be found. If it does raise a
# NameError, then let response get response_not_found()
# instead of response_ok()
try:
body, mimetype = resolve_uri(uri)
except NameError:
response = response_not_found()
else:
response = response_ok(body=body, mimetype=mimetype)
conn.sendall(response)
finally:
conn.close()
except KeyboardInterrupt:
sock.close()
return
if __name__ == '__main__':
server()
sys.exit(0)