import std/exitprocs
import std/nativesockets
import std/net
import std/options
import std/os
import std/posix
import std/selectors
import std/streams
import std/strutils
import std/tables
import std/unicode
import bindings/constcharp
import bindings/quickjs
import config/config
import html/chadombuilder
import html/dom
import html/event
import html/formdata
import html/xmlhttprequest
import io/bufstream
import io/bufwriter
import io/dynstream
import io/filestream
import io/posixstream
import io/promise
import io/serversocket
import io/socketstream
import js/base64
import js/console
import js/domexception
import js/encoding
import js/jserror
import js/fromjs
import js/intl
import js/javascript
import js/jstypes
import js/jsutils
import js/jsmodule
import js/jsopaque
import js/timeout
import js/tojs
import loader/headers
import loader/loader
import loader/request
import local/container
import local/lineedit
import local/pager
import local/term
import server/buffer
import server/forkserver
import types/blob
import types/cookie
import types/opt
import types/url
import utils/twtstr
import chagashi/charset
type
Client* = ref object
alive: bool
config {.jsget.}: Config
consoleWrapper: ConsoleWrapper
fdmap: Table[int, Container]
feednext: bool
jsctx: JSContext
jsrt: JSRuntime
pager {.jsget.}: Pager
timeouts: TimeoutState
pressed: tuple[col: int; row: int]
exitCode: int
inEval: bool
ConsoleWrapper = object
console: Console
container: Container
prev: Container
func console(client: Client): Console {.jsfget.} =
return client.consoleWrapper.console
template selector(client: Client): Selector[int] =
client.pager.selector
template loader(client: Client): FileLoader =
client.pager.loader
template forkserver(client: Client): ForkServer =
client.pager.forkserver
template readChar(client: Client): char =
client.pager.term.readChar()
proc fetch[T: Request|string](client: Client; req: T;
init = none(RequestInit)): JSResult[FetchPromise] {.jsfunc.} =
let req = ?newRequest(client.jsctx, req, init)
return ok(client.loader.fetch(req))
proc interruptHandler(rt: JSRuntime; opaque: pointer): cint {.cdecl.} =
let client = cast[Client](opaque)
if client.console == nil or client.pager.term.istream == nil:
return 0
try:
let c = client.pager.term.istream.sreadChar()
if c == char(3): #C-c
client.pager.term.ibuf = ""
return 1
else:
client.pager.term.ibuf &= c
except IOError:
discard
return 0
proc runJSJobs(client: Client) =
while true:
let r = client.jsrt.runJSJobs()
if r.isSome:
break
let ctx = r.error
ctx.writeException(client.console.err)
proc cleanup(client: Client) =
if client.alive:
client.alive = false
client.pager.quit()
for val in client.config.cmd.map.values:
JS_FreeValue(client.jsctx, val)
for fn in client.config.jsvfns:
JS_FreeValue(client.jsctx, fn)
assert not client.inEval
client.jsctx.free()
client.jsrt.free()
proc quit(client: Client; code = 0) =
client.cleanup()
quit(code)
proc evalJS(client: Client; src, filename: string; module = false): JSValue =
client.pager.term.unblockStdin()
let flags = if module:
JS_EVAL_TYPE_MODULE
else:
JS_EVAL_TYPE_GLOBAL
let wasInEval = client.inEval
client.inEval = true
result = client.jsctx.eval(src, filename, flags)
client.inEval = false
client.pager.term.restoreStdin()
if client.exitCode != -1:
# if we are in a nested eval, then just wait until we are not.
if not wasInEval:
client.quit(client.exitCode)
else:
client.runJSJobs()
proc evalJSFree(client: Client; src, filename: string) =
JS_FreeValue(client.jsctx, client.evalJS(src, filename))
proc command0(client: Client; src: string; filename = "<command>";
silence = false; module = false) =
let ret = client.evalJS(src, filename, module = module)
if JS_IsException(ret):
client.jsctx.writeException(client.console.err)
else:
if not silence:
let str = fromJS[string](client.jsctx, ret)
if str.isSome:
client.console.log(str.get)
JS_FreeValue(client.jsctx, ret)
proc command(client: Client; src: string) =
client.command0(src)
let container = client.consoleWrapper.container
if container != nil:
container.tailOnLoad = true
proc suspend(client: Client) {.jsfunc.} =
client.pager.term.quit()
discard kill(0, cint(SIGTSTP))
client.pager.term.restart()
proc jsQuit(client: Client; code: uint32 = 0): JSValue {.jsfunc: "quit".} =
client.exitCode = int(code)
let ctx = client.jsctx
let ctor = ctx.getOpaque().errCtorRefs[jeInternalError]
let err = JS_CallConstructor(ctx, ctor, 0, nil)
JS_SetUncatchableError(ctx, err, true);
return JS_Throw(ctx, err)
proc feedNext(client: Client) {.jsfunc.} =
client.feednext = true
proc alert(client: Client; msg: string) {.jsfunc.} =
client.pager.alert(msg)
proc handlePagerEvents(client: Client) =
let container = client.pager.container
if container != nil:
client.pager.handleEvents(container)
proc evalActionJS(client: Client; action: string): JSValue =
client.config.cmd.map.withValue(action, p):
return JS_DupValue(client.jsctx, p[])
return client.evalJS(action, "<command>")
# Warning: this is not re-entrant.
proc evalAction(client: Client; action: string; arg0: int32): EmptyPromise =
var ret = client.evalActionJS(action)
let ctx = client.jsctx
var p = EmptyPromise()
p.resolve()
if JS_IsFunction(ctx, ret):
if arg0 != 0:
let arg0 = toJS(ctx, arg0)
let ret2 = JS_Call(ctx, ret, JS_UNDEFINED, 1, arg0.toJSValueArray())
JS_FreeValue(ctx, arg0)
JS_FreeValue(ctx, ret)
ret = ret2
else: # no precnum
let ret2 = JS_Call(ctx, ret, JS_UNDEFINED, 0, nil)
JS_FreeValue(ctx, ret)
ret = ret2
if client.exitCode != -1:
assert not client.inEval
client.quit(client.exitCode)
if JS_IsException(ret):
client.jsctx.writeException(client.console.err)
elif JS_IsObject(ret):
let maybep = fromJSEmptyPromise(ctx, ret)
if maybep.isSome:
p = maybep.get
JS_FreeValue(ctx, ret)
return p
# The maximum number we are willing to accept.
# This should be fine for 32-bit signed ints (which precnum currently is).
# We can always increase it further (e.g. by switching to uint32, uint64...) if
# it proves to be too low.
const MaxPrecNum = 100000000
proc handleCommandInput(client: Client; c: char): EmptyPromise =
if client.config.input.vi_numeric_prefix and not client.pager.notnum:
if client.pager.precnum != 0 and c == '0' or c in '1' .. '9':
if client.pager.precnum < MaxPrecNum: # better ignore than eval...
client.pager.precnum *= 10
client.pager.precnum += cast[int32](decValue(c))
return
else:
client.pager.notnum = true
client.pager.inputBuffer &= c
let action = getNormalAction(client.config, client.pager.inputBuffer)
if action != "":
let p = client.evalAction(action, client.pager.precnum)
if not client.feednext:
client.pager.precnum = 0
client.pager.notnum = false
client.handlePagerEvents()
return p
if client.config.input.use_mouse:
if client.pager.inputBuffer == "\e[<":
let input = client.pager.term.parseMouseInput()
if input.isSome:
let input = input.get
let container = client.pager.container
if container != nil:
case input.button
of mibLeft:
case input.t
of mitPress:
client.pressed = (input.col, input.row)
of mitRelease:
#TODO this does not work very well with double width chars,
# because pressed could be equivalent to two separate cells
if client.pressed == (input.col, input.row):
if input.col == container.acursorx and
input.row == container.acursory:
container.click()
else:
container.setCursorXY(container.fromx + input.col,
container.fromy + input.row)
else:
let diff = (input.col - client.pressed.col,
input.row - client.pressed.row)
if diff[0] > 0:
container.scrollLeft(diff[0])
else:
container.scrollRight(-diff[0])
if diff[1] > 0:
container.scrollUp(diff[1])
else:
container.scrollDown(-diff[1])
client.pressed = (-1, -1)
else: discard
of mibMiddle:
if input.t == mitRelease: # release, to emulate w3m
client.pager.discardBuffer()
of mibWheelUp:
if input.t == mitPress:
container.scrollUp(5)
of mibWheelDown:
if input.t == mitPress:
container.scrollDown(5)
of mibWheelLeft:
if input.t == mitPress:
container.scrollLeft(5)
of mibWheelRight:
if input.t == mitPress:
container.scrollRight(5)
of mibThumbInner:
if input.t == mitPress:
discard client.pager.prevBuffer()
of mibThumbTip:
if input.t == mitPress:
discard client.pager.nextBuffer()
else: discard
client.pager.inputBuffer = ""
elif "\e[<".startsWith(client.pager.inputBuffer):
client.feednext = true
return nil
proc input(client: Client): EmptyPromise =
var p: EmptyPromise = nil
client.pager.term.restoreStdin()
var buf: string
while true:
let c = client.readChar()
if client.pager.askpromise != nil:
if c == 'y':
client.pager.fulfillAsk(true)
elif c == 'n':
client.pager.fulfillAsk(false)
elif client.pager.askcharpromise != nil:
buf &= c
if buf.validateUtf8() != -1:
continue
client.pager.fulfillCharAsk(buf)
elif client.pager.lineedit.isSome:
client.pager.inputBuffer &= c
let edit = client.pager.lineedit.get
if edit.escNext:
edit.escNext = false
if edit.write(client.pager.inputBuffer, client.pager.term.cs):
client.pager.inputBuffer = ""
else:
let action = getLinedAction(client.config, client.pager.inputBuffer)
if action == "":
if edit.write(client.pager.inputBuffer, client.pager.term.cs):
client.pager.inputBuffer = ""
else:
client.feednext = true
elif not client.feednext:
discard client.evalAction(action, 0)
if not client.feednext:
client.pager.updateReadLine()
else:
p = client.handleCommandInput(c)
if not client.feednext:
client.pager.inputBuffer = ""
client.pager.refreshStatusMsg()
break
#TODO this is not perfect, because it results in us never displaying
# lone escape. maybe a timeout for escape display would be useful
if not "\e[<".startsWith(client.pager.inputBuffer):
client.pager.refreshStatusMsg()
client.pager.draw()
if not client.feednext:
client.pager.inputBuffer = ""
break
else:
client.feednext = false
client.pager.inputBuffer = ""
if p == nil:
p = EmptyPromise()
p.resolve()
return p
proc setTimeout[T: JSValue|string](client: Client; handler: T;
timeout = 0i32): int32 {.jsfunc.} =
return client.timeouts.setTimeout(handler, timeout)
proc setInterval[T: JSValue|string](client: Client; handler: T;
interval = 0i32): int32 {.jsfunc.} =
return client.timeouts.setInterval(handler, interval)
proc clearTimeout(client: Client; id: int32) {.jsfunc.} =
client.timeouts.clearTimeout(id)
proc clearInterval(client: Client; id: int32) {.jsfunc.} =
client.timeouts.clearInterval(id)
let SIGWINCH {.importc, header: "<signal.h>", nodecl.}: cint
proc showConsole(client: Client) {.jsfunc.} =
let container = client.consoleWrapper.container
if client.pager.container != container:
client.consoleWrapper.prev = client.pager.container
client.pager.setContainer(container)
proc hideConsole(client: Client) {.jsfunc.} =
if client.pager.container == client.consoleWrapper.container:
client.pager.setContainer(client.consoleWrapper.prev)
proc consoleBuffer(client: Client): Container {.jsfget.} =
return client.consoleWrapper.container
proc acceptBuffers(client: Client) =
let pager = client.pager
while pager.unreg.len > 0:
let container = pager.unreg.pop()
if container.iface != nil: # fully connected
let stream = container.iface.stream
let fd = int(stream.source.fd)
client.selector.unregister(fd)
client.fdmap.del(fd)
stream.sclose()
elif container.process != -1: # connecting to buffer process
let i = pager.findProcMapItem(container.process)
pager.procmap.del(i)
elif (let i = pager.findConnectingContainer(container); i != -1):
# connecting to URL
let stream = pager.connectingContainers[i].stream
client.selector.unregister(stream.fd)
stream.sclose()
pager.connectingContainers.del(i)
let registerFun = proc(fd: int) =
client.selector.unregister(fd)
client.selector.registerHandle(fd, {Read, Write}, 0)
for item in pager.procmap:
let container = item.container
let stream = connectSocketStream(client.config.external.tmpdir,
client.loader.sockDirFd, container.process)
# unlink here; on Linux we can't unlink from the buffer :/
discard tryRemoveFile(getSocketPath(client.config.external.tmpdir,
container.process))
if stream == nil:
pager.alert("Error: failed to set up buffer")
continue
let key = pager.addLoaderClient(container.process, container.loaderConfig)
stream.withPacketWriter w:
w.swrite(key)
let loader = pager.loader
if item.fdin != -1:
let outputId = item.istreamOutputId
if container.cacheId == -1:
(container.cacheId, container.cacheFile) =
loader.addCacheFile(outputId, loader.clientPid)
var outCacheId = container.cacheId
let pid = container.process
if item.fdout == item.fdin:
loader.shareCachedItem(container.cacheId, pid)
loader.resume(@[item.istreamOutputId])
else:
outCacheId = loader.addCacheFile(item.ostreamOutputId, pid).outputId
loader.resume(@[item.istreamOutputId, item.ostreamOutputId])
w.swrite(outCacheId)
if item.fdin != -1:
# pass down fdout
# must come after the previous block so the first packet is flushed
stream.sendFileHandle(item.fdout)
discard close(item.fdout)
container.setStream(stream, registerFun)
else:
# buffer is cloned, no need to cache anything
container.setCloneStream(stream, registerFun)
let fd = int(stream.fd)
client.fdmap[fd] = container
client.selector.registerHandle(fd, {Read}, 0)
pager.handleEvents(container)
pager.procmap.setLen(0)
proc handleRead(client: Client; fd: int) =
if client.pager.term.istream != nil and fd == client.pager.term.istream.fd:
client.input().then(proc() =
client.handlePagerEvents()
)
elif (let i = client.pager.findConnectingContainer(fd); i != -1):
client.pager.handleConnectingContainer(i)
elif fd == client.forkserver.estream.fd:
const BufferSize = 4096
const prefix = "STDERR: "
var buffer {.noinit.}: array[BufferSize, char]
let estream = client.forkserver.estream
var hadlf = true
while true:
try:
let n = estream.recvData(addr buffer[0], BufferSize)
if n == 0:
break
var i = 0
while i < n:
var j = n
var found = false
for k in i ..< n:
if buffer[k] == '\n':
j = k + 1
found = true
break
if hadlf:
client.console.err.write(prefix)
if j - i > 0:
client.console.err.write(buffer.toOpenArray(i, j - 1))
i = j
hadlf = found
except ErrorAgain:
break
if not hadlf:
client.console.err.write('\n')
client.console.err.sflush()
elif fd in client.loader.connecting:
client.loader.onConnected(fd)
client.runJSJobs()
elif fd in client.loader.ongoing:
client.loader.onRead(fd)
elif fd in client.loader.unregistered:
discard # ignore
else:
let container = client.fdmap[fd]
client.pager.handleEvent(container)
proc handleWrite(client: Client; fd: int) =
let container = client.fdmap[fd]
if container.iface.stream.flushWrite():
client.selector.unregister(fd)
client.selector.registerHandle(fd, {Read}, 0)
proc flushConsole*(client: Client) {.jsfunc.} =
if client.console == nil:
# hack for when client crashes before console has been initialized
client.consoleWrapper = ConsoleWrapper(
console: newConsole(newDynFileStream(stderr))
)
client.handleRead(client.forkserver.estream.fd)
proc handleError(client: Client; fd: int) =
if client.pager.term.istream != nil and fd == client.pager.term.istream.fd:
#TODO do something here...
stderr.write("Error in tty\n")
client.quit(1)
elif fd == client.forkserver.estream.fd:
#TODO do something here...
stderr.write("Fork server crashed :(\n")
client.quit(1)
elif fd in client.loader.connecting:
#TODO handle error?
discard
elif fd in client.loader.ongoing:
client.loader.onError(fd)
elif fd in client.loader.unregistered:
discard # already unregistered...
elif (let i = client.pager.findConnectingContainer(fd); i != -1):
client.pager.handleConnectingContainerError(i)
else:
if fd in client.fdmap:
let container = client.fdmap[fd]
if container != client.consoleWrapper.container:
client.console.log("Error in buffer", $container.url)
else:
client.consoleWrapper.container = nil
client.selector.unregister(fd)
client.fdmap.del(fd)
if client.consoleWrapper.container != nil:
client.showConsole()
else:
doAssert false
proc inputLoop(client: Client) =
let selector = client.selector
selector.registerHandle(int(client.pager.term.istream.fd), {Read}, 0)
let sigwinch = selector.registerSignal(int(SIGWINCH), 0)
while true:
let events = client.selector.select(-1)
for event in events:
if Read in event.events:
client.handleRead(event.fd)
if Write in event.events:
client.handleWrite(event.fd)
if Error in event.events:
client.handleError(event.fd)
if Signal in event.events:
assert event.fd == sigwinch
client.pager.windowChange()
if selectors.Event.Timer in event.events:
let r = client.timeouts.runTimeoutFd(event.fd)
assert r
let container = client.consoleWrapper.container
if container != nil:
container.tailOnLoad = true
client.runJSJobs()
client.loader.unregistered.setLen(0)
client.acceptBuffers()
if client.pager.scommand != "":
client.command(client.pager.scommand)
client.pager.scommand = ""
client.handlePagerEvents()
if client.pager.container == nil and client.pager.lineedit.isNone:
# No buffer to display.
if not client.pager.hasload:
# Failed to load every single URL the user passed us. We quit, and that
# will dump all alerts to stderr.
client.quit(1)
else:
# At least one connection has succeeded, but we have nothing to display.
# Normally, this means that the input stream has been redirected to a
# file or to an external program. That also means we can't just exit
# without potentially interrupting that stream.
#TODO: a better UI would be querying the number of ongoing streams in
# loader, and then asking for confirmation if there is at least one.
client.pager.term.setCursor(0, client.pager.term.attrs.height - 1)
client.pager.term.anyKey("Hit any key to quit Chawan:")
client.quit(0)
client.pager.showAlerts()
client.pager.draw()
func hasSelectFds(client: Client): bool =
return not client.timeouts.empty or
client.pager.numload > 0 or
client.loader.connecting.len > 0 or
client.loader.ongoing.len > 0 or
client.pager.procmap.len > 0
proc headlessLoop(client: Client) =
while client.hasSelectFds():
let events = client.selector.select(-1)
for event in events:
if Read in event.events:
client.handleRead(event.fd)
if Write in event.events:
client.handleWrite(event.fd)
if Error in event.events:
client.handleError(event.fd)
if selectors.Event.Timer in event.events:
let r = client.timeouts.runTimeoutFd(event.fd)
assert r
client.runJSJobs()
client.loader.unregistered.setLen(0)
client.acceptBuffers()
proc clientLoadJSModule(ctx: JSContext; module_name: cstringConst;
opaque: pointer): JSModuleDef {.cdecl.} =
let global = JS_GetGlobalObject(ctx)
JS_FreeValue(ctx, global)
var x: Option[URL]
if module_name[0] == '/' or module_name[0] == '.' and
(module_name[1] == '/' or
module_name[1] == '.' and module_name[2] == '/'):
let cur = getCurrentDir()
x = parseURL($module_name, parseURL("file://" & cur & "/"))
else:
x = parseURL($module_name)
if x.isNone or x.get.scheme != "file":
JS_ThrowTypeError(ctx, "Invalid URL: %s", module_name)
return nil
try:
let f = readFile($x.get.path)
return finishLoadModule(ctx, f, cstring(module_name))
except IOError:
JS_ThrowTypeError(ctx, "Failed to open file %s", module_name)
return nil
proc readBlob(client: Client; path: string): Option[WebFile] {.jsfunc.} =
try:
return some(newWebFile(path))
except IOError:
discard
#TODO this is dumb
proc readFile(client: Client; path: string): string {.jsfunc.} =
try:
return readFile(path)
except IOError:
discard
#TODO ditto
proc writeFile(client: Client; path, content: string) {.jsfunc.} =
writeFile(path, content)
const ConsoleTitle = "Browser Console"
proc addConsole(pager: Pager; interactive: bool; clearFun, showFun, hideFun:
proc()): ConsoleWrapper =
if interactive and pager.config.start.console_buffer:
var pipefd: array[0..1, cint]
if pipe(pipefd) == -1:
raise newException(Defect, "Failed to open console pipe.")
let url = newURL("stream:console").get
let container = pager.readPipe0("text/plain", CHARSET_UNKNOWN, pipefd[0],
url, ConsoleTitle, {})
let err = newPosixStream(pipefd[1])
err.write("Type (M-c) console.hide() to return to buffer mode.\n")
let console = newConsole(err, clearFun, showFun, hideFun)
return ConsoleWrapper(console: console, container: container)
else:
let err = newPosixStream(stderr.getFileHandle())
return ConsoleWrapper(console: newConsole(err))
proc clearConsole(client: Client) =
var pipefd: array[0..1, cint]
if pipe(pipefd) == -1:
raise newException(Defect, "Failed to open console pipe.")
let url = newURL("stream:console").get
let pager = client.pager
let replacement = pager.readPipe0("text/plain", CHARSET_UNKNOWN, pipefd[0],
url, ConsoleTitle, {})
replacement.replace = client.consoleWrapper.container
pager.replace(client.consoleWrapper.container, replacement)
client.consoleWrapper.container = replacement
let console = client.consoleWrapper.console
console.err.sclose()
console.err = newPosixStream(pipefd[1])
proc dumpBuffers(client: Client) =
client.headlessLoop()
let ostream = newFileStream(stdout)
for container in client.pager.containers:
try:
client.pager.drawBuffer(container, ostream)
client.pager.handleEvents(container)
except IOError:
client.console.log("Error in buffer", $container.url)
# check for errors
client.handleRead(client.forkserver.estream.fd)
client.quit(1)
proc launchClient*(client: Client; pages: seq[string];
contentType: Option[string]; cs: Charset; dump: bool) =
var istream: PosixStream
var dump = dump
if not dump:
if stdin.isatty():
istream = newPosixStream(stdin.getFileHandle())
dump = istream == nil
if stdout.isatty():
istream = newPosixStream("/dev/tty", O_RDONLY, 0)
if istream != nil:
dump = false
else:
dump = true
let selector = newSelector[int]()
let efd = int(client.forkserver.estream.fd)
selector.registerHandle(efd, {Read}, 0)
client.loader.registerFun = proc(fd: int) =
selector.registerHandle(fd, {Read}, 0)
client.loader.unregisterFun = proc(fd: int) =
selector.unregister(fd)
client.pager.launchPager(istream, selector)
let clearFun = proc() =
client.clearConsole()
let showFun = proc() =
client.showConsole()
let hideFun = proc() =
client.hideConsole()
client.consoleWrapper = client.pager.addConsole(interactive = istream != nil,
clearFun, showFun, hideFun)
#TODO passing console.err here makes it impossible to change it later. maybe
# better associate it with jsctx
client.timeouts = newTimeoutState(client.selector, client.jsctx,
client.console.err, proc(src, file: string) = client.evalJSFree(src, file))
addExitProc((proc() = client.cleanup()))
if client.config.start.startup_script != "":
let s = if fileExists(client.config.start.startup_script):
readFile(client.config.start.startup_script)
else:
client.config.start.startup_script
let ismodule = client.config.start.startup_script.endsWith(".mjs")
client.command0(s, client.config.start.startup_script, silence = true,
module = ismodule)
if not stdin.isatty():
# stdin may very well receive ANSI text
let contentType = contentType.get("text/x-ansi")
client.pager.readPipe(contentType, cs, stdin.getFileHandle(), "*stdin*")
for page in pages:
client.pager.loadURL(page, ctype = contentType, cs = cs)
client.pager.showAlerts()
client.acceptBuffers()
if not dump:
client.inputLoop()
else:
client.dumpBuffers()
if client.config.start.headless:
client.headlessLoop()
proc nimGCStats(client: Client): string {.jsfunc.} =
return GC_getStatistics()
proc jsGCStats(client: Client): string {.jsfunc.} =
return client.jsrt.getMemoryUsage()
proc nimCollect(client: Client) {.jsfunc.} =
GC_fullCollect()
proc jsCollect(client: Client) {.jsfunc.} =
JS_RunGC(client.jsrt)
proc sleep(client: Client; millis: int) {.jsfunc.} =
sleep millis
proc atob(client: Client; data: string): DOMResult[NarrowString] {.jsfunc.} =
return atob(data)
proc btoa(ctx: JSContext; client: Client; data: JSValue): DOMResult[string]
{.jsfunc.} =
return btoa(ctx, data)
func line(client: Client): LineEdit {.jsfget.} =
return client.pager.lineedit.get(nil)
proc addJSModules(client: Client; ctx: JSContext) =
ctx.addDOMExceptionModule()
ctx.addConsoleModule()
ctx.addCookieModule()
ctx.addURLModule()
ctx.addEventModule()
ctx.addDOMModule()
ctx.addHTMLModule()
ctx.addIntlModule()
ctx.addBlobModule()
ctx.addFormDataModule()
ctx.addXMLHttpRequestModule()
ctx.addHeadersModule()
ctx.addRequestModule()
ctx.addResponseModule()
ctx.addLineEditModule()
ctx.addConfigModule()
ctx.addPagerModule()
ctx.addContainerModule()
ctx.addEncodingModule()
func getClient(client: Client): Client {.jsfget: "client".} =
return client
proc newClient*(config: Config; forkserver: ForkServer; jsctx: JSContext;
warnings: seq[string]): Client =
setControlCHook(proc() {.noconv.} = quit(1))
let jsrt = JS_GetRuntime(jsctx)
JS_SetModuleLoaderFunc(jsrt, normalizeModuleName, clientLoadJSModule, nil)
let pager = newPager(config, forkserver, jsctx, warnings)
let loaderPid = forkserver.forkLoader(LoaderConfig(
urimethodmap: config.external.urimethodmap,
w3mCGICompat: config.external.w3m_cgi_compat,
cgiDir: seq[string](config.external.cgi_dir),
tmpdir: config.external.tmpdir
))
let loader = FileLoader(process: loaderPid, clientPid: getCurrentProcessId())
loader.setSocketDir(config.external.tmpdir)
pager.setLoader(loader)
let client = Client(
config: config,
jsrt: jsrt,
jsctx: jsctx,
pager: pager,
exitCode: -1,
alive: true
)
jsrt.setInterruptHandler(interruptHandler, cast[pointer](client))
jsctx.registerType(Client, asglobal = true)
jsctx.setGlobal(client)
let global = JS_GetGlobalObject(jsctx)
jsctx.definePropertyE(global, "cmd", config.cmd.jsObj)
JS_FreeValue(jsctx, global)
config.cmd.jsObj = JS_NULL
client.addJSModules(jsctx)
return client