tvl-depot/nix/stateMonad/default.nix
sterni 70427fd934 refactor(nix/stateMonad): optimize after
This should save on one function application which can be a big deal for
bigger for_ loops, I suspect. It's not really complicated, so why not.

Change-Id: I2bfcd254e55f1bea366b09de294b2bef9f5b5dda
Reviewed-on: https://cl.tvl.fyi/c/depot/+/6834
Reviewed-by: grfn <grfn@gws.fyi>
Tested-by: BuildkiteCI
2022-10-08 10:59:45 +00:00

76 lines
1.7 KiB
Nix

# Simple state monad represented as
#
# stateMonad s a = s -> { state : s; value : a }
#
{ ... }:
rec {
#
# Monad
#
# Type: stateMonad s a -> (a -> stateMonad s b) -> stateMonad s b
bind = action: f: state:
let
afterAction = action state;
in
(f afterAction.value) afterAction.state;
# Type: stateMonad s a -> stateMonad s b -> stateMonad s b
after = action1: action2: state: action2 (action1 state).state;
# Type: stateMonad s (stateMonad s a) -> stateMonad s a
join = action: bind action (action': action');
# Type: [a] -> (a -> stateMonad s b) -> stateMonad s null
for_ = xs: f:
builtins.foldl'
(laterAction: x:
after (f x) laterAction
)
(pure null)
xs;
#
# Applicative
#
# Type: a -> stateMonad s a
pure = value: state: { inherit state value; };
# TODO(sterni): <*>, lift2, …
#
# Functor
#
# Type: (a -> b) -> stateMonad s a -> stateMonad s b
fmap = f: action: bind action (result: pure (f result));
#
# State Monad
#
# Type: (s -> s) -> stateMonad s null
modify = f: state: { value = null; state = f state; };
# Type: stateMonad s s
get = state: { value = state; inherit state; };
# Type: s -> stateMonad s null
set = new: modify (_: new);
# Type: str -> stateMonad set set.${str}
getAttr = attr: fmap (state: state.${attr}) get;
# Type: str -> (any -> any) -> stateMonad s null
modifyAttr = attr: f: modify (state: state // {
${attr} = f state.${attr};
});
# Type: str -> any -> stateMonad s null
setAttr = attr: value: modifyAttr attr (_: value);
# Type: s -> stateMonad s a -> a
run = state: action: (action state).value;
}