-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathhttpd.gd
237 lines (217 loc) · 6.68 KB
/
httpd.gd
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
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
# file serving http server using godot
# execute this using godot -s httpd.gd
# and open http://localhost:40004 in a browser
# set data_dir to what you desire, it points at
# the directory served to the public
extends SceneTree
var srv = TCP_Server.new()
var data_dir # data_dir is set below in the _init() method.
func write_str(con, stri):
#print(str("writing string ", stri))
return con.put_data(stri.to_utf8())
# decodes the percent encoding in urls
func decode_percent_url(url):
var arr = url.split("%")
var first = true
var in_seq = false
var encod_seq
var ret = arr[0]
#print(str("URL: ", url))
for stri in arr:
if (not first):
var hex = stri.substr(0, 2)
var hi = str("0x", hex).hex_to_int()
if (in_seq):
encod_seq.push_back(hi)
if (stri.length() == 2):
if (in_seq == false):
in_seq = true
encod_seq = [hi]
else:
if (in_seq):
in_seq = false
var encoded = PoolByteArray(encod_seq).get_string_from_utf8()
ret = str(ret, encoded, stri.substr(2, stri.length()))
else:
first = false
#the url can end with a percent encoded part
if (in_seq):
var encoded = PoolByteArray(encod_seq).get_string_from_utf8()
ret = str(ret, encoded)
return ret
# reads (and blocks) until the first \n, and perhaps more.
# you can feed the "more" part to the startstr arg
# of subsequent calls
func read_line(con, startstr):
var first = true
var pdata
var pdatastr
var retstr = startstr
if (startstr.find("\n") != -1):
return startstr
while (first or (pdatastr.find("\n") == -1)):
first = false
pdata = con.get_partial_data(64)
if (pdata[0] != OK):
return false
if (pdata[1].size() != 0):
pdatastr = pdata[1].get_string_from_ascii()
else:
pdata = con.get_data(8) # force block
if (pdata[0] != OK):
return false
pdatastr = pdata[1].get_string_from_ascii()
retstr = str(retstr, pdatastr)
return retstr
func get_mime(path):
var arr = path.split(".")
var ext = arr[arr.size() - 1]
if (not ext):
return false
elif (ext == "cpp" or ext == "h" or ext == "txt" or ext == "conf"):
return "text/plain; charset=utf-8"
elif (ext == "html" or ext == "htm"):
return "text/html; charset=utf-8"
#elif (ext == "png"):
# return "image/png"
#elif (ext == "gif"):
# return "image/gif"
#elif (ext == "jpg" or ext == "jpeg" or ext == "jpe"):
# return "image/jpeg"
func write_error(con, error, content):
var cont_data = content.to_utf8()
write_str(con, str("HTTP/1.0 ", error, "\n"))
write_str(con, str("Content-Length: ", cont_data.size(), "\n"))
write_str(con, "Connection: close\n")
write_str(con, "\n")
con.put_data(cont_data)
func write_dir_contents(con, path, dir):
#print("Sending directory contents")
var rethtml = "<html><head></head><body>\n"
dir.list_dir_begin()
var dirname = dir.get_next()
while (dirname != ""):
var href
if ((path != "") and (path != "/")):
href = str(path , "/", dirname)
else:
href = dirname
rethtml = str(rethtml, "<a href ='", href, "'>", dirname, "</a><br>\n")
dirname = dir.get_next()
rethtml = str(rethtml, "</body></html>")
var ret_data = rethtml.to_utf8()
write_str(con, "HTTP/1.0 200 OK\n")
write_str(con, str("Content-Length: ", ret_data.size(), "\n"))
write_str(con, str("Content-Type: text/html; charset=utf-8\n")) # its utf8 at least for unix
write_str(con, "Connection: close\n")
write_str(con, "\n")
con.put_data(ret_data)
func write_file(con, path):
var f = File.new()
print(str("Sending file ", path, " to ", con.get_connected_host()))
if (f.open(str(data_dir, path), File.READ) != OK):
var dir = Directory.new()
if (dir.open(str(data_dir, path)) != OK):
write_error(con, "404 Not found", str("File ", str(data_dir, path), " not found!"))
else:
write_dir_contents(con, path, dir)
return
var filesiz = f.get_len()
write_str(con, "HTTP/1.0 200 OK\n")
write_str(con, str("Content-Length: ", filesiz, "\n"))
write_str(con, "Connection: close\n")
var mime = get_mime(path)
if (mime):
write_str(con, str("Content-Type: ", mime, "\n"))
write_str(con, "\n")
var buf
var first = true
var sum = 0
while (first or (buf.size() > 0)):
first = false
var am = min(filesiz - sum, 1048576)
buf = f.get_buffer(am)
sum = sum + am
con.put_data(buf)
f.close()
func file_exists(path):
var f = File.new()
var ret = false
if (f.open(path, File.READ) == OK):
ret = true
else:
var dir = Directory.new()
ret = (dir.open(path) == OK)
f.close()
return ret
func write_options(con, path):
var allow
if (path == "*"):
allow = "OPTIONS, GET"
else:
var dir = Directory.new()
if (file_exists(str(data_dir, path))):
allow = "OPTIONS, GET"
else:
allow = "OPTIONS"
write_str(con, "HTTP/1.0 200\n")
write_str(con, "Content-Length: 0\n")
write_str(con, str("Allow: ", allow, "\n"))
write_str(con, "Connection: close\n")
write_str(con, "\n")
# returns the path and method if no error, sends error and false if error
func parse_request(con):
var st_line = read_line(con, "")
if (not st_line):
write_error(con, "500 Server error", "Error while reading.")
return false
var lines = st_line.split("\n")
var arr = lines[0].split(" ")
if (arr.size() != 3):
write_error(con, "400 Forbidden", "Invalid request!")
return false
var mth = arr[0]
var url = decode_percent_url(arr[1])
if ((url.find("\\") != -1) or (url.find("../") != -1)):
write_error(con, "403 Forbidden", "Forbidden URL!")
return false
else:
return [mth, url]
func run_thrd(params):
var con = params.con
#if (con.is_connected()):
# print("connection is connected")
#else:
# print("connection is NOT connected")
var req = parse_request(con)
if (typeof(req) == TYPE_ARRAY):
var mth = req[0]
var path = req[1]
if (mth == "GET"):
write_file(con, path)
elif (mth == "OPTIONS"):
write_options(con, path)
else:
write_error(con, "501 Not Implemented", str("HTTP method '", mth, "' not supported!"))
con.disconnect()
# hack to free the thread reference after it has exited
# godot has no native protection here, and can
# free a running thread if all references are lost
# The call below saves the reference until the method
# can be called, and gives additional safety by calling
# wait_to_finish and not some arbitrary method, to account for
# the engine or the OS doing other tasks on the thread
# before actually declaring a thread to be "finished"
params.thread.call_deferred("wait_to_finish")
func _init():
var port = 40004
srv.listen(port)
print(str("Server listening at http://localhost:", port))
data_dir = "/var/www/" # has to end with an "/"
while (true):
while (!srv.is_connection_available()): # TODO replace this with actual blocking
OS.delay_msec(100)
var cn = srv.take_connection()
var thread = Thread.new()
thread.start(self, "run_thrd", {con=cn, thread=thread})
quit()