Haskell アドベントカレンダー (その1)の9日目の記事です. Backpack の話の予定でしたが,先にこっちが書けたのでその3のと入れ替えました. 複数とっておくとこういうことが出来るよね(いいの?).

さて本題. タイトルからお察しの通り,以下の記事のオマージュです.

いきさつ

最近アルバイトで,モナドトランスフォーマーの代わりにextensible パッケージの extensible effects でモナドスタックを作ってました. (理論的な部分は置いておいて)使い方の部分は概ね分かったので記事にしたいと思い,昔々にモナドトランスフォーマーでお世話になった上の記事を extensible effects で追ってみようと思ったわけです.

なので,extensible effects に関する Haskell のパッケージはいくつかあるんですが,今回は extensible パッケージを使います

Extensible Effects

とは何かについては特に言及しません. extensible の作者さんの記事やググるなりしてください.

概要

上述した記事を参考にして,extensible effects によつモンドスタックをステップバイステップに構築してみる. 最終的には, エラー処理(Either),環境渡し(Reader),状態(State),ログ(Writer),入出力(IO)といった機能をモナドスタックに積む.

ちなみに,モナドトランスフォーマーと extensible effects を比較したりは基本的にしない. そもそも,オリジナルなモナドを導入しない限りは(パフォーマンス以外に)大きな差は生まれないと思う(たぶん).

あと,必要なモジュールについては基本的に省く. (おまけを除く)全てのコードは以下のリポジトリに置いたので適宜参照してください.

Step 0. イントロダクション

とりあえず,実行するプログラムの例だけ示しておく. モナドトランスフォーマーの記事と同じようにシンプルなプログラミング言語(?)のインタープリタを使う.

type Name   = String                -- variable names
data Exp    = Lit Integer           -- expressions
            | Var Name
            | Plus Exp Exp
            | Abs Name Exp
            | App Exp Exp
            deriving (Show)
data Value  = IntVal Integer        -- values
            | FunVal Env Name Exp
            deriving (Show)
type Env    = Map.Map Name Value    -- mapping from names to values

eval0 :: Env -> Exp -> Value
eval0 env (Lit i)       = IntVal i
eval0 env (Var n)       = fromJust (Map.lookup n env)
eval0 env (Plus e1 e2)  =
    let
        IntVal i1 = eval0 env e1
        IntVal i2 = eval0 env e2
    in  
        IntVal (i1 + i2)
eval0 env (Abs n e)     = FunVal env n e
eval0 env (App e1 e2)   =
    let
        val1 = eval0 env e1
        val2 = eval0 env e2
    in
        case val1 of
            FunVal env' n body -> eval0 (Map.insert n val2 env') body

評価関数 eval0 をステップバイステップに拡張していくといった感じ. let ... in の辺りとか冗長な気もするが,モナドを導入したときに変更が楽になるようにこう書かれている.

これを実行するには ghci で以下のように書けばよい.

> exampleExp = Lit 12 `Plus` (App (Abs "x" (Var "x")) (Lit 4 `Plus` Lit 2))
> eval0 Map.empty exampleExp
IntVal 18

ちなみに,exampleExp12 + ((λx -> x) (4+2)) という式を表現している.

Step 1. モナドスタイルに

まずはモナドスタイルに変更する. 本来であれば,Identity モナドを使えばよいのだが,今回は次のように書き換えた.

{-# LANGUAGE DataKinds #-}

type Eval1 a = Eff '[] a

runEval1 :: Eval1 a -> a
runEval1 ev = leaveEff ev

extensible ではモナドを型レベルリスト '[] に入れていくことで,モナドスタックを表現する. なので,空にしてしまえばなにも積まれていないモナドの完成である(Eff xsMonad のインスタンスになっている).

そして eval0 もモナドスタイルに書き直す(eval1 とする).

eval1 :: Env -> Exp -> Eval1 Value
eval1 env (Lit i)       = return $ IntVal i
eval1 env (Var n)       =
    maybe (fail ("undefined variable: " ++ n)) return $ Map.lookup n env
eval1 env (Plus e1 e2)  = do
    IntVal i1 <- eval1 env e1
    IntVal i2 <- eval1 env e2
    return $ IntVal (i1 + i2)
eval1 env (Abs n e)     = return $ FunVal env n e
eval1 env (App e1 e2)   = do
    val1 <- eval1 env e1
    val2 <- eval1 env e2
    case val1 of
      FunVal env' n body -> eval1 (Map.insert n val2 env') body

無駄に return とバインドを呼んでるだけ. ちなみに,この eval1 関数はモナドトランスフォーマーのモノと何ら変わらない.

ghci で試すには次のようにする(exampleExp は前と同じ).

> runEval1 $ eval1 Map.empty exampleExp
IntVal 18

