import std/options
import std/streams
import std/strutils
import std/tables
import bindings/quickjs
import js/error
import js/fromjs
import js/javascript
import js/jstypes
import loader/headers
import types/blob
import types/formdata
import types/referer
import types/url
type
HttpMethod* = enum
HTTP_GET = "GET"
HTTP_CONNECT = "CONNECT"
HTTP_DELETE = "DELETE"
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"
JSON = "json"
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*: Opt[string]
multipart*: Opt[FormData]
referer*: URL
mode* {.jsget.}: RequestMode
destination* {.jsget.}: RequestDestination
credentialsMode* {.jsget.}: CredentialsMode
proxy*: URL #TODO do something with this
canredir*: bool
ReadableStream* = ref object of Stream
isource*: Stream
buf: string
isend: bool
jsDestructor(Request)
proc js_url(this: Request): string {.jsfget: "url".} =
return $this.url
#TODO pretty sure this is incorrect
proc js_referrer(this: Request): string {.jsfget: "referrer".} =
if this.referer != nil:
return $this.referer
return ""
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 =
var len: int
isource.read(len)
result = ReadableStream(
isource: isource,
readDataImpl: rsReadData,
atEndImpl: rsAtEnd,
closeImpl: rsClose,
isend: len == 0
)
if len != 0:
isource.readStr(len, result.buf)
func newRequest*(url: URL, httpMethod = HTTP_GET, headers = newHeaders(),
body = opt(string), multipart = opt(FormData), mode = RequestMode.NO_CORS,
credentialsMode = CredentialsMode.SAME_ORIGIN,
destination = RequestDestination.NO_DESTINATION, proxy: URL = nil,
referrer: URL = nil, canredir = false): Request =
return Request(
url: url,
httpMethod: httpMethod,
headers: headers,
body: body,
multipart: multipart,
mode: mode,
credentialsMode: credentialsMode,
destination: destination,
referer: referrer,
proxy: proxy
)
func newRequest*(url: URL, httpMethod = HTTP_GET,
headers: seq[(string, string)] = @[], body = opt(string),
multipart = opt(FormData), mode = RequestMode.NO_CORS, proxy: URL = nil,
canredir = false):
Request =
let hl = newHeaders()
for pair in headers:
let (k, v) = pair
hl.table[k] = @[v]
return newRequest(url, httpMethod, hl, body, multipart, 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)
type
BodyInitType = enum
BODY_INIT_BLOB, BODY_INIT_FORM_DATA, BODY_INIT_URL_SEARCH_PARAMS,
BODY_INIT_STRING
BodyInit = object
#TODO ReadableStream, BufferSource
case t: BodyInitType
of BODY_INIT_BLOB:
blob: Blob
of BODY_INIT_FORM_DATA:
formData: FormData
of BODY_INIT_URL_SEARCH_PARAMS:
searchParams: URLSearchParams
of BODY_INIT_STRING:
str: string
RequestInit* = object of JSDict
#TODO aliasing in dicts
`method`: HttpMethod # default: GET
headers: Opt[HeadersInit]
body: Opt[BodyInit]
referrer: Opt[string]
referrerPolicy: Opt[ReferrerPolicy]
credentials: Opt[CredentialsMode]
proxyUrl: URL
mode: Opt[RequestMode]
proc fromJS2*(ctx: JSContext, val: JSValue, res: var JSResult[BodyInit]) =
if JS_IsUndefined(val) or JS_IsNull(val):
res.err(nil)
return
block formData:
let x = fromJS[FormData](ctx, val)
if x.isSome:
res.ok(BodyInit(t: BODY_INIT_FORM_DATA, formData: x.get))
return
block blob:
let x = fromJS[Blob](ctx, val)
if x.isSome:
res.ok(BodyInit(t: BODY_INIT_BLOB, blob: x.get))
return
block searchParams:
let x = fromJS[URLSearchParams](ctx, val)
if x.isSome:
res.ok(BodyInit(t: BODY_INIT_URL_SEARCH_PARAMS, searchParams: x.get))
return
block str:
let x = fromJS[string](ctx, val)
if x.isSome:
res.ok(BodyInit(t: BODY_INIT_STRING, str: x.get))
return
res.err(newTypeError("Invalid body init type"))
func newRequest*[T: string|Request](ctx: JSContext, resource: T,
init = none(RequestInit)): JSResult[Request] {.jsctor.} =
when T is string:
let url = ?newURL(resource)
if url.username != "" or url.password != "":
return err(newTypeError("Input URL contains a username or password"))
var httpMethod = HTTP_GET
var headers = newHeaders()
let referer: URL = nil
var credentials = CredentialsMode.SAME_ORIGIN
var body: Opt[string]
var multipart: Opt[FormData]
var proxyUrl: URL #TODO?
let fallbackMode = opt(RequestMode.CORS)
else:
let url = resource.url
var httpMethod = resource.httpMethod
var headers = resource.headers.clone()
let referer = resource.referer
var credentials = resource.credentialsMode
var body = resource.body
var multipart = resource.multipart
var proxyUrl = resource.proxy #TODO?
let fallbackMode = opt(RequestMode)
#TODO window
var mode = fallbackMode.get(RequestMode.NO_CORS)
let destination = NO_DESTINATION
#TODO origin, window
if init.isSome:
if mode == RequestMode.NAVIGATE:
mode = RequestMode.SAME_ORIGIN
#TODO flags?
#TODO referrer
let init = init.get
httpMethod = init.`method`
if init.body.isSome:
let ibody = init.body.get
case ibody.t
of BODY_INIT_FORM_DATA:
multipart = opt(ibody.formData)
of BODY_INIT_STRING:
body = opt(ibody.str)
else:
discard #TODO
if httpMethod in {HTTP_GET, HTTP_HEAD}:
return err(newTypeError("HEAD or GET Request cannot have a body."))
if init.headers.isSome:
headers.fill(init.headers.get)
if init.credentials.isSome:
credentials = init.credentials.get
if init.mode.isSome:
mode = init.mode.get
#TODO find a standard compatible way to implement this
proxyUrl = init.proxyUrl
return ok(Request(
url: url,
httpMethod: httpMethod,
headers: headers,
body: body,
multipart: multipart,
mode: mode,
credentialsMode: credentials,
destination: destination,
proxy: proxyUrl,
referer: referer
))
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)