1
1
mirror of https://github.com/github/semantic.git synced 2024-12-22 22:31:36 +03:00
semantic/src/Interpreter.hs
2015-11-17 17:50:15 -08:00

35 lines
1.4 KiB
Haskell

module Interpreter (interpret) where
import Algorithm
import Control.Monad.Free
import Control.Comonad.Cofree
import Operation
import Diff
import Syntax
import Data.Map
constructAndRun :: Term a Info -> Term a Info -> Maybe (Diff a)
constructAndRun a b =
run algorithm where
algorithm = Free $ Recursive a b Pure
run :: Algorithm a (Diff a) -> Maybe (Diff a)
run (Pure diff) = Just diff
run (Free (Recursive a b f)) = recur a b where
recur (_ :< Indexed a') (_ :< Indexed b') | length a' == length b' =
run $ f $ Free $ Indexed $ zipWith interpret a' b'
recur (_ :< Fixed a') (_ :< Fixed b') | length a' == length b' =
run $ f $ Free $ Fixed $ zipWith interpret a' b'
recur (_ :< Keyed a') (_ :< Keyed b') | keys a' == keys b' =
run $ f $ Free $ Keyed $ fromList $ fmap (\ x -> (x, interpretInBoth x a' b')) $ keys b' where
interpretInBoth :: String -> Map String (Term a Info) -> Map String (Term a Info) -> Diff a
interpretInBoth key a' b' = maybeInterpret (Data.Map.lookup key a') (Data.Map.lookup key b')
maybeInterpret :: Maybe (Term a Info) -> Maybe (Term a Info) -> Diff a
maybeInterpret (Just a) (Just b) = interpret a b
recur _ _ = run $ f $ Pure Patch { old = Just a, new = Just b }
interpret :: Term a Info -> Term a Info -> Diff a
interpret a b = maybeReplace $ constructAndRun a b where
maybeReplace (Just a) = a
maybeReplace Nothing = Just a </> Just b