# Architecture of Chawan This document describes some aspects of how Chawan works. **Table of contents** * [Module organization](#module-organization) * [Process model](#process-model) * [Main process](#main-process) * [Forkserver](#forkserver) * [Loader](#loader) * [Buffer](#buffer) * [Opening buffers](#opening-buffers) * [Parsing HTML](#parsing-html) * [JavaScript](#javascript) * [General](#general) * [JS in the pager](#js-in-the-pager) * [JS in the buffer](#js-in-the-buffer) * [Styling](#styling) * [Layout](#layout) * [Rendering](#rendering) ## Module organization Explanation for the separate directories found in `src/`: * bindings: bindings for various C libraries. (Not wrappers, just the plain C API as Nim procedure prototypes.) * config: configuration-related code. Mainly parsers for config files. * css: styling-related code; CSS parsing and cascading. * html: DOM building, DOM functions, the DOM itself, forms, etc. Note that it does not include the [HTML parser](https://git.sr.ht/~bptato/chame) itself. * img: image-related code. Mostly useless because we can't draw images to the screen yet. (One day...) * io: code for IPC, interaction with the file system, etc. * js: wrappers for QuickJS. * layout: the layout engine and its renderer. * loader: code for the file loader server (?). * local: code for the main process (i.e. the pager). * server: code for processes other than the main process, e.g. buffer and forkserver. (Why is loader not here? idk) * types: mainly definitions of data types and things I didn't know where to put. * utils: things I didn't know where to put part 2 Additionally, "adapters" of various protocols and file formats can be found in `adapter/`: * protocol: includes support for every protocol supported by Chawan. * format: HTML converters for various text-based file formats. * img: image decoders and encoders. In general, these just read and output RGBA data through standard I/O. (This is not very efficient, and therefore subject to change.) ## Process model Described as a tree: * cha (main process) - forkserver (forked immediately at startup) * loader - local CGI scripts * buffer(s) - mailcap processes (e.g. md2html, feh, ...) - editor (e.g. vi) ### Main process The main process runs code related to the pager. This includes processing user input, printing buffer contents to the screen, and managing buffers in general. The complete list of buffers is only known to the main process. Mailcap commands are executed by the main process. This depends on knowing the content type of the resource, so the main process also reads in all network headers of navigation responses before launching a buffer process. More on this in [Opening buffers](#opening-buffers). ### Forkserver For forking buffer and loader processes, a forkserver process is launched at the very beginning of every 'cha' invocation. The fork server is responsible for forking the loader process, and also buffer processes. We use a fork server for two reasons: 1. It helps clean up child processes when the main process crashes. (We open pipes between the main process and the fork server, and kill all child processes from the fork server on EOF.) 2. It allows us to start new processes without cloning the pager's entire address space. This reduces the impact of memory bugs somewhat, and also our memory usage. The fork server is not used for mailcap or CGI processes, because their address space is replaced by exec anyway. (Also, it would be slow.) ### Loader The loader process takes requests from the main process and the buffer processes. Then, depending on the scheme, it responds by performing one of the following steps: * `cgi-bin:` Start a CGI script, and read out its stdout into the response body. For the main contents of text-based buffers, it also saves the response into the cache. This is also used for schemes like http/s, ftp, etc. by internally rewriting them into the appropriate `cgi-bin:` URL. * `stream:` Do the same thing as above, but read from a file descriptor passed to the loader beforehand. This is used when stdin is a file, e.g. `echo test | cha`. It is also used for mailcap entries with an x-htmloutput field. * `cache:` Read the file from the cache. This is used by the pager for the "view source" operation, and by buffers in the rare situation where their initial character encoding guess proves to be incorrect and they need to rewind the source. The loader process distinguishes between clients (i.e the main process or buffers) through client keys. In theory this should help against rogue clients, though in practice it is still trivial to crash the loader as a client. It also helps us block further requests from buffers that have been discarded by the pager, but still haven't found out yet that their life time has ended. ### Buffer Buffer processes parse HTML, optionally query external resources from loader, run styling, JS, and finally render the page to an internal canvas. Buffers are managed by the pager through Container objects. A UNIX domain socket is established between each buffer and the pager to enable communication between them. ## Opening buffers Scenario: the user attempts to navigate to . 1. pager creates a new container for the target URL. 2. pager sends a request for "https://example.org" to the loader. Then, it registers the file descriptor in its selector, and does something else until select() reports activity on the file descriptor. 3. loader rewrites "https://example.org" into "cgi-bin:http". It then runs the http CGI script with the appropriate environment variables set. 4. The http CGI script opens a connection to example.org. When connected, it starts printing out headers it receives to stdout. 5. loader parses these headers, and sends them to pager. 6. pager reads in the headers, and decides what to do based on the Content-Type. * If Content-Type is found in mailcap, then the command in that mailcap entry is executed, with the response body dup2'd onto its stdin. If the entry has x-htmloutput, then the command's stdout is taken instead of the response body, and Content-Type is set to text/html. Otherwise, the container is discarded. * If Content-Type is text/html, then a new buffer process is created, which then parses the response body as HTML. If it is any `text/*` subtype, then the response is simply inserted into a `` tag. * If Content-Type is not a `text/*` type, and no mailcap entry for it is found, then the user is prompted about where they wish to save the file. ## Parsing HTML The character decoder and the HTML parser are implementations of the WHATWG standard, and are available as [separate](https://git.sr.ht/~bptato/chagashi) [libraries](https://git.sr.ht/~bptato/chame). The decoding and parsing of HTML documents happens in buffer processes. This operation is asynchronous; when bytes from the network are exhausted, the buffer will 1) partially render the current document as-is, 2) return it to the pager so that the user can interact with the document. Character encoding detection is rather primitive; the list specified in `encoding.document-charset` is enumerated until either no errors are produced by the decoder, or no more charsets exist. In some edge cases, the document must be (and is) re-downloaded from the cache, but this pretty much never happens in real-world scenarios. (The most common case is that the UTF-8 validator just runs through the entire document without reporting errors.) The HTML parser then consumes the input buffer, which on the happy path (valid UTF-8) is just whatever we pulled from the network as-is. In some cases, a script calls document.write and then the parser is called re-entrantly. (Debugging this is not very fun.) ## JavaScript QuickJS is used by both the pager as a scripting language, and by buffers for running on-page scripts when JavaScript is enabled. The core JS related functionality can be found in the js/ module. (One day I hope to split it out into a standalone library.) ### General To avoid having to type out all the type conversion & error handling code manually, we have JS pragmas to automagically turn Nim procedures into JavaScript functions. An explanation of what these pragmas are & what they do can be found in the header of js/javascript.nim. The type conversion itself is handled by the overloaded toJS function and the generic fromJS function. toJS returns a JSValue, the native data type of QuickJS. fromJS returns a Result[T, JSError], which is interpreted as follows: * ok(T) is successful conversion. * err(JSError) is an error in the conversion. * ok(nil) for reference types is null. For non-nullable types, null is ok(none(T)). * err(nil) is JS_EXCEPTION, i.e. an exception has been thrown and is being propagated. An additional point of interest is reference types: ref types registered with the registerType macro can be freely passed to JS, and the function- defining macros set functions on their JS prototypes. When a ref type is passed to JS, a shim JS object is associated with the Nim object, and will remain in memory until neither Nim nor JS has references to it. Effectively, this means that you can expose Nim objects to JS and take Nim objects as arguments through the automagical .jsfunc pragma (& friends) without having to bother with error-prone manual reference counting. How this is achieved is detailed below. (You generally don't need the following info unless you're debugging the JS type conversion logic, in which case I offer my condolences.) In fact, there is a complication in this system: QuickJS has a reference- counting GC, but Nim also has a reference-counting GC. Associating two objects that are managed by two separate GCs is problematic, because even if you can freely manage the references on both objects, you now have a cycle that only a cycle collector can break up. A cross-GC cycle collector is obviously out of question; then it would be easier to just replace the entire GC in one of the runtimes. So instead, we hook into the QuickJS cycle collector (through a custom patch). Every time a JS companion object of a Nim object would be freed, we first check if the Nim object still has references from Nim, and if yes, prevent the JS object from being freed by "moving" a reference to the JS object (i.e. unref Nim, ref JS). Then, if we want to pass the object to JS again, we add no references to the JS object, only to the Nim object. By this, we "moved" the reference back to JS. This way, the Nim cycle collector can destroy the object without problems if no more references to it exist. But also, if you set some properties on the JS companion object, it will remain even if no more references exist to it in JS for some time, only in Nim. i.e. this works: ```js document.querySelector("html").canary = "chirp"; console.log(document.querySelector("html").canary); /* chirp */ ``` ### JS in the pager Keybindings can be assigned JavaScript functions in the config, and then the pager executes those when the keybindings are pressed. Also, contents of the start.startup-script option are executed at startup. This is used when `cha` is called with the `-r` flag. There *is* an API, described at [api.md](api.md). Web APIs are exposed to pager too, but you cannot operate on the DOMs themselves from the pager, unless you create one yourself with DOMParser.parseFromString. [config.md](config.md) describes all commands that are used in the default config. ### JS in the buffer The DOM is implemented through the same wrappers as those in pager. (Obviously, the pager modules are not exposed to buffer JS.) Aside from document.write, it is mostly straightforward, and usually works OK, though too many things are missing to really make it useful. As for document.write: don't ask. It works as far as I can tell, but I wouldn't know why. ## Styling css/ contains everything related to styling: CSS parsing and cascading. The parser is not very interesting, it's just an implementation of the CSS 3 parsing module. The latest iteration of the selector parser is pretty good. The media query parser is horrible and should be rewritten. And the CSS value parser works OK, but is missing features like variables. Cascading is slow, though it could be slower. Chawan has style caching, so re-styles are normally very fast. Also, a hash map is used for reducing initial style calculation times. However, we don't have a Bloom filter yet. ## Layout Layout can be found in the layout/ module. It has some problems: * CSS was designed for pixel-based displays, not for character-based ones. So we have to round a lot, and sometimes this goes wrong. (This is mostly solved by some basic heuristics inside the layout engine.) * Some (now) commonly used features like grid are not implemented, so websites using those look ugly. * It's slow on large documents, because we don't have partial layouting capabilities. A good layout engine would be able to skip layouting of contents before/after the currently visible part of the screen, especially in plain text files. Sadly we don't have this functionality; everything is laid out again from scratch every time we may have to re-layout. Our layout engine is a rather simple procedural layout implementation. It runs in two passes. In the first pass, it generates the layout tree; this is important because rules for generating anonymous boxes are surprisingly involved. (Specifically, anonymous inline box handling is kind of a mess.) The second pass then does the actual arrangement of the boxes on the screen. The output tree uses relative coordinates; that is, every box is positioned relative to its parent. Layout is fully recursive. This means that after a certain nesting depth, the buffer will run out of stack space and promptly crash. ### Rendering After layout is finished, the document is rendered onto a text-based canvas, which is represented as a sequence of strings associated with their formatting. Again, the entire document is rendered, which is the main reason why Chawan performs poorly on large documents. The positive side of this is that search is very simple (and fast), since we are just running regexes over a linear sequence of strings.