Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/zhxxch/effectfulcode

algebraic effects for F#
https://github.com/zhxxch/effectfulcode

algebraic-effects computation-expressions fsharp

Last synced: about 2 months ago
JSON representation

algebraic effects for F#

Awesome Lists containing this project

README

        

# Algebraic effects for F#

This library (EffectfulCode.fs) implements **one-shot algebraic effects** in F# using [F# 6 Resumable Code](https://learn.microsoft.com/dotnet/fsharp/whats-new/fsharp-6#resumable-code); (when possible,) effectful codes are compiled to highly efficient state machines.

How to using this library: see below demonstration.

## Examples

### Example 1

run `dotnet fsi -O example/example-1.fsx` to interact with this example program.

```F#
#load "../src/EffectfulCode.fs"

// declear computation expression (CE) builder
let ef = new EffectfulCode.CodeBuilder()
let withHandler = new EffectfulCode.AddHandlerBuilder()

// declear effects
type AskInputEffect = AskInput of string
type WriteLogEffect = WriteLog of string
type MultiplyEffect = Mul of float * float
type AddEffect = Add of float * float

// define effectful code
let inline example1 () =
ef {
do! ef { yield WriteLog "effectful code start" }
let! x = ef { yield AskInput "x = " }
printfn "variable x = %g" x

let! y = ef { yield AskInput "y = " }
printfn "variable y = %g" y

let! xy = ef { yield Add(x, y) }
printfn "x + y = %g" xy

let! z = ef { yield AskInput "z = " }
printfn "variable z = %g" z

let! xyz = ef { yield Mul(xy, z) }
printfn "(x + y) * z = %g" xyz

do! ef { yield WriteLog "effectful code about to return" }

let xyz' = (x + y) * z
return (xyz, xyz')
}

// define handlers
let inline h_log (WriteLog s) = ef { return printfn "-- log: %s" s }

let inline h_input (AskInput question) =
ef {
printf "%s" question
let input_str = System.Console.ReadLine()
let x = System.Double.Parse input_str
do! ef { yield WriteLog(sprintf "resume `AskInput` with %g" x) }
return x
}

let inline h_add (Add(x, y)) =
ef {
do! ef { yield WriteLog(sprintf "resume `Add` with %g" (x + y)) }
return x + y
}

let inline h_mul (Mul(x, y)) =
ef {
do! ef { yield WriteLog(sprintf "resume `Mul` with %g" (x * y)) }
return x * y
}
// define alternative handlers
let inline h_add_tropical (Add(x, y)) =
ef {
do! ef { yield WriteLog(sprintf "resume `Add` with %g" (min x y)) }
return min x y
}

let inline h_mul_tropical (Mul(x, y)) =
ef {
do! ef { yield WriteLog(sprintf "resume `Mul` with %g" (x + y)) }
return x + y
}

// add handlers
withHandler {
return example1 ()
yield h_input
yield h_add
yield h_mul
yield h_log
} |> fun f -> f () |> fun (a, a') -> printfn "(x + y) * z = %g (%g)" a a'

// add handlers(alternative syntax & different handlers)
let inline example1' () =
example1 () |> withHandler{yield h_input}

let code = withHandler {
yield h_log
yield h_add_tropical
yield h_mul_tropical
return example1' ()
}

code () |> fun (a, a') -> printfn "(x + y) * z = %g (%g)" a a'

// trying to run effectful code with any unhandled effect raises an exception
withHandler {
return example1' ()
yield h_input
yield h_log
yield h_add_tropical
yield h_mul_tropical
} |> fun code -> code () |> fun (a, a') -> printfn "(x + y) * z = %g (%g)" a a'

```

## Licence

GPLv3

## Annex 1 - about algebraic effects

Readings:

- Dan Abramov. [Algebraic Effects for the Rest of Us — overreacted](https://overreacted.io/algebraic-effects-for-the-rest-of-us/#a-function-has-no-color)
- Nathanaël Courant. [Gagallium : Safely typing algebraic effects](https://gallium.inria.fr/blog/safely-typing-algebraic-effects/)
- [A Neighborhood of Infinity: The Mother of all Monads](http://blog.sigfpe.com/2008/12/mother-of-all-monads.html)