aboutsummaryrefslogtreecommitdiff
path: root/samples/state_lambda.js
blob: 8cebcecd1a0a90dc3f17faa856bd44aa32ad5680 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
// translating de Bruijn levels in lambda terms to unique variables
import * as state from '../src/state.js'

const Abs = n => m => abs => app => lvl => abs(n)(m)
const App = f => x => abs => app => lvl => app(f)(x)
const Lvl =      l => abs => app => lvl => lvl(l)

const transform = term => term
    // case: Abstraction
    (_ => m => state.DO(function* () {
        const {ctr, stk} = yield state.get
        yield state.put({ctr: ctr + 1, stk: [ctr, ...stk]})
        const _m = yield transform(m)
        yield state.modify(({ctr}) => ({ctr, stk}))
        return Abs(ctr)(_m)
    }))
    // case: Application
    (f => x => state.ap
        (state.fmap(App)(transform(f)))
        (transform(x))
    )
    // case: de Bruijn level
    (l => state.DO(function* () {
        const {stk} = yield state.get
        return Lvl(stk[stk.length - l - 1])
    }))

const prettyTerm = term => term
    (n => m => `λ${n}.${prettyTerm(m)}`)
    (f => x => `(${prettyTerm(f)} ${prettyTerm(x)})`)
    (l => l)

// (λλ(0 0) λ0)
const term = App(Abs()(Abs()(App(Lvl(0))(Lvl(0)))))(Abs()(Lvl(0)))

console.log(transform(term)({ctr: 0, stk: []})(_ => t => prettyTerm(t)))
// (λ0.λ1.(0 0) λ2.2)