{-# LANGUAGE PatternGuards #-}
module Data.Derive.DSL.Derive(derive) where
import Control.Arrow
import Data.Derive.DSL.HSE
import Data.Derive.DSL.DSL
import Data.Derive.DSL.Apply
import Data.List
import Data.Char
import Data.Maybe
data Guess = Guess DSL
| GuessFld Int DSL
| GuessCtr Int Bool DSL
deriving Show
ctrNames = map ctorName $ dataCtors sample
derive :: Out -> [DSL]
derive x = [simplifyDSL y | Guess y <- guess $ toOutput x]
guess :: Output -> [Guess]
guess (OApp "InstDecl" [_,OApp "Nothing" [],rule,decls])
| OApp "IRule" [_, OApp "Nothing" [], ctx, ihead] <- unparen rule
, (name, types) <- unInstHead ihead
, OApp "UnQual" [_, OApp "Ident" [_, OString name]] <- name
, [OApp "TyParen" [_,OApp "TyApp"
[_,OApp "TyCon" [_,OApp "UnQual" [_,OApp "Ident" [_,OString nam]]]
,OApp "TyVar" [_,OApp "Ident" [_,OString var]]]]] <- types
, nam == dataName sample
= [Guess $ Instance (unContext ctx) name y | Guess y <- guess decls]
where
unContext (OApp "Just" [x])
| OApp "CxSingle" [_,x] <- x = unClass x
| OApp "CxTuple" [_,OList xs] <- x = concatMap unClass xs
unContext x = []
unClass (OApp "ClassA" [_,OApp "UnQual" [_,OApp "Ident" [_,OString x]],_]) = [x]
unClass _ = []
unInstHead (OApp "IHCon" [_, name]) = (name, [])
unInstHead (OApp "IHInfix" [_, ty, name]) = (name, [ty])
unInstHead (OApp "IHParen" [_, x]) = unInstHead x
unInstHead (OApp "IHApp" [_, hd, ty]) = second (++[ty]) $ unInstHead hd
unparen (OApp p [_, x]) | "Paren" `isInfixOf` p = unparen x
unparen x = x
guess (OList xs) = guessList xs
guess o@(OApp op xs) = gssFold o ++ gssApp o ++ map (lift (App op)) (guessList xs)
guess (OString x)
| Just i <- findIndex (`isSuffixOf` x) ctrNames = [GuessCtr i True $ String (take (length x - length (ctrNames !! i)) x) `append` CtorName]
| "Sample" `isSuffixOf` x = [Guess $ String (take (length x - 6) x) `append` DataName]
| otherwise =
[lift (\d -> append (String $ init x) (ShowInt d)) g | x /= "", isDigit (last x), g <- guess $ OInt $ read [last x]] ++
[Guess $ String x]
guess (OInt i) =
[GuessFld (fromInteger i) FieldIndex | i `elem` [1,2]] ++
[GuessCtr 1 False CtorIndex | i == 1] ++
[GuessCtr 1 False CtorArity | i == 2] ++
[Guess $ Int i]
guess x = error $ show ("fallthrough",x)
guessList :: [Output] -> [Guess]
guessList xs = mapMaybe sames $ map diffs $ sequence $ map guess xs
where
sames :: [Guess] -> Maybe Guess
sames xs = do
let (is,fs) = unzip $ map fromGuess xs
i <- maxim is
return $ toGuess i $ Concat $ List fs
diffs :: [Guess] -> [Guess]
diffs (GuessCtr 0 True x0:GuessCtr 1 True x1:GuessCtr 2 True x2:xs)
| f 0 x0 == f 0 x1 && f 2 x2 == f 2 x1 = Guess (MapCtor x1) : diffs xs
where f i x = applyEnv x env{envInput=sample, envCtor=dataCtors sample !! i}
diffs (GuessCtr 2 True x2:GuessCtr 1 True x1:GuessCtr 0 True x0:xs)
| f 0 x0 == f 0 x1 && f 2 x2 == f 2 x1 = Guess (Reverse $ MapCtor x1) : diffs xs
where f i x = applyEnv x env{envInput=sample, envCtor=dataCtors sample !! i}
diffs (GuessFld 1 x1:GuessFld 2 x2:xs)
| f 1 x1 == f 1 x2 = GuessCtr 1 False (MapField x2) : diffs xs
where f i x = applyEnv x env{envInput=sample, envField=i}
diffs (GuessFld 2 x2:GuessFld 1 x1:xs)
| f 1 x1 == f 1 x2 = GuessCtr 1 False (Reverse $ MapField x2) : diffs xs
where f i x = applyEnv x env{envInput=sample, envField=i}
diffs (x:xs) = lift box x : diffs xs
diffs [] = []
gssFold o@(OApp op [pre,x,m,y]) = f True (x : follow True y) ++ f False (y : follow False x)
where
follow dir (OApp op2 [pre2,a,m2,b]) | op == op2 && pre == pre2 && m == m2 = a2 : follow dir b2
where (a2,b2) = if dir then (a,b) else (b,a)
follow dir x = [x]
f dir xs | length xs <= 2 = []
f dir xs | pre:_ <- [d | Guess d <- guess pre] = map (lift $ g pre) $ guess $ OList xs
where g pre = Fold (App op $ List [pre,h,fromOut m,t])
(h,t) = if dir then (Head,Tail) else (Tail,Head)
gssFold _ = []
gssApp (OApp "App" [_,OApp "App" [_,x,y],z]) = map (lift Application) $ guess $ OList $ fromApp x ++ [y,z]
where fromApp (OApp "App" [_,x,y]) = fromApp x ++ [y]
fromApp x = [x]
gssApp _ = []
lift :: (DSL -> DSL) -> Guess -> Guess
lift f x = toGuess a (f b)
where (a,b) = fromGuess x
type GuessState = Maybe (Either Int (Int,Bool))
fromGuess :: Guess -> (GuessState, DSL)
fromGuess (Guess x) = (Nothing, x)
fromGuess (GuessFld i x) = (Just (Left i), x)
fromGuess (GuessCtr i b x) = (Just (Right (i,b)), x)
toGuess :: GuessState -> DSL -> Guess
toGuess Nothing = Guess
toGuess (Just (Left i)) = GuessFld i
toGuess (Just (Right (i,b))) = GuessCtr i b
maxim :: [GuessState] -> Maybe GuessState
maxim [] = Just Nothing
maxim [x] = Just x
maxim (Nothing:xs) = maxim xs
maxim (x:Nothing:xs) = maxim $ x:xs
maxim (x1:x2:xs) | x1 == x2 = maxim $ x1:xs
maxim (Just (Right (i1,b1)):Just (Right (i2,b2)):xs) | i1 == i2 = maxim $ Just (Right (i1,max b1 b2)) : xs
maxim _ = Nothing