LoginSignup
3
0

More than 5 years have passed since last update.

SakuraScriptインタプリタをFreeモナドで実装する

Last updated at Posted at 2016-12-10

SakuraScriptインタプリタをFreeモナドで実装する

この記事は 伺か Advent Calendar 2016の4日目の記事です。

Free モナドを使えばインタプリタが簡単に実装できると聞いて PureScript で Aff を使いがてら簡易的な SakuraScript インタプリタを書いてみた。

Free モナドとはなんなのかについては参考リンクを参照。

コード

module Main where

import Prelude 

import Control.Monad.Eff (Eff)
import Control.Monad.Eff.Exception (EXCEPTION)
import Control.Monad.Eff.Console (CONSOLE)
import Control.Monad.Eff.Class (liftEff)

import Test.Unit.Console (TESTOUTPUT, print)

import Control.Monad.Aff (Aff, forkAff, launchAff, later')
import Control.Monad.Aff.Console (log)

import Control.Monad.Free (Free, foldFree, liftF)

import Data.Maybe (Maybe(..))
import Data.String (split, Pattern(..))
import Data.Array (foldM)
import Data.NaturalTransformation (NaturalTransformation)



data SakuraScriptF a =
    ChangeScope Int a
  | ChangeSurface Int a
  | Talk String a
  | Wait Int a
  | YenE a

type SakuraScript a = Free SakuraScriptF a

--                           + type NaturalTransformation f g = forall a. f a -> g a
--                           |                                    + data Aff :: # ! -> * -> *
--                           |                                    |
sakuraScriptN :: forall eff. NaturalTransformation SakuraScriptF (Aff (console :: CONSOLE, testOutput :: TESTOUTPUT | eff))
sakuraScriptN (Wait n a)          = const a <$> do
                                      log ("\\_w[" <> (show n) <> "]")
                                      later' n do pure unit
--                                    |
--                                    + later' :: forall e a. Int -> Aff e a -> Aff e a
--                                    : Runs the specified asynchronous computation later, by the specified number of milliseconds.
sakuraScriptN (Talk s a)          = const a <$> do
                                      let arr = (split (Pattern "") s)
                                      foldM wait unit arr -- 1文字づつ出力
                                      log "" -- 改行
                                      where 
                                        wait _ c = later' 50 do liftEff $ print c -- 改行せず文字出力
sakuraScriptN (ChangeScope n a)   = const a <$> log ("\\p[" <> (show n) <> "]")
sakuraScriptN (ChangeSurface n a) = const a <$> log ("\\s[" <> (show n) <> "]")
--                                  |       |   |                     |
--                                  |       |   |                     + append :: a -> a -> a
--                                  |       |   + log :: forall eff. String -> Aff (console :: CONSOLE | eff) Unit
--                                  |       + map :: forall a b. (a -> b) -> f a -> f b
--                                  + const :: forall a b. a -> b -> a
--                                  : Returns its first argument and ignores its second.
sakuraScriptN (YenE a)            = const a <$> log "\\e"


run :: forall eff. NaturalTransformation SakuraScript (Aff (console :: CONSOLE, testOutput :: TESTOUTPUT | eff))
run = foldFree sakuraScriptN
--    |
--    + foldFree :: forall f m. MonadRec m => (f ~> m) -> (Free f) ~> m

onBoot :: SakuraScript Unit
onBoot = do
  liftF (ChangeScope 1 unit)
  liftF (ChangeSurface 10 unit)
  liftF (Talk "なんや、えらい難しい言語やな。" unit)
  liftF (Wait 1000 unit)
  liftF (ChangeScope 0 unit)
  liftF (ChangeSurface 0 unit)
  liftF (Talk "そうだねー。" unit)
  liftF (YenE unit)
-- |
-- + liftF :: forall f. f ~> Free f


main :: forall eff. Eff ( console :: CONSOLE , err :: EXCEPTION, testOutput :: TESTOUTPUT | eff ) Unit
main = do
  void $ launchAff do
-- |     |
-- |     + launchAff :: forall e a. Aff e a -> Eff (err :: EXCEPTION | e) (Canceler e)
-- |     : Converts the asynchronous computation into a synchronous one. All values are ignored, and if the computation produces an error, it is thrown.
-- + void :: forall f a. Functor f => f a -> f Unit
    forkAff do
      run $ onBoot
--  |
--  + forkAff :: forall e a. Aff e a -> Aff e (Canceler e)
--  : Forks the specified asynchronous computation so subsequent computations will not block on the result of the computation.
--  : Returns a canceler that can be used to attempt cancellation of the forked computation.

結果

$ psc --version
0.10.2
$ pulp run
* Building project in ***/purs-free
Compiling Main
* Build successful.
\p[1]
\s[10]
なんや、えらい難しい言語やな。
\_w[1000]
\p[0]
\s[0]
そうだねー。
\e

小学生並みの感想

  • 文字列結合は <> で Semigroup にある
  • ~> とは一体何なのか?(Naturalで、自然変換です。型オペレータも定義できるので
  • YenE で処理を打ち切りたかったが方法が分からなかった。 Nothing とか使えばいいのだろうか?
  • が使えるようになったようだ?
  • 改行なし出力のために Test.Unit.Console を使った

参考

3
0
4

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
3
0