Reactivity reading

Modern frontend frameworks

Self-adjusting computation (SAC)

Self-adjusting computation is a framework for incremental computation, where the goal is to efficiently recompute a program's output across changes to its input by reusing unchanged intermediate results.

  • Self adjusting computation papers
    • (there are many good papers here, but the original dissertation and the imperative one are recommended)
  • Incremental (talk, library docs). A production-quality OCaml library for SAC, which tackles a ton of performance and usability issues not dealt with in the academic literature. Highlights:
    • GC safety without manual disposal or finalizers, by making sure "unnecessary" nodes (those that aren't currently being transitively observed) aren't retained by the framework.
    • A clever way of efficiently maintaining a topological ordering on a dependency graph that can change dynamically, assigning each node a height that can increase but never decrease.
    • Dynamic dependencies are opt-in, which they found important for performance (no tracking overhead when rerunning nodes whose dependencies won't ever change).
  • Adapton (website, paper): rust, ocaml, and even python versions.
    • Nominal Adapton introduces "names" as a mechanism to reuse previous results even if the structure of the computation changes and shuffles things around.
    • Fungi (paper) provides a type system for Nominal Adapton, helpful because the restrictions on the correct use of names are rather tricky.
  • Salsa (docs, talk) the reactive system powering some of Rust's IDE support.
    • Doing this in Rust has some complications (have to declare interfaces for everything) and some neat benefits (macros; automatically deriving traits; letting the type system forbid setting a signal from within a computation).
    • Everything is declared up front (no closures or anonymous computations) so serialization is on the table.
  • Anchors (explainer post)
    • Tries to combine some of the implementation tricks of Incremental and Adapton. Offers a straightforward explanation of the design space with illustrations.

The most widely applicable takeaway from this line of research is a definition of "from-scratch consistency" as the notion of correctness for incremental computation.

Build systems

Somewhat like self-adjusting computation, a build system (think: make) is concerned with reusing past work to deliver a final set of outputs.

  • Build Systems à la Carte (paper, talk, slides). Develops a pluggable framework for decomposing the design space of build systems (defined very loosely: Excel is a build system in their framework) into orthogonal choices. The two major choices they consider:
    • Deciding what order to rerun/recheck tasks (the "scheduler")
    • Determining whether or not a given task needs to be rerun (the "rebuilder").

Many concepts here are cross-applicable to other types of reactivity, and the idea of a pluggable framework is very powerful for unifying different systems into a broader umbrella.

Functional Reactive Programming (FRP)

FRP is a broad class of dataflow-centered approaches to interactive/reactive programming. It treats inputs and outputs as streams of events rather than as individual values, and provides combinators for producing, consuming, and transforming such streams. FRP systems often (but not always) include a distinction between time-varying "signals" (or "behaviors") whose current value can be observed at any time, and "events" which occur only at discrete times.

FRP systems -- as expected of a stream-processing formalism -- allow future outputs to depend on the entire history of the program, not just the current inputs.

Implicits & scope-respecting operations

Practical reactive systems sometimes include features that involve propagating information down or up the nesting hierarchy, such as Context, Suspense, or error handlers. Reactivity brings some unique complications here, but the non-reactive versions of these features are reasonably well-studied and can provide a behavioral guideline.

Algebraic Effects

Might be tangential; might be very relevant if JS had algebraic effects as a feature. The concept can nevertheless guide certain bits of reactive system design, if you restrict yourself to "immediately tail-resumptive" effects (that don't require suspending the call stack to resume it later).