summaryrefslogtreecommitdiff
path: root/edit-lens/src/Data/String/DFST.hs
blob: 54a133617bf63ae96231688136c16be35f0a5ca1 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
{-# LANGUAGE ScopedTypeVariables
#-}

{-|
Description: Deterministic finite state transducers
-}
module Data.String.DFST
  ( DFST(..)
  , runDFST, runDFST'
  ) where

import Data.Map.Strict (Map, (!?))
import qualified Data.Map.Strict as Map

import Data.Set (Set)
import qualified Data.Set as Set

import Control.Monad

data DFST state = DFST
  { stInitial :: state
  , stTransition :: Map (state, Char) (state, String)
    -- ^ All @(s, c)@-combinations not mapped are assumed to map to @(s, Nothing)@
  , stAccept :: Set state
  }
  
runDFST :: forall state. Ord state => DFST state -> String -> Maybe String
runDFST dfst@DFST{..} str = let (finalState, str') = runDFST' dfst stInitial str id
                            in str' "" <$ guard (finalState `Set.member` stAccept)

runDFST' :: forall state. Ord state
         => DFST state
         -> state -- ^ Current state
         -> String -- ^ Remaining input
         -> (String -> String) -- ^ Output as difference list
         -> (state, (String -> String)) -- ^ Next state, altered output
runDFST' _ st [] acc = (st, acc)
runDFST' dfst@DFST{..} st (c:cs) acc
  | Just (st', mc') <- stTransition !? (st, c)
  = runDFST' dfst st' cs $ acc . (mc' ++)
  | otherwise
  = runDFST' dfst st cs acc