-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathtw5server.nim
396 lines (335 loc) · 10.4 KB
/
tw5server.nim
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
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
import
asynchttpserver,
asyncdispatch,
os,
strutils,
times,
parseopt,
sequtils,
algorithm,
sets,
uri,
zip/zlib,
zippy,
mimetypes
import strformat
import tables, strtabs
from httpcore import HttpMethod, HttpHeaders
from parseBody import parseMPFD
import json
const
name = "TW5 server"
version = "1.5.2"
style = staticRead("style.css")
temp = staticRead("template.html")
js = staticRead("main.js")
const usage = fmt"""
{name} {version}
Usage:
tw5server -a:localhost -p:8000 -d:dir -b:backup
-h this help
-c config file, json format, default tw5server.json
-a address, defautl "127.0.0.1"
-p port, default 8000
-d directory to serve, default `current dir`
-b backup directory, default `backup` in serve dir. `backup/` or `backup\\` for a backup path.
-l show log message
-m max size of uploaded file (MB), default 100
--autoclean if auto clean backups.
Backups auto-clean strategy:
Keep all backups in current month, keep only the newest one for previous months.
"""
proc time_now(): string =
return now().format("yyyyMMddHHmmss")
type
NimHttpResponse = tuple[
code: HttpCode,
content: string,
headers: HttpHeaders
]
NimHttpSettings = object
directory: string
mimes: MimeDb
port: Port
title: string
address: string
name: string
version: string
# TODO: update web page after upload
proc h_page(settings:NimHttpSettings, content, title, subtitle: string): string =
var footer = """<div id="footer">$1 v$2</div>""" % [settings.name, settings.version]
result = temp % [title, style, subtitle, content, footer, js]
proc relativePath(path, cwd: string): string =
var path2 = path
var wd = cwd
if wd.endsWith("/"):
wd.removeSuffix("/")
if wd == "/":
return wd
elif wd == path2:
return "/"
else:
path2.removePrefix(wd)
var relpath = path2.replace("\\", "/")
if (not relpath.endsWith("/")) and (not path.fileExists):
relpath = relpath&"/"
if not relpath.startsWith("/"):
relpath = "/"&relpath
return relpath
proc relativeParent(path, cwd: string): string =
var relparent = path.parentDir.relativePath(cwd)
if relparent == "":
return "/"
else:
return relparent
proc sendNotFound(settings: NimHttpSettings, path: string): NimHttpResponse =
var content = "<p>The page you requested cannot be found.<p>"
return (code: Http404, content: h_page(settings, content, $int(Http404), "Not Found"), headers: {"Content-Type": "text/html"}.newHttpHeaders())
proc sendStaticFile(settings: NimHttpSettings, path: string): NimHttpResponse =
let mimes = settings.mimes
var ext = path.splitFile.ext
if ext == "":
ext = ".txt"
ext = ext[1 .. ^1]
let mimetype = mimes.getMimetype(ext.toLowerAscii)
var file = path.readFile
return (code: Http200, content: file, headers: {"Content-Type": mimetype}.newHttpHeaders)
proc fileSort(a, b: tuple[kind: PathComponent, path: string]): int =
return cmpIgnoreCase(a.path, b.path)
proc sendDirContents(settings: NimHttpSettings, dir: string): NimHttpResponse =
let cwd = settings.directory.absolutePath
var res: NimHttpResponse
var files = newSeq[string](0)
var path = dir.absolutePath
if not path.startsWith(cwd):
path = cwd
if path != cwd and path != cwd&"/" and path != cwd&"\\":
files.add """<li class="i-back entypo"><a href="$1">..</a></li>""" % [path.relativeParent(cwd)]
var title = settings.title
let subtitle = path.relativePath(cwd)
for i in sorted(walkDir(path).toSeq, fileSort):
let name = i.path.extractFilename
let relpath = i.path.relativePath(path).strip(chars = {'/'}, trailing = false)
if name == "index.html" or name == "index.htm":
return sendStaticFile(settings, i.path)
if i.path.dirExists:
files.add """<li class="i-folder entypo"><a href="$1">$2</a></li>""" % [relpath, name]
else:
files.add """<li class="i-file entypo"><a href="$1">$2</a></li>""" % [relpath, name]
let ul = """
<ul>
$1
</ul>
""" % [files.join("\n")]
res = (code: Http200, content: h_page(settings, ul, title, subtitle), headers: {"Content-Type": "text/html"}.newHttpHeaders())
return res
proc sendOptions(): NimHttpResponse =
var header = {"status": "ok", "dav": "tw5/put", "allow": "GET,HEAD,POST,OPTIONS,CONNECT,PUT,DAV,dav", "x-api-access-type": "file"}
return (code: Http200, content: "", headers: header.newHttpHeaders())
proc logmsg(msg: string, log: bool) =
if log:
echo msg
proc zip(c: string): string =
when defined(macosx) and defined(arm64):
return zlib.compress(c, stream=GZIP_STREAM)
else:
return zippy.compress(c, BestCompression)
proc getPut(req: Request, path, backup: string, log: bool): NimHttpResponse =
let content = req.body
writeFile(path, content)
logmsg("Update: " & path, log)
let (_, name, _) = splitFile(path)
let backup_name = backup / name & "-" & time_now() & ".html.gz"
let compressed = zip(content)
writeFile(backup_name, compressed)
logmsg("Backup to: " & backup_name, log)
return (code: Http200, content: "saved", headers: {"Content-Type": "text;charset=UTF-8"}.newHttpHeaders())
proc savePost(req: Request, path, url_path: string, log: bool): NimHttpResponse =
let
header = req.headers
contentType = header.getOrDefault("Content-Type")
body = parseMPFD(contentType, req.body)
file = body["file"]
filename = file.fields["filename"]
file_body = file.body
overwrite = body.getOrDefault("overwrite").body
var
rsp_content = ""
code = Http400
if fileExists(path / filename) and "yes" != overwrite:
let
(_, base, ext) = filename.splitFile()
newName = base & "-" & time_now() & ext
writeFile(path / newName, file_body)
let
rsp_msg = "Save file to " & newName
msg = rsp_msg & " in " & path
rsp_content = newName
code = Http200
logmsg(msg, log)
else:
writeFile(path / filename, file_body)
let
rsp_msg = "Save file to " & filename
msg = rsp_msg & " in " & path
rsp_content = filename
code = Http200
logmsg(msg, log)
return (code: code, content: rsp_content, headers: {"status": "ok"}.newHttpHeaders())
proc serve(settings: NimHttpSettings, backup: string, log: bool, maxbody: int) =
var server = newAsyncHttpServer(maxBody = maxbody)
proc handleHttpRequest(req: Request): Future[void] {.async.} =
let
url_path = req.url.path.replace("%20", " ").decodeUrl()
path = settings.directory / url_path
var res: NimHttpResponse
case req.reqMethod:
of HttpGet:
if path.dirExists:
res = sendDirContents(settings, path)
elif path.fileExists:
res = sendStaticFile(settings, path)
else:
res = sendNotFound(settings, path)
of HttpPut:
res = getPut(req, path, backup, log)
of HttpOptions:
res = sendOptions()
of HttpHead:
res = sendOptions()
of HttpPost:
res = savePost(req, path, url_path, log)
else:
echo(req.reqMethod)
await req.respond(res.code, res.content, res.headers)
asyncCheck server.serve(settings.port, handleHttpRequest, settings.address)
proc currentMonth(ymd: string): string =
# 20230123 -> 202301
return ymd[0..<6]
proc old_backups(path, name: string): seq[string] =
let
now = time_now()
c_year_month = currentMonth(now)
backup_name = path / name
all_backup_unsort = toSeq(walkPattern(backup_name & "*.html.gz"))
all_backup = sorted(all_backup_unsort, Descending)
var
to_be_removed: seq[string]
saved_y_m = ""
for i in all_backup:
let
date = i[^22..^1]
y_m = currentMonth(date)
if y_m >= c_year_month:
continue
if saved_y_m == y_m:
to_be_removed.add(i)
else:
saved_y_m = y_m
return to_be_removed
proc backupFileName(name: string): string =
# backup name: name-timestamp.html.gz, e.g, test-20230227142037.html.gz
return name[0..^21]
proc clean_backup(backup: string): int =
var names: HashSet[string]
let all_backups = toSeq(walkPattern(backup / "*.html.gz"))
for i in all_backups:
let (_, name, _) = splitFile(i)
names.incl(backupFileName(name))
var count = 0
for i in names:
for old in old_backups(backup, i):
removeFile(old)
count += 1
return count
var
port = 8000
address = "127.0.0.1"
dir = getCurrentDir()
backup = "backup"
title = "TW5 server"
log = false
maxbody = 100 # max body length (MB)
configFile = "tw5server.json"
configStr = "{}"
autoclean = false
for kind, key, val in parseopt.getopt():
case kind
of cmdArgument:
continue
of cmdShortOption, cmdLongOption:
case key
of "h", "help":
echo usage
quit()
of "c":
configFile = val
of "a", "address":
address = val
of "p", "port":
port = parseInt(val)
of "d", "dir":
dir = val
of "b", "backup":
backup = val
of "autoclean":
autoclean = true
of "l", "log":
log = true
of "m", "max":
maxbody = parseInt(val)
of "v", "version":
echo version
quit()
else:
assert(false)
if configFile.fileExists:
configStr = configFile.readFile
let config = parseJson(configStr)
dir = config{"server_path"}.getStr(dir)
address = config{"address"}.getStr(address)
port = config{"port"}.getInt(port)
title = config{"title"}.getStr(title)
backup = config{"backup"}.getStr(backup)
autoclean = config{"autoclean"}.getBool(autoclean)
var settings: NimHttpSettings
settings.directory = dir
settings.mimes = newMimetypes()
settings.mimes.register("htm", "text/html")
settings.address = address
settings.name = name
settings.title = title
settings.version = version
settings.port = Port(port)
echo(" Serving url: ", address, ":", port)
echo("Serving path: ", dir)
if not ("/" in backup or "\\" in backup):
backup = dir / backup
echo(" Backup dir: ", backup)
createDir(backup)
proc auto_clean_backup(folder: string) =
var cleaned = 0
cleaned = clean_backup(folder)
if cleaned > 0:
echo(cleaned, " backup(s) cleaned")
else:
echo("No backups were cleaned")
proc handleCtrlC() {.noconv.} =
if autoclean:
auto_clean_backup(backup)
else:
write(stdout, "\rClean backups (y to clean): ")
let clean = readLine(stdin)
if "y" == clean:
auto_clean_backup(backup)
echo("Bye ~")
quit()
if autoclean:
auto_clean_backup(backup)
setControlCHook(handleCtrlC)
else:
setControlCHook(handleCtrlC)
maxbody = config{"max_body"}.getInt(maxbody)
log = config{"log"}.getBool(log)
serve(settings, backup, log, maxbody = maxbody * 1024 * 1024)
runForever()