import streams import unicode import bindings/quickjs import io/promise import js/error import js/javascript import loader/headers import loader/request import types/blob import types/url import chakasu/charset import chakasu/decoderstream import chakasu/encoderstream type Response* = ref object res*: int fd*: int body*: Stream bodyUsed* {.jsget.}: bool contentType*: string status* {.jsget.}: uint16 headers* {.jsget.}: Headers redirect*: Request url*: URL #TODO should be urllist? unregisterFun*: proc() bodyRead*: Promise[string] charset*: Charset jsDestructor(Response) proc newResponse*(res: int, request: Request, fd = -1, stream: Stream = nil): Response = return Response( res: res, url: request.url, body: stream, bodyRead: Promise[string](), fd: fd ) func sok(response: Response): bool {.jsfget: "ok".} = return response.status in 200u16 .. 299u16 func surl(response: Response): string {.jsfget: "url".} = return $response.url #TODO: this should be a property of body proc close*(response: Response) {.jsfunc.} = response.bodyUsed = true if response.unregisterFun != nil: response.unregisterFun() if response.body != nil: response.body.close() proc text*(response: Response): Promise[JSResult[string]] {.jsfunc.} = if response.bodyRead == nil: let p = newPromise[JSResult[string]]() let err = JSResult[string] .err(newTypeError("Body has already been consumed")) p.resolve(err) return p let bodyRead = response.bodyRead response.bodyRead = nil return bodyRead.then(proc(s: string): JSResult[string] = let cs = if response.charset == CHARSET_UNKNOWN: CHARSET_UTF_8 else: response.charset if cs == CHARSET_UTF_8 and s.validateUtf8() == -1: ok(s) else: let ss = newStringStream(s) let ds = newDecoderStream(ss, cs) let es = newEncoderStream(ds, CHARSET_UTF_8) return ok(es.readAll()) ) proc blob*(response: Response): Promise[JSResult[Blob]] {.jsfunc.} = if response.bodyRead == nil: let p = newPromise[JSResult[Blob]]() let err = JSResult[Blob] .err(newTypeError("Body has already been consumed")) p.resolve(err) return p let bodyRead = response.bodyRead response.bodyRead = nil return bodyRead.then(proc(s: string): JSResult[Blob] = if s.len == 0: return ok(newBlob(nil, 0, response.contentType, nil)) GC_ref(s) let deallocFun = proc() = GC_unref(s) let blob = newBlob(unsafeAddr s[0], s.len, response.contentType, deallocFun) ok(blob)) proc json(ctx: JSContext, this: Response): Promise[JSResult[JSValue]] {.jsfunc.} = return this.text().then(proc(s: JSResult[string]): JSResult[JSValue] = let s = ?s return ok(JS_ParseJSON(ctx, cstring(s), cast[csize_t](s.len), cstring""))) proc addResponseModule*(ctx: JSContext) = ctx.registerType(Response)