import std/nativesockets import std/net import std/options import std/os import std/selectors import std/streams import std/strutils import std/tables import std/unicode when defined(posix): import std/posix import std/exitprocs import bindings/constcharp import bindings/quickjs import config/config import display/lineedit import display/term import html/chadombuilder import html/dom import html/event import io/posixstream import io/promise import io/socketstream import js/base64 import js/console import js/domexception import js/encoding import js/error import js/fromjs import js/intl import js/javascript import js/jstypes import js/module import js/timeout import js/tojs import loader/headers import loader/loader import loader/request import local/container import local/pager import server/forkserver import types/blob import types/cookie import types/opt import types/url import utils/twtstr import xhr/formdata import xhr/xmlhttprequest import chagashi/charset type Client* = ref object alive: bool config {.jsget.}: Config consoleWrapper: ConsoleWrapper fdmap: Table[int, Container] feednext: bool forkserver: ForkServer ibuf: string jsctx: JSContext jsrt: JSRuntime loader: FileLoader pager {.jsget.}: Pager selector: Selector[int] timeouts: TimeoutState pressed: tuple[col: int, row: int] ConsoleWrapper = object console: Console container: Container prev: Container jsDestructor(Client) func console(client: Client): Console {.jsfget.} = return client.consoleWrapper.console proc readChar(client: Client): char = if client.ibuf == "": try: return client.pager.infile.readChar() except EOFError: quit(1) else: result = client.ibuf[0] client.ibuf.delete(0..0) proc finalize(client: Client) {.jsfin.} = if client.jsctx != nil: free(client.jsctx) if client.jsrt != nil: free(client.jsrt) 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.infile == nil: return try: let c = client.pager.infile.readChar() if c == char(3): #C-c client.ibuf = "" return 1 else: client.ibuf &= c except IOError: discard return 0 proc runJSJobs(client: Client) = client.jsrt.runJSJobs(client.console.err) 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 result = client.jsctx.eval(src, filename, flags) client.runJSJobs() client.pager.term.restoreStdin() proc evalJSFree(client: Client, src, filename: string) = JS_FreeValue(client.jsctx, client.evalJS(src, filename)) proc command0(client: Client, src: string, filename = "", 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 container.tailOnLoad = true proc suspend(client: Client) {.jsfunc.} = client.pager.term.quit() discard kill(0, cint(SIGTSTP)) client.pager.term.restart() proc quit(client: Client, code = 0) {.jsfunc.} = if client.alive: client.alive = false client.pager.quit() let ctx = client.jsctx var global = JS_GetGlobalObject(ctx) JS_FreeValue(ctx, global) if client.jsctx != nil: free(client.jsctx) #TODO #if client.jsrt != nil: # free(client.jsrt) quit(code) 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 evalAction(client: Client, action: string, arg0: int32): EmptyPromise = var ret = client.evalJS(action, "") let ctx = client.jsctx var p = EmptyPromise() p.resolve() if JS_IsFunction(ctx, ret): if arg0 != 0: var arg0 = toJS(ctx, arg0) let ret2 = JS_Call(ctx, ret, JS_UNDEFINED, 1, addr arg0) JS_FreeValue(ctx, arg0) JS_FreeValue(ctx, ret) ret = ret2 JS_FreeValue(ctx, arg0) else: # no precnum let ret2 = JS_Call(ctx, ret, JS_UNDEFINED, 0, nil) JS_FreeValue(ctx, ret) ret = ret2 if JS_IsException(ret): client.jsctx.writeException(client.console.err) if JS_IsObject(ret): let maybep = fromJS[EmptyPromise](ctx, ret) if maybep.isOk: p = maybep.get JS_FreeValue(ctx, ret) return p type MouseInputType = enum mitPress = "press", mitRelease = "release", mitMove = "move" MouseInputMod = enum mimShift = "shift", mimCtrl = "ctrl", mimMeta = "meta" MouseInputButton = enum mibLeft = (0, "left") mibMiddle = (1, "middle") mibRight = (2, "right") mibWheelUp = (3, "wheelUp") mibWheelDown = (4, "wheelDown") mibButton6 = (5, "button6") mibButton7 = (6, "button7") mibButton8 = (7, "button8") mibButton9 = (8, "button9") mibButton10 = (9, "button10") mibButton11 = (10, "button11") MouseInput = object t: MouseInputType button: MouseInputButton mods: set[MouseInputMod] col: int row: int proc parseMouseInput(client: Client): Opt[MouseInput] = template fail = return err() var btn = 0 while (let c = client.readChar(); c != ';'): let n = decValue(c) if n == -1: fail btn *= 10 btn += n var mods: set[MouseInputMod] = {} if (btn and 4) != 0: mods.incl(mimShift) if (btn and 8) != 0: mods.incl(mimCtrl) if (btn and 16) != 0: mods.incl(mimMeta) var px = 0 while (let c = client.readChar(); c != ';'): let n = decValue(c) if n == -1: fail px *= 10 px += n var py = 0 var c: char while (c = client.readChar(); c notin {'m', 'M'}): let n = decValue(c) if n == -1: fail py *= 10 py += n var t = if c == 'M': mitPress else: mitRelease if (btn and 32) != 0: t = mitMove var button = btn and 3 if (btn and 64) != 0: button += 3 if (btn and 128) != 0: button += 6 if button notin int(MouseInputButton.low)..int(MouseInputButton.high): return err() ok(MouseInput( t: t, mods: mods, button: MouseInputButton(button), col: px - 1, row: py - 1 )) # 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.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 mibWheelUp: container.scrollUp(5) of mibWheelDown: container.scrollDown(5) of mibButton6, mibButton8: if input.t == mitPress: discard client.pager.nextBuffer() of mibButton7, mibButton9: if input.t == mitPress: discard client.pager.prevBuffer() 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: "", 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) container.requestLines() 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) = while client.pager.unreg.len > 0: let (pid, stream) = client.pager.unreg.pop() let fd = int(stream.fd) if fd in client.fdmap: client.selector.unregister(fd) client.fdmap.del(fd) else: client.pager.procmap.del(pid) stream.close() var accepted: seq[Pid] for pid, container in client.pager.procmap: let stream = connectSocketStream(pid, buffered = false, blocking = true) if stream == nil: client.pager.alert("Error: failed to set up buffer") continue container.setStream(stream) let fd = int(stream.fd) client.fdmap[fd] = container client.selector.registerHandle(fd, {Read}, 0) client.pager.handleEvents(container) accepted.add(pid) client.pager.procmap.clear() proc c_setvbuf(f: File, buf: pointer, mode: cint, size: csize_t): cint {. importc: "setvbuf", header: "", tags: [].} proc handleRead(client: Client, fd: int) = if client.pager.infile != nil and fd == client.pager.infile.getFileHandle(): client.input().then(proc() = client.handlePagerEvents() ) 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.writeData(addr buffer[i], j - i) i = j hadlf = found except ErrorAgain: break if not hadlf: client.console.err.write('\n') client.console.err.flush() 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 flushConsole*(client: Client) {.jsfunc.} = if client.console == nil: # hack for when client crashes before console has been initialized client.consoleWrapper = ConsoleWrapper( console: newConsole(newFileStream(stderr)) ) client.handleRead(client.forkserver.estream.fd) proc handleError(client: Client, fd: int) = if client.pager.infile != nil and fd == client.pager.infile.getFileHandle(): #TODO do something here... stderr.write("Error in tty\n") quit(1) elif fd == client.forkserver.estream.fd: #TODO do something here... stderr.write("Fork server crashed :(\n") 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... else: if fd in client.fdmap: let container = client.fdmap[fd] if container != client.consoleWrapper.container: client.console.log("Error in buffer", $container.location) 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 discard c_setvbuf(client.pager.infile, nil, IONBF, 0) selector.registerHandle(int(client.pager.infile.getFileHandle()), {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 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 client.pager.container.requestLines().then(proc() = client.pager.container.cursorLastLine()) 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: # No buffer to display. quit(1) 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 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: string, content: string) {.jsfunc.} = writeFile(path, content) const ConsoleTitle = "Browser Console" proc addConsole(pager: Pager, interactive: bool, clearFun, showFun, hideFun: proc()): ConsoleWrapper = if interactive: 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], some(url), ConsoleTitle, canreinterpret = false) let err = newPosixStream(pipefd[1]) err.writeLine("Type (M-c) console.hide() to return to buffer mode.") err.flush() pager.registerContainer(container) let console = newConsole( err, clearFun = clearFun, showFun = showFun, hideFun = hideFun ) return ConsoleWrapper( console: console, container: container ) else: let err = newFileStream(stderr) 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], some(url), ConsoleTitle, canreinterpret = false) replacement.replace = client.consoleWrapper.container pager.registerContainer(replacement) client.consoleWrapper.container = replacement let console = client.consoleWrapper.console console.err.close() 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.location) # check for errors client.handleRead(client.forkserver.estream.fd) quit(1) stdout.close() proc launchClient*(client: Client, pages: seq[string], contentType: Option[string], cs: Charset, dump: bool) = var infile: File var dump = dump if not dump: if stdin.isatty(): infile = stdin if stdout.isatty(): if infile == nil: dump = not open(infile, "/dev/tty", fmRead) 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.selector = selector client.pager.launchPager(infile) let clearFun = proc() = client.clearConsole() let showFun = proc() = client.showConsole() let hideFun = proc() = client.hideConsole() client.consoleWrapper = addConsole(client.pager, interactive = infile != 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)) client.alive = true addExitProc((proc() = client.quit())) 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() client.quit() 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(client: Client, data: JSString): DOMResult[string] {.jsfunc.} = return btoa(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): Client = setControlCHook(proc() {.noconv.} = quit(1)) let jsrt = newJSRuntime() JS_SetModuleLoaderFunc(jsrt, normalizeModuleName, clientLoadJSModule, nil) let jsctx = jsrt.newJSContext() let pager = newPager(config, forkserver, jsctx) let client = Client( config: config, forkserver: forkserver, loader: forkserver.newFileLoader( defaultHeaders = config.getDefaultHeaders(), proxy = config.getProxy(), urimethodmap = config.getURIMethodMap(), cgiDir = pager.cgiDir, acceptProxy = true, w3mCGICompat = config.external.w3m_cgi_compat ), jsrt: jsrt, jsctx: jsctx, pager: pager ) jsrt.setInterruptHandler(interruptHandler, cast[pointer](client)) var global = JS_GetGlobalObject(jsctx) jsctx.registerType(Client, asglobal = true) setGlobal(jsctx, global, client) JS_FreeValue(jsctx, global) client.addJSModules(jsctx) return client