Step 2. エラー処理を加える

さて,ここからが楽しくなってくる. エラー処理を加えるために Either e モナドをモナドスタックに積もう.

{-# LANGUAGE TypeOperators #-}

type Eval2 a = Eff '[ "Either" >: Either String ] a

runEval2 :: Eval2 a -> Either String a
runEval2 ev = retractEff ev

(>:) 型演算子を使って,モナドの Key (ここでいう "Either")と積みたいモナドの,いわゆるタプルを作る. モナドスタックを表現する型レベルリストは,モナドの型を要素に持つのではなく,モナドの型とその Key のタプルを持つのだ.

さて,これに合わせて eval1 関数も書き換える.

eval2 :: Env -> Exp -> Eval2 Value
eval2 env (Lit i)       = return $ IntVal i
eval2 env (Var n)       =
    case Map.lookup n env of
        Nothing  -> throwError ("unbound variable: " `mappend` n)
        Just val -> return val
eval2 env (Plus e1 e2)  = do
    e1' <- eval2 env e1
    e2' <- eval2 env e2
    case (e1', e2') of
        (IntVal i1, IntVal i2) -> return $ IntVal (i1 + i2)
        _                      -> throwError "type error in addition"
eval2 env (Abs n e)     = return $ FunVal env n e
eval2 env (App e1 e2)   = do
    val1 <- eval2 env e1
    val2 <- eval2 env e2
    case val1 of
        FunVal env' n body -> eval2 (Map.insert n val2 env') body
        _                  -> throwError "type error in application"

throwError :: String -> Eval2 a
throwError err = liftEff (Proxy :: Proxy "Either") $ Left err

実はモナドトランスフォーマーのモノと全く同じ. 但し,MonadError 型クラスのthrowError 関数は呼べないので(Eval2 はインスタンスではない),代わりに自分で定義した.

liftEff 関数で持ち上げてやる. ただし,どこに持ち上げればいいかを明示してやるために,Proxy :: Proxy "Either" を引数として与えている. どこに持ち上げるかは,この Proxy で指定するので,例えばモナドスタックがどんどん積まれて行っても,lift を何回も呼び出す必要は全くない.

ghci で実行してみる.

> runEval2 $ eval2 Map.empty exampleExp
Right (IntVal 18)
> runEval2 $ eval2 Map.empty (Plus (Lit 1) (Abs "x" (Var "x")))
Left "type error in addition"
> runEval2 $ eval2 Map.empty (Var "x")
Left "unbound variable: x"

組込みのモナドを使う

実は,extensible にはいくつかのモナドが extensible effects 用に用意されている. Either の場合は EitherEff を使う.

type Eval2 a = Eff '[ "Either" >: EitherEff String ] a

throwError :: String -> Eval2 a
throwError err = throwEff (Proxy :: Proxy "Either") err

runEval2 :: Eval2 a -> Either String a
runEval2 ev = leaveEff $ runEitherEff ev

現状だとあまりありがたみは無いが,2つ以上のモナドを積んだときに runEval2 を書くのが非常に楽になる. というか,2つより上にある run 系の関数を自分で定義するのは大変なのだ.

MonadError を使う

実は,MonadError のインスタンスも用意してある. EitherEff の代わりに EitherDef を使えば良い.

type Eval2 a = Eff '[ EitherDef String ] a

runEval2 :: Eval2 a -> Either String a
runEval2 ev = leaveEff $ runEitherDef ev

throwErrorMonadError 型クラスのを使うのでもう要らない.

注意点として,EitherDef が定義されている Data.Extensible.Effect.Default モジュールは,大本のモジュールである Data.Extensible にエクスポートされていないので,別途インポートする必要がある. おそらく,MonadError のインスタンス宣言が広範囲に影響するのを懸念してだろう(Eff xs 全部に影響しているので).

また,以降は Data.Extensible.Effect.Default モジュールにあるモナド型をスタックに積んでいくことにする.

Step 3. 環境を隠す

eval2 関数では,環境 Env を明示的に引数としていて渡しているが,これを Reader r モナドを使って隠蔽する. Reader r モナドの Eff 版として ReaderDef を用いる.

type Eval3 a = Eff '[ ReaderDef Env, EitherDef String ] a

runEval3 :: Env -> Eval3 a -> Either String a
runEval3 env ev = leaveEff . runEitherDef . flip runReaderDef env $ ev

既に,MonadReader の型クラスのインスタンスになっているので ask 関数や loacal 関数を利用できる.

eval3 :: Exp -> Eval3 Value
eval3 (Lit i)       = return $ IntVal i
eval3 (Var n)       = do
    env <- ask
    case Map.lookup n env of
        Nothing  -> throwError ("unbound variable: " `mappend` n)
        Just val -> return val
eval3 (Plus e1 e2)  = do
    e1' <- eval3 e1
    e2' <- eval3 e2
    case (e1', e2') of
        (IntVal i1, IntVal i2) -> return $ IntVal (i1 + i2)
        _                      -> throwError "type error in addition"
eval3 (Abs n e)     = do
    env <- ask
    return $ FunVal env n e
eval3 (App e1 e2)   = do
    val1 <- eval3 e1
    val2 <- eval3 e2
    case val1 of
        FunVal env' n body ->
            local (const (Map.insert n val2 env')) (eval3 body)
        _                  -> throwError "type error in application"

eval3 (Var n)eval3 (Abs n e) の最初に ask を使って環境を参照している. また eval3 (App e1 e2) の最後に,local 関数で変数に値を束縛した環境で更新している.

Step 4. 状態を加える

簡約回数を状態として引き回してみる. 状態を引き回すには State s モナドを使う. わざわざ説明することではないが,State s as -> (a, s) と同義である.

State s モナドの Eff 版として StateDef を用いる.

type Eval4 a =
    Eff '[ ReaderDef Env, EitherDef String, StateDef Integer ] a

runEval4 :: Env -> Integer -> Eval4 a -> (Either String a, Integer)
runEval4 env st ev =
    leaveEff . flip runStateDef st . runEitherDef . flip runReaderDef env $ ev

(モナドトランスフォーマーの記事でも書いてあるように) EitherState のような 最終的な結果に影響を与えるモナドを組み合わせる場合には順番が重要になってくる. 今回は Either の外に State を置きたいので,(Either String a, Integer) となるようにモナドスタックを積んだ. Eff の場合は型レベルリストの左から剥がしていき,剥がした結果を右に渡すイメージなので注意してください(要するに左畳み込み?).

eval3 関数は以下のように書き換える.

tick :: (Num s, MonadState s m) => m ()
tick = do
    st <- get
    put (st + 1)

eval4 :: Exp -> Eval4 Value
eval4 (Lit i)       = do
    tick
    return $ IntVal i
eval4 (Var n)       = do
    tick
    env <- ask
    case Map.lookup n env of
        Nothing  -> throwError ("unbound variable: " `mappend` n)
        Just val -> return val
eval4 (Plus e1 e2)  = do
    tick
    e1' <- eval4 e1
    e2' <- eval4 e2
    case (e1', e2') of
        (IntVal i1, IntVal i2) -> return $ IntVal (i1 + i2)
        _                      -> throwError "type error in addition"
eval4 (Abs n e)     = do
    tick
    env <- ask
    return $ FunVal env n e
eval4 (App e1 e2)   = do
    tick
    val1 <- eval4 e1
    val2 <- eval4 e2
    case val1 of
        FunVal env' n body ->
            local (const (Map.insert n val2 env')) (eval4 body)
        _                  -> throwError "type error in application"

ghci の結果は次のようになる.

> runEval4 Map.empty 0 $ eval4 exampleExp
(Right (IntVal 18),8)

Step 5. ログを加える

ログとしてモナドスタックに Writer w モナドを積む. Writer w モナドの Eff 版として WriterDef を使う. State のときと同様に,最後に影響を与えるモナドなので,積む順番に注意が必要だ.

type Eval5 a =
    Eff '[ ReaderDef Env, EitherDef String, WriterDef [String], StateDef Integer ] a

runEval5 :: Env -> Integer -> Eval5 a -> ((Either String a, [String]), Integer)
runEval5 env st ev =
    leaveEff . flip runStateDef st . runWriterDef . runEitherDef . flip runReaderDef env $ ev

ちなみに,Writer wwMonoid 型クラスのインスタンスである必要がある.

ほぼほぼ意味は無いんだけど,今回は評価中に遭遇した変数名を書き出すというログをとることにする.

eval5 :: Exp -> Eval5 Value
eval5 (Lit i)       = do
    tick
    return $ IntVal i
eval5 (Var n)       = do
    tick
    tell [n]
    env <- ask
    case Map.lookup n env of
        Nothing  -> throwError ("unbound variable: " `mappend` n)
        Just val -> return val
eval5 (Plus e1 e2)  = do
    tick
    e1' <- eval5 e1
    e2' <- eval5 e2
    case (e1', e2') of
        (IntVal i1, IntVal i2) -> return $ IntVal (i1 + i2)
        _                      -> throwError "type error in addition"
eval5 (Abs n e)     = do
    tick
    env <- ask
    return $ FunVal env n e
eval5 (App e1 e2)   = do
    tick
    val1 <- eval5 e1
    val2 <- eval5 e2
    case val1 of
        FunVal env' n body ->
            local (const (Map.insert n val2 env')) (eval5 body)
        _                  -> throwError "type error in application"

eval5 (Var n)tell [n] が追加されただけである(tell 関数は Monoid で合成して追記する).

ghci で実行すると次のようになる.

> runEval5 Map.empty 0 $ eval5 exampleExp
((Right (IntVal 18),["x"]),8)

Step 6. IOはどうすんの?

さぁ最後は IO だ. 実は IOEff 版は用意されていない. しかし,Data.Extensible.Effect.Data モジュールの中を見てみると,一番下の Orphan instances のところに MonadIO もある. 定義より,次のようにすれば,MonadIO のインスタンスとなり liftIO が使えるようになる.

type Eval6 a =
    Eff '[ ReaderDef Env, EitherDef String, WriterDef [String], StateDef Integer, "IO" >: IO ] a

runEval6 :: Env -> Integer -> Eval6 a -> IO ((Either String a, [String]), Integer)
runEval6 env st ev =
    retractEff . flip runStateDef st . runWriterDef . runEitherDef . flip runReaderDef env $ ev

runEval の方は leaveEff から retractEff にしただけだ.

今回は次のようにいわゆる printf デバッグを eval6 (Lit i) のところに入れている.

eval6 :: Exp -> Eval6 Value
eval6 (Lit i)       = do
    tick
    liftIO $ print i
    return $ IntVal i
eval6 (Var n)       = do
    tick
    tell [n]
    env <- ask
    case Map.lookup n env of
        Nothing  -> throwError ("unbound variable: " `mappend` n)
        Just val -> return val
eval6 (Plus e1 e2)  = do
    tick
    e1' <- eval6 e1
    e2' <- eval6 e2
    case (e1', e2') of
        (IntVal i1, IntVal i2) -> return $ IntVal (i1 + i2)
        _                      -> throwError "type error in addition"
eval6 (Abs n e)     = do
    tick
    env <- ask
    return $ FunVal env n e
eval6 (App e1 e2)   = do
    tick
    val1 <- eval6 e1
    val2 <- eval6 e2
    case val1 of
        FunVal env' n body ->
            local (const (Map.insert n val2 env')) (eval6 body)
        _                  -> throwError "type error in application"

これを ghci で実行すると次のようになる.

> runEval6 Map.empty 0 $ eval6 exampleExp
12
4
2
((Right (IntVal 18),["x"]),8)

おまけ

組み合わせる前提のモナドはどうすれば良いだろうか? 例えば,monad-logger パッケージの LoggingT モナドとか(これは下に IO を持ってないといけない).

実はぼくも良く分かっていない… が,とりあえず思いつく方法を書いておく.

まずは IOLoggingT IO に置き換える(ごちゃごちゃするので IO 以外は消した).

type Eval a = Eff '[ "Logger" >: LoggingT IO ] a

runEval :: Eval a -> IO a
runEval = runStdoutLoggingT . retractEff

こうすると Data.Extensible.Effect.DefaultMonadIO のインスタンスは使えないので,自分で liftIO のような関数を定義する必要がある.

liftIO' :: IO a -> MarketM a
liftIO' = liftEff (Proxy :: Proxy "Logger") . liftIO

ちなみに,MonadIO のインスタンスには できない . 既に Eff xsMonadIO のインスタンスは Data.Extensible.Effect.Default に定義してあるからだ(なのでコレをインポートしなければできるよ).

正直,これで十分だがどーしても共存させたい,ないしは LoggingT 以外の IO 前提のモナドと共存させたいときにはどうするか. イロイロ頑張って結果以下のようにできた.

type Eval a = Eff '[ LoggerDef, "IO" >: IO ] a

runEval :: Eval a -> IO a
runEval = retractEff . runLoggerDef

type LoggerDef = "Logger" >: Logging
type Logging = LoggingT IO

runLoggerDef :: (MonadIO (Eff xs)) => Eff (LoggerDef ': xs) a -> Eff xs a
runLoggerDef = pealEff0 pure $ \m k -> k =<< liftIO (runStdoutLoggingT m)

class (Associate "Logger" Logging xs) => MonadLogger (Eff xs) where
    monadLoggerLog loc ls level msg =
        liftEff (Proxy :: Proxy "Logger") $ monadLoggerLog loc ls level msg

重要なのは runLoggerDef の部分. (2つ目以上に積んだ extensible にないモナドは)こういった剥がす関数を自分で定義する必要があり,ここが一番難しい. 自分より下に MonadIO のインスタンス,ようするに "IO" >: IO がある前提で,一度 IO にしてから,再度持ち上げるというずるいやり方をした.

いまいちパッとしないけど,これで一応目的のものはできる…

おしまい

eval 関数の実装はモナドトランスフォーマーのときと変わらないので,あんまり面白くなかったですね…すいません.