import std/deques import std/net import std/options import std/os import std/osproc import std/posix import std/selectors import std/sets import std/tables import std/unicode import chagashi/charset import config/chapath import config/config import config/mailcap import img/bitmap import io/bufreader import io/dynstream import io/promise import io/stdio import io/tempfile import io/urlfilter import js/timeout import layout/renderdocument import loader/connecterror import loader/headers import loader/loader import loader/request import local/container import local/lineedit import local/term import monoucha/fromjs import monoucha/javascript import monoucha/jserror import monoucha/jsregex import monoucha/jstypes import monoucha/jsutils import monoucha/libregexp import monoucha/quickjs import monoucha/tojs import server/buffer import server/forkserver import types/blob import types/cell import types/color import types/cookie import types/opt import types/url import types/winattrs import utils/luwrap import utils/mimeguess import utils/regexutils import utils/strwidth import utils/twtstr type LineMode* = enum lmLocation = "URL: " lmUsername = "Username: " lmPassword = "Password: " lmCommand = "COMMAND: " lmBuffer = "(BUFFER) " lmSearchF = "/" lmSearchB = "?" lmISearchF = "/" lmISearchB = "?" lmGotoLine = "Goto line: " lmDownload = "(Download)Save file to: " lmBufferFile = "(Upload)Filename: " # fdin is the original fd; fdout may be the same, or different if mailcap # is used. ProcMapItem = object container*: Container fdin*: FileHandle fdout*: FileHandle istreamOutputId*: int ostreamOutputId*: int PagerAlertState = enum pasNormal, pasAlertOn, pasLoadInfo ContainerConnectionState = enum ccsBeforeResult, ccsBeforeStatus, ccsBeforeHeaders ConnectingContainerItem = ref object state: ContainerConnectionState container: Container stream*: SocketStream res: int outputId: int status: uint16 LineData = ref object of RootObj LineDataDownload = ref object of LineData outputId: int stream: DynStream LineDataAuth = ref object of LineData url: URL NavDirection = enum ndPrev = "prev" ndNext = "next" ndPrevSibling = "prev-sibling" ndNextSibling = "next-sibling" ndParent = "parent" ndFirstChild ndAny = "any" Surface = object redraw: bool grid: FixedGrid Pager* = ref object alertState: PagerAlertState alerts*: seq[string] askcharpromise*: Promise[string] askcursor: int askpromise*: Promise[bool] askprompt: string commandMode {.jsget.}: bool config*: Config connectingContainers*: seq[ConnectingContainerItem] container*: Container cookiejars: Table[string, CookieJar] devRandom: PosixStream display: Surface forkserver*: ForkServer formRequestMap*: Table[string, FormRequestType] hasload*: bool # has a page been successfully loaded since startup? imageId: int # hack to allocate a new ID for canvas each frame, TODO remove inputBuffer*: string # currently uninterpreted characters iregex: Result[Regex, string] isearchpromise: EmptyPromise jsctx: JSContext lineData: LineData lineedit*: LineEdit linehist: array[LineMode, LineHistory] linemode: LineMode loader*: FileLoader luctx: LUContext navDirection {.jsget.}: NavDirection notnum*: bool # has a non-numeric character been input already? numload*: int # number of pages currently being loaded precnum*: int32 # current number prefix (when vi-numeric-prefix is true) procmap*: seq[ProcMapItem] refreshAllowed: HashSet[string] regex: Opt[Regex] reverseSearch: bool scommand*: string selector*: Selector[int] status: Surface term*: Terminal timeouts*: TimeoutState unreg*: seq[Container] jsDestructor(Pager) # Forward declarations proc alert*(pager: Pager; msg: string) template attrs(pager: Pager): WindowAttributes = pager.term.attrs func loaderPid(pager: Pager): int64 {.jsfget.} = int64(pager.loader.process) func getRoot(container: Container): Container = var c = container while c.parent != nil: c = c.parent return c func bufWidth(pager: Pager): int = return pager.attrs.width func bufHeight(pager: Pager): int = return pager.attrs.height - 1 # depth-first descendant iterator iterator descendants(parent: Container): Container {.inline.} = var stack = newSeqOfCap[Container](parent.children.len) for i in countdown(parent.children.high, 0): stack.add(parent.children[i]) while stack.len > 0: let c = stack.pop() # add children first, so that deleteContainer works on c for i in countdown(c.children.high, 0): stack.add(c.children[i]) yield c iterator containers*(pager: Pager): Container {.inline.} = if pager.container != nil: let root = getRoot(pager.container) yield root for c in root.descendants: yield c proc clearDisplay(pager: Pager) = pager.display = Surface( grid: newFixedGrid(pager.bufWidth, pager.bufHeight), redraw: true ) proc clearStatus(pager: Pager) = pager.status = Surface( grid: newFixedGrid(pager.attrs.width), redraw: true ) proc setContainer*(pager: Pager; c: Container) {.jsfunc.} = if pager.term.imageMode != imNone and pager.container != nil: pager.container.cachedImages.setLen(0) pager.container = c if c != nil: c.queueDraw() pager.term.setTitle(c.getTitle()) proc hasprop(ctx: JSContext; pager: Pager; s: string): bool {.jshasprop.} = result = false if pager.container != nil: let cval = toJS(ctx, pager.container) let val = JS_GetPropertyStr(ctx, cval, s) if val != JS_UNDEFINED: result = true JS_FreeValue(ctx, val) proc reflect(ctx: JSContext; this_val: JSValue; argc: cint; argv: ptr UncheckedArray[JSValue]; magic: cint; func_data: ptr UncheckedArray[JSValue]): JSValue {.cdecl.} = let obj = func_data[0] let fun = func_data[1] return JS_Call(ctx, fun, obj, argc, argv) proc getter(ctx: JSContext; pager: Pager; a: JSAtom): JSValue {.jsgetprop.} = if pager.container != nil: let cval = ctx.toJS(pager.container) let val = JS_GetProperty(ctx, cval, a) if JS_IsFunction(ctx, val): let func_data = @[cval, val] let fun = JS_NewCFunctionData(ctx, reflect, 1, 0, 2, func_data.toJSValueArray()) JS_FreeValue(ctx, cval) JS_FreeValue(ctx, val) return fun JS_FreeValue(ctx, cval) if not JS_IsUndefined(val): return val return JS_NULL proc searchNext(pager: Pager; n = 1) {.jsfunc.} = if pager.regex.isSome: let wrap = pager.config.search.wrap pager.container.markPos0() if not pager.reverseSearch: pager.container.cursorNextMatch(pager.regex.get, wrap, true, n) else: pager.container.cursorPrevMatch(pager.regex.get, wrap, true, n) pager.container.markPos() else: pager.alert("No previous regular expression") proc searchPrev(pager: Pager; n = 1) {.jsfunc.} = if pager.regex.isSome: let wrap = pager.config.search.wrap pager.container.markPos0() if not pager.reverseSearch: pager.container.cursorPrevMatch(pager.regex.get, wrap, true, n) else: pager.container.cursorNextMatch(pager.regex.get, wrap, true, n) pager.container.markPos() else: pager.alert("No previous regular expression") proc getLineHist(pager: Pager; mode: LineMode): LineHistory = if pager.linehist[mode] == nil: pager.linehist[mode] = newLineHistory() return pager.linehist[mode] proc setLineEdit(pager: Pager; mode: LineMode; current = ""; hide = false; extraPrompt = "") = let hist = pager.getLineHist(mode) if pager.term.isatty() and pager.config.input.use_mouse: pager.term.disableMouse() pager.lineedit = readLine($mode & extraPrompt, current, pager.attrs.width, {}, hide, hist) pager.linemode = mode proc clearLineEdit(pager: Pager) = pager.lineedit = nil if pager.term.isatty() and pager.config.input.use_mouse: pager.term.enableMouse() proc searchForward(pager: Pager) {.jsfunc.} = pager.setLineEdit(lmSearchF) proc searchBackward(pager: Pager) {.jsfunc.} = pager.setLineEdit(lmSearchB) proc isearchForward(pager: Pager) {.jsfunc.} = pager.container.pushCursorPos() pager.isearchpromise = newResolvedPromise() pager.container.markPos0() pager.setLineEdit(lmISearchF) proc isearchBackward(pager: Pager) {.jsfunc.} = pager.container.pushCursorPos() pager.isearchpromise = newResolvedPromise() pager.container.markPos0() pager.setLineEdit(lmISearchB) proc gotoLine(ctx: JSContext; pager: Pager; val = JS_UNDEFINED): Opt[void] {.jsfunc.} = var n: int if ctx.fromJS(val, n).isSome: pager.container.gotoLine(n) elif JS_IsUndefined(val): pager.setLineEdit(lmGotoLine) else: var s: string ?ctx.fromJS(val, s) pager.container.gotoLine(s) proc dumpAlerts*(pager: Pager) = for msg in pager.alerts: stderr.write("cha: " & msg & '\n') proc quit*(pager: Pager) = pager.term.quit() pager.dumpAlerts() proc newPager*(config: Config; forkserver: ForkServer; ctx: JSContext; alerts: seq[string]): Pager = return Pager( config: config, forkserver: forkserver, term: newTerminal(stdout, config), alerts: alerts, jsctx: ctx, luctx: LUContext() ) proc genClientKey(pager: Pager): ClientKey = var key: ClientKey pager.devRandom.recvDataLoop(key) return key proc addLoaderClient*(pager: Pager; pid: int; config: LoaderClientConfig; clonedFrom = -1): ClientKey = var key = pager.genClientKey() while unlikely(not pager.loader.addClient(key, pid, config, clonedFrom)): key = pager.genClientKey() return key proc setLoader*(pager: Pager; loader: FileLoader) = pager.devRandom = newPosixStream("/dev/urandom", O_RDONLY, 0) pager.loader = loader let config = LoaderClientConfig( defaultHeaders: newHeaders(pager.config.network.default_headers), proxy: pager.config.network.proxy, filter: newURLFilter(default = true), ) loader.key = pager.addLoaderClient(pager.loader.clientPid, config) proc launchPager*(pager: Pager; istream: PosixStream; selector: Selector[int]) = pager.selector = selector case pager.term.start(istream) of tsrSuccess: discard of tsrDA1Fail: pager.alert("Failed to query DA1, please set display.query-da1 = false") pager.clearDisplay() pager.clearStatus() proc buffer(pager: Pager): Container {.jsfget, inline.} = return pager.container # Note: this function does not work correctly if start < i of last written char proc writeStatusMessage(pager: Pager; str: string; format = Format(); start = 0; maxwidth = -1; clip = '$'): int {.discardable.} = var maxwidth = maxwidth if maxwidth == -1: maxwidth = pager.status.grid.len var i = start let e = min(start + maxwidth, pager.status.grid.width) if i >= e: return i pager.status.redraw = true for r in str.runes: let w = r.width() if i + w >= e: pager.status.grid[i].format = format pager.status.grid[i].str = $clip inc i # Note: we assume `clip' is 1 cell wide break if r.isControlChar(): pager.status.grid[i].str = "^" pager.status.grid[i + 1].str = $getControlLetter(char(r)) pager.status.grid[i + 1].format = format else: pager.status.grid[i].str = $r pager.status.grid[i].format = format i += w result = i var def = Format() while i < e: pager.status.grid[i].str = "" pager.status.grid[i].format = def inc i # Note: should only be called directly after user interaction. proc refreshStatusMsg*(pager: Pager) = let container = pager.container if container == nil: return if pager.askpromise != nil: return if pager.precnum != 0: pager.writeStatusMessage($pager.precnum & pager.inputBuffer) elif pager.inputBuffer != "": pager.writeStatusMessage(pager.inputBuffer) elif pager.alerts.len > 0: pager.alertState = pasAlertOn pager.writeStatusMessage(pager.alerts[0]) pager.alerts.delete(0) else: var format = Format(flags: {ffReverse}) pager.alertState = pasNormal container.clearHover() var msg = $(container.cursory + 1) & "/" & $container.numLines & " (" & $container.atPercentOf() & "%)" let mw = pager.writeStatusMessage(msg, format) let title = " <" & container.getTitle() & ">" let hover = container.getHoverText() if hover.len == 0: pager.writeStatusMessage(title, format, mw) else: let hover2 = " " & hover let maxwidth = pager.status.grid.width - hover2.width() - mw let tw = pager.writeStatusMessage(title, format, mw, maxwidth, '>') pager.writeStatusMessage(hover2, format, tw) # Call refreshStatusMsg if no alert is being displayed on the screen. # Alerts take precedence over load info, but load info is preserved when no # pending alerts exist. proc showAlerts*(pager: Pager) = if (pager.alertState == pasNormal or pager.alertState == pasLoadInfo and pager.alerts.len > 0) and pager.inputBuffer == "" and pager.precnum == 0: pager.refreshStatusMsg() proc drawBuffer*(pager: Pager; container: Container; ofile: File) = var format = Format() container.readLines(proc(line: SimpleFlexibleLine) = if line.formats.len == 0: ofile.write(line.str & "\n") else: var x = 0 var w = 0 var i = 0 var s = "" for f in line.formats: let si = i while x < f.pos: var r: Rune fastRuneAt(line.str, i, r) x += r.width() let outstr = line.str.substr(si, i - 1) s &= pager.term.processOutputString(outstr, w) s &= pager.term.processFormat(format, f.format) if i < line.str.len: s &= pager.term.processOutputString(line.str.substr(i), w) s &= pager.term.processFormat(format, Format()) & "\n" ofile.write(s)) ofile.flushFile() proc redraw(pager: Pager) {.jsfunc.} = pager.term.clearCanvas() pager.display.redraw = true pager.status.redraw = true if pager.container != nil: pager.container.redraw = true if pager.container.select != nil: pager.container.select.redraw = true proc loadCachedImage(pager: Pager; container: Container; image: PosBitmap) = let bmp = NetworkBitmap() bmp[] = NetworkBitmap(image.bmp)[] let request = newRequest(newURL("cache:" & $bmp.cacheId).get) let cachedImage = CachedImage( bmp: bmp, width: image.width, height: image.height ) pager.loader.shareCachedItem(bmp.cacheId, pager.loader.clientPid, container.process) let imageMode = pager.term.imageMode pager.loader.fetch(request).then(proc(res: JSResult[Response]): Promise[JSResult[Response]] = if res.isNone: pager.loader.removeCachedItem(bmp.cacheId) return let response = res.get let headers = newHeaders() if uint64(image.width) != bmp.width or uint64(image.height) != bmp.height: headers.add("Cha-Image-Target-Dimensions", $image.width & 'x' & $image.height) let request = newRequest( newURL("img-codec+" & bmp.contentType.after('/') & ":decode").get, httpMethod = hmPost, headers = headers, body = RequestBody(t: rbtOutput, outputId: response.outputId), ) let r = pager.loader.fetch(request) response.resume() response.unregisterFun() response.body.sclose() return r ).then(proc(res: JSResult[Response]) = if res.isNone: pager.loader.removeCachedItem(bmp.cacheId) return let response = res.get # take target sizes bmp.width = uint64(image.width) bmp.height = uint64(image.height) case imageMode of imSixel: #TODO we should only cache the final output in memory, not the full # bitmap. response.saveToBitmap(bmp).then(proc() = container.redraw = true cachedImage.loaded = true pager.loader.removeCachedItem(bmp.cacheId) ) of imKitty: let headers = newHeaders({ "Cha-Image-Dimensions": $image.width & 'x' & $image.height }) let request = newRequest( newURL("img-codec+png:encode").get, httpMethod = hmPost, headers = headers, body = RequestBody(t: rbtOutput, outputId: response.outputId), ) let r = pager.loader.fetch(request) response.resume() response.unregisterFun() response.body.sclose() r.then(proc(res: JSResult[Response]): Promise[JSResult[Blob]] = return res.get.blob() ).then(proc(res: JSResult[Blob]) = container.redraw = true cachedImage.data = res.get cachedImage.loaded = true pager.loader.removeCachedItem(bmp.cacheId) ) of imNone: assert false ) container.cachedImages.add(cachedImage) proc initImages(pager: Pager; container: Container) = var newImages: seq[CanvasImage] = @[] for image in container.images: var imageId = -1 var data: Blob = nil var bmp0 = image.bmp if image.bmp of NetworkBitmap: let bmp = NetworkBitmap(image.bmp) let cached = container.findCachedImage(image) imageId = bmp.imageId if cached == nil: pager.loadCachedImage(container, image) continue bmp0 = cached.bmp data = cached.data if not cached.loaded: continue # loading else: imageId = pager.imageId inc pager.imageId let canvasImage = pager.term.loadImage(bmp0, data, container.process, imageId, image.x - container.fromx, image.y - container.fromy, image.x, image.y, pager.bufWidth, pager.bufHeight) if canvasImage != nil: newImages.add(canvasImage) pager.term.clearImages(pager.bufHeight) pager.term.canvasImages = newImages proc draw*(pager: Pager) = var redraw = false var imageRedraw = false let container = pager.container if container != nil: if container.redraw: pager.clearDisplay() let hlcolor = cellColor(pager.config.display.highlight_color) container.drawLines(pager.display.grid, hlcolor) if pager.config.display.highlight_marks: container.highlightMarks(pager.display.grid, hlcolor) container.redraw = false pager.display.redraw = true imageRedraw = true if container.select != nil: container.select.redraw = true if (let select = container.select; select != nil and select.redraw): select.drawSelect(pager.display.grid) select.redraw = false pager.display.redraw = true if pager.display.redraw: pager.term.writeGrid(pager.display.grid) pager.display.redraw = false redraw = true if pager.askpromise != nil or pager.askcharpromise != nil: pager.term.writeGrid(pager.status.grid, 0, pager.attrs.height - 1) pager.status.redraw = false redraw = true elif pager.lineedit != nil: if pager.lineedit.redraw: let x = pager.lineedit.generateOutput() pager.term.writeGrid(x, 0, pager.attrs.height - 1) pager.lineedit.redraw = false redraw = true elif pager.status.redraw: pager.term.writeGrid(pager.status.grid, 0, pager.attrs.height - 1) pager.status.redraw = false redraw = true if imageRedraw and pager.term.imageMode != imNone: # init images only after term canvas has been finalized pager.initImages(container) if redraw: pager.term.hideCursor() pager.term.outputGrid() if pager.term.imageMode != imNone: pager.term.outputImages() if pager.askpromise != nil: pager.term.setCursor(pager.askcursor, pager.attrs.height - 1) elif pager.lineedit != nil: pager.term.setCursor(pager.lineedit.getCursorX(), pager.attrs.height - 1) elif container != nil: if (let select = container.select; select != nil): pager.term.setCursor(select.getCursorX(), select.getCursorY()) else: pager.term.setCursor(container.acursorx, container.acursory) if redraw: pager.term.showCursor() pager.term.flush() proc writeAskPrompt(pager: Pager; s = "") = let maxwidth = pager.status.grid.width - s.len let i = pager.writeStatusMessage(pager.askprompt, maxwidth = maxwidth) pager.askcursor = pager.writeStatusMessage(s, start = i) proc ask(pager: Pager; prompt: string): Promise[bool] {.jsfunc.} = pager.askprompt = prompt pager.writeAskPrompt(" (y/n)") pager.askpromise = Promise[bool]() return pager.askpromise proc askChar(pager: Pager; prompt: string): Promise[string] {.jsfunc.} = pager.askprompt = prompt pager.writeAskPrompt() pager.askcharpromise = Promise[string]() return pager.askcharpromise proc fulfillAsk*(pager: Pager; y: bool) = pager.askpromise.resolve(y) pager.askpromise = nil pager.askprompt = "" proc fulfillCharAsk*(pager: Pager; s: string) = pager.askcharpromise.resolve(s) pager.askcharpromise = nil pager.askprompt = "" proc addContainer*(pager: Pager; container: Container) = container.parent = pager.container if pager.container != nil: pager.container.children.insert(container, 0) pager.setContainer(container) proc onSetLoadInfo(pager: Pager; container: Container) = if pager.alertState != pasAlertOn: if container.loadinfo == "": pager.alertState = pasNormal else: pager.writeStatusMessage(container.loadinfo) pager.alertState = pasLoadInfo proc newContainer(pager: Pager; bufferConfig: BufferConfig; loaderConfig: LoaderClientConfig; request: Request; title = ""; redirectDepth = 0; flags = {cfCanReinterpret, cfUserRequested}; contentType = none(string); charsetStack: seq[Charset] = @[]; url = request.url): Container = let stream = pager.loader.startRequest(request, loaderConfig) pager.loader.registerFun(stream.fd) let cacheId = if request.url.scheme == "cache": parseInt32(request.url.pathname).get(-1) else: -1 let container = newContainer( bufferConfig, loaderConfig, url, request, pager.luctx, pager.term.attrs, title, redirectDepth, flags, contentType, charsetStack, cacheId, pager.config ) pager.connectingContainers.add(ConnectingContainerItem( state: ccsBeforeResult, container: container, stream: stream )) pager.onSetLoadInfo(container) return container proc newContainerFrom(pager: Pager; container: Container; contentType: string): Container = let url = newURL("cache:" & $container.cacheId).get return pager.newContainer( container.config, container.loaderConfig, newRequest(url), contentType = some(contentType), charsetStack = container.charsetStack, url = container.url ) func findConnectingContainer*(pager: Pager; fd: int): int = for i, item in pager.connectingContainers: if item.stream.fd == fd: return i -1 func findConnectingContainer*(pager: Pager; container: Container): int = for i, item in pager.connectingContainers: if item.container == container: return i -1 func findProcMapItem*(pager: Pager; pid: int): int = for i, item in pager.procmap: if item.container.process == pid: return i -1 proc dupeBuffer(pager: Pager; container: Container; url: URL) = let p = container.clone(url, pager.loader) if p == nil: pager.alert("Failed to duplicate buffer.") else: p.then(proc(container: Container) = if container == nil: pager.alert("Failed to duplicate buffer.") else: pager.addContainer(container) pager.procmap.add(ProcMapItem( container: container, fdin: -1, fdout: -1, istreamOutputId: -1, ostreamOutputId: -1 )) ) proc dupeBuffer(pager: Pager) {.jsfunc.} = pager.dupeBuffer(pager.container, pager.container.url) func findPrev(container: Container): Container = if container.parent == nil: return nil let n = container.parent.children.find(container) assert n != -1, "Container not a child of its parent" if n == 0: return container.parent var container = container.parent.children[n - 1] while container.children.len > 0: container = container.children[^1] return container func findNext(container: Container): Container = if container.children.len > 0: return container.children[0] var container = container while container.parent != nil: let n = container.parent.children.find(container) assert n != -1, "Container not a child of its parent" if n < container.parent.children.high: return container.parent.children[n + 1] container = container.parent return nil func findPrevSibling(container: Container): Container = if container.parent == nil: return nil var n = container.parent.children.find(container) assert n != -1, "Container not a child of its parent" if n == 0: n = container.parent.children.len return container.parent.children[n - 1] func findNextSibling(container: Container): Container = if container.parent == nil: return nil var n = container.parent.children.find(container) assert n != -1, "Container not a child of its parent" if n == container.parent.children.high: n = -1 return container.parent.children[n + 1] func findParent(container: Container): Container = return container.parent func findFirstChild(container: Container): Container = if container.children.len == 0: return nil return container.children[0] func findAny(container: Container): Container = let prev = container.findPrev() if prev != nil: return prev return container.findNext() func opposite(dir: NavDirection): NavDirection = const Map = [ ndPrev: ndNext, ndNext: ndPrev, ndPrevSibling: ndNextSibling, ndNextSibling: ndPrevSibling, ndParent: ndFirstChild, ndFirstChild: ndParent, ndAny: ndAny ] return Map[dir] func find(container: Container; dir: NavDirection): Container = return case dir of ndPrev: container.findPrev() of ndNext: container.findNext() of ndPrevSibling: container.findPrevSibling() of ndNextSibling: container.findNextSibling() of ndParent: container.findParent() of ndFirstChild: container.findFirstChild() of ndAny: container.findAny() # The prevBuffer and nextBuffer procedures emulate w3m's PREV and NEXT # commands by traversing the container tree in a depth-first order. proc prevBuffer*(pager: Pager): bool {.jsfunc.} = pager.navDirection = ndPrev if pager.container == nil: return false let prev = pager.container.findPrev() if prev == nil: return false pager.setContainer(prev) return true proc nextBuffer*(pager: Pager): bool {.jsfunc.} = pager.navDirection = ndNext if pager.container == nil: return false let next = pager.container.findNext() if next == nil: return false pager.setContainer(next) return true proc parentBuffer(pager: Pager): bool {.jsfunc.} = pager.navDirection = ndParent if pager.container == nil: return false let parent = pager.container.findParent() if parent == nil: return false pager.setContainer(parent) return true proc prevSiblingBuffer(pager: Pager): bool {.jsfunc.} = pager.navDirection = ndPrevSibling if pager.container == nil: return false if pager.container.parent == nil: return false var n = pager.container.parent.children.find(pager.container) assert n != -1, "Container not a child of its parent" if n == 0: n = pager.container.parent.children.len pager.setContainer(pager.container.parent.children[n - 1]) return true proc nextSiblingBuffer(pager: Pager): bool {.jsfunc.} = pager.navDirection = ndNextSibling if pager.container == nil: return false if pager.container.parent == nil: return false var n = pager.container.parent.children.find(pager.container) assert n != -1, "Container not a child of its parent" if n == pager.container.parent.children.high: n = -1 pager.setContainer(pager.container.parent.children[n + 1]) return true proc alert*(pager: Pager; msg: string) {.jsfunc.} = pager.alerts.add(msg) # replace target with container in the tree proc replace*(pager: Pager; target, container: Container) = let n = target.children.find(container) if n != -1: target.children.delete(n) container.parent = nil let n2 = container.children.find(target) if n2 != -1: container.children.delete(n2) target.parent = nil container.children.add(target.children) for child in container.children: child.parent = container target.children.setLen(0) if target.parent != nil: container.parent = target.parent let n = target.parent.children.find(target) assert n != -1, "Container not a child of its parent" container.parent.children[n] = container target.parent = nil if pager.container == target: pager.setContainer(container) proc deleteContainer(pager: Pager; container, setTarget: Container) = if container.loadState == lsLoading: container.cancel() if container.replaceBackup != nil: pager.setContainer(container.replaceBackup) elif container.replace != nil: pager.replace(container, container.replace) if container.sourcepair != nil: container.sourcepair.sourcepair = nil container.sourcepair = nil if container.replaceRef != nil: container.replaceRef.replace = nil container.replaceRef.replaceBackup = nil container.replaceRef = nil if container.parent != nil: let parent = container.parent let n = parent.children.find(container) assert n != -1, "Container not a child of its parent" for i in countdown(container.children.high, 0): let child = container.children[i] child.parent = container.parent parent.children.insert(child, n + 1) parent.children.delete(n) elif container.children.len > 0: let parent = container.children[0] parent.parent = nil for i in 1..container.children.high: container.children[i].parent = parent parent.children.add(container.children[i]) container.parent = nil container.children.setLen(0) if container.replace != nil: container.replace = nil elif container.replaceBackup != nil: container.replaceBackup = nil elif pager.container == container: pager.setContainer(setTarget) pager.unreg.add(container) if container.process != -1: pager.loader.removeCachedItem(container.cacheId) pager.forkserver.removeChild(container.process) pager.loader.removeClient(container.process) proc discardBuffer*(pager: Pager; container = none(Container); dir = none(NavDirection)) {.jsfunc.} = if dir.isSome: pager.navDirection = dir.get.opposite() let container = container.get(pager.container) let dir = pager.navDirection.opposite() let setTarget = container.find(dir) if container == nil or setTarget == nil: pager.alert("No buffer in direction: " & $dir) else: pager.deleteContainer(container, setTarget) proc discardTree(pager: Pager; container = none(Container)) {.jsfunc.} = let container = container.get(pager.container) if container != nil: for c in container.descendants: pager.deleteContainer(container, nil) else: pager.alert("Buffer has no children!") proc c_system(cmd: cstring): cint {.importc: "system", header: "".} # Run process (without suspending the terminal controller). proc runProcess(cmd: string): bool = let wstatus = c_system(cstring(cmd)) if wstatus == -1: result = false else: result = WIFEXITED(wstatus) and WEXITSTATUS(wstatus) == 0 if not result: # Hack. #TODO this is a very bad idea, e.g. say the editor is writing into the # file, then receives SIGINT, now the file is corrupted but Chawan will # happily read it as if nothing happened. # We should find a proper solution for this. result = WIFSIGNALED(wstatus) and WTERMSIG(wstatus) == SIGINT # Run process (and suspend the terminal controller). proc runProcess(term: Terminal; cmd: string; wait = false): bool = term.quit() result = runProcess(cmd) if wait: term.anyKey() term.restart() # Run process, and capture its output. proc runProcessCapture(cmd: string; outs: var string): bool = let file = popen(cmd, "r") if file == nil: return false outs = file.readAll() let rv = pclose(file) if rv == -1: return false return rv == 0 # Run process, and write an arbitrary string into its standard input. proc runProcessInto(cmd, ins: string): bool = let file = popen(cmd, "w") if file == nil: return false file.write(ins) let rv = pclose(file) if rv == -1: return false return rv == 0 template myExec(cmd: string) = discard execl("/bin/sh", "sh", "-c", cstring(cmd), nil) exitnow(127) proc toggleSource(pager: Pager) {.jsfunc.} = if cfCanReinterpret notin pager.container.flags: return if pager.container.sourcepair != nil: pager.setContainer(pager.container.sourcepair) else: let ishtml = cfIsHTML notin pager.container.flags #TODO I wish I could set the contentType to whatever I wanted, not just HTML let contentType = if ishtml: "text/html" else: "text/plain" let container = pager.newContainerFrom(pager.container, contentType) if container != nil: container.sourcepair = pager.container pager.container.sourcepair = container pager.addContainer(container) proc getCacheFile(pager: Pager; cacheId: int): string {.jsfunc.} = return pager.loader.getCacheFile(cacheId) proc cacheFile(pager: Pager): string {.jsfget.} = if pager.container != nil: return pager.getCacheFile(pager.container.cacheId) return "" proc getEditorCommand(pager: Pager; file: string; line = 1): string {.jsfunc.} = var editor = pager.config.external.editor if (let uqEditor = ChaPath(editor).unquote(); uqEditor.isSome): if uqEditor.get in ["vi", "nvi", "vim", "nvim"]: editor = uqEditor.get & " +%d" var canpipe = true var s = unquoteCommand(editor, "", file, nil, canpipe, line) if canpipe: # %s not in command; add file name ourselves if s[^1] != ' ': s &= ' ' s &= quoteFile(file, qsNormal) return s proc openInEditor(pager: Pager; input: var string): bool = try: let tmpf = getTempFile(pager.config.external.tmpdir) if input != "": writeFile(tmpf, input) let cmd = pager.getEditorCommand(tmpf) if pager.term.runProcess(cmd): if fileExists(tmpf): input = readFile(tmpf) removeFile(tmpf) return true except IOError: discard return false proc windowChange*(pager: Pager) = let oldAttrs = pager.attrs pager.term.windowChange() if pager.attrs == oldAttrs: #TODO maybe it's more efficient to let false positives through? return if pager.lineedit != nil: pager.lineedit.windowChange(pager.attrs) pager.clearDisplay() pager.clearStatus() for container in pager.containers: container.windowChange(pager.attrs) if pager.askprompt != "": pager.writeAskPrompt() pager.showAlerts() # Apply siteconf settings to a request. # Note that this may modify the URL passed. proc applySiteconf(pager: Pager; url: var URL; charsetOverride: Charset; loaderConfig: var LoaderClientConfig): BufferConfig = let host = url.host let ctx = pager.jsctx var res = BufferConfig( userstyle: pager.config.css.stylesheet, refererFrom: pager.config.buffer.referer_from, scripting: pager.config.buffer.scripting, charsets: pager.config.encoding.document_charset, images: pager.config.buffer.images, styling: pager.config.buffer.styling, autofocus: pager.config.buffer.autofocus, isdump: pager.config.start.headless, charsetOverride: charsetOverride, protocol: pager.config.protocol, metaRefresh: pager.config.buffer.meta_refresh ) loaderConfig = LoaderClientConfig( defaultHeaders: newHeaders(pager.config.network.default_headers), cookiejar: nil, proxy: pager.config.network.proxy, filter: newURLFilter( scheme = some(url.scheme), allowschemes = @["data", "cache"], default = true ), insecureSSLNoVerify: false ) for sc in pager.config.siteconf: if sc.url.isSome and not sc.url.get.match($url): continue elif sc.host.isSome and not sc.host.get.match(host): continue if sc.rewrite_url.isSome: let fun = sc.rewrite_url.get var arg0 = ctx.toJS(url) let ret = JS_Call(ctx, fun, JS_UNDEFINED, 1, arg0.toJSValueArray()) if not JS_IsException(ret): # Warning: we must only print exceptions if the *call* returned one. # Conversion may simply error out because the function didn't return a # new URL, and that's fine. var nu: URL if ctx.fromJS(ret, nu).isSome and nu != nil: url = nu else: #TODO should writeException the message to console pager.alert("Error rewriting URL: " & ctx.getExceptionMsg()) JS_FreeValue(ctx, arg0) JS_FreeValue(ctx, ret) if sc.cookie.isSome: if sc.cookie.get: # host/url might have changed by now let jarid = sc.share_cookie_jar.get(url.host) if jarid notin pager.cookiejars: pager.cookiejars[jarid] = newCookieJar(url, sc.third_party_cookie) loaderConfig.cookieJar = pager.cookiejars[jarid] else: loaderConfig.cookieJar = nil # override if sc.scripting.isSome: res.scripting = sc.scripting.get if sc.referer_from.isSome: res.refererFrom = sc.referer_from.get if sc.document_charset.len > 0: res.charsets = sc.document_charset if sc.images.isSome: res.images = sc.images.get if sc.stylesheet.isSome: res.userstyle &= "\n" res.userstyle &= sc.stylesheet.get if sc.proxy.isSome: loaderConfig.proxy = sc.proxy.get if sc.default_headers != nil: loaderConfig.defaultHeaders = newHeaders(sc.default_headers[]) if sc.insecure_ssl_no_verify.isSome: loaderConfig.insecureSSLNoVerify = sc.insecure_ssl_no_verify.get if sc.autofocus.isSome: res.autofocus = sc.autofocus.get if sc.meta_refresh.isSome: res.metaRefresh = sc.meta_refresh.get if res.images: loaderConfig.filter.allowschemes .add(pager.config.external.urimethodmap.imageProtos) return res # Load request in a new buffer. proc gotoURL(pager: Pager; request: Request; prevurl = none(URL); contentType = none(string); cs = CHARSET_UNKNOWN; replace: Container = nil; replaceBackup: Container = nil; redirectDepth = 0; referrer: Container = nil; save = false; url: URL = nil): Container = pager.navDirection = ndNext if referrer != nil and referrer.config.refererFrom: request.referrer = referrer.url let url = if url != nil: url else: request.url var loaderConfig: LoaderClientConfig var bufferConfig = pager.applySiteconf(request.url, cs, loaderConfig) if prevurl.isNone or not prevurl.get.equals(request.url, true) or request.url.hash == "" or request.httpMethod != hmGet: # Basically, we want to reload the page *only* when # a) we force a reload (by setting prevurl to none) # b) or the new URL isn't just the old URL + an anchor # I think this makes navigation pretty natural, or at least very close to # what other browsers do. Still, it would be nice if we got some visual # feedback on what is actually going to happen when typing a URL; TODO. if referrer != nil: loaderConfig.referrerPolicy = referrer.loaderConfig.referrerPolicy var flags = {cfCanReinterpret, cfUserRequested} if save: flags.incl(cfSave) let container = pager.newContainer( bufferConfig, loaderConfig, request, redirectDepth = redirectDepth, contentType = contentType, flags = flags, url = url ) if replace != nil: pager.replace(replace, container) if replaceBackup == nil: container.replace = replace replace.replaceRef = container else: container.replaceBackup = replaceBackup replaceBackup.replaceRef = container container.copyCursorPos(replace) else: pager.addContainer(container) inc pager.numload return container else: pager.container.findAnchor(request.url.anchor) return nil proc omniRewrite(pager: Pager; s: string): string = for rule in pager.config.omnirule: if rule.match.match(s): let fun = rule.substitute_url.get let ctx = pager.jsctx var arg0 = ctx.toJS(s) let jsRet = JS_Call(ctx, fun, JS_UNDEFINED, 1, arg0.toJSValueArray()) defer: JS_FreeValue(ctx, jsRet) defer: JS_FreeValue(ctx, arg0) var res: string if ctx.fromJS(jsRet, res).isSome: return res pager.alert("Error in substitution of " & $rule.match & " for " & s & ": " & ctx.getExceptionMsg()) return s # When the user has passed a partial URL as an argument, they might've meant # either: # * file://$PWD/ # * https:// # So we attempt to load both, and see what works. proc loadURL*(pager: Pager; url: string; ctype = none(string); cs = CHARSET_UNKNOWN) = let url0 = pager.omniRewrite(url) let url = if url[0] == '~': expandPath(url0) else: url0 let firstparse = parseURL(url) if firstparse.isSome: let prev = if pager.container != nil: some(pager.container.url) else: none(URL) discard pager.gotoURL(newRequest(firstparse.get), prev, ctype, cs) return var urls: seq[URL] if pager.config.network.prepend_https and pager.config.network.prepend_scheme != "" and url[0] != '/': let pageurl = parseURL(pager.config.network.prepend_scheme & url) if pageurl.isSome: # attempt to load remote page urls.add(pageurl.get) let cdir = parseURL("file://" & percentEncode(getCurrentDir(), LocalPathPercentEncodeSet) & DirSep) let localurl = percentEncode(url, LocalPathPercentEncodeSet) let newurl = parseURL(localurl, cdir) if newurl.isSome: urls.add(newurl.get) # attempt to load local file if urls.len == 0: pager.alert("Invalid URL " & url) else: let container = pager.gotoURL(newRequest(urls.pop()), contentType = ctype, cs = cs) if container != nil: container.retry = urls proc readPipe0*(pager: Pager; contentType: string; cs: Charset; fd: FileHandle; url: URL; title: string; flags: set[ContainerFlag]): Container = var url = url pager.loader.passFd(url.pathname, fd) safeClose(fd) var loaderConfig: LoaderClientConfig let bufferConfig = pager.applySiteconf(url, cs, loaderConfig) return pager.newContainer( bufferConfig, loaderConfig, newRequest(url), title = title, flags = flags, contentType = some(contentType) ) proc readPipe*(pager: Pager; contentType: string; cs: Charset; fd: FileHandle; title: string) = let url = newURL("stream:-").get let container = pager.readPipe0(contentType, cs, fd, url, title, {cfCanReinterpret, cfUserRequested}) inc pager.numload pager.addContainer(container) proc command(pager: Pager) {.jsfunc.} = pager.setLineEdit(lmCommand) proc commandMode(pager: Pager; val: bool) {.jsfset.} = pager.commandMode = val if val: pager.command() proc checkRegex(pager: Pager; regex: Result[Regex, string]): Opt[Regex] = if regex.isNone: pager.alert("Invalid regex: " & regex.error) return err() return ok(regex.get) proc compileSearchRegex(pager: Pager; s: string): Result[Regex, string] = return compileSearchRegex(s, pager.config.search.ignore_case) proc updateReadLineISearch(pager: Pager; linemode: LineMode) = let lineedit = pager.lineedit pager.isearchpromise = pager.isearchpromise.then(proc(): EmptyPromise = case lineedit.state of lesCancel: pager.iregex.err() pager.container.popCursorPos() pager.container.clearSearchHighlights() pager.container.redraw = true pager.isearchpromise = nil of lesEdit: if lineedit.news != "": pager.iregex = pager.compileSearchRegex(lineedit.news) pager.container.popCursorPos(true) pager.container.pushCursorPos() if pager.iregex.isSome: pager.container.hlon = true let wrap = pager.config.search.wrap return if linemode == lmISearchF: pager.container.cursorNextMatch(pager.iregex.get, wrap, false, 1) else: pager.container.cursorPrevMatch(pager.iregex.get, wrap, false, 1) of lesFinish: if lineedit.news != "": pager.regex = pager.checkRegex(pager.iregex) else: pager.searchNext() pager.reverseSearch = linemode == lmISearchB pager.container.markPos() pager.container.clearSearchHighlights() pager.container.sendCursorPosition() pager.container.redraw = true pager.isearchpromise = nil ) proc saveTo(pager: Pager; data: LineDataDownload; path: string) = if pager.loader.redirectToFile(data.outputId, path): pager.alert("Saving file to " & path) pager.loader.resume(data.outputId) data.stream.sclose() pager.lineData = nil else: pager.ask("Failed to save to " & path & ". Retry?").then( proc(x: bool) = if x: pager.setLineEdit(lmDownload, path) else: data.stream.sclose() pager.lineData = nil ) proc updateReadLine*(pager: Pager) = let lineedit = pager.lineedit if pager.linemode in {lmISearchF, lmISearchB}: pager.updateReadLineISearch(pager.linemode) else: case lineedit.state of lesEdit: discard of lesFinish: case pager.linemode of lmLocation: pager.loadURL(lineedit.news) of lmUsername: LineDataAuth(pager.lineData).url.username = lineedit.news pager.setLineEdit(lmPassword, hide = true) of lmPassword: let url = LineDataAuth(pager.lineData).url url.password = lineedit.news discard pager.gotoURL(newRequest(url), some(pager.container.url), replace = pager.container, referrer = pager.container) pager.lineData = nil of lmCommand: pager.scommand = lineedit.news if pager.commandMode: pager.command() of lmBuffer: pager.container.readSuccess(lineedit.news) of lmBufferFile: let ps = newPosixStream(lineedit.news, O_RDONLY, 0) if ps == nil: pager.alert("File not found") pager.container.readCanceled() else: var stats: Stat if fstat(ps.fd, stats) < 0 or S_ISDIR(stats.st_mode): pager.alert("Not a file: " & lineedit.news) else: let name = lineedit.news.afterLast('/') pager.container.readSuccess(name, ps.fd) ps.sclose() of lmSearchF, lmSearchB: if lineedit.news != "": let regex = pager.compileSearchRegex(lineedit.news) pager.regex = pager.checkRegex(regex) pager.reverseSearch = pager.linemode == lmSearchB pager.searchNext() of lmGotoLine: pager.container.gotoLine(lineedit.news) of lmDownload: let data = LineDataDownload(pager.lineData) if fileExists(lineedit.news): pager.ask("Override file " & lineedit.news & "?").then( proc(x: bool) = if x: pager.saveTo(data, lineedit.news) else: pager.setLineEdit(lmDownload, lineedit.news) ) else: pager.saveTo(data, lineedit.news) of lmISearchF, lmISearchB: discard of lesCancel: case pager.linemode of lmUsername, lmPassword: pager.discardBuffer() of lmBuffer: pager.container.readCanceled() of lmCommand: pager.commandMode = false of lmDownload: let data = LineDataDownload(pager.lineData) data.stream.sclose() else: discard pager.lineData = nil if lineedit.state in {lesCancel, lesFinish} and pager.lineedit == lineedit: pager.clearLineEdit() # Same as load(s + '\n') proc loadSubmit(pager: Pager; s: string) {.jsfunc.} = pager.loadURL(s) # Open a URL prompt and visit the specified URL. proc load(pager: Pager; s = "") {.jsfunc.} = if s.len > 0 and s[^1] == '\n': if s.len > 1: pager.loadURL(s[0..^2]) elif s == "": pager.setLineEdit(lmLocation, $pager.container.url) else: pager.setLineEdit(lmLocation, s) # Go to specific URL (for JS) type GotoURLDict = object of JSDict contentType {.jsdefault.}: Option[string] replace {.jsdefault.}: Container proc jsGotoURL(pager: Pager; v: JSValue; t = GotoURLDict()): JSResult[void] {.jsfunc: "gotoURL".} = var request: Request = nil var jsRequest: JSRequest = nil if pager.jsctx.fromJS(v, jsRequest).isSome: request = jsRequest.request else: var url: URL = nil if pager.jsctx.fromJS(v, url).isNone: var s: string ?pager.jsctx.fromJS(v, s) url = ?newURL(s) request = newRequest(url) discard pager.gotoURL(request, contentType = t.contentType, replace = t.replace) return ok() # Reload the page in a new buffer, then kill the previous buffer. proc reload(pager: Pager) {.jsfunc.} = discard pager.gotoURL(newRequest(pager.container.url), none(URL), pager.container.contentType, replace = pager.container) proc setEnvVars(pager: Pager) {.jsfunc.} = try: putEnv("CHA_URL", $pager.container.url) putEnv("CHA_CHARSET", $pager.container.charset) except OSError: pager.alert("Warning: failed to set some environment variables") type ExternDict = object of JSDict setenv {.jsdefault: true.}: bool suspend {.jsdefault: true.}: bool wait {.jsdefault: false.}: bool #TODO we should have versions with retval as int? proc extern(pager: Pager; cmd: string; t = ExternDict(setenv: true, suspend: true)): bool {.jsfunc.} = if t.setenv: pager.setEnvVars() if t.suspend: return runProcess(pager.term, cmd, t.wait) else: return runProcess(cmd) proc externCapture(pager: Pager; cmd: string): Option[string] {.jsfunc.} = pager.setEnvVars() var s: string if not runProcessCapture(cmd, s): return none(string) return some(s) proc externInto(pager: Pager; cmd, ins: string): bool {.jsfunc.} = pager.setEnvVars() return runProcessInto(cmd, ins) proc externFilterSource(pager: Pager; cmd: string; c: Container = nil; contentType = none(string)) {.jsfunc.} = let fromc = if c != nil: c else: pager.container let fallback = pager.container.contentType.get("text/plain") let contentType = contentType.get(fallback) let container = pager.newContainerFrom(fromc, contentType) if contentType == "text/html": container.flags.incl(cfIsHTML) else: container.flags.excl(cfIsHTML) pager.addContainer(container) container.filter = BufferFilter(cmd: cmd) type CheckMailcapResult = object fdout: int ostreamOutputId: int connect: bool ishtml: bool found: bool template myFork(): cint = stdout.flushFile() stderr.flushFile() fork() # Pipe output of an x-ansioutput mailcap command to the text/x-ansi handler. proc ansiDecode(pager: Pager; url: URL; ishtml: var bool; fdin: cint): cint = let entry = pager.config.external.mailcap.getMailcapEntry("text/x-ansi", "", url) var canpipe = true let cmd = unquoteCommand(entry.cmd, "text/x-ansi", "", url, canpipe) if not canpipe: pager.alert("Error: could not pipe to text/x-ansi, decoding as text/plain") return -1 var pipefdOutAnsi: array[2, cint] if pipe(pipefdOutAnsi) == -1: pager.alert("Error: failed to open pipe") return case myFork() of -1: pager.alert("Error: failed to fork ANSI decoder process") discard close(pipefdOutAnsi[0]) discard close(pipefdOutAnsi[1]) return -1 of 0: # child process discard close(pipefdOutAnsi[0]) discard dup2(fdin, stdin.getFileHandle()) discard close(fdin) discard dup2(pipefdOutAnsi[1], stdout.getFileHandle()) discard close(pipefdOutAnsi[1]) closeStderr() myExec(cmd) else: discard close(pipefdOutAnsi[1]) discard close(fdin) ishtml = mfHtmloutput in entry.flags return pipefdOutAnsi[0] # Pipe input into the mailcap command, then read its output into a buffer. # needsterminal is ignored. proc runMailcapReadPipe(pager: Pager; stream: SocketStream; cmd: string; pipefdOut: array[2, cint]): int = let pid = myFork() if pid == -1: pager.alert("Error: failed to fork mailcap read process") return -1 elif pid == 0: # child process discard close(pipefdOut[0]) discard dup2(stream.fd, stdin.getFileHandle()) stream.sclose() discard dup2(pipefdOut[1], stdout.getFileHandle()) closeStderr() discard close(pipefdOut[1]) myExec(cmd) # parent pid # Pipe input into the mailcap command, and discard its output. # If needsterminal, leave stderr and stdout open and wait for the process. proc runMailcapWritePipe(pager: Pager; stream: SocketStream; needsterminal: bool; cmd: string) = if needsterminal: pager.term.quit() let pid = myFork() if pid == -1: pager.alert("Error: failed to fork mailcap write process") elif pid == 0: # child process discard dup2(stream.fd, stdin.getFileHandle()) stream.sclose() if not needsterminal: closeStdout() closeStderr() myExec(cmd) else: # parent stream.sclose() if needsterminal: var x: cint discard waitpid(pid, x, 0) pager.term.restart() proc writeToFile(istream: SocketStream; outpath: string): bool = let ps = newPosixStream(outpath, O_WRONLY or O_CREAT, 0o600) if ps == nil: return false var buffer: array[4096, uint8] while true: let n = istream.recvData(buffer) if n == 0: break ps.sendDataLoop(buffer.toOpenArray(0, n - 1)) ps.sclose() true # Save input in a file, run the command, and redirect its output to a # new buffer. # needsterminal is ignored. proc runMailcapReadFile(pager: Pager; stream: SocketStream; cmd, outpath: string; pipefdOut: array[2, cint]): int = let pid = myFork() if pid == 0: # child process discard close(pipefdOut[0]) discard dup2(pipefdOut[1], stdout.getFileHandle()) discard close(pipefdOut[1]) closeStderr() if not stream.writeToFile(outpath): #TODO print error message quit(1) stream.sclose() let ret = execCmd(cmd) discard tryRemoveFile(outpath) quit(ret) # parent pid # Save input in a file, run the command, and discard its output. # If needsterminal, leave stderr and stdout open and wait for the process. proc runMailcapWriteFile(pager: Pager; stream: SocketStream; needsterminal: bool; cmd, outpath: string) = if needsterminal: pager.term.quit() if not stream.writeToFile(outpath): pager.term.restart() pager.alert("Error: failed to write file for mailcap process") else: discard execCmd(cmd) discard tryRemoveFile(outpath) pager.term.restart() else: # don't block let pid = myFork() if pid == 0: # child process closeStdin() closeStdout() closeStderr() if not stream.writeToFile(outpath): #TODO print error message (maybe in parent?) quit(1) stream.sclose() let ret = execCmd(cmd) discard tryRemoveFile(outpath) quit(ret) # parent stream.sclose() proc filterBuffer(pager: Pager; stream: SocketStream; cmd: string; ishtml: bool): CheckMailcapResult = pager.setEnvVars() var pipefd_out: array[2, cint] if pipe(pipefd_out) == -1: pager.alert("Error: failed to open pipe") return CheckMailcapResult(connect: false, fdout: -1) let pid = myFork() if pid == -1: pager.alert("Error: failed to fork buffer filter process") return CheckMailcapResult(connect: false, fdout: -1) elif pid == 0: # child discard close(pipefd_out[0]) discard dup2(stream.fd, stdin.getFileHandle()) stream.sclose() discard dup2(pipefd_out[1], stdout.getFileHandle()) closeStderr() discard close(pipefd_out[1]) myExec(cmd) # parent discard close(pipefd_out[1]) let fdout = pipefd_out[0] let url = parseURL("stream:" & $pid).get pager.loader.passFd(url.pathname, FileHandle(fdout)) safeClose(fdout) let response = pager.loader.doRequest(newRequest(url)) return CheckMailcapResult( connect: true, fdout: response.body.fd, ostreamOutputId: response.outputId, ishtml: ishtml, found: true ) # Search for a mailcap entry, and if found, execute the specified command # and pipeline the input and output appropriately. # There are four possible outcomes: # * pipe stdin, discard stdout # * pipe stdin, read stdout # * write to file, run, discard stdout # * write to file, run, read stdout # If needsterminal is specified, and stdout is not being read, then the # pager is suspended until the command exits. #TODO add support for edit/compose, better error handling proc checkMailcap(pager: Pager; container: Container; stream: SocketStream; istreamOutputId: int; contentType: string): CheckMailcapResult = if container.filter != nil: return pager.filterBuffer( stream, container.filter.cmd, cfIsHTML in container.flags ) # contentType must exist, because we set it in applyResponse let shortContentType = container.contentType.get if shortContentType == "text/html": # We support text/html natively, so it would make little sense to execute # mailcap filters for it. return CheckMailcapResult( connect: true, fdout: stream.fd, ishtml: true, found: true ) if shortContentType == "text/plain": # text/plain could potentially be useful. Unfortunately, many mailcaps # include a text/plain entry with less by default, so it's probably better # to ignore this. return CheckMailcapResult(connect: true, fdout: stream.fd, found: true) #TODO callback for outpath or something let url = container.url let entry = pager.config.external.mailcap.getMailcapEntry(contentType, "", url) if entry == nil: return CheckMailcapResult(connect: true, fdout: stream.fd, found: false) let ext = url.pathname.afterLast('.') let tempfile = getTempFile(pager.config.external.tmpdir, ext) let outpath = if entry.nametemplate != "": unquoteCommand(entry.nametemplate, contentType, tempfile, url) else: tempfile var canpipe = true let cmd = unquoteCommand(entry.cmd, contentType, outpath, url, canpipe) var ishtml = mfHtmloutput in entry.flags let needsterminal = mfNeedsterminal in entry.flags putEnv("MAILCAP_URL", $url) block needsConnect: if entry.flags * {mfCopiousoutput, mfHtmloutput, mfAnsioutput} == {}: # No output. Resume here, so that blocking needsterminal filters work. pager.loader.resume(istreamOutputId) if canpipe: pager.runMailcapWritePipe(stream, needsterminal, cmd) else: pager.runMailcapWriteFile(stream, needsterminal, cmd, outpath) # stream is already closed break needsConnect # never connect here, since there's no output var pipefdOut: array[2, cint] if pipe(pipefdOut) == -1: pager.alert("Error: failed to open pipe") stream.sclose() # connect: false implies that we consumed the stream break needsConnect let pid = if canpipe: pager.runMailcapReadPipe(stream, cmd, pipefdOut) else: pager.runMailcapReadFile(stream, cmd, outpath, pipefdOut) discard close(pipefdOut[1]) # close write let fdout = if not ishtml and mfAnsioutput in entry.flags: pager.ansiDecode(url, ishtml, pipefdOut[0]) else: pipefdOut[0] delEnv("MAILCAP_URL") let url = parseURL("stream:" & $pid).get pager.loader.passFd(url.pathname, FileHandle(fdout)) safeClose(cint(fdout)) let response = pager.loader.doRequest(newRequest(url)) return CheckMailcapResult( connect: true, fdout: response.body.fd, ostreamOutputId: response.outputId, ishtml: ishtml, found: true ) delEnv("MAILCAP_URL") return CheckMailcapResult(connect: false, fdout: -1, found: true) proc redirectTo(pager: Pager; container: Container; request: Request) = let replaceBackup = if container.replaceBackup != nil: container.replaceBackup else: container.find(ndAny) let nc = pager.gotoURL(request, some(container.url), replace = container, replaceBackup = replaceBackup, redirectDepth = container.redirectDepth + 1, referrer = container) nc.loadinfo = "Redirecting to " & $request.url pager.onSetLoadInfo(nc) dec pager.numload proc fail(pager: Pager; container: Container; errorMessage: string) = dec pager.numload pager.deleteContainer(container, container.find(ndAny)) if container.retry.len > 0: discard pager.gotoURL(newRequest(container.retry.pop()), contentType = container.contentType) else: pager.alert("Can't load " & $container.url & " (" & errorMessage & ")") proc redirect(pager: Pager; container: Container; response: Response; request: Request) = # if redirection fails, then we need some other container to move to... let failTarget = container.find(ndAny) # still need to apply response, or we lose cookie jars. container.applyResponse(response, pager.config.external.mime_types) if container.redirectDepth < pager.config.network.max_redirect: if container.url.scheme == request.url.scheme or container.url.scheme == "cgi-bin" or container.url.scheme == "http" and request.url.scheme == "https" or container.url.scheme == "https" and request.url.scheme == "http": pager.redirectTo(container, request) #TODO perhaps make following behavior configurable? elif request.url.scheme == "cgi-bin": pager.alert("Blocked redirection attempt to " & $request.url) else: let url = request.url pager.ask("Warning: switch protocols? " & $url).then(proc(x: bool) = if x: pager.redirectTo(container, request) ) else: pager.alert("Error: maximum redirection depth reached") pager.deleteContainer(container, failTarget) proc askDownloadPath(pager: Pager; container: Container; response: Response) = var buf = pager.config.external.download_dir let pathname = container.url.pathname if pathname[^1] == '/': buf &= "index.html" else: buf &= container.url.pathname.afterLast('/').percentDecode() pager.setLineEdit(lmDownload, buf) pager.lineData = LineDataDownload( outputId: response.outputId, stream: response.body ) pager.deleteContainer(container, container.find(ndAny)) pager.refreshStatusMsg() dec pager.numload proc connected(pager: Pager; container: Container; response: Response) = let istream = response.body container.applyResponse(response, pager.config.external.mime_types) if response.status == 401: # unauthorized pager.setLineEdit(lmUsername, container.url.username) pager.lineData = LineDataAuth(url: newURL(container.url)) istream.sclose() return # This forces client to ask for confirmation before quitting. # (It checks a flag on container, because console buffers must not affect this # variable.) if cfUserRequested in container.flags: pager.hasload = true if cfSave in container.flags: # download queried by user pager.askDownloadPath(container, response) return let realContentType = if "Content-Type" in response.headers: response.headers["Content-Type"] else: # both contentType and charset must be set by applyResponse. container.contentType.get & ";charset=" & $container.charset let mailcapRes = pager.checkMailcap(container, istream, response.outputId, realContentType) let shortContentType = container.contentType.get if not mailcapRes.found and not shortContentType.startsWithIgnoreCase("text/") and not shortContentType.isJavaScriptType(): pager.askDownloadPath(container, response) return if mailcapRes.connect: if mailcapRes.ishtml: container.flags.incl(cfIsHTML) else: container.flags.excl(cfIsHTML) # buffer now actually exists; create a process for it var attrs = pager.attrs # subtract status line height attrs.height -= 1 attrs.heightPx -= attrs.ppl container.process = pager.forkserver.forkBuffer( container.config, container.url, attrs, mailcapRes.ishtml, container.charsetStack ) if mailcapRes.fdout != istream.fd: # istream has been redirected into a filter istream.sclose() pager.procmap.add(ProcMapItem( container: container, fdout: FileHandle(mailcapRes.fdout), fdin: FileHandle(istream.fd), ostreamOutputId: mailcapRes.ostreamOutputId, istreamOutputId: response.outputId )) if container.replace != nil: pager.deleteContainer(container.replace, container.find(ndAny)) container.replace = nil else: dec pager.numload pager.deleteContainer(container, container.find(ndAny)) pager.refreshStatusMsg() proc unregisterFd(pager: Pager; fd: int) = pager.selector.unregister(fd) pager.loader.unregistered.add(fd) # true if done, false if keep proc handleConnectingContainer*(pager: Pager; i: int) = let item = pager.connectingContainers[i] let container = item.container let stream = item.stream case item.state of ccsBeforeResult: var r = stream.initPacketReader() var res: int r.sread(res) if res == 0: r.sread(item.outputId) inc item.state container.loadinfo = "Connected to " & $container.url & ". Downloading..." pager.onSetLoadInfo(container) # continue else: var msg: string r.sread(msg) if msg == "": msg = getLoaderErrorMessage(res) pager.fail(container, msg) # done pager.connectingContainers.del(i) pager.unregisterFd(int(item.stream.fd)) stream.sclose() of ccsBeforeStatus: var r = stream.initPacketReader() r.sread(item.status) inc item.state # continue of ccsBeforeHeaders: let response = newResponse(item.res, container.request, stream, item.outputId, item.status) var r = stream.initPacketReader() r.sread(response.headers) # done pager.connectingContainers.del(i) pager.unregisterFd(int(item.stream.fd)) let redirect = response.getRedirect(container.request) if redirect != nil: stream.sclose() pager.redirect(container, response, redirect) else: pager.connected(container, response) proc handleConnectingContainerError*(pager: Pager; i: int) = let item = pager.connectingContainers[i] pager.fail(item.container, "loader died while loading") pager.unregisterFd(int(item.stream.fd)) item.stream.sclose() pager.connectingContainers.del(i) proc metaRefresh(pager: Pager; container: Container; n: int; url: URL) = let ctx = pager.jsctx let fun = ctx.newFunction(["url", "replace"], "pager.gotoURL(url, {replace: replace})") let args = [ctx.toJS(url), ctx.toJS(container)] discard pager.timeouts.setTimeout(ttTimeout, fun, int32(n), args) JS_FreeValue(ctx, fun) for arg in args: JS_FreeValue(ctx, arg) proc handleEvent0(pager: Pager; container: Container; event: ContainerEvent): bool = case event.t of cetLoaded: dec pager.numload of cetAnchor: let url2 = newURL(container.url) url2.setHash(event.anchor) pager.dupeBuffer(container, url2) of cetNoAnchor: pager.alert("Couldn't find anchor " & event.anchor) of cetReadLine: if container == pager.container: pager.setLineEdit(lmBuffer, event.value, hide = event.password, extraPrompt = event.prompt) of cetReadArea: if container == pager.container: var s = event.tvalue if pager.openInEditor(s): pager.container.readSuccess(s) else: pager.container.readCanceled() of cetReadFile: if container == pager.container: pager.setLineEdit(lmBufferFile, "") of cetOpen: let url = event.request.url let sameScheme = container.url.scheme == url.scheme if event.request.httpMethod != hmGet and not sameScheme and not (container.url.scheme in ["http", "https"] and url.scheme in ["http", "https"]): pager.alert("Blocked cross-scheme POST: " & $url) return #TODO this is horrible UX, async actions shouldn't block input if pager.container != container or not event.save and not container.isHoverURL(url): pager.ask("Open pop-up? " & $url).then(proc(x: bool) = if x: discard pager.gotoURL(event.request, some(container.url), referrer = pager.container, save = event.save) ) else: let url = if event.url != nil: event.url else: event.request.url discard pager.gotoURL(event.request, some(container.url), referrer = pager.container, save = event.save, url = url) of cetStatus: if pager.container == container: pager.showAlerts() of cetSetLoadInfo: if pager.container == container: pager.onSetLoadInfo(container) of cetTitle: if pager.container == container: pager.showAlerts() pager.term.setTitle(container.getTitle()) of cetAlert: if pager.container == container: pager.alert(event.msg) of cetCancel: let i = pager.findConnectingContainer(container) if i == -1: # whoops. we tried to cancel, but the event loop did not favor us... # at least cancel it in the buffer container.remoteCancel() else: let item = pager.connectingContainers[i] dec pager.numload pager.deleteContainer(container, container.find(ndAny)) pager.connectingContainers.del(i) pager.unregisterFd(int(item.stream.fd)) item.stream.sclose() of cetMetaRefresh: let url = event.refreshURL let n = event.refreshIn case container.config.metaRefresh of mrNever: assert false of mrAlways: pager.metaRefresh(container, n, url) of mrAsk: let surl = $url if surl in pager.refreshAllowed: pager.metaRefresh(container, n, url) else: pager.ask("Redirect to " & $url & " (in " & $n & "ms)?") .then(proc(x: bool) = if x: pager.refreshAllowed.incl($url) pager.metaRefresh(container, n, url) ) return true proc handleEvents*(pager: Pager; container: Container) = while container.events.len > 0: let event = container.events.popFirst() if not pager.handleEvent0(container, event): break proc handleEvents*(pager: Pager) = if pager.container != nil: pager.handleEvents(pager.container) proc handleEvent*(pager: Pager; container: Container) = try: container.handleEvent() pager.handleEvents(container) except IOError: discard proc addPagerModule*(ctx: JSContext) = ctx.registerType(Pager)