aboutsummaryrefslogtreecommitdiffhomepage
path: root/docs/wiki_src/coding
diff options
context:
space:
mode:
Diffstat (limited to 'docs/wiki_src/coding')
-rw-r--r--docs/wiki_src/coding/IO.md29
-rw-r--r--docs/wiki_src/coding/REPL.md100
-rw-r--r--docs/wiki_src/coding/combinators.md0
-rw-r--r--docs/wiki_src/coding/compilation.md44
-rw-r--r--docs/wiki_src/coding/currying.md13
-rw-r--r--docs/wiki_src/coding/data-structures.md111
-rw-r--r--docs/wiki_src/coding/examples.md0
-rw-r--r--docs/wiki_src/coding/laziness.md52
-rw-r--r--docs/wiki_src/coding/meta-programming.md72
-rw-r--r--docs/wiki_src/coding/mixfix.md33
-rw-r--r--docs/wiki_src/coding/performance.md0
-rw-r--r--docs/wiki_src/coding/prefix.md25
-rw-r--r--docs/wiki_src/coding/recursion.md78
-rw-r--r--docs/wiki_src/coding/style.md36
-rw-r--r--docs/wiki_src/coding/test-driven-development.md42
-rw-r--r--docs/wiki_src/coding/typing.md0
-rw-r--r--docs/wiki_src/coding/uniform-function-call-syntax.md0
17 files changed, 635 insertions, 0 deletions
diff --git a/docs/wiki_src/coding/IO.md b/docs/wiki_src/coding/IO.md
new file mode 100644
index 0000000..974f98e
--- /dev/null
+++ b/docs/wiki_src/coding/IO.md
@@ -0,0 +1,29 @@
+# IO
+
+Bruijn supports a variant of John Tromp's monadic IO[^1].
+
+Every program's `main`{.bruijn} function has an additional abstraction
+that gets applied with a lazy list of input bytes. These bytes are
+encoded as the syntactic sugar encoding of binary numbers, which can be
+manipulated with [`std/Number/Binary`](/std/Number_Binary.bruijn.html).
+
+You can use [`std/Monad`](/std/Monad.bruijn.html) to interact with the
+input monadically, or simply use [`std/List`](/std/List.bruijn.html)
+operations to work with the input as a normal list.
+
+## Example
+
+``` bruijn
+:import std/List .
+
+# reverse the input list
+main [<~>0]
+```
+
+``` bash
+$ printf "tacocat" | bruijn reverse.bruijn
+tacocat
+```
+
+[^1]: [Tromp, John. "Functional Bits: Lambda Calculus based Algorithmic
+ Information Theory." (2023).](https://tromp.github.io/cl/LC.pdf)
diff --git a/docs/wiki_src/coding/REPL.md b/docs/wiki_src/coding/REPL.md
new file mode 100644
index 0000000..a801ef9
--- /dev/null
+++ b/docs/wiki_src/coding/REPL.md
@@ -0,0 +1,100 @@
+# REPL
+
+The REPL is a very helpful feature for functional programming languages
+like bruijn. You can use it to continuously test or execute parts of
+your code.
+
+You can start the REPL using `stack run`{.bash} or (if installed)
+`bruijn`{.bash}.
+
+Any valid term will get reduced to normal form after pressing enter.
+
+## Definitions
+
+Since everything you type will get evaluated, definitions (compared to
+definitions in files) require an equal sign:
+
+``` bruijn
+> id = [0]
+> id
+[0]
+```
+
+## Commands
+
+### `:import`{.bruijn}/`:input`{.bruijn}
+
+Equivalent to the [respective commands in
+files](../introduction/syntax.md#imports).
+
+``` bruijn
+> :import std/Math .
+```
+
+### `:test`{.bruijn}
+
+Equivalent to the [test command in
+files](../introduction/syntax.md#tests).
+
+``` bruijn
+> :test ([0]) ([[1]])
+ERROR test failed: [0] = [[1]]
+ reduced to [0] = [[1]]
+```
+
+### `:watch`{.bruijn}
+
+`:watch`{.bruijn} re-imports the file automatically after every saved
+change. It will rerun any test the watched file contains.
+
+``` bruijn
+> :watch collatz-proof
+```
+
+This can be very helpful for [test driven
+development](test-driven-development.md).
+
+### `:time`{.bruijn}
+
+Measures the time from start of reduction to its end (normal form) in
+seconds.
+
+``` bruijn
+> :time fac (+30)
+0.15 seconds
+```
+
+### `:blc`{.bruijn}
+
+Translates both the unreduced and the reduced expression to binary
+lambda calculus. Helpful for golfed [compilation](compilation.md).
+
+``` bruijn
+> :blc [0] [0]
+0100100010
+0010
+```
+
+### `:length`{.bruijn}
+
+Measures the length of the binary lambda calculus encoding of both the
+unreduced and the reduced expression. Helpful for golfed
+[compilation](compilation.md).
+
+``` bruijn
+> :blc [0] [0]
+0100100010
+0010
+```
+
+### `:free`{.bruijn}
+
+The `free` command frees the current environment including all defined
+identifiers and imported files.
+
+``` bruijn
+> id = [0]
+> :free
+> id
+ERROR undefined identifier id
+```
diff --git a/docs/wiki_src/coding/combinators.md b/docs/wiki_src/coding/combinators.md
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/docs/wiki_src/coding/combinators.md
diff --git a/docs/wiki_src/coding/compilation.md b/docs/wiki_src/coding/compilation.md
new file mode 100644
index 0000000..9863c21
--- /dev/null
+++ b/docs/wiki_src/coding/compilation.md
@@ -0,0 +1,44 @@
+# Compilation
+
+Bruijn can be compiled to John Tromp’s binary lambda calculus (BLC).
+
+BLC uses the following encoding:
+
+| term | lambda | bruijn | BLC |
+|:-------------|:-------|:--------|:-----------------|
+| abstraction | λM | `[M]` | 00M |
+| application | (MN) | `(M N)` | 01MN |
+| bruijn index | i | `i` | 1<sup>i+1</sup>0 |
+
+There are two modes of compilation:
+
+- **Bitwise** compiles to BLC and encodes every bit as 1 bit and pads
+ the last remaining byte: `bruijn -b path`
+- **ASCII** compiles to BLC and encodes every bit as 1 ASCII character
+ (`'0'`/`'1'`): `bruijn -B path`
+
+## Compilation overhead
+
+Typical compilation to BLC results in much redundant code, since every
+used function gets substituted and translated separately. In
+`((+3) + (+4) + (+3))`, for example, `add` gets compiled to BLC two
+times, resulting in a redundant overhead of around 3500 bits.
+
+This is because BLC was never intended for compilation of normal
+programs, but mainly as an academic encoding model. This also means that
+it’s quite good for writing very expressive and minimal programs
+(i.e. obfuscated code golfing, see [John Tromp’s
+IOCCC](https://ioccc.org/2012/tromp/hint.html)).
+
+Most programs, however, won’t be golfed and can result in rather large
+compiled programs. While there’s not really any practical need for
+compilation aside from golfing, you could still use the
+[BLoC](https://github.com/marvinborner/bloc) project to optimize
+redundant terms.
+
+Typical workflow:
+
+``` bash
+$ bruijn -B program.bruijn | bloc --from-blc -i - -o out.bloc
+$ cat input | bruijn -E <(bloc --from-bloc -i out.bloc)
+```
diff --git a/docs/wiki_src/coding/currying.md b/docs/wiki_src/coding/currying.md
new file mode 100644
index 0000000..d84eb0a
--- /dev/null
+++ b/docs/wiki_src/coding/currying.md
@@ -0,0 +1,13 @@
+# Currying
+
+Lambda calculus naturally supports currying -- that is, only partially
+applying a function. In fact *any* function can be applied with *any*
+amount of arguments!
+
+In bruijn, currying is a great way to make functions even more elegant.
+
+Partially applying the `mul`{.bruijn} function:
+
+``` bruijn
+six [0 (+3)] (mul (+2))
+```
diff --git a/docs/wiki_src/coding/data-structures.md b/docs/wiki_src/coding/data-structures.md
new file mode 100644
index 0000000..384ac29
--- /dev/null
+++ b/docs/wiki_src/coding/data-structures.md
@@ -0,0 +1,111 @@
+# Data structures
+
+Bruijn's standard library defines several common data structures.
+
+Relevant blog post: [Data structures in pure lambda
+calculus](https://text.marvinborner.de/2023-04-07-01.html).
+
+## States
+
+For storing states (i.e. enums), you can use the available libraries and
+syntactic sugar.
+
+### Booleans/bits [`std/Logic`](/std/Logic.bruijn.html)
+
+- typical [Church
+ booleans](https://en.wikipedia.org/wiki/Church_encoding#Church_Booleans)
+ -- fast and reliable
+- encoding: `true`{.bruijn}=`[[1]]`{.bruijn},
+ `false`{.bruijn}=`[[0]]`{.bruijn}
+
+### Unary numbers [`std/Number/Unary`](/std/Number_Unary.bruijn.html)
+
+- `u` suffix for syntactic sugar, e.g. `(+3u)`{.bruijn}
+- encoding: `(+4u)`{.bruijn}=`[[(1 (1 (1 (1 0))))]]`{.bruijn}
+- typical [Church
+ numerals](https://en.wikipedia.org/wiki/Church_encoding#Church_numerals),
+ simple but high space/time complexity
+- only positive numbers
+
+### Binary numbers [`std/Number/Binary`](/std/Number_Binary.bruijn.html)
+
+- `b` suffix for syntactic sugar, e.g. `(+3b)`{.bruijn}
+- encoding: `(+4b)`{.bruijn}=`[[[0 (0 (1 2))]]]`{.bruijn}
+- encoding for chars/strings, e.g. `'0'`{.bruijn}=`(+48b)`{.bruijn}
+- faster and more compact than unary
+- only positive numbers (excluding two's complement)
+
+### Balanced ternary [`std/Number/Ternary`](/std/Number_Ternary.bruijn.html)
+
+- default syntactic sugar for numbers (optional suffix `t`),
+ e.g. `(+3)`{.bruijn}
+- encoding: `(+4)`{.bruijn}=`[[[[(1 (1 3))]]]]`,
+ `(-4)`{.bruijn}=`[[[[(2 (2 3))]]]]`{.bruijn}
+- faster and more compact than binary[^1]
+- positive and negative numbers
+
+## Boxes [`std/Box`](/std/Box.bruijn.html)
+
+Boxes are good for storing single values as immutable object with an
+empty/full state.
+
+Example:
+
+``` bruijn
+a-box <>'a'
+
+:test (set? a-box) (true)
+:test (get 'b' a-box) ('a')
+:test (get 'b' empty) ('b')
+:test (store! a-box 'b') (<>'b')
+```
+
+## Pairs [`std/Pair`](/std/Pair.bruijn.html)
+
+Pairs (tuples) can store any two terms.
+
+Example:
+
+``` bruijn
+one-two (+1) : (+2)
+
+:test (^one-two) ((+1))
+:test (~one-two) ((+2))
+:test (inc <$> one-two) ((+2) : (+3))
+:test (uncurry add one-two) ((+3))
+```
+
+## Lists [`std/List`](/std/List.bruijn.html)
+
+Lists are a repeated composition (right-associative) of pairs with a
+`empty`{.bruijn} ending symbol `[[1]]`{.bruijn}. They can store any
+(heterogeneous) values and are recursively iterable. The call-by-need
+reduction order of bruijn allows lazy evaluation (i.e. infinite lists).
+
+Example:
+
+``` bruijn
+:test (length (take (+3) (repeat (+4)))) ((+3))
+:test (take (+5) (iterate ++‣ (+0))) (((+0) : ((+1) : ((+2) : ((+3) : {}(+4))))))
+:test ((foldr …+… (+0) ((+1) : ((+2) : {}(+3)))) =? (+6)) (true)
+```
+
+## Strings [`std/String`](/std/String.bruijn.html)
+
+Strings are just a list of binary encoded bytes. You may use
+[`std/List`](/std/List.bruijn.html) in combinatoin with
+[`std/Number/Binary`](/std/Binary.bruijn.html) to interact with them.
+
+Example:
+
+``` bruijn
+:test (lines "abc\ndef") ("abc" : {}"def")
+:test ("ab" =? "ab") (true)
+```
+
+[^1]: [Mogensen, Torben Æ. "An investigation of compact and efficient
+ number representations in the pure lambda calculus." Perspectives of
+ System Informatics: 4th International Andrei Ershov Memorial
+ Conference, PSI 2001 Akademgorodok, Novosibirsk, Russia, July 2--6,
+ 2001 Revised Papers 4. Springer Berlin Heidelberg,
+ 2001.](https://doi.org/10.1007/3-540-45575-2_20)
diff --git a/docs/wiki_src/coding/examples.md b/docs/wiki_src/coding/examples.md
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/docs/wiki_src/coding/examples.md
diff --git a/docs/wiki_src/coding/laziness.md b/docs/wiki_src/coding/laziness.md
new file mode 100644
index 0000000..feec480
--- /dev/null
+++ b/docs/wiki_src/coding/laziness.md
@@ -0,0 +1,52 @@
+# Laziness
+
+Due to the call-by-need reduction order of bruijn, several operations
+are lazily evaluated (automagically!).
+
+## Infinite lists
+
+You can use infinite list generators like `repeat`{.bruijn} or
+`iterate`{.bruijn} to lazily interact with list elements.
+
+``` bruijn
+:import std/List .
+
+:test (length (take (+3) (repeat (+4)))) ((+3))
+:test (take (+5) (iterate ++‣ (+0))) (((+0) : ((+1) : ((+2) : ((+3) : {}(+4))))))
+```
+
+## Math
+
+``` bruijn
+# power function
+:test ((iterate (…⋅… (+2)) (+1)) !! (+3)) ((+8))
+
+# prime numbers
+primes nub ((…≠?… (+1)) ∘∘ gcd) (iterate ++‣ (+2)) ⧗ (List Number)
+
+:test (take (+4) primes) ((+2) : ((+3) : ((+5) : {}(+7))))
+
+# fibonacci
+fibs head <$> (iterate [~0 : (^0 + ~0)] ((+0) : (+1))) ⧗ (List Number)
+
+:test (take (+4) primes) ((+0) : ((+1) : ((+1) : {}(+2))))
+```
+
+## Optimization
+
+Laziness can (in some cases) produce huge performance boosts. For
+example:
+
+``` bruijn
+# 11 seconds
+:time (+10) ** (+500)
+
+# 0.1 seconds
+:time ((+10) ** (+500)) =? (+400)
+```
+
+This works because a ternary number is just a list of trits which (in
+this case) gets recursively generated by the `pow`{.bruijn} function.
+The `eq?`{.bruijn} function just throws away the first argument if it's
+already clear that the numbers can't be equal (in this case after the
+first argument got bigger than `(+400)`{.bruijn}).
diff --git a/docs/wiki_src/coding/meta-programming.md b/docs/wiki_src/coding/meta-programming.md
new file mode 100644
index 0000000..44f5ea2
--- /dev/null
+++ b/docs/wiki_src/coding/meta-programming.md
@@ -0,0 +1,72 @@
+# Meta programming
+
+TODO: more (blog)
+
+Bruijn has a homoiconic meta encoding similar to Lisp's quoting feature.
+
+Blog post with more details: [Homoiconic self interpretation of lambda
+calculus](https://text.marvinborner.de/2023-09-03-21.html).
+
+## Encoding
+
+``` bruijn
+`X ⤳ [[[2 (+Xu)]]]
+`(M N) ⤳ [[[1 `M `N]]]
+`[M] ⤳ [[[0 `M]]]
+```
+
+Any quoted term gets converted to this encoding:
+
+``` bruijn
+# example quotations
+:test (`0) ([[[2 (+0u)]]])
+:test (`[0]) ([[[0 [[[2 (+0u)]]]]]])
+:test (`'0') ([[[0 [[[0 [[[0 [[[1 [[[2 (+0u)]]] [[[1 [[[2 (+0u)]]] [[[1 [[[2 (+0u)]]] [[[1 [[[2 (+0u)]]] [[[1 [[[2 (+1u)]]] [[[1 [[[2 (+1u)]]] [[[1 [[[2 (+0u)]]] [[[1 [[[2 (+0u)]]] [[[2 (+2u)]]]]]]]]]]]]]]]]]]]]]]]]]]]]]]]]]]]])
+
+# quotes are nestable!
+:test (``0) ([[[0 [[[0 [[[0 [[[1 [[[2 (+2u)]]] [[[0 [[[0 [[[2 (+0u)]]]]]]]]]]]]]]]]]]]]])
+:test (`[0 `0]) ([[[0 [[[1 [[[2 (+0u)]]] [[[0 [[[0 [[[0 [[[1 [[[2 (+2u)]]] [[[0 [[[0 [[[2 (+0u)]]]]]]]]]]]]]]]]]]]]]]]]]]])
+```
+
+## Quasiquotation
+
+Quoted terms can be escaped (*unquoted*) using the comma symbol.
+Unquoted terms will be fully evaluated first before getting quoted
+again.
+
+``` bruijn
+:test (```,[0]) (``[0])
+:test (`,`,[0]) ([0])
+:test (`[0 `,[0]]) (`[0 [0]])
+```
+
+Unquoted De Bruijn indices will get bound to the respective abstraction
+outside of its meta encoding.
+
+``` bruijn
+# adds two using normal quotation
+add-two `[0 + (+2u)]
+
+:test (!add-two (+2u)) ((+4u))
+
+# adds two using a reaching De Bruijn index
+add-two* [`(,0 + (+2u))]
+
+:test (!(add-two* (+2u))) ((+4u))
+```
+
+## Meta library [`std/Meta`](/std/Meta.bruijn.html)
+
+The meta library enables simple interaction with the meta encoding.
+
+Examples:
+
+``` bruijn
+# testing equivalence
+:test (α-eq? `[0 0] `[0 0]) (true)
+:test (α-eq? `α-eq? `α-eq?) (true)
+
+# BLC length of meta term
+:test (length `[0]) ((+4u))
+:test (length `[[1 1]]) ((+12u))
+```
diff --git a/docs/wiki_src/coding/mixfix.md b/docs/wiki_src/coding/mixfix.md
new file mode 100644
index 0000000..c85b936
--- /dev/null
+++ b/docs/wiki_src/coding/mixfix.md
@@ -0,0 +1,33 @@
+# Mixfix
+
+Mixfix functions allow arbitrary infix operations based on "substitution
+holes" by using the `…` symbol in (special character) definitions. The
+symbols and terms always need to be delimited by a space character, else
+they get interpreted as a [prefix](prefix.md).
+
+Example:
+
+``` bruijn
+…+… add
+
+# the "holes" get applied in normal order
+:test ((+4) + (+3)) (add (+4) (+3))
+```
+
+You can define as many holes as you like. Make sure to place parenthesis
+for applications inside substitution holes.
+
+``` bruijn
+{…<$>…|… [[[2 - 1 + 0]]]
+
+# evaluated as (5 - 2) + 1 = 4
+:test ({ (+5) <$> (+2) | (+1)) ((+4))
+:test ({ ((+3) + (+1)) <$> (+2) | (+1)) ((+4))
+```
+
+You can use them as normal functions by writing the identifier
+literally:
+
+``` bruijn
+:test (…+… (+4) (+3)) (add (+4) (+3))
+```
diff --git a/docs/wiki_src/coding/performance.md b/docs/wiki_src/coding/performance.md
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/docs/wiki_src/coding/performance.md
diff --git a/docs/wiki_src/coding/prefix.md b/docs/wiki_src/coding/prefix.md
new file mode 100644
index 0000000..589e081
--- /dev/null
+++ b/docs/wiki_src/coding/prefix.md
@@ -0,0 +1,25 @@
+# Prefix
+
+Prefix functions are symbols written directly in front of another term
+(without space). The term gets applied as an argument to the prefix
+function. Use [mixfix functions](mixfix.md) if the function has more
+than one argument.
+
+They are defined by the `‣`{.bruijn} suffix.
+
+Example:
+
+``` bruijn
+# defines a negation prefix function called '-'
+-‣ [(+0) - 0]
+
+# returns 0 - 10 = -10
+:test (-(+10)) ((-10))
+```
+
+You can use them as normal functions by writing the identifier
+literally:
+
+``` bruijn
+:test (-‣ (+10)) ((-10))
+```
diff --git a/docs/wiki_src/coding/recursion.md b/docs/wiki_src/coding/recursion.md
new file mode 100644
index 0000000..68148e5
--- /dev/null
+++ b/docs/wiki_src/coding/recursion.md
@@ -0,0 +1,78 @@
+# Recursion
+
+Just as normal lambda calculus, bruijn does *not* support typical
+recursion.
+
+If you want to recursively call a function (or imitate `for`/`while`
+loops), you need to use *fixed-point combinators* like `y`{.bruijn} from
+[`std/Combinator`](/std/Combinator.bruijn.html).
+
+Fixed-point combinators have the fascinating property of inducing
+recursive behaviour in programming languages without support for
+recursion.
+
+Say we want a function `g`{.bruijn} to be able to call itself. With the
+`y`{.bruijn} combinator the following equivalence is obtained:
+
+``` bruijn
+ (y g)
+⤳ [[1 (0 0)] [1 (0 0)]] g
+⤳ [g (0 0)] [g (0 0)]
+⤳ g ([g (0 0)] [g (0 0)])
+≡ g (y g)
+```
+
+With this equivalence, `g`{.bruijn} is able to call itself since its
+outer argument is the initial function again.
+
+Example for using `y`{.bruijn} to find the factorial of 2:
+
+``` bruijn
+# here, `1` is the outer argument (y g)
+# `0` is the accumulator (the argument of `factorial`)
+g [[=?0 (+1) (0 ⋅ (1 --0))]]
+
+factorial y g ⧗ Number → Number
+
+:test ((factorial (+3)) =? (+6)) (true)
+```
+
+In-the-wild, this could look like this.
+
+``` bruijn
+# 3 abstractions => two arguments
+# 2 is recursive call
+# 1 is accumulator (+0)
+# 0 is argument (list)
+length z [[[rec]]] (+0) ⧗ (List a) → Number
+ rec 0 [[[case-inc]]] case-end
+ case-inc 5 ++4 1
+ case-end 1
+```
+
+Also see [coding style](style.md) for other style suggestions.
+
+## Mutual recurrence relations
+
+For solving mutual recurrence relations, you can use the *variadic
+fixed-point combinator* `y*`{.bruijn} from
+[`std/List`](/std/List.bruijn.html). This combinator produces all the
+fixed points of a function as an iterable [list](data-structures.md).
+
+Example `even?`{.bruijn}/`odd?`{.bruijn} implementation using
+`y*`{.bruijn}:
+
+``` bruijn
+# the odd? recursive call will be the second argument (1)
+g [[[=?0 true (1 --0)]]]
+
+# the even? recursive call will be the first argument (2)
+h [[[=?0 false (2 --0)]]]
+
+even? head (y* g h) ⧗ Number → Bool
+
+odd? tail (y* g h) ⧗ Number → Bool
+```
+
+Read more about this in the blog post [Variadic fixed-point
+combinators](https://text.marvinborner.de/2023-06-18-15.html).
diff --git a/docs/wiki_src/coding/style.md b/docs/wiki_src/coding/style.md
new file mode 100644
index 0000000..d0ec7a9
--- /dev/null
+++ b/docs/wiki_src/coding/style.md
@@ -0,0 +1,36 @@
+# Coding style
+
+## Scoping
+
+## If/else
+
+redundant
+
+## Head/tail
+
+redundant
+
+## Type signatures
+
+## Recursion
+
+[Recursion](recursion.md) should almost always be achieved with the
+`y`{.bruijn} or `z`{.bruijn} combinators.
+
+A common coding style in bruijn's standard library is to use the scoped
+`rec` function to indicate recursion. You would then use `n+1`
+abstraction around `rec` to indicate `n` arguments and the additionally
+induced recursive call.
+
+Example of the `length`{.bruijn} function for lists:
+
+``` bruijn
+# 3 abstractions => two arguments
+# 2 is recursive call
+# 1 is accumulator (+0)
+# 0 is argument (list)
+length z [[[rec]]] (+0) ⧗ (List a) → Number
+ rec 0 [[[case-inc]]] case-end
+ case-inc 5 ++4 1
+ case-end 1
+```
diff --git a/docs/wiki_src/coding/test-driven-development.md b/docs/wiki_src/coding/test-driven-development.md
new file mode 100644
index 0000000..f62c448
--- /dev/null
+++ b/docs/wiki_src/coding/test-driven-development.md
@@ -0,0 +1,42 @@
+# Test driven development (TDD)
+
+The suggested technique for bruijn development is the TDD method. When
+creating functions, we suggest the following procedure:
+
+- Write a comment, a type signature, and the head of the function
+
+``` bruijn
+# measures the length of a list
+length y [[[rec]]] (+0) ⧗ (List a) → Number
+```
+
+- Write several tests including all edge cases
+
+``` bruijn
+# measures the length of a list
+length y [[[rec]]] (+0) ⧗ (List a) → Number
+
+:test (length empty) ((+0))
+:test (length "a") ((+1))
+:test (length ({}empty)) ((+1))
+:test (length (empty : {}empty)) ((+2))
+:test (length ("abc")) ((+3))
+```
+
+- Finish the implementation until all tests pass (e.g. using the
+ [`:watch`{.bruijn} repl command](REPL.md#watch))
+- Refactor and clean up the definition
+
+``` bruijn
+# measures the length of a list
+length y [[[rec]]] (+0) ⧗ (List a) → Number
+ rec 0 [[[case-inc]]] case-end
+ case-inc 5 ++4 1
+ case-end 1
+
+:test (length empty) ((+0))
+:test (length "a") ((+1))
+:test (length ({}empty)) ((+1))
+:test (length (empty : {}empty)) ((+2))
+:test (length ("abc")) ((+3))
+```
diff --git a/docs/wiki_src/coding/typing.md b/docs/wiki_src/coding/typing.md
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/docs/wiki_src/coding/typing.md
diff --git a/docs/wiki_src/coding/uniform-function-call-syntax.md b/docs/wiki_src/coding/uniform-function-call-syntax.md
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/docs/wiki_src/coding/uniform-function-call-syntax.md