On this page:
Qi:   An Embeddable Flow-Oriented Language
8.16.0.1

Qi: An Embeddable Flow-Oriented Language🔗ℹ

 (require qi) package: qi-lib

An embeddable, general-purpose language to allow convenient framing of programming logic in terms of functional flows. A flow is a function from inputs to outputs, and Qi provides compact notation for describing complex flows.

Tired of writing long functional pipelines with nested syntax like this?

(map f (filter g (vector->list my-awesome-data)))

Then Qi is for you!

(~> (my-awesome-data) vector->list (filter g) (map f))

But wait, there’s more: Qi isn’t just a turbo-charged threading language. It supports multiple values and a suite of other operators for describing computations:
(define-flow average
  (~> (-< + count) /))

Start by getting your bearings. For an overview of the language, continue to Introduction and Usage. For a thorough orientation, watch the original video from RacketCon 2021.

    1 Using These Docs

    2 Introduction and Usage

      2.1 Overview

      2.2 Installation

      2.3 Using Qi

      2.4 Using Qi as a Dependency

      2.5 Relationship to the Threading Macro

    3 Tutorial

      3.1 Interactive Tutorial

        3.1.1 Installation

        3.1.2 Setup

          3.1.2.1 DrRacket

          3.1.2.2 Emacs

          3.1.2.3 Vim

        3.1.3 Starting the Tutorial

      3.2 Online Tutorial

    4 Language Interface

      4.1 Using Qi from the Host Language

        4.1.1 Core

        4.1.2 Threading

        4.1.3 Conditionals

        4.1.4 Lambdas

        4.1.5 Definitions

      4.2 Using the Host Language from Qi

        4.2.1 Using Racket Values in Qi Flows

        4.2.2 Using Racket to Define Flows

        4.2.3 Using Racket Macros as Flows

      4.3 Using Qi with Another DSL

        4.3.1 Using Qi Directly

        4.3.2 Using a Macro Bridge

        4.3.3 Writing a Qi Dialect

    5 The Qi Language

      5.1 Syntax

      5.2 Basic

      5.3 Predicates

      5.4 Boolean Algebra

      5.5 Routing

      5.6 Conditionals

      5.7 Loops

      5.8 Exceptions

      5.9 Higher-order Flows

      5.10 Binding

        5.10.1 Variable Scope

      5.11 Identifiers

      5.12 Literals

      5.13 Templates and Partial Application

      5.14 Utilities

      5.15 Language Extension

    6 List Operations

      6.1 Producers

      6.2 Transformers

      6.3 Consumers

    7 Qi Macros

      7.1 Defining Macros

      7.2 Using Macros

        7.2.1 Racket Version Compatibility

      7.3 Adding New Language Features

        7.3.1 Write Yourself a Maybe Monad for Great Good

        7.3.2 Translating Foreign Macros

      7.4 Writing Languages in Qi

        7.4.1 Embedded Languages

        7.4.2 Hosted Languages

        7.4.3 Embedding a Hosted Language

          7.4.3.1 Exercise: Pattern Matching

    8 Field Guide

      8.1 Writing Flows

        8.1.1 Start by Drawing a Circuit Diagram

        8.1.2 Use Small Building Blocks

        8.1.3 Carry Your Toolbox

        8.1.4 Separate Effects from Other Computations

      8.2 Debugging

        8.2.1 Using Side Effects

        8.2.2 Using a Probe

        8.2.3 Using Fixtures

        8.2.4 Common Errors and What They Mean

          8.2.4.1 Expected Number of Values Not Received

          8.2.4.2 Wildcard Not Allowed as an Expression

          8.2.4.3 ~@ Not Allowed as an Expression

          8.2.4.4 Bad Syntax

          8.2.4.5 Use Does Not Match Pattern

          8.2.4.6 Expected Identifier Not Starting With Character

          8.2.4.7 Identifier’s Binding is Ambiguous

          8.2.4.8 Not Defined as Syntax Class

          8.2.4.9 Too Many Ellipses in Template

          8.2.4.10 Syntax: Unbound Identifier

          8.2.4.11 Undefined

          8.2.4.12 map/filter Contract Violation

          8.2.4.13 Compose: Contract Violation

          8.2.4.14 List Arity Mismatch

          8.2.4.15 Fancy-app Arity Mismatch

          8.2.4.16 Application: Not a Procedure

        8.2.5 Gotchas

          8.2.5.1 null is Not a Literal

          8.2.5.2 There’s No Escaping esc

          8.2.5.3 Mutable Values Defy the Laws of Flows

          8.2.5.4 Order of Effects

      8.3 Effectively Using Feedback Loops

        8.3.1 Control Values and Data Values

        8.3.2 Input Tracing

        8.3.3 Keeping It Tidy

      8.4 Idioms and Transforms

        8.4.1 Nested Applications are Sequential Flows

        8.4.2 Converting a Function to a Closure

          8.4.2.1 Basic Recipe

          8.4.2.2 Definition vs Invocation Inputs

        8.4.3 Converting a Macro to a Flow

        8.4.4 Bindings are an Alternative to Nonlinearity

    9 Principles of Qi

      9.1 What is a Flow?

      9.2 Values and Flows

      9.3 Flows as Graphs

      9.4 Values are Not Collections

      9.5 Counting Flows

      9.6 Flowy Logic

      9.7 Phrases

      9.8 Identities

      9.9 Flows and Arrows

      9.10 It’s Languages All the Way Down

      9.11 The Qi Core Language

    10 When Should I Use Qi?

      10.1 Hadouken!

        10.1.1 Super Smush Numbers

        10.1.2 Root-Mean-Square

      10.2 The Science of Deduction

        10.2.1 Compound Predicates

        10.2.2 abs

        10.2.3 Range

        10.2.4 Length

      10.3 Don’t Stop Me Now

      10.4 Curbing Curries and Losing Lambdas

      10.5 The Value in Values

      10.6 Making the Switch

      10.7 The Structure and Interpretation of Flows

      10.8 Using the Right Tool for the Job

    11 Input Methods

      11.1 Unicode Support

      11.2 DrRacket

      11.3 Vim/Emacs

        11.3.1 Keybindings

        11.3.2 Indentation