import options import streams import strutils import tables import bindings/quickjs import types/url import js/javascript import utils/twtstr type HttpMethod* = enum HTTP_CONNECT = "CONNECT" HTTP_DELETE = "DELETE" HTTP_GET = "GET" HTTP_HEAD = "HEAD" HTTP_OPTIONS = "OPTIONS" HTTP_PATCH = "PATCH" HTTP_POST = "POST" HTTP_PUT = "PUT" HTTP_TRACE = "TRACE" RequestMode* = enum NO_CORS = "no-cors" SAME_ORIGIN = "same-origin" CORS = "cors" NAVIGATE = "navigate" WEBSOCKET = "websocket" RequestDestination* = enum NO_DESTINATION = "" AUDIO = "audio" AUDIOWORKLET = "audioworklet" DOCUMENT = "document" EMBED = "embed" FONT = "font" FRAME = "frame" IFRAME = "iframe" IMAGE = "image" MANIFEST = "manifest" OBJECT = "object" PAINTWORKLET = "paintworklet" REPORT = "report" SCRIPT = "script" SERVICEWORKER = "serviceworker" SHAREDWORKER = "sharedworker" STYLE = "style" TRACK = "track" WORKER = "worker" XSLT = "xslt" CredentialsMode* = enum SAME_ORIGIN = "same-origin" OMIT = "omit" INCLUDE = "include" CORSAttribute* = enum NO_CORS = "no-cors" ANONYMOUS = "anonymous" USE_CREDENTIALS = "use-credentials" type Request* = ref RequestObj RequestObj* = object httpmethod*: HttpMethod url*: Url headers* {.jsget.}: Headers body*: Option[string] multipart*: Option[MimeData] referer*: URL mode* {.jsget.}: RequestMode destination* {.jsget.}: RequestDestination credentialsMode* {.jsget.}: CredentialsMode proxy*: URL #TODO do something with this Response* = ref object body*: Stream bodyUsed* {.jsget.}: bool res* {.jsget.}: int contenttype* {.jsget.}: string status* {.jsget.}: int headers* {.jsget.}: Headers redirect*: Request url*: URL #TODO should be urllist? unregisterFun*: proc() ReadableStream* = ref object of Stream isource*: Stream buf: string isend: bool Headers* = ref object table* {.jsget.}: Table[string, seq[string]] # Originally from the stdlib MimePart* = object name*, content*: string case isFile*: bool of true: filename*, contentType*: string fileSize*: int64 isStream*: bool else: discard MimeData* = object content*: seq[MimePart] proc Request_url(ctx: JSContext, this: JSValue, magic: cint): JSValue {.cdecl.} = let op = getOpaque0(this) if unlikely(not ctx.isInstanceOf(this, "Request") or op == nil): return JS_ThrowTypeError(ctx, "Value is not an instance of %s", "Request") let request = cast[Request](op) return toJS(ctx, $request.url) proc Request_referrer(ctx: JSContext, this: JSValue, magic: cint): JSValue {.cdecl.} = let op = getOpaque0(this) if unlikely(not ctx.isInstanceOf(this, "Request") or op == nil): return JS_ThrowTypeError(ctx, "Value is not an instance of %s", "Request") let request = cast[Request](op) return toJS(ctx, $request.referer) iterator pairs*(headers: Headers): (string, string) = for k, vs in headers.table: for v in vs: yield (k, v) proc rsReadData(s: Stream, buffer: pointer, bufLen: int): int = var s = ReadableStream(s) if s.atEnd: return 0 while s.buf.len < bufLen: var len: int s.isource.read(len) if len == 0: result = s.buf.len copyMem(buffer, addr(s.buf[0]), result) s.buf = s.buf.substr(result) s.isend = true return var nbuf: string s.isource.readStr(len, nbuf) s.buf &= nbuf assert s.buf.len >= bufLen result = bufLen copyMem(buffer, addr(s.buf[0]), result) s.buf = s.buf.substr(result) if s.buf.len == 0: var len: int s.isource.read(len) if len == 0: s.isend = true else: s.isource.readStr(len, s.buf) proc rsAtEnd(s: Stream): bool = ReadableStream(s).isend proc rsClose(s: Stream) = {.cast(tags: [WriteIOEffect]).}: #TODO TODO TODO ew. var s = ReadableStream(s) if s.isend: return s.buf = "" while true: var len: int s.isource.read(len) if len == 0: s.isend = true break s.isource.setPosition(s.isource.getPosition() + len) proc newReadableStream*(isource: Stream): ReadableStream = new(result) result.isource = isource result.readDataImpl = rsReadData result.atEndImpl = rsAtEnd result.closeImpl = rsClose var len: int result.isource.read(len) if len == 0: result.isend = true else: result.isource.readStr(len, result.buf) proc fill(headers: Headers, ctx: JSContext, val: JSValue) = if isSequence(ctx, val): let x = fromJS[seq[(string, string)]](ctx, val) if x.isSome: for (k, v) in x.get: if k in headers.table: headers.table[k].add(v) else: headers.table[k] = @[v] else: let x = fromJS[Table[string, string]](ctx, val) if x.isSome: for k, v in x.get: if k in headers.table: headers.table[k].add(v) else: headers.table[k] = @[v] func newHeaders*(obj = none(JSObject)): Headers {.jsctor.} = new(result) if obj.isSome: result.fill(obj.get.ctx, obj.get.val) func newHeaders*(table: Table[string, string]): Headers = new(result) for k, v in table: let k = k.toHeaderCase() if k in result.table: result.table[k].add(v) else: result.table[k] = @[v] func newRequest*(url: URL, httpmethod = HTTP_GET, headers = newHeaders(), body = none(string), # multipart = none(MimeData), mode = RequestMode.NO_CORS, credentialsMode = CredentialsMode.SAME_ORIGIN, destination = RequestDestination.NO_DESTINATION, proxy: URL = nil): Request = return Request( url: url, httpmethod: httpmethod, headers: headers, body: body, #multipart: multipart, mode: mode, credentialsMode: credentialsMode, destination: destination, proxy: proxy ) func newRequest*(url: URL, httpmethod = HTTP_GET, headers: seq[(string, string)] = @[], body = none(string), # multipart = none(MimeData), TODO TODO TODO multipart mode = RequestMode.NO_CORS, proxy: URL = nil): Request = let hl = newHeaders() for pair in headers: let (k, v) = pair hl.table[k] = @[v] return newRequest(url, httpmethod, hl, body, mode, proxy = proxy) func createPotentialCORSRequest*(url: URL, destination: RequestDestination, cors: CORSAttribute, fallbackFlag = false): Request = var mode = if cors == NO_CORS: RequestMode.NO_CORS else: RequestMode.CORS if fallbackFlag and mode == NO_CORS: mode = SAME_ORIGIN let credentialsMode = if cors == ANONYMOUS: CredentialsMode.SAME_ORIGIN else: CredentialsMode.INCLUDE return newRequest(url, destination = destination, mode = mode, credentialsMode = credentialsMode) #TODO resource as Request #TODO also, I'm not sure what to do with init. For now I've re-introduced # JSObject to make this work, but I really wish we had a better solution. func newRequest*(resource: string, init: JSObject): Request {.jserr, jsctor.} = let x = parseURL(resource) if x.isNone: JS_ERR JS_TypeError, resource & " is not a valid URL" if x.get.username != "" or x.get.password != "": JS_ERR JS_TypeError, resource & " is not a valid URL" let url = x.get let ctx = init.ctx let fallbackMode = some(RequestMode.CORS) #TODO none if resource is request #TODO fallback mode, origin, window, request mode, ... let httpMethod = fromJS[HttpMethod](ctx, JS_GetPropertyStr(ctx, init.val, "method")).get(HTTP_GET) let body = fromJS[string](ctx, JS_GetPropertyStr(ctx, init.val, "body")) let jheaders = JS_GetPropertyStr(ctx, init.val, "headers") let hl = newHeaders() hl.fill(ctx, jheaders) let credentials = fromJS[CredentialsMode](ctx, JS_GetPropertyStr(ctx, init.val, "credentials")) .get(CredentialsMode.SAME_ORIGIN) let mode = fromJS[RequestMode](ctx, JS_GetPropertyStr(ctx, init.val, "mode")) .get(fallbackMode.get(RequestMode.NO_CORS)) #TODO find a standard compatible way to implement this let proxyUrl = fromJS[URL](ctx, JS_GetPropertyStr(ctx, init.val, "proxyUrl")) return newRequest(url, httpMethod, hl, body, mode, credentials, proxy = proxyUrl.get(nil)) proc `[]=`*(multipart: var MimeData, k, v: string) = multipart.content.add(MimePart(name: k, content: v)) proc add*(headers: var Headers, k, v: string) = let k = k.toHeaderCase() if k notin headers.table: headers.table[k] = @[v] else: headers.table[k].add(v) proc `[]=`*(headers: var Headers, k, v: string) = headers.table[k.toHeaderCase()] = @[v] func getOrDefault*(headers: Headers, k: string, default = ""): string = let k = k.toHeaderCase() if k in headers.table: headers.table[k][0] else: default #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): string {.jsfunc.} = if response.body == nil: return "" result = response.body.readAll() response.close() #TODO: get rid of this proc readAll*(response: Response): string {.jsfunc.} = return response.text() proc Response_json(ctx: JSContext, this: JSValue, argc: cint, argv: ptr JSValue): JSValue {.cdecl.} = let op = getOpaque0(this) if unlikely(not ctx.isInstanceOf(this, "Response") or op == nil): return JS_ThrowTypeError(ctx, "Value is not an instance of %s", "Response") let response = cast[Response](op) var s = response.text() return JS_ParseJSON(ctx, addr s[0], cast[csize_t](s.len), cstring"") func credentialsMode*(attribute: CORSAttribute): CredentialsMode = case attribute of NO_CORS, ANONYMOUS: return SAME_ORIGIN of USE_CREDENTIALS: return INCLUDE proc addRequestModule*(ctx: JSContext) = ctx.registerType(Request, extra_getset = [ TabGetSet(name: "url", get: Request_url), TabGetSet(name: "referrer", get: Request_referrer) ]) ctx.registerType(Response, extra_funcs = [TabFunc(name: "json", fun: Response_json)]) ctx.registerType(Headers)