core/server.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 |
import net, asynchttpserver, asyncdispatch, httpclient, streams, critbits, pegs, strutils import regex, types, parser, interpreter, utils proc validUrl(req: Request, url: string): bool = return req.url.path == url or req.url.path == url & "/" proc validMethod(req: Request, meth: string): bool = return req.reqMethod == meth or req.reqMethod == meth.toLower proc parseException(host: string): string = let e = getCurrentException() result = """($1 ("$2" "$3"))""" % [host, regex.replace($e.name, ":.+$", ""), e.msg] proc exec(req: Request, interpreter: MinInterpreter, hosts: CritBitTree[string]): string {.gcsafe.}= let filename = "request" let s = newStringStream(req.body) var i = interpreter i.open(s, filename) discard i.parser.getToken() try: i.interpret() result = i.dump() except: echo getCurrentExceptionMsg() result = i.link.name.parseException finally: i.close() proc process(req: Request, i: MinInterpreter, hosts: CritBitTree[string]): string {.gcsafe.} = var matches = @["", "", ""] template route(req, peg: expr, op: stmt): stmt {.immediate.}= if req.url.path.find(peg, matches) != -1: op req.route peg"^\/?$": if not req.validMethod("GET"): raiseServer(Http405, "Method Not Allowed: " & req.reqMethod) return "MiNiM Host '$1'" % [i.link.name] req.route peg"^\/exec\/?$": if not req.validMethod("POST"): raiseServer(Http405, "Method Not Allowed: " & req.reqMethod) return exec(req, i, hosts) raiseServer(Http400, "Bad Request: POST "& req.url.path) proc init*(link: ref MinLink) {.thread.} = proc handleHttpRequest(req: Request): Future[void] {.async.} = var res: string var code: HttpCode = Http200 try: res = req.process(link.interpreter, link.hosts) except MinServerError: let e: MinServerError = (MinServerError)getCurrentException() res = e.msg code = e.code await req.respond(code, res) asyncCheck link.server.serve(link.port, handleHttpRequest, link.address) proc remoteExec*(i: var MinInterpreter, host, content: string): string {.gcsafe.}= if i.link.hosts.hasKey(host): let url = "http://" & i.link.hosts[host] & "/exec" try: result = "($1 $2)" % [host, url.postContent(body = content, sslContext = nil)] except: result = host.parseException else: raiseServer(Http404, "Not Found: Host '$1'" % [host]) proc syncHosts*(i: In): CritBitTree[string] {.gcsafe.}= var cmd = "" for key, val in i.link.hosts.pairs: cmd = cmd & """ ($1 "$2")""" % [key, val] cmd = "(" & cmd.strip & ") set-hosts" for key, val in i.link.hosts.pairs: if key != i.link.name: result[key] = i.remoteExec(key, cmd) proc executeOnHost*(i: var MinInterpreter, host: string, q: MinValue) = if not i.link.hosts.hasKey(host): raiseInvalid("Unknown host: " & host) let res = i.remoteExec(host, $q & " unquote") i.open(newStringStream(res), "remote-exec") discard i.parser.getToken() i.interpret() proc newMinLink*(name, address: string, port: int, i: var MinInterpreter): ref MinLink = var link: ref MinLink = new MinLink result = link result.server = newAsyncHttpServer() result.name = name result.address = address result.port = port.Port i.link = result result.interpreter = i |