A pure, splittable, value supply for Purescript
spago install supply
Then you can use the functions in Supply
module. Example -
main :: Effect Unit
main = withIntSupply \intSupply -> do
let v = map (\i -> "v" <> show i) intSupply
let (v1 /\ v2 /\ v3 /\ v4 /\ _) = split4 v
let (v5 /\ v6 /\ v7 /\ v8 /\ _) = split4 v2
log $ supplyValue v1
log $ supplyValue v1
log $ supplyValue v8
log $ supplyValue v3
log $ supplyValue v8
The supplied values are sequential integers that are lazily resolved when they are used. So this should print -
v0
v0
v1
v2
v1
Effectful interface.
newSupply :: forall s a. a -> (a -> a) -> Effect (Supply s a)
Pure interface. The phantom type parameter prevents supplies from being used outside of the handler functions so as not to break referential transparency.
withSupply :: forall r a. a -> (a -> a) -> (forall s. Supply s a -> r) -> r
supplyValue :: forall s a. Supply a -> a
Each supply can only provide a single value. Repeatedly calling supplyValue
on the same supply will give the same result.
To get multiple values, supplies can be split any number of times.
-- | Split a supply into two different supplies.
-- The resulting supplies are different from the input supply.
split2 :: forall s a. Supply s a -> Tuple (Supply s a) (Supply s a)
-- | Split a supply into three different supplies.
split3 :: forall s a. Supply a -> Tuple3 (Supply s a) (Supply s a) (Supply s a)
-- | Split a supply into four different supplies.
split4 :: forall s a. Supply s a -> Tuple4 (Supply s a) (Supply s a) (Supply s a) (Supply s a)
Supply
has a Functor
instance.
map :: forall s a b. (a -> b) -> Supply s a -> Supply s b
Also supplies have a Comonad
instance so you can use extend
(or modifySupply
which is the flipped version).
-- | Generate a new supply by systematically applying a function to an existing supply
extend :: forall s a b. (Supply s a -> b) -> Supply s a -> Supply s b
-- | Generate a new supply by systematically applying a function to an existing supply
modifySupply :: forall s a b. Supply s a -> (Supply s a -> b) -> Supply s b
For convenience, functions are provided to directly construct a supply of integers.
Create a new integer supply.
newIntSupply :: forall s. Effect (Supply s Int)
newIntSupply = newSupply 0 (_+1)
Create a new integer supply and use it without effects. The phantom type parameter prevents supplies from being used outside of the handler functions so as not to break referential transparency.
withIntSupply :: forall r. (forall s. Supply s Int -> r) -> r
withIntSupply = withSupply 0 (_+1)