diff --git a/qi-doc/scribblings/field-guide.scrbl b/qi-doc/scribblings/field-guide.scrbl index 53a1726c..d0212f6c 100644 --- a/qi-doc/scribblings/field-guide.scrbl +++ b/qi-doc/scribblings/field-guide.scrbl @@ -31,9 +31,9 @@ A journeyman of one's craft -- a woodworker, electrician, or a plumber, say -- a @subsection{Separate Effects from Other Computations} -In functional programming, "effects" refer to anything a function does that is not captured in its inputs and outputs. This could include things like printing to the screen, writing to a file, or mutating a global variable. +In functional programming, @deftech{effects} refer to anything a function does that is not captured in its @tech{inputs} and @tech{outputs}. This could include things like printing to the screen, writing to a file, or mutating a global variable. -In general, pure functions (that is, functions free of such effects) are easier to understand and easier to reuse, and favoring their use is considered good functional style. But of course, it's necessary for your code to actually do things besides compute values, too! There are many ways in which you might combine effects and pure functions, from mixing them freely, as you might in Racket, to extracting them completely using monads, as you might in Haskell. Qi encourages using pure functions side by side with what we could call "pure effects." +In general, pure functions (that is, functions free of such effects) are easier to understand and easier to reuse, and favoring their use is considered good functional style. But of course, it's necessary for your code to actually do things besides compute values, too! There are many ways in which you might combine effects and pure functions, from mixing them freely, as you might in Racket, to extracting them completely using monads, as you might in Haskell. Qi encourages using pure functions side by side with what we could call @deftech{pure effects}. If you have a function with ordinary inputs and outputs that also performs an effect, then, to adopt this style, decouple the effect from the rest of the function (@seclink["Use_Small_Building_Blocks"]{splitting it into smaller functions}, as necessary) and then invoke it via an explicit use of the @racket[effect] form, thus neatly separating the functional computation from the effect. @@ -99,10 +99,12 @@ To use it, first wrap the entire expression @emph{invoking} the flow with a @rac @defform[(probe flo)] @defidform[readout] )]{ - @racket[probe] simply marks a @tech{flow} invocation for debugging, and does not change its functionality. Then, when evaluation encounters the first occurrence of @racket[readout] within @racket[flo], the values at that point are immediately returned as the value of the entire @racket[flo]. This is done via a @tech/reference{continuation}, so that you may precede it with whatever flows you like that might help you understand what's happening at that point, and you don't have to worry about it affecting downstream flows during the process of debugging since those flows would simply never be hit. Additionally, readouts may be placed @emph{anywhere} within the flow, and not necessarily on the main stream -- it will always return the values observed at the specific point where you place the readout. + @racket[probe] on its own simply marks a @tech{flow} invocation for debugging, and does not change its functionality. Then, when evaluation encounters the first occurrence of @racket[readout] within @racket[flo], the values at that point are immediately returned as the value of the entire @racket[flo]. This is done via a @tech/reference{continuation}, so that you may precede it with whatever flows you like that might help you understand what's happening at that point, and you don't have to worry about it affecting downstream flows during the process of debugging since those flows would simply never be hit. Additionally, readouts may be placed @emph{anywhere} within the flow, and not necessarily on the main stream -- it will always return the values observed at the specific point where you place the readout. Note that @racket[probe] is a Racket (rather than Qi) form, and it must wrap a flow @emph{invocation} rather than a flow @emph{definition}. The @racket[readout], on the other hand, is a Qi expression and must be placed somewhere within the flow @emph{definition}. + Finally, it's important to know that placing a readout represents a change to the program, and it could mean that the running program you are observing is subtly different from the original one. See @secref["Schrodinger_s_Probe"] to understand this phenomenon. + @racketblock[ (~> (5) sqr (* 2) add1) (probe (~> (5) readout sqr (* 2) add1)) @@ -411,17 +413,99 @@ So in general, use mutable values with caution. Such values can be useful as sid @subsubsection{Order of Effects} - Qi @tech{flows} may exhibit a different order of effects (in the @seclink["Separate_Effects_from_Other_Computations"]{functional programming sense}) than equivalent Racket functions. +In general, the behavior of @emph{pure} (in the @seclink["Separate_Effects_from_Other_Computations"]{functional programming sense}) Qi @tech{flows} is the same as that of equivalent Racket expressions, but effectful flows may exhibit a different order of effects. Consider the Racket expression: @racket[(map sqr (filter odd? (list 1 2 3 4 5)))]. As this invokes @racket[odd?] on all of the elements of the input list, followed by @racket[sqr] on all of the elements of the intermediate list, if we imagine that @racket[odd?] and @racket[sqr] print their inputs as a side effect before producing their results, then executing this program would print the numbers in the sequence @racket[1,2,3,4,5,1,3,5]. -The equivalent Qi flow is @racket[(~> ((list 1 2 3 4 5)) (filter odd?) (map sqr))]. As this sequence is @seclink["Don_t_Stop_Me_Now"]{deforested by Qi's compiler} to avoid multiple passes over the data and the memory overhead of intermediate representations, it invokes the functions in sequence @emph{on each element} rather than @emph{on all of the elements of each list in turn}. The printed sequence with Qi would be @racket[1,1,2,3,3,4,5,5]. +The equivalent Qi flow is @racket[(~>> ((list 1 2 3 4 5)) (filter odd?) (map sqr))]. As this sequence is @seclink["Don_t_Stop_Me_Now"]{deforested by Qi's compiler} to avoid multiple passes over the data and the memory overhead of intermediate representations, it invokes the functions in sequence @emph{on each element} rather than @emph{on all of the elements of each list in turn}. The printed sequence with Qi would be @racket[1,1,2,3,3,4,5,5]. + +Yet, in this case, either implementation produces the same output: @racket[(list 1 9 25)]. Often, as we see here, exhibiting a different order of effects does not make a difference to the @tech{output} of the program. + +But in the case where the output of such effectful flows is dependent on those effects (such as by incorporating mutable state), these flows could produce different output than otherwise equivalent Racket code, as this next example shows. + +@racketblock[ + (define add-count + (let ([v 0]) + (lambda (arg) + (set! v (+ v 1)) + (+ arg v)))) + + (~>> ((list 1 2 3)) (filter odd?) (map add-count) (map add-count)) +] + +Here, the unoptimized program would be equivalent to: + +@racketblock[ + ((lambda (lst) + (map add-count + (map add-count + (filter odd? lst)))) + (list 1 2 3)) +] + +… which produces the output @racket[(list 5 9)]. + +The optimized program deforests the sequence of functional operations, interleaving the effects (as discussed above), producing a different result, @racket[(list 4 10)]. + +From the perspective of Qi, such programs are poorly defined, and it is better to @seclink["Designing_Effects"]{design effects} to avoid such nonlocal interactions. If you'd like to employ such effects all the same, it would be advisable to write the program in Racket, encapsulating such behavior in a flow that could be used at a higher level without nonlocal effects, for instance @seclink["Using_Racket_to_Define_Flows"]{by using a wrapping @racket[esc]}. A flow specified with @racket[esc] follows Racket's order of effects since it is a Racket program. + +See @secref["Effect_Locality"] for more insights into Qi's handling of effects and its implications for the design of effects in your flows. + +@subsubsection{Schrodinger's Probe} + +Another curious thing to watch out for is that use of the @seclink["Using_a_Probe"]{probe debugger} can affect the @seclink["Order_of_Effects"]{order of effects}, as it could suppress optimizations that would otherwise be performed if the @tech{flow} were unobserved. + +Consider this example: + +@racketblock[ +(define-flow foo + (~> (pass odd?) (>< sqr))) +] + +This program would be optimized by the Qi compiler to: -Yet, either implementation produces the same output: @racket[(list 1 9 25)]. +@racketblock[ + (>< (if odd? sqr ⏚)) +] + +If we placed a readout here: + +@racketblock[ +(define-flow foo + (~> (pass odd?) readout (>< sqr))) +] + +… then for an input list @racket[(list 1 2 3)], the readout would show @racket[(list 1 3)]. But in the optimized program above, the readout would not even represent a valid point in the program (where should it be placed?). Thus, the readout is showing values that are flowing in the original program rather than the one that would actually have been executed in the absence of the readout. + +Yet, the optimization (@seclink["Qi_s_Guarantee_on_Effects"]{by requirement}) does not change the meaning of the program in the absence of effects, and so the actual @tech{output} of the program is consistent with the intermediate values that are read out. + +If there are @tech{effects} present, however, then the situation gets more spooky. + +The first program would look something like this: + +@racketblock[ +(define-flow foo + (~> (pass (effect E₁ odd?))) (>< (effect E₂ sqr))) +] + +… where all the effects E₁ would happen before any of the effects E₂. And the second program would look like: + +@racketblock[ + (>< (if (effect E₁ odd?) (effect E₂ sqr) ⏚)) +] + +… where the effects E₁ and E₂ would be interleaved. Though it changes the order of effects, the optimization is still valid because it preserves @tech{well-ordering}. The second program here represents what will actually be executed when the first program is written. + +But what happens when we place a @racket[readout] in the source program, this time? + +@racketblock[ +(define-flow foo + (~> (pass (effect E₁ odd?))) readout (>< (effect E₂ sqr))) +] -So, to reiterate, while the behavior of @emph{pure} Qi flows will be the same as that of equivalent Racket expressions, effectful flows may exhibit a different order of effects. In the case where the output of such effectful flows is dependent on those effects (such as relying on a mutable global variable), these flows could even produce different output than otherwise equivalent (from the perspective of inputs and outputs, disregarding effects) Racket code. +Here, with the @racket[readout], the program once again would not be optimized, and thus, all the effects E₁ would occur first before the values are read out. Without the @racket[readout], the flow would be optimized, as we have just seen, and the effects E₁ and E₂ would be interleaved, so that the effects observed in the presence of the readout are different from what would be observed without it! -If you'd like to use Racket's order of effects in any flow, @seclink["Using_Racket_to_Define_Flows"]{write the flow in Racket} by using a wrapping @racket[esc]. +So it's important to bear in mind that one cannot observe a flow using @racket[probe] without changing the program being observed, a change which in some cases has no observable impact, and which in other cases (i.e. when there are effects involved) could be significant. But now that you understand this phenomenon, you can develop intuition for the nature of such changes, and how best to use the tool to find the answers you are looking for. @section{Effectively Using Feedback Loops} diff --git a/qi-doc/scribblings/principles.scrbl b/qi-doc/scribblings/principles.scrbl index 09afcdde..60e7a6d5 100644 --- a/qi-doc/scribblings/principles.scrbl +++ b/qi-doc/scribblings/principles.scrbl @@ -3,12 +3,28 @@ scribble-abbrevs/manual scribble/example racket/sandbox + scribble/core scribble-math @for-label[qi racket]] @(use-mathjax) +@; based on scribble-abbrevs/latex +@(define (definition term . defn*) + (make-paragraph plain + (list + (bold "Definition") + (element #f (list " (" (deftech term) "). ")) + defn*))) + +@(define (theorem . thm*) + (make-paragraph plain + (list + (bold "Theorem") + (element #f (list ". ")) + thm*))) + @title{Principles of Qi} After many patient hours meticulously crafting Qi flows, you may find that you seek a deeper understanding; insight into guiding principles and inner workings, so that you can hone your skills on firmer ground. @@ -21,7 +37,7 @@ A @deftech{flow} is either made up of flows, or is a native (e.g. Racket) @seclink["lambda" #:doc '(lib "scribblings/guide/guide.scrbl")]{function}. Flows may be composed using a number of combinators that could yield either linear or nonlinear composite flows. - A flow in general accepts @code{m} inputs and yields @code{n} outputs, for arbitrary non-negative integers @code{m} and @code{n}. We say that such a flow is @code{m × n}. + A flow in general accepts @code{m} @deftech{inputs} and yields @code{n} @deftech{outputs}, for arbitrary non-negative integers @code{m} and @code{n}. We say that such a flow is @code{m × n}. Inputs and outputs are ordinary @tech/reference{values}. The semantics of a flow is function invocation -- simply invoke a flow with inputs (i.e. ordinary arguments) to obtain the outputs. @@ -41,7 +57,7 @@ @section{Values are Not Collections} - The things that flow are values. Individual values may happen to be collections such as lists, but the values that are flowing are not, together, a collection of any kind. + The things that flow are @tech/reference{values}. Individual values may happen to be collections such as @tech/guide{lists}, but the values that are flowing are not, together, a collection of any kind. To understand this with an example: when we employ a tee junction in a @tech{flow}, colloquially, we might say that the junction "divides the flow into two," which might suggest that there are now two flows. But in fact, there is just one flow that divides @emph{values} down two separate flows which are part of its makeup. More precisely, @racket[-<] composes two flows to yield a single composite flow. Like any flow, this composite flow accepts values and produces values, not collections of values. There is no way to differentiate, at the output end, which values came from the first channel of the junction and which ones came from the second, since downstream flows have no idea about the structure of upstream flows and only see the values they receive. @@ -59,6 +75,171 @@ Consider this example: There are six @tech{flows} here, in all: the entire one, each component of the thread, and each component of the tee junction. +@section{Effect Locality} + +Qi programs provide weaker guarantees on @seclink["Order_of_Effects"]{order of effects} than do otherwise equivalent Racket programs. + +For instance, this Qi flow: + +@codeblock{ + (~>> (filter my-odd?) (map my-sqr)) +} + +is roughly equivalent to this Racket expression: + +@codeblock{ + (lambda (vs) + (map my-sqr + (filter my-odd? vs))) +} + +But if @racket[my-odd?] and @racket[my-sqr] exhibit any side effects, such as printing their inputs to the screen, then the behavior of these two expressions is not quite the same. Racket guarantees that @emph{all} of the @racket[my-odd?] effects will occur before @emph{any} of the @racket[my-sqr] effects. Qi provides a more minimal (and ultimately very simple) guarantee that can be summarized as (1) effects are only defined in association with function invocations, and (2) they will occur at the same time as the function invocation with which they are associated. We call this property "effect locality." + +To understand what this means, we will need to develop some concepts. + +@subsection{Functional Effects} + +First, regarding the definability of effects, as far as Qi is concerned, they are only well-defined in connection with some @tech{flow}, and are not independently conceivable. + +@definition["Associated effect"]{If a flow @${f} either includes an effect @${e} in its primitive definition or has one declared using @racket[effect], then @${e} is said to be an effect "on" @${f}. We denote an arbitrary effect on @${f} by @${ε(f)}.} + +In the case where we use the @racket[effect] form on its own as in @racket[(effect displayln)], the implicit associated function is the identity flow, @racket[_]. + +Note that @tech{effects} are a distinct concept from program @tech{inputs} and @tech{outputs}. + +@subsection{Upstream and Downstream} + +We already saw how a flow can be thought of as a @seclink["Flows_as_Graphs"]{directed graph}. This naturally suggests that some flows are upstream (or downstream) of others in terms of this directionality. Let's define this relation more precisely, as it will be useful to us. + +@definition["Upstream and downstream"]{A flow invocation @${f} is @deftech{upstream} of another invocation @${g} if the output of @${f} is @emph{necessary} to determining the input of @${g}. Conversely, @${g} is @deftech{downstream} of @${f}. We will denote this relation @${f < g}.} + +@; This relation, like the usual strict order relation on numbers, is +@; irreflexive (f isn't upstream of itself, not f < f), +@; asymmetric (f < g ⇒ not g < f) +@; and transitive (f < g and g < h ⇒ f < h) + +Note that this definition relates flow @emph{invocations} rather than @tech{flows} themselves. For now, we need not worry about this distinction, but we will soon see why it matters. + +In terms of the @tech{paths} that could be traced over a flow, the ordering implied by @racket[~>] naturally shows us many members of this relation: flows that come later in the sequence in a @racket[~>] form are downstream of those that come earlier, because the output of earlier flows is needed to determine the input to later flows. + +In the above example, @racket[filter] and @racket[map] are obviously ordered by @racket[~>] in this way, so that @racket[(filter my-odd?)] is upstream of @racket[(map my-sqr)]. But it's not so obvious how @racket[my-odd?] and @racket[my-sqr] should be treated. These are employed "internally" by the higher-order flows @racket[filter] and @racket[map], and are not directly ordered by the @racket[~>] form. Should @racket[my-odd?] be considered to be upstream of @racket[my-sqr] here? + +This is where the distinction between flows and flow invocations comes into play. In fact, not all invocations of @racket[my-odd?] are upstream of any particular invocation of @racket[my-sqr]. Rather, specific invocations of @racket[my-sqr] that use values computed by individual invocations of @racket[my-odd?] are downstream of those invocations, and notably, these invocations involve the individual elements of the input list rather than the entire list, so that the computational dependency expressed by this relation is as fine-grained as possible. + +For instance, for an input list @racket[(list 1 2 3)], @racket[(my-odd? 1)] is @tech{upstream} of @racket[(my-sqr 1)], and likewise, @racket[(my-odd? 3)] is @tech{upstream} of @racket[(my-sqr 3)], but @racket[(my-odd? 3)] is not upstream of @racket[(my-sqr 1)], and @racket[(my-odd? 2)] isn't upstream of anything. + +That brings us to the guarantee that Qi provides in this case (and in general). + +@subsection{Qi's Guarantee on Effects} + +@definition["Well-ordering"]{For @tech{flow} invocations @${f} and @${g} and corresponding effects @${ε(f)} and @${ε(g)}, + +@$${f \lt g \implies \epsilon(f) \lt \epsilon(g)} + +where @${<} on the left denotes the relation of being upstream, and @${<} on the right denotes one effect happening before another. Such effects are said to be @emph{well-ordered}. +} + +Well-ordering is defined in relation to a source program encoding the intended meaning of the flow, which serves as the point of reference for program translations. Qi guarantees that effects will remain well-ordered through any such translations of the source program that are undertaken during @seclink["It_s_Languages_All_the_Way_Down"]{optimization}. As we will soon see, this guarantee assumes, and prescribes, that effects employed in flows be @tech{local}. + +@definition["Effect locality"]{@tech{Effects} in a flow F are said to be @deftech{local} if the @tech{output} of F is invariant under all @tech{well-orderings} of effects. Specifically, if a @techlink[#:key "well-ordering"]{well-ordered} program translation causes a program to produce different @tech{output}, then the program contains @deftech{nonlocal} effects.} + +For example, effects that mutate shared state serving as the input to other flows are often nonlocal in this way. The section on @secref["Order_of_Effects"] elaborates on this example. + +We will discuss the practicalities of these in more detail shortly, but first, it's worth noting that although well-ordered effects seem natural for flows, the property does not necessarily hold under arbitrary program translations without an explicit compiler guarantee (as Qi provides). We can see this in terms of the underlying pure flow that is free of effects. + +@definition["Pure projection"]{The pure projection of a flow @${f} is @${f} with all effects removed. We'll denote this @${π(f)}. For flows @${f_{1}} and @${f_{2}}, @${π(f_{1})} is @emph{equivalent} to @${π(f_{2})} if they produce the same @tech{output} given the same @tech{input}.} + +@theorem{For a @tech{flow} @${f}, not every flow @${f′} such that @${π(f′)} is equivalent to @${π(f)} preserves @tech{well-ordering} of effects in relation to @${f}.} + +For instance, the compiler could accumulate all effects and execute them in an arbitrary order at the end of execution of the flow. For at least some subset of local effects (say, effects that simply print their inputs), the output remains the same even if the effects are not well-ordered. + +Locality of effects does not imply well-ordering of effects under program translation, nor vice versa – these are independent. + +In sum, @emph{Qi guarantees that the @tech{output} of execution of the compiled program is the same as that of the source program, assuming @tech{effects} are @tech{local}, and further, it guarantees that the effects will be @techlink[#:key "well-ordering"]{well-ordered} in relation to the source program.} + +This has a few implications of note. + +@subsubsection{The @racket[effect] Form} + +First, for the @racket[effect] form, this implies that Qi considers an effectful flow @${f} performing an effect @${e} to be indistinguishable from a flow @racket[(effect e f′)], where @${f′} is @${f} without @${e} (and therefore pure), and effects declared in this way will never be separated from the associated flows. Thus, Qi @seclink["Separate_Effects_from_Other_Computations"]{encourages writing pure functions} while preserving the intuitive association of effects with functions. + +@subsubsection{The @racket[esc] Form} + +Qi will not optimize a flow that is wrapped with @racket[esc]. Thus, such flows will exhibit @seclink["Racket_vs_Qi"]{Racket's order of effects} (which, as we'll discuss below, also satisfies the requirements of locality). + +For more on how @racket[esc] is handled, see @secref["Using_Racket_to_Define_Flows"]. + +@subsubsection{Truncating Effects} + +Next, for a flow like this one: + +@racketblock[ + (~>> (filter my-odd?) (map my-sqr) car) +] + +… when it is invoked with a large input list, Qi in fact @seclink["Don_t_Stop_Me_Now"]{only processes the very first value} of the list, since it determines, at the end, that no further elements are needed in order to generate the final result. This means that all effects on would-be subsequent invocations of @racket[my-odd?] and @racket[my-sqr] would simply not be performed. Yet, @tech{well-ordering} is preserved here, since the @techlink[#:key "well-ordering"]{defining implication} holds for every flow invocation that actually happens. Well-ordering is about effects being guided by the @emph{necessity} of @techlink[#:key "associated effect"]{associated} computations to the final result. + +@subsubsection{Independent Effects} + +For a nonlinear flow like this one: + +@racketblock[ + +(~>> (filter my-odd?) + (-< (map my-sqr __) + (map my-add1 __)) + (map my-*) + (foldl + 0)) +] + +… as invocations of neither @racket[my-sqr] nor @racket[my-add1] are @tech{upstream} of the other, there is no guarantee on the mutual order of effects either. For instance, the effects @techlink[#:key "associated effect"]{on} @racket[my-sqr] may happen first, or those @techlink[#:key "associated effect"]{on} @racket[my-add1] may happen first, or they may be interleaved. + +But both of these effects would occur before the one on the corresponding @racket[my-*] invocation, since this is @tech{downstream} of them both. + +@subsubsection{Designing Effects} + +The guarantee of @tech{well-ordering} of effects provided by the compiler represents a prescription for the design of @tech{effects} by users. + +As @secref["Order_of_Effects"] elaborates on, it's possible that the @tech{output} of effectful flows will differ from that of seemingly equivalent Racket programs. In fact, it's precisely in the cases where the program contains @tech{nonlocal} effects that the output could differ. + +From Qi's perspective, such effects are poorly defined as they are either too broadly scoped or likely have the scope of their operation diffused over multiple function invocations in a way that is not neatly captured by their composition. + +In general, Qi encourages designing your programs so that effects are @tech{local}. + +@subsubsection{A Natural Order of Effects} + +By being as fine-grained as possible in expressing computational dependencies, and in tying the execution of effects to such computational dependencies, @tech{well-ordering} is in some sense the minimum well-formed guarantee on effects, and a natural one for functional languages to provide. + +@subsection{Racket vs Qi} + +In the earlier example, reproduced here for convenience: + +@racketblock[ + (lambda (vs) + (map my-sqr + (filter my-odd? vs))) +] + +@racketblock[ + (~>> (filter my-odd?) (map my-sqr)) +] + +… with an input list @racket[(list 1 2 3)], Racket's order of effects follows the invocation order: + +@racketblock[ + (my-odd? 1) (my-odd? 2) (my-odd? 3) (my-sqr 1) (my-sqr 3) +] + +Qi's order of effects is: + +@racketblock[ + (my-odd? 1) (my-sqr 1) (my-odd? 2) (my-odd? 3) (my-sqr 3) +] + +Either of these orders @emph{satisfies} @tech{well-ordering}, but as we saw earlier, Racket guarantees something more than this minimum, or, in mathematical terms, Racket's guarantees on effects are @emph{stronger} than Qi's. + +In principle, this allows Qi to offer @seclink["Don_t_Stop_Me_Now"]{faster performance} in some cases. + @section{Flowy Logic} Qi's design is inspired by buddhist śūnyatā logic. To understand it holistically would require a history lesson to put the sunyata development in context, and that would be quite a digression. But in essence, sunyata is about transcension of context or viewpoint. A viewpoint is identifiable with a logical span of possibilities (@emph{catuṣkoṭi}) in terms of which assertions may be made. Sunyata is the rejection of @emph{all} of the available logical possibilities, thus transcending the very framing of the problem (this is signified by the word @emph{mu} in Zen). This kind of transcension could suggest alternative points of view, but more precisely, does not indicate a point of view (which isn't the same as being ambivalent or even agnostic). This idea has implications not just for formal logical systems but also for everyday experience and profound metaphysical questions alike. @@ -129,7 +310,7 @@ Therefore, any theoretical results about arrows should generally apply to Qi as @section{It's Languages All the Way Down} -Qi is a language implemented on top of another language, Racket, by means of a macro called @racket[flow]. All of the other macros that serve as Qi's @seclink["Embedding_a_Hosted_Language"]{embedding} into Racket, such as (the Racket macros) @racket[~>] and @racket[switch], expand to a use of @racket[flow]. +Qi is a language implemented on top of another language, Racket, by means of a @tech/reference{macro} called @racket[flow]. All of the other macros that serve as Qi's @seclink["Embedding_a_Hosted_Language"]{embedding} into Racket, such as (the Racket macros) @racket[~>] and @racket[switch], expand to a use of @racket[flow]. The @racket[flow] form accepts Qi syntax and (like any @tech/reference{macro}) produces Racket syntax. It does this in two stages: