/- Copyright (c) 2016 Gabriel Ebner. All rights reserved. Released under Apache 2.0 license as described in the file LICENSE. Authors: Gabriel Ebner, Sebastian Ullrich Classy functions for lifting monadic actions of different shapes. This theory is roughly modeled after the Haskell 'layers' package https://hackage.haskell.org/package/layers-0.1. Please see https://hackage.haskell.org/package/layers-0.1/docs/Documentation-Layers-Overview.html for an exhaustive discussion of the different approaches to lift functions. -/ prelude import init.function init.coe import init.control.monad universes u v w /-- A function for lifting a computation from an inner monad to an outer monad. Like [MonadTrans](https://hackage.haskell.org/package/transformers-0.5.5.0/docs/Control-Monad-Trans-Class.html), but `n` does not have to be a monad transformer. Alternatively, an implementation of [MonadLayer](https://hackage.haskell.org/package/layers-0.1/docs/Control-Monad-Layer.html#t:MonadLayer) without `layerInvmap` (so far). -/ class has_monad_lift (m : Type u → Type v) (n : Type u → Type w) := (monad_lift {} : ∀ {α}, m α → n α) /-- The reflexive-transitive closure of `has_monad_lift`. `monad_lift` is used to transitively lift monadic computations such as `state_t.get` or `state_t.put s`. Corresponds to [MonadLift](https://hackage.haskell.org/package/layers-0.1/docs/Control-Monad-Layer.html#t:MonadLift). -/ class has_monad_lift_t (m : Type u → Type v) (n : Type u → Type w) := (monad_lift {} : ∀ {α}, m α → n α) export has_monad_lift_t (monad_lift) /-- A coercion that may reduce the need for explicit lifting. Because of [limitations of the current coercion resolution](https://github.com/leanprover/lean/issues/1402), this definition is not marked as a global instance and should be marked locally instead. -/ @[reducible] def has_monad_lift_to_has_coe {m n} [has_monad_lift_t m n] {α} : has_coe (m α) (n α) := ⟨monad_lift⟩ instance has_monad_lift_t_trans (m n o) [has_monad_lift n o] [has_monad_lift_t m n] : has_monad_lift_t m o := ⟨λ α ma, has_monad_lift.monad_lift (monad_lift ma : n α)⟩ instance has_monad_lift_t_refl (m) : has_monad_lift_t m m := ⟨λ α, id⟩ lemma monad_lift_refl {m : Type u → Type v} {α} : (monad_lift : m α → m α) = id := rfl /-- A functor in the control of monads. Can be used to lift monad-transforming functions. Based on pipes' [MFunctor](https://hackage.haskell.org/package/pipes-2.4.0/docs/Control-MFunctor.html), but not restricted to monad transformers. Alternatively, an implementation of [MonadTransFunctor](http://duairc.netsoc.ie/layers-docs/Control-Monad-Layer.html#t:MonadTransFunctor). Remark: other libraries equate `m` and `m'`, and `n` and `n'`. We need to distinguish them to be able to implement gadgets such as `monad_state_adapter` and `monad_reader_adapter`. -/ class monad_functor (m m' : Type u → Type v) (n n' : Type u → Type w) := (monad_map {} {α : Type u} : (∀ {β}, m β → m' β) → n α → n' α) /-- The reflexive-transitive closure of `monad_functor`. `monad_map` is used to transitively lift monad morphisms such as `state_t.zoom`. A generalization of [MonadLiftFunctor](http://duairc.netsoc.ie/layers-docs/Control-Monad-Layer.html#t:MonadLiftFunctor), which can only lift endomorphisms (i.e. m = m', n = n'). -/ class monad_functor_t (m m' : Type u → Type v) (n n' : Type u → Type w) := (monad_map {} {α : Type u} : (∀ {β}, m β → m' β) → n α → n' α) export monad_functor_t (monad_map) instance monad_functor_t_trans (m m' n n' o o') [monad_functor n n' o o'] [monad_functor_t m m' n n'] : monad_functor_t m m' o o' := ⟨λ α f, monad_functor.monad_map (λ β, (monad_map @f : n β → n' β))⟩ instance monad_functor_t_refl (m m') : monad_functor_t m m' m m' := ⟨λ α f, f⟩ lemma monad_map_refl {m m' : Type u → Type v} (f : ∀ {β}, m β → m' β) {α} : (monad_map @f : m α → m' α) = f := rfl /-- Run a monad stack to completion. `run` should be the composition of the transformers' individual `run` functions. This class mostly saves some typing when using highly nested monad stacks: ``` @[reducible] def my_monad := reader_t my_cfg $ state_t my_state $ except_t my_err id -- def my_monad.run {α : Type} (x : my_monad α) (cfg : my_cfg) (st : my_state) := ((x.run cfg).run st).run def my_monad.run {α : Type} (x : my_monad α) := monad_run.run x ``` -/ class monad_run (out : out_param $ Type u → Type v) (m : Type u → Type v) := (run {} {α : Type u} : m α → out α) export monad_run (run)