{-# LANGUAGE CPP, TupleSections #-}
module Conjure.Expr
( module Data.Express
, module Data.Express.Fixtures
, rehole
, funToVar
, recursexpr
, apparentlyTerminates
, mayNotEvaluateArgument
, compareSimplicity
, ifFor
, caseForOrd
, valuesBFS
, holesBFS
, fillBFS
, ($$**)
, ($$|<)
, possibleHoles
, revaluate
, reval
, useMatches
, deholings
, varToConst
, hasAppInstanceOf
, enumerateAppsFor
, enumerateFillings
, digApp
, extractApp
, updateAppAt
, ($!!)
, conflicts
, listConflicts
, module Conjure.Utils
)
where
import Conjure.Utils
import Data.Express
import Data.Express.Utils.Typeable
import Data.Express.Fixtures hiding ((-==-))
import Data.Dynamic
import Control.Applicative ((<$>))
import Test.LeanCheck (mapT, filterT, (\/), delay, productWith, productMaybeWith)
import Test.LeanCheck.Tiers (products)
import Test.LeanCheck.Utils.Types (A, B, C, D, E, F)
compareSimplicity :: Expr -> Expr -> Ordering
compareSimplicity :: Expr -> Expr -> Ordering
compareSimplicity = (Int -> Int -> Ordering
forall a. Ord a => a -> a -> Ordering
compare (Int -> Int -> Ordering)
-> (Expr -> Int) -> Expr -> Expr -> Ordering
forall b c a. (b -> b -> c) -> (a -> b) -> a -> a -> c
`on` [Expr] -> Int
forall a. [a] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length ([Expr] -> Int) -> (Expr -> [Expr]) -> Expr -> Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Expr -> [Expr]
values)
(Expr -> Expr -> Ordering)
-> (Expr -> Expr -> Ordering) -> Expr -> Expr -> Ordering
forall a. Semigroup a => a -> a -> a
<> (Int -> Int -> Ordering
forall a. Ord a => a -> a -> Ordering
compare (Int -> Int -> Ordering)
-> (Expr -> Int) -> Expr -> Expr -> Ordering
forall b c a. (b -> b -> c) -> (a -> b) -> a -> a -> c
`on` [Expr] -> Int
forall a. [a] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length ([Expr] -> Int) -> (Expr -> [Expr]) -> Expr -> Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Expr -> [Expr]
vars)
(Expr -> Expr -> Ordering)
-> (Expr -> Expr -> Ordering) -> Expr -> Expr -> Ordering
forall a. Semigroup a => a -> a -> a
<> (Int -> Int -> Ordering
forall a. Ord a => a -> a -> Ordering
compare (Int -> Int -> Ordering)
-> (Expr -> Int) -> Expr -> Expr -> Ordering
forall b c a. (b -> b -> c) -> (a -> b) -> a -> a -> c
`on` [Expr] -> Int
forall a. [a] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length ([Expr] -> Int) -> (Expr -> [Expr]) -> Expr -> Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Expr -> [Expr]
nubConsts)
funToVar :: Expr -> Expr
funToVar :: Expr -> Expr
funToVar (Expr
ef :$ Expr
ex) = Expr -> Expr
funToVar Expr
ef Expr -> Expr -> Expr
:$ Expr
ex
funToVar ef :: Expr
ef@(Value String
nm Dynamic
_) = String
nm String -> Expr -> Expr
`varAsTypeOf` Expr
ef
varToConst :: Expr -> Expr
varToConst :: Expr -> Expr
varToConst (Value (Char
'_':String
nm) Dynamic
dyn) = String -> Dynamic -> Expr
Value String
nm Dynamic
dyn
varToConst Expr
_ = String -> Expr
forall a. HasCallStack => String -> a
error String
"varToConst: can only be applied to variables"
hasAppInstanceOf :: Expr -> Expr -> Bool
Expr
e hasAppInstanceOf :: Expr -> Expr -> Bool
`hasAppInstanceOf` Expr
efxs = Expr -> Expr
constApp Expr
e Expr -> Expr -> Bool
`hasInstanceOf` Expr -> Expr
constApp Expr
efxs
where
constApp :: Expr -> Expr
constApp Expr
e = Expr
e Expr -> [(Expr, Expr)] -> Expr
//- [(Expr
ef,Expr -> Expr
varToConst Expr
ef)]
(Expr
ef:[Expr]
_) = Expr -> [Expr]
unfoldApp Expr
efxs
recursexpr :: Int -> Expr -> Expr -> Expr
recursexpr :: Int -> Expr -> Expr -> Expr
recursexpr Int
sz Expr
epat = Expr -> Expr
re
where
err :: a
err = String -> a
forall a. HasCallStack => String -> a
error String
"recursexpr: pattern must contain an application of variables"
(Expr
erf:[Expr]
vs) = Expr -> [Expr]
unfoldApp Expr
epat
re :: Expr -> Expr
re Expr
e' | Bool -> Bool
not ((Expr -> Bool) -> [Expr] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
all Expr -> Bool
isVar (Expr
erfExpr -> [Expr] -> [Expr]
forall a. a -> [a] -> [a]
:[Expr]
vs)) = Expr
forall {a}. a
err
| Expr
e Expr -> Expr -> Bool
forall a. Eq a => a -> a -> Bool
== Expr
e' Bool -> Bool -> Bool
|| Expr -> Int
size Expr
e Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
> Int
sz = Expr
e
| Bool
otherwise = Expr -> Expr
re Expr
e
where
e :: Expr
e = Expr -> Expr
re1 Expr
e'
re1 :: Expr -> Expr
re1 Expr
e = case Expr -> [Expr]
unfoldApp Expr
e of
[Expr
e] -> Expr
e
(Expr
ef:[Expr]
exs) | Expr
ef Expr -> Expr -> Bool
forall a. Eq a => a -> a -> Bool
== Expr
erf -> Expr
e' Expr -> [(Expr, Expr)] -> Expr
//- [Expr] -> [Expr] -> [(Expr, Expr)]
forall a b. [a] -> [b] -> [(a, b)]
zip [Expr]
vs [Expr]
exs
| Bool
otherwise -> [Expr] -> Expr
foldApp ((Expr -> Expr) -> [Expr] -> [Expr]
forall a b. (a -> b) -> [a] -> [b]
map Expr -> Expr
re1 (Expr
efExpr -> [Expr] -> [Expr]
forall a. a -> [a] -> [a]
:[Expr]
exs))
apparentlyTerminates :: Expr -> Expr -> Bool
apparentlyTerminates :: Expr -> Expr -> Bool
apparentlyTerminates Expr
eRecursiveCall = Expr -> Bool
at
where
at :: Expr -> Bool
at (Expr
e1 :$ Expr
e2) = (Expr -> Bool
mayNotEvaluateArgument Expr
e1 Bool -> Bool -> Bool
|| Expr -> Bool
at Expr
e2) Bool -> Bool -> Bool
&& Expr -> Bool
at Expr
e1
at Expr
e = Expr
e Expr -> Expr -> Bool
forall a. Eq a => a -> a -> Bool
/= Expr
eRecursiveCall
mayNotEvaluateArgument :: Expr -> Bool
mayNotEvaluateArgument :: Expr -> Bool
mayNotEvaluateArgument (Value String
"if" Dynamic
ce :$ Expr
_ :$ Expr
_) = Bool
True
mayNotEvaluateArgument (Value String
"&&" Dynamic
ce :$ Expr
_) = Bool
True
mayNotEvaluateArgument (Value String
"||" Dynamic
ce :$ Expr
_) = Bool
True
mayNotEvaluateArgument Expr
_ = Bool
False
ifFor :: Typeable a => a -> Expr
ifFor :: forall a. Typeable a => a -> Expr
ifFor a
a = String -> (Bool -> a -> a -> a) -> Expr
forall a. Typeable a => String -> a -> Expr
value String
"if" (\Bool
p a
x a
y -> if Bool
p then a
x else a
y a -> a -> a
forall a. a -> a -> a
`asTypeOf` a
a)
caseForOrd :: Typeable a => a -> Expr
caseForOrd :: forall a. Typeable a => a -> Expr
caseForOrd a
a = String -> (Ordering -> a -> a -> a -> a) -> Expr
forall a. Typeable a => String -> a -> Expr
value String
"case" (\Ordering
o a
x a
y a
z -> case Ordering
o of Ordering
LT -> a
x; Ordering
EQ -> a
y; Ordering
GT -> a
z a -> a -> a
forall a. a -> a -> a
`asTypeOf` a
a)
valuesBFS :: Expr -> [Expr]
valuesBFS :: Expr -> [Expr]
valuesBFS = [[Expr]] -> [Expr]
forall (t :: * -> *) a. Foldable t => t [a] -> [a]
concat ([[Expr]] -> [Expr]) -> (Expr -> [[Expr]]) -> Expr -> [Expr]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Expr -> [[Expr]]
bfs
where
bfs :: Expr -> [[Expr]]
bfs :: Expr -> [[Expr]]
bfs (Expr
ef :$ Expr
ex) = [] [Expr] -> [[Expr]] -> [[Expr]]
forall a. a -> [a] -> [a]
: [[Expr]] -> [[Expr]] -> [[Expr]]
forall a. Monoid a => [a] -> [a] -> [a]
mzip (Expr -> [[Expr]]
bfs Expr
ef) (Expr -> [[Expr]]
bfs Expr
ex)
bfs Expr
e = [[Expr
e]]
holesBFS :: Expr -> [Expr]
holesBFS :: Expr -> [Expr]
holesBFS = (Expr -> Bool) -> [Expr] -> [Expr]
forall a. (a -> Bool) -> [a] -> [a]
filter Expr -> Bool
isHole ([Expr] -> [Expr]) -> (Expr -> [Expr]) -> Expr -> [Expr]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Expr -> [Expr]
valuesBFS
fillBFS :: Expr -> Expr -> Expr
fillBFS :: Expr -> Expr -> Expr
fillBFS Expr
e Expr
e' = (Expr, Maybe Int) -> Expr
forall a b. (a, b) -> a
fst (Expr -> (Expr, Maybe Int)
f Expr
e)
where
f :: Expr -> (Expr,Maybe Int)
f :: Expr -> (Expr, Maybe Int)
f (Expr
ef :$ Expr
ex) = case (Maybe Int
mf, Maybe Int
mx) of
(Maybe Int
Nothing, Maybe Int
Nothing) -> (Expr
ef Expr -> Expr -> Expr
:$ Expr
ex, Maybe Int
forall a. Maybe a
Nothing)
(Just Int
lf, Maybe Int
Nothing) -> (Expr
ef' Expr -> Expr -> Expr
:$ Expr
ex, Int -> Maybe Int
forall a. a -> Maybe a
Just (Int -> Maybe Int) -> Int -> Maybe Int
forall a b. (a -> b) -> a -> b
$ Int
lfInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1)
(Maybe Int
Nothing, Just Int
lx) -> (Expr
ef Expr -> Expr -> Expr
:$ Expr
ex', Int -> Maybe Int
forall a. a -> Maybe a
Just (Int -> Maybe Int) -> Int -> Maybe Int
forall a b. (a -> b) -> a -> b
$ Int
lxInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1)
(Just Int
lf, Just Int
lx) | Int
lf Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
lx -> (Expr
ef' Expr -> Expr -> Expr
:$ Expr
ex, Int -> Maybe Int
forall a. a -> Maybe a
Just (Int -> Maybe Int) -> Int -> Maybe Int
forall a b. (a -> b) -> a -> b
$ Int
lfInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1)
| Bool
otherwise -> (Expr
ef Expr -> Expr -> Expr
:$ Expr
ex', Int -> Maybe Int
forall a. a -> Maybe a
Just (Int -> Maybe Int) -> Int -> Maybe Int
forall a b. (a -> b) -> a -> b
$ Int
lxInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1)
where
(Expr
ef', Maybe Int
mf) = Expr -> (Expr, Maybe Int)
f Expr
ef
(Expr
ex', Maybe Int
mx) = Expr -> (Expr, Maybe Int)
f Expr
ex
f Expr
e | Expr -> Bool
isHole Expr
e Bool -> Bool -> Bool
&& Expr -> TypeRep
typ Expr
e TypeRep -> TypeRep -> Bool
forall a. Eq a => a -> a -> Bool
== Expr -> TypeRep
typ Expr
e' = (Expr
e', Int -> Maybe Int
forall a. a -> Maybe a
Just Int
0)
| Bool
otherwise = (Expr
e, Maybe Int
forall a. Maybe a
Nothing)
($$**) :: Expr -> Expr -> Maybe Expr
Expr
e1 $$** :: Expr -> Expr -> Maybe Expr
$$** Expr
e2 = Expr -> Maybe Expr
forall a. a -> Maybe a
Just (Expr -> Maybe Expr) -> Expr -> Maybe Expr
forall a b. (a -> b) -> a -> b
$ Expr
e1 Expr -> Expr -> Expr
:$ Expr
e2
($$|<) :: Expr -> Expr -> Maybe Expr
Expr
e1 $$|< :: Expr -> Expr -> Maybe Expr
$$|< Expr
e2 = if TypeRep -> Bool
isFunTy TypeRep
t1 Bool -> Bool -> Bool
&& TypeRep -> Int
tyArity (TypeRep -> TypeRep
argumentTy TypeRep
t1) Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== TypeRep -> Int
tyArity TypeRep
t2
then Expr -> Maybe Expr
forall a. a -> Maybe a
Just (Expr -> Maybe Expr) -> Expr -> Maybe Expr
forall a b. (a -> b) -> a -> b
$ Expr
e1 Expr -> Expr -> Expr
:$ Expr
e2
else Maybe Expr
forall a. Maybe a
Nothing
where
t1 :: TypeRep
t1 = Expr -> TypeRep
ktyp Expr
e1
t2 :: TypeRep
t2 = Expr -> TypeRep
ktyp Expr
e2
ktyp :: Expr -> TypeRep
ktyp :: Expr -> TypeRep
ktyp (Expr
e1 :$ Expr
e2) = TypeRep -> TypeRep
resultTy (Expr -> TypeRep
ktyp Expr
e1)
ktyp Expr
e = Expr -> TypeRep
typ Expr
e
possibleHoles :: [Expr] -> [Expr]
possibleHoles :: [Expr] -> [Expr]
possibleHoles = [Expr] -> [Expr]
forall a. Ord a => [a] -> [a]
nubSort ([Expr] -> [Expr]) -> ([Expr] -> [Expr]) -> [Expr] -> [Expr]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Expr] -> [Expr]
ph ([Expr] -> [Expr]) -> ([Expr] -> [Expr]) -> [Expr] -> [Expr]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Expr] -> [Expr]
forall a. Ord a => [a] -> [a]
nubSort ([Expr] -> [Expr]) -> ([Expr] -> [Expr]) -> [Expr] -> [Expr]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Expr -> Expr) -> [Expr] -> [Expr]
forall a b. (a -> b) -> [a] -> [b]
map Expr -> Expr
holeAsTypeOf
where
ph :: [Expr] -> [Expr]
ph [Expr]
hs = case [Expr] -> [Expr]
forall a. Ord a => [a] -> [a]
nubSort ([Expr] -> [Expr]) -> [Expr] -> [Expr]
forall a b. (a -> b) -> a -> b
$ [Expr]
hs [Expr] -> [Expr] -> [Expr]
forall a. [a] -> [a] -> [a]
++ [Expr -> Expr
holeAsTypeOf Expr
hfx | Expr
hf <- [Expr]
hs, Expr
hx <- [Expr]
hs, Just Expr
hfx <- [Expr
hf Expr -> Expr -> Maybe Expr
$$ Expr
hx]] of
[Expr]
hs' | [Expr]
hs' [Expr] -> [Expr] -> Bool
forall a. Eq a => a -> a -> Bool
== [Expr]
hs -> [Expr]
hs
| Bool
otherwise -> [Expr] -> [Expr]
ph [Expr]
hs'
enumerateAppsFor :: Expr -> (Expr -> Bool) -> [Expr] -> [[Expr]]
enumerateAppsFor :: Expr -> (Expr -> Bool) -> [Expr] -> [[Expr]]
enumerateAppsFor Expr
h Expr -> Bool
keep [Expr]
es = Expr -> [[Expr]]
for Expr
h
where
hs :: [Expr]
hs :: [Expr]
hs = [Expr] -> [Expr]
possibleHoles [Expr]
es
for :: Expr -> [[Expr]]
for :: Expr -> [[Expr]]
for Expr
h = (Expr -> Bool) -> [Expr] -> [Expr]
forall a. (a -> Bool) -> [a] -> [a]
filter (\Expr
e -> Expr -> TypeRep
typ Expr
h TypeRep -> TypeRep -> Bool
forall a. Eq a => a -> a -> Bool
== Expr -> TypeRep
typ Expr
e) [Expr]
es [Expr] -> [[Expr]] -> [[Expr]]
forall a. a -> [a] -> [a]
: [[Expr]]
apps
where
apps :: [[Expr]]
apps = ([[Expr]] -> [[Expr]] -> [[Expr]])
-> [[Expr]] -> [[[Expr]]] -> [[Expr]]
forall a b. (a -> b -> b) -> b -> [a] -> b
forall (t :: * -> *) a b.
Foldable t =>
(a -> b -> b) -> b -> t a -> b
foldr [[Expr]] -> [[Expr]] -> [[Expr]]
forall a. [[a]] -> [[a]] -> [[a]]
(\/) []
[ (Expr -> Bool) -> [[Expr]] -> [[Expr]]
forall a. (a -> Bool) -> [[a]] -> [[a]]
filterT Expr -> Bool
keep ([[Expr]] -> [[Expr]]) -> [[Expr]] -> [[Expr]]
forall a b. (a -> b) -> a -> b
$ (Expr -> Expr -> Expr) -> [[Expr]] -> [[Expr]] -> [[Expr]]
forall a b c. (a -> b -> c) -> [[a]] -> [[b]] -> [[c]]
fliproductWith Expr -> Expr -> Expr
(:$) (Expr -> [[Expr]]
for Expr
hf) (Expr -> [[Expr]]
for Expr
hx)
| Expr
hf <- [Expr]
hs
, Expr
hx <- [Expr]
hs
, Just Expr
hfx <- [Expr
hf Expr -> Expr -> Maybe Expr
$$ Expr
hx]
, Expr -> TypeRep
typ Expr
h TypeRep -> TypeRep -> Bool
forall a. Eq a => a -> a -> Bool
== Expr -> TypeRep
typ Expr
hfx
]
enumerateFillings :: Expr -> [[Expr]] -> [[Expr]]
enumerateFillings :: Expr -> [[Expr]] -> [[Expr]]
enumerateFillings Expr
e = ([Expr] -> Expr) -> [[[Expr]]] -> [[Expr]]
forall a b. (a -> b) -> [[a]] -> [[b]]
mapT (Expr -> [Expr] -> Expr
fill Expr
e)
([[[Expr]]] -> [[Expr]])
-> ([[Expr]] -> [[[Expr]]]) -> [[Expr]] -> [[Expr]]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [[[Expr]]] -> [[[Expr]]]
forall a. [[[a]]] -> [[[a]]]
products
([[[Expr]]] -> [[[Expr]]])
-> ([[Expr]] -> [[[Expr]]]) -> [[Expr]] -> [[[Expr]]]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Int -> [[Expr]] -> [[[Expr]]]
forall a. Int -> a -> [a]
replicate ([Expr] -> Int
forall a. [a] -> Int
forall (t :: * -> *) a. Foldable t => t a -> Int
length ([Expr] -> Int) -> [Expr] -> Int
forall a b. (a -> b) -> a -> b
$ Expr -> [Expr]
holes Expr
e)
recursiveToDynamic :: (Expr,Expr) -> Int -> Expr -> Maybe Dynamic
recursiveToDynamic :: (Expr, Expr) -> Int -> Expr -> Maybe Dynamic
recursiveToDynamic (Expr
efxs, Expr
ebody) Int
n = ((Int, Int, Dynamic) -> Dynamic)
-> Maybe (Int, Int, Dynamic) -> Maybe Dynamic
forall a b. (a -> b) -> Maybe a -> Maybe b
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
fmap (\(Int
_,Int
_,Dynamic
d) -> Dynamic
d) (Maybe (Int, Int, Dynamic) -> Maybe Dynamic)
-> (Expr -> Maybe (Int, Int, Dynamic)) -> Expr -> Maybe Dynamic
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Int -> Int -> Expr -> Maybe (Int, Int, Dynamic)
re (Int
n Int -> Int -> Int
forall a. Num a => a -> a -> a
* Expr -> Int
size Expr
ebody) Int
n
where
(Expr
ef':[Expr]
exs') = Expr -> [Expr]
unfoldApp Expr
efxs
rev :: Typeable a => Int -> Int -> Expr -> Maybe (Int, Int, a)
rev :: forall a. Typeable a => Int -> Int -> Expr -> Maybe (Int, Int, a)
rev Int
m Int
n Expr
e = case Int -> Int -> Expr -> Maybe (Int, Int, Dynamic)
re Int
m Int
n Expr
e of
Maybe (Int, Int, Dynamic)
Nothing -> Maybe (Int, Int, a)
forall a. Maybe a
Nothing
Just (Int
m,Int
n,Dynamic
d) -> case Dynamic -> Maybe a
forall a. Typeable a => Dynamic -> Maybe a
fromDynamic Dynamic
d of
Maybe a
Nothing -> Maybe (Int, Int, a)
forall a. Maybe a
Nothing
Just a
x -> (Int, Int, a) -> Maybe (Int, Int, a)
forall a. a -> Maybe a
Just (Int
m, Int
n, a
x)
re :: Int -> Int -> Expr -> Maybe (Int, Int, Dynamic)
re :: Int -> Int -> Expr -> Maybe (Int, Int, Dynamic)
re Int
m Int
n Expr
_ | Int
n Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
0 = String -> Maybe (Int, Int, Dynamic)
forall a. HasCallStack => String -> a
error String
"recursiveToDynamic: recursion limit reached"
re Int
m Int
n Expr
_ | Int
m Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
0 = String -> Maybe (Int, Int, Dynamic)
forall a. HasCallStack => String -> a
error String
"recursiveToDynamic: evaluation limit reached"
re Int
m Int
n (Value String
"if" Dynamic
_ :$ Expr
ec :$ Expr
ex :$ Expr
ey) = case Int -> Int -> Expr -> Maybe (Int, Int, Bool)
forall a. Typeable a => Int -> Int -> Expr -> Maybe (Int, Int, a)
rev Int
m Int
n Expr
ec of
Maybe (Int, Int, Bool)
Nothing -> Maybe (Int, Int, Dynamic)
forall a. Maybe a
Nothing
Just (Int
m,Int
n,Bool
True) -> Int -> Int -> Expr -> Maybe (Int, Int, Dynamic)
re Int
m Int
n Expr
ex
Just (Int
m,Int
n,Bool
False) -> Int -> Int -> Expr -> Maybe (Int, Int, Dynamic)
re Int
m Int
n Expr
ey
re Int
m Int
n (Value String
"||" Dynamic
_ :$ Expr
ep :$ Expr
eq) = case Int -> Int -> Expr -> Maybe (Int, Int, Bool)
forall a. Typeable a => Int -> Int -> Expr -> Maybe (Int, Int, a)
rev Int
m Int
n Expr
ep of
Maybe (Int, Int, Bool)
Nothing -> Maybe (Int, Int, Dynamic)
forall a. Maybe a
Nothing
Just (Int
m,Int
n,Bool
True) -> (Int
m,Int
n,) (Dynamic -> (Int, Int, Dynamic))
-> Maybe Dynamic -> Maybe (Int, Int, Dynamic)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Expr -> Maybe Dynamic
toDynamic (Bool -> Expr
forall a. (Typeable a, Show a) => a -> Expr
val Bool
True)
Just (Int
m,Int
n,Bool
False) -> Int -> Int -> Expr -> Maybe (Int, Int, Dynamic)
re Int
m Int
n Expr
eq
re Int
m Int
n (Value String
"&&" Dynamic
_ :$ Expr
ep :$ Expr
eq) = case Int -> Int -> Expr -> Maybe (Int, Int, Bool)
forall a. Typeable a => Int -> Int -> Expr -> Maybe (Int, Int, a)
rev Int
m Int
n Expr
ep of
Maybe (Int, Int, Bool)
Nothing -> Maybe (Int, Int, Dynamic)
forall a. Maybe a
Nothing
Just (Int
m,Int
n,Bool
True) -> Int -> Int -> Expr -> Maybe (Int, Int, Dynamic)
re Int
m Int
n Expr
eq
Just (Int
m,Int
n,Bool
False) -> (Int
m,Int
n,) (Dynamic -> (Int, Int, Dynamic))
-> Maybe Dynamic -> Maybe (Int, Int, Dynamic)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Expr -> Maybe Dynamic
toDynamic (Bool -> Expr
forall a. (Typeable a, Show a) => a -> Expr
val Bool
False)
re Int
m Int
n Expr
e = case Expr -> [Expr]
unfoldApp Expr
e of
[] -> String -> Maybe (Int, Int, Dynamic)
forall a. HasCallStack => String -> a
error String
"recursiveToDynamic: empty application unfold"
[Expr
e] -> (Int
mInt -> Int -> Int
forall a. Num a => a -> a -> a
-Int
1,Int
n,) (Dynamic -> (Int, Int, Dynamic))
-> Maybe Dynamic -> Maybe (Int, Int, Dynamic)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Expr -> Maybe Dynamic
toDynamic Expr
e
(Expr
ef:[Expr]
exs) | Expr
ef Expr -> Expr -> Bool
forall a. Eq a => a -> a -> Bool
== Expr
ef' -> Int -> Int -> Expr -> Maybe (Int, Int, Dynamic)
re Int
m (Int
nInt -> Int -> Int
forall a. Num a => a -> a -> a
-Int
1) (Expr -> Maybe (Int, Int, Dynamic))
-> Expr -> Maybe (Int, Int, Dynamic)
forall a b. (a -> b) -> a -> b
$ Expr
ebody Expr -> [(Expr, Expr)] -> Expr
//- [Expr] -> [Expr] -> [(Expr, Expr)]
forall a b. [a] -> [b] -> [(a, b)]
zip [Expr]
exs' [Expr]
exs
| Bool
otherwise -> (Maybe (Int, Int, Dynamic) -> Expr -> Maybe (Int, Int, Dynamic))
-> Maybe (Int, Int, Dynamic) -> [Expr] -> Maybe (Int, Int, Dynamic)
forall b a. (b -> a -> b) -> b -> [a] -> b
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl Maybe (Int, Int, Dynamic) -> Expr -> Maybe (Int, Int, Dynamic)
($$) (Int -> Int -> Expr -> Maybe (Int, Int, Dynamic)
re Int
m Int
n Expr
ef) [Expr]
exs
Just (Int
m,Int
n,Dynamic
d1) $$ :: Maybe (Int, Int, Dynamic) -> Expr -> Maybe (Int, Int, Dynamic)
$$ Expr
e2 = case Int -> Int -> Expr -> Maybe (Int, Int, Dynamic)
re Int
m Int
n Expr
e2 of
Maybe (Int, Int, Dynamic)
Nothing -> Maybe (Int, Int, Dynamic)
forall a. Maybe a
Nothing
Just (Int
m', Int
n', Dynamic
d2) -> (Int
m',Int
n',) (Dynamic -> (Int, Int, Dynamic))
-> Maybe Dynamic -> Maybe (Int, Int, Dynamic)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Dynamic -> Dynamic -> Maybe Dynamic
dynApply Dynamic
d1 Dynamic
d2
Maybe (Int, Int, Dynamic)
_ $$ Expr
_ = Maybe (Int, Int, Dynamic)
forall a. Maybe a
Nothing
revaluate :: Typeable a => (Expr,Expr) -> Int -> Expr -> Maybe a
revaluate :: forall a. Typeable a => (Expr, Expr) -> Int -> Expr -> Maybe a
revaluate (Expr, Expr)
dfn Int
n Expr
e = (Expr, Expr) -> Int -> Expr -> Maybe Dynamic
recursiveToDynamic (Expr, Expr)
dfn Int
n Expr
e Maybe Dynamic -> (Dynamic -> Maybe a) -> Maybe a
forall a b. Maybe a -> (a -> Maybe b) -> Maybe b
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= Dynamic -> Maybe a
forall a. Typeable a => Dynamic -> Maybe a
fromDynamic
reval :: Typeable a => (Expr,Expr) -> Int -> a -> Expr -> a
reval :: forall a. Typeable a => (Expr, Expr) -> Int -> a -> Expr -> a
reval (Expr, Expr)
dfn Int
n a
x = a -> Maybe a -> a
forall a. a -> Maybe a -> a
fromMaybe a
x (Maybe a -> a) -> (Expr -> Maybe a) -> Expr -> a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Expr, Expr) -> Int -> Expr -> Maybe a
forall a. Typeable a => (Expr, Expr) -> Int -> Expr -> Maybe a
revaluate (Expr, Expr)
dfn Int
n
fliproductWith :: (a->b->c) -> [[a]] -> [[b]] -> [[c]]
fliproductWith :: forall a b c. (a -> b -> c) -> [[a]] -> [[b]] -> [[c]]
fliproductWith a -> b -> c
_ [] [[b]]
_ = []
fliproductWith a -> b -> c
_ [[a]]
_ [] = []
fliproductWith a -> b -> c
f [[a]]
xss ([b]
ys:[[b]]
yss) = ([a] -> [c]) -> [[a]] -> [[c]]
forall a b. (a -> b) -> [a] -> [b]
map ([a] -> [b] -> [c]
** [b]
ys) [[a]]
xss
[[c]] -> [[c]] -> [[c]]
forall a. [[a]] -> [[a]] -> [[a]]
\/ [[c]] -> [[c]]
forall a. [[a]] -> [[a]]
delay ((a -> b -> c) -> [[a]] -> [[b]] -> [[c]]
forall a b c. (a -> b -> c) -> [[a]] -> [[b]] -> [[c]]
productWith a -> b -> c
f [[a]]
xss [[b]]
yss)
where
[a]
xs ** :: [a] -> [b] -> [c]
** [b]
ys = [a
x a -> b -> c
`f` b
y | a
x <- [a]
xs, b
y <- [b]
ys]
useMatches :: [Expr] -> [Expr] -> [[(Expr,Expr)]]
useMatches :: [Expr] -> [Expr] -> [[(Expr, Expr)]]
useMatches [] [] = [[]]
useMatches [] [Expr]
es = []
useMatches [Expr]
es [] = []
useMatches (Expr
e:[Expr]
es) [Expr]
es' = [[[(Expr, Expr)]]] -> [[(Expr, Expr)]]
forall (t :: * -> *) a. Foldable t => t [a] -> [a]
concat
[ ([(Expr, Expr)] -> [(Expr, Expr)])
-> [[(Expr, Expr)]] -> [[(Expr, Expr)]]
forall a b. (a -> b) -> [a] -> [b]
map ((Expr
e,Expr
e')(Expr, Expr) -> [(Expr, Expr)] -> [(Expr, Expr)]
forall a. a -> [a] -> [a]
:) ([Expr] -> [Expr] -> [[(Expr, Expr)]]
useMatches [Expr]
es [Expr]
es')
| (Expr
e',[Expr]
es') <- (Expr -> [Expr] -> Bool) -> [Expr] -> [(Expr, [Expr])]
forall a. (a -> [a] -> Bool) -> [a] -> [(a, [a])]
choicesThat (\Expr
e' [Expr]
_ -> (Expr -> Bool) -> [Expr] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
any (Expr -> [Expr] -> Bool
forall a. Eq a => a -> [a] -> Bool
forall (t :: * -> *) a. (Foldable t, Eq a) => a -> t a -> Bool
`elem` Expr -> [Expr]
vars Expr
e') (Expr -> [Expr]
vars Expr
e)) [Expr]
es'
]
rehole :: Expr -> Expr
rehole :: Expr -> Expr
rehole (Expr
e1 :$ Expr
e2) = Expr -> Expr
rehole Expr
e1 Expr -> Expr -> Expr
:$ Expr -> Expr
rehole Expr
e2
rehole Expr
e | Expr -> Bool
isVar Expr
e = String
"" String -> Expr -> Expr
`varAsTypeOf` Expr
e
| Bool
otherwise = Expr
e
deholings :: Expr -> Expr -> [Expr]
deholings :: Expr -> Expr -> [Expr]
deholings Expr
e' = Expr -> [Expr]
deh
where
deh :: Expr -> [Expr]
deh (Expr
e1 :$ Expr
e2) = (Expr -> Expr) -> [Expr] -> [Expr]
forall a b. (a -> b) -> [a] -> [b]
map (Expr -> Expr -> Expr
:$ Expr
e2) (Expr -> [Expr]
deh Expr
e1)
[Expr] -> [Expr] -> [Expr]
forall a. [a] -> [a] -> [a]
++ (Expr -> Expr) -> [Expr] -> [Expr]
forall a b. (a -> b) -> [a] -> [b]
map (Expr
e1 Expr -> Expr -> Expr
:$) (Expr -> [Expr]
deh Expr
e2)
deh Expr
e = if Expr -> TypeRep
typ Expr
e TypeRep -> TypeRep -> Bool
forall a. Eq a => a -> a -> Bool
== Expr -> TypeRep
typ Expr
e' Bool -> Bool -> Bool
&& Expr -> Bool
isHole Expr
e
then [Expr
e']
else []
digApp :: Int -> Expr -> Expr
digApp :: Int -> Expr -> Expr
digApp Int
i = Int -> (Expr -> Expr) -> Expr -> Expr
updateAppAt Int
i Expr -> Expr
holeAsTypeOf
updateAppAt :: Int -> (Expr -> Expr) -> Expr -> Expr
updateAppAt :: Int -> (Expr -> Expr) -> Expr -> Expr
updateAppAt Int
i Expr -> Expr
f = [Expr] -> Expr
foldApp ([Expr] -> Expr) -> (Expr -> [Expr]) -> Expr -> Expr
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Int -> (Expr -> Expr) -> [Expr] -> [Expr]
forall a. Int -> (a -> a) -> [a] -> [a]
updateAt Int
i Expr -> Expr
f ([Expr] -> [Expr]) -> (Expr -> [Expr]) -> Expr -> [Expr]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Expr -> [Expr]
unfoldApp
($!!) :: Expr -> Int -> Expr
Expr
e $!! :: Expr -> Int -> Expr
$!! Int
i = Expr -> [Expr]
unfoldApp Expr
e [Expr] -> Int -> Expr
forall a. HasCallStack => [a] -> Int -> a
!! Int
i
extractApp :: Int -> Expr -> (Expr,Expr)
Int
i Expr
efxs = ([Expr] -> Expr
foldApp ([Expr] -> Expr) -> [Expr] -> Expr
forall a b. (a -> b) -> a -> b
$ Int -> (Expr -> Expr) -> [Expr] -> [Expr]
forall a. Int -> (a -> a) -> [a] -> [a]
updateAt Int
i Expr -> Expr
holeAsTypeOf [Expr]
es, [Expr]
es [Expr] -> Int -> Expr
forall a. HasCallStack => [a] -> Int -> a
!! Int
i)
where
es :: [Expr]
es = Expr -> [Expr]
unfoldApp Expr
efxs
conflicts :: Expr -> Expr -> [(Expr,Expr)]
conflicts :: Expr -> Expr -> [(Expr, Expr)]
conflicts Expr
e1 Expr
e2 | Expr -> TypeRep
typ Expr
e1 TypeRep -> TypeRep -> Bool
forall a. Eq a => a -> a -> Bool
/= Expr -> TypeRep
typ Expr
e2 = [(Expr
e1,Expr
e2)]
conflicts (Expr
ef :$ Expr
ex) (Expr
eg :$ Expr
ey) = Expr -> Expr -> [(Expr, Expr)]
conflicts Expr
ef Expr
eg [(Expr, Expr)] -> [(Expr, Expr)] -> [(Expr, Expr)]
forall a. Ord a => [a] -> [a] -> [a]
+++ Expr -> Expr -> [(Expr, Expr)]
conflicts Expr
ex Expr
ey
conflicts Expr
e1 Expr
e2 = [(Expr
e1,Expr
e2) | Expr
e1 Expr -> Expr -> Bool
forall a. Eq a => a -> a -> Bool
/= Expr
e2]
listConflicts :: [Expr] -> [[Expr]]
listConflicts :: [Expr] -> [[Expr]]
listConflicts [Expr]
es
| Bool -> Bool
not ((Expr -> TypeRep) -> [Expr] -> Bool
forall b a. Eq b => (a -> b) -> [a] -> Bool
allEqualOn Expr -> TypeRep
typ [Expr]
es) = [[Expr]
es]
| (Expr -> Bool) -> [Expr] -> Bool
forall (t :: * -> *) a. Foldable t => (a -> Bool) -> t a -> Bool
all Expr -> Bool
isApp [Expr]
es = [Expr] -> [[Expr]]
listConflicts [Expr
ef | Expr
ef :$ Expr
_ <- [Expr]
es]
[[Expr]] -> [[Expr]] -> [[Expr]]
forall a. Ord a => [a] -> [a] -> [a]
+++ [Expr] -> [[Expr]]
listConflicts [Expr
ex | Expr
_ :$ Expr
ex <- [Expr]
es]
| Bool
otherwise = [[Expr]
es | Bool -> Bool
not ([Expr] -> Bool
forall a. Eq a => [a] -> Bool
allEqual [Expr]
es)]
where
fun :: Expr -> Expr
fun (Expr
ef :$ Expr
_) = Expr
ef
arg :: Expr -> Expr
arg (Expr
_ :$ Expr
ex) = Expr
ex
instance Express A where expr :: A -> Expr
expr = A -> Expr
forall a. (Typeable a, Show a) => a -> Expr
val
instance Express B where expr :: B -> Expr
expr = B -> Expr
forall a. (Typeable a, Show a) => a -> Expr
val
instance Express C where expr :: C -> Expr
expr = C -> Expr
forall a. (Typeable a, Show a) => a -> Expr
val
instance Express D where expr :: D -> Expr
expr = D -> Expr
forall a. (Typeable a, Show a) => a -> Expr
val
instance Express E where expr :: E -> Expr
expr = E -> Expr
forall a. (Typeable a, Show a) => a -> Expr
val
instance Express F where expr :: F -> Expr
expr = F -> Expr
forall a. (Typeable a, Show a) => a -> Expr
val