init
authorJulien Moutinho <julm+symantic@autogeree.net>
Thu, 27 Oct 2016 20:54:52 +0000 (22:54 +0200)
committerJulien Moutinho <julm+symantic@autogeree.net>
Sun, 30 Oct 2016 10:45:31 +0000 (11:45 +0100)
30 files changed:
Language/Symantic/AST/Test.hs
Language/Symantic/Expr.hs
Language/Symantic/Expr/Applicative.hs
Language/Symantic/Expr/Applicative/Test.hs
Language/Symantic/Expr/Bool/Test.hs
Language/Symantic/Expr/Either.hs [new file with mode: 0644]
Language/Symantic/Expr/Eq/Test.hs
Language/Symantic/Expr/From.hs
Language/Symantic/Expr/Functor.hs
Language/Symantic/Expr/Functor/Test.hs
Language/Symantic/Expr/If/Test.hs
Language/Symantic/Expr/Int/Test.hs
Language/Symantic/Expr/Lambda.hs
Language/Symantic/Expr/List.hs
Language/Symantic/Expr/Map.hs
Language/Symantic/Expr/Maybe.hs
Language/Symantic/Expr/Maybe/Test.hs
Language/Symantic/Expr/Monad.hs [new file with mode: 0644]
Language/Symantic/Expr/Traversable.hs [new file with mode: 0644]
Language/Symantic/Expr/Traversable/HLint.hs [new symlink]
Language/Symantic/Expr/Traversable/Test.hs [new file with mode: 0644]
Language/Symantic/Expr/Tuple.hs
Language/Symantic/Repr/Dup.hs
Language/Symantic/Repr/Host.hs
Language/Symantic/Repr/Host/Test.hs
Language/Symantic/Repr/Text.hs
Language/Symantic/Type.hs
Language/Symantic/Type/Either.hs [new file with mode: 0644]
Language/Symantic/Type/Tuple.hs
symantic.cabal

index 4299ccb655c45ae4ee372f8fb3b6f856afb9f140..82079f51fad31d2e1b3100db7546b5766742e163 100644 (file)
@@ -643,8 +643,6 @@ instance -- Expr_from AST Expr_Functor
  ( Eq_Type (Type_Root_of_Expr root)
  , Type_from AST (Type_Root_of_Expr root)
  , Expr_from AST root
- , String_from_Type (Type_Root_of_Expr root)
- -- , Lift_Type   Type_Var1      (Type_of_Expr root)
  , Lift_Type   (Type_Fun lam) (Type_of_Expr root)
  , Unlift_Type (Type_Fun lam) (Type_of_Expr root)
  , Unlift_Type1 (Type_of_Expr root)
@@ -662,8 +660,6 @@ instance -- Expr_from AST Expr_Applicative
  ( Eq_Type (Type_Root_of_Expr root)
  , Type1_from AST (Type_Root_of_Expr root)
  , Expr_from AST root
- , String_from_Type (Type_Root_of_Expr root)
- -- , Lift_Type   Type_Var1      (Type_of_Expr root)
  , Lift_Type   (Type_Fun lam) (Type_of_Expr root)
  , Unlift_Type (Type_Fun lam) (Type_of_Expr root)
  , Eq_Type1 (Type_Root_of_Expr root)
@@ -678,3 +674,21 @@ instance -- Expr_from AST Expr_Applicative
                 AST "pure" asts -> from_ast2 asts pure_from     ex ast ctx k
                 AST "<*>"  asts -> from_ast2 asts ltstargt_from ex ast ctx k
                 _ -> Left $ error_expr_unsupported ex ast
+instance -- Expr_from AST Expr_Traversable
+ ( Eq_Type (Type_Root_of_Expr root)
+ , Type1_from AST (Type_Root_of_Expr root)
+ , Expr_from AST root
+ , Lift_Type   (Type_Fun lam) (Type_of_Expr root)
+ , Unlift_Type (Type_Fun lam) (Type_of_Expr root)
+ , Eq_Type1 (Type_Root_of_Expr root)
+ , Unlift_Type1 (Type_of_Expr root)
+ , Lift_Error_Expr (Error_Expr_of_Root AST root) (Error_of_Expr AST root)
+ , Constraint_Type1 Applicative (Type_Root_of_Expr root)
+ , Constraint_Type1 Traversable (Type_Root_of_Expr root)
+ , Root_of_Expr root ~ root
+ , IBool (Is_Last_Expr (Expr_Traversable lam root) root)
+ ) => Expr_from AST (Expr_Traversable lam root) where
+       expr_from ex ast ctx k =
+               case ast of
+                AST "traverse"  asts -> from_ast2 asts traverse_from ex ast ctx k
+                _ -> Left $ error_expr_unsupported ex ast
index eed673cc65f6a5dd4a5acb86125241649b1cdf04..bd0cb08bf17fbefc5d817c28a956579cf2b219fb 100644 (file)
@@ -3,6 +3,7 @@ module Language.Symantic.Expr
  ( module Language.Symantic.Expr.Alt
  , module Language.Symantic.Expr.Applicative
  , module Language.Symantic.Expr.Bool
+ , module Language.Symantic.Expr.Either
  , module Language.Symantic.Expr.Eq
  , module Language.Symantic.Expr.Error
  , module Language.Symantic.Expr.From
@@ -13,14 +14,17 @@ module Language.Symantic.Expr
  , module Language.Symantic.Expr.List
  , module Language.Symantic.Expr.Map
  , module Language.Symantic.Expr.Maybe
+ , module Language.Symantic.Expr.Monad
  , module Language.Symantic.Expr.Ord
  , module Language.Symantic.Expr.Root
+ , module Language.Symantic.Expr.Traversable
  , module Language.Symantic.Expr.Tuple
  ) where
 
 import Language.Symantic.Expr.Alt
 import Language.Symantic.Expr.Applicative
 import Language.Symantic.Expr.Bool
+import Language.Symantic.Expr.Either
 import Language.Symantic.Expr.Eq
 import Language.Symantic.Expr.Error
 import Language.Symantic.Expr.From
@@ -31,6 +35,8 @@ import Language.Symantic.Expr.Lambda
 import Language.Symantic.Expr.List
 import Language.Symantic.Expr.Map
 import Language.Symantic.Expr.Maybe
+import Language.Symantic.Expr.Monad
 import Language.Symantic.Expr.Ord
 import Language.Symantic.Expr.Root
+import Language.Symantic.Expr.Traversable
 import Language.Symantic.Expr.Tuple
index 71620214ba3f66f46b4e131cbfa0e79ecd4d72f0..cf56c0b135a8a48d74d56039c330ba55f4fd1ece 100644 (file)
@@ -7,15 +7,15 @@
 {-# LANGUAGE ScopedTypeVariables #-}
 {-# LANGUAGE TypeFamilies #-}
 {-# LANGUAGE TypeOperators #-}
-{-# LANGUAGE UndecidableInstances #-}
 {-# OPTIONS_GHC -fno-warn-orphans #-}
 -- | Expression for 'Applicative'.
 module Language.Symantic.Expr.Applicative where
 
 import Control.Applicative (Applicative)
+-- import qualified Control.Applicative as Applicative
 import Data.Proxy (Proxy(..))
 import Data.Type.Equality ((:~:)(Refl))
-import Prelude hiding (Applicative(..))
+import Prelude hiding ((<$>), Applicative(..))
 
 import Language.Symantic.Type
 import Language.Symantic.Trans.Common
@@ -27,24 +27,36 @@ import Language.Symantic.Expr.Functor
 
 -- * Class 'Sym_Applicative'
 -- | Symantic.
-class Sym_Applicative repr where
-       pure  :: Applicative f => repr a -> repr (f a)
-       -- (*>)  :: Applicative f => repr (f a) -> repr (f b) -> repr (f b)
-       -- (<*)  :: Applicative f => repr (f a) -> repr (f b) -> repr (f a)
+class
+ ( Sym_Applicative_Lam (Lambda_from_Repr repr) repr
+ -- , Applicative (Lambda_from_Repr repr)
+ -- , Applicative repr
+ ) => Sym_Applicative repr where
+       pure :: Applicative f => repr a -> repr (f a)
        
        default pure :: (Trans t repr, Applicative f) => t repr a -> t repr (f a)
        pure = trans_map1 pure
+       {-
+       (*>) :: Applicative f => repr (f a) -> repr (f b) -> repr (f b)
+       (<*) :: Applicative f => repr (f a) -> repr (f b) -> repr (f a)
+       (*>) fa = (g <$> fa <*>)
+               where
+                       g :: repr (Lambda (Lambda_from_Repr repr) a (Lambda (Lambda_from_Repr repr) b b))
+                       g = Applicative.pure $ Lambda $ const $ Applicative.pure $ Lambda id
+       (<*) fa = (g <$> fa <*>)
+               where
+                       g :: repr (Lambda (Lambda_from_Repr repr) a (Lambda (Lambda_from_Repr repr) b a))
+                       g = Applicative.pure $ Lambda $ \a -> Applicative.pure $ Lambda (const a)
+infixl 4  *>
+infixl 4 <*
+-}
 
 -- * Class 'Sym_Applicative_Lam'
 -- | Symantic requiring a 'Lambda'.
 class Sym_Functor lam repr => Sym_Applicative_Lam lam repr where
        (<*>) :: Applicative f => repr (f (Lambda lam a b)) -> repr (f a) -> repr (f b)
-       -- (*>)  :: Applicative f => repr (f a) -> repr (f b) -> repr (f b)
-       -- (<*)  :: Applicative f => repr (f a) -> repr (f b) -> repr (f a)
-       
-       default (<*>) :: (Trans t repr, Applicative f) => t repr (f (Lambda lam a b)) -> t repr (f a) -> t repr (f b)
-       -- default (*>)  :: (Trans t, Applicative f) => t repr (f a) -> t repr (f b) -> t repr (f b)
-       -- default (<*)  :: (Trans t, Applicative f) => t repr (f a) -> t repr (f b) -> t repr (f a)
+       default (<*>) :: (Trans t repr, Applicative f)
+        => t repr (f (Lambda lam a b)) -> t repr (f a) -> t repr (f b)
        (<*>) = trans_map2 (<*>)
 infixl 4 <*>
 
@@ -86,7 +98,6 @@ pure_from ast_f ast_a ex ast ctx k =
 ltstargt_from
  :: forall root ty lam ast hs ret.
  ( ty ~ Type_Root_of_Expr (Expr_Applicative lam root)
- , String_from_Type ty
  , Eq_Type (Type_Root_of_Expr root)
  , Eq_Type1 (Type_Root_of_Expr root)
  , Expr_from ast root
index 9017910080c855fec6c639df11d93c1071a54ab3..d40f656782d54fc8c91b6288154ed1e64ab621e9 100644 (file)
@@ -11,7 +11,8 @@ module Expr.Applicative.Test where
 import Test.Tasty
 import Test.Tasty.HUnit
 
-import Control.Arrow (left)
+import qualified Control.Arrow as Arrow
+import qualified Control.Monad as Monad
 import Data.Functor.Identity
 import Data.Proxy (Proxy(..))
 import Data.Text (Text)
@@ -47,26 +48,26 @@ ex_from = root_expr_from (Proxy::Proxy (Ex lam)) (Proxy::Proxy lam)
 (==>) ast expected =
        testCase (show ast) $
        case ex_from ast of
-        Left err -> Left err @?= snd `left` expected
+        Left err -> Left err @?= snd `Arrow.left` expected
         Right (Exists_Type_and_Repr ty (Forall_Repr r)) ->
                case expected of
                 Left (_, err) -> Right ("…"::String) @?= Left err
                 Right (ty_expected::Type_Root_of_Expr (Ex Identity) h, _::h, _::Text) ->
-                       (>>= (@?= (\(_::Proxy h, err) -> err) `left` expected)) $
+                       (Monad.>>= (@?= (\(_::Proxy h, err) -> err) `Arrow.left` expected)) $
                        case ty `eq_type` ty_expected of
-                        Nothing -> return $ Left $
+                        Nothing -> Monad.return $ Left $
                                error_expr (Proxy::Proxy (Ex Identity)) $
                                Error_Expr_Type_mismatch ast
                                 (Exists_Type ty)
                                 (Exists_Type ty_expected)
                         Just Refl -> do
                                let h = runIdentity $ host_from_expr r
-                               return $
+                               Monad.return $
                                        Right
                                         ( ty
                                         , h
                                         , text_from_expr r
-                                        -- , (text_from_expr :: Repr_String Identity h -> Text) r
+                                        -- , (text_from_expr :: Repr_Text Identity h -> Text) r
                                         )
 
 tests :: TestTree
index 851ed370fe721536032de4d82f81bfd5fa9682e1..67123a0238f7df6f37f7ad95aa7b9d3d78dd57c6 100644 (file)
@@ -10,8 +10,9 @@ module Expr.Bool.Test where
 import Test.Tasty
 import Test.Tasty.HUnit
 
-import Control.Arrow (left)
--- import Data.Functor.Identity (Identity)
+import qualified Control.Arrow as Arrow
+import qualified Control.Monad as Monad
+import Data.Functor.Identity
 -- import Control.Monad.IO.Class (MonadIO(..))
 import Data.Proxy (Proxy(..))
 import Data.Text (Text)
@@ -65,26 +66,26 @@ ex_from = root_expr_from (Proxy::Proxy (Ex lam)) (Proxy::Proxy lam)
 (==>) ast expected =
        testCase (show ast) $
        case ex_from ast of
-        Left err -> Left err @?= snd `left` expected
+        Left err -> Left err @?= snd `Arrow.left` expected
         Right (Exists_Type_and_Repr ty (Forall_Repr r)) ->
                case expected of
                 Left (_, err) -> Right ("…"::String) @?= Left err
-                Right (ty_expected::Type_Root_of_Expr (Ex IO) h, _::h, _::Text) ->
-                       (>>= (@?= (\(_::Proxy h, err) -> err) `left` expected)) $
+                Right (ty_expected::Type_Root_of_Expr (Ex Identity) h, _::h, _::Text) ->
+                       (Monad.>>= (@?= (\(_::Proxy h, err) -> err) `Arrow.left` expected)) $
                        case ty `eq_type` ty_expected of
-                        Nothing -> return $ Left $
-                               error_expr (Proxy::Proxy (Ex IO)) $
+                        Nothing -> Monad.return $ Left $
+                               error_expr (Proxy::Proxy (Ex Identity)) $
                                Error_Expr_Type_mismatch ast
                                 (Exists_Type ty)
                                 (Exists_Type ty_expected)
                         Just Refl -> do
-                               h <- host_from_expr r
-                               return $
+                               let h = runIdentity $ host_from_expr r
+                               Monad.return $
                                        Right
                                         ( ty
                                         , h
                                         , text_from_expr r
-                                        -- , (text_from_expr :: Repr_Text IO h -> String) r
+                                        -- , (text_from_expr :: Repr_Text Identity h -> Text) r
                                         )
 
 tests :: TestTree
@@ -121,8 +122,8 @@ tests = testGroup "Bool" $
        Error_Expr_Alt_Curr $
        Error_Expr_Type_mismatch ast
         (Exists_Type (type_var0 SZero `type_fun` type_var0 (SSucc SZero)
-        ::Type_Root (Type_Var0 :|: Type_Var1 :|: Type_Fun IO :|: Type_Bool)
-                    (Lambda IO Var0 Var0)))
+        ::Type_Root (Type_Var0 :|: Type_Var1 :|: Type_Fun Identity :|: Type_Bool)
+                    (Lambda Identity Var0 Var0)))
         (Exists_Type type_bool))
  , AST "app"
         [ AST "val"
diff --git a/Language/Symantic/Expr/Either.hs b/Language/Symantic/Expr/Either.hs
new file mode 100644 (file)
index 0000000..9519efe
--- /dev/null
@@ -0,0 +1,112 @@
+{-# LANGUAGE DefaultSignatures #-}
+{-# LANGUAGE FlexibleContexts #-}
+{-# LANGUAGE FlexibleInstances #-}
+{-# LANGUAGE MultiParamTypeClasses #-}
+{-# LANGUAGE ScopedTypeVariables #-}
+{-# LANGUAGE TypeFamilies #-}
+{-# OPTIONS_GHC -fno-warn-orphans #-}
+-- | Expression for 'Either'.
+module Language.Symantic.Expr.Either where
+
+import Data.Proxy (Proxy(..))
+import Prelude hiding (maybe)
+
+import Language.Symantic.Type
+import Language.Symantic.Trans.Common
+import Language.Symantic.Expr.Root
+import Language.Symantic.Expr.Error
+import Language.Symantic.Expr.From
+
+-- * Class 'Sym_Tuple_Lam'
+-- | Symantic.
+class Sym_Either repr where
+       left  :: repr l -> repr (Either l r)
+       right :: repr r -> repr (Either l r)
+       default left  :: Trans t repr => t repr l -> t repr (Either l r)
+       default right :: Trans t repr => t repr r -> t repr (Either l r)
+       left  = trans_map1 left
+       right = trans_map1 right
+
+-- * Type 'Expr_Either'
+-- | Expression.
+data Expr_Either (lam:: * -> *) (root:: *)
+type instance Root_of_Expr      (Expr_Either lam root)      = root
+type instance Type_of_Expr      (Expr_Either lam root)      = Type_Either
+type instance Sym_of_Expr       (Expr_Either lam root) repr = (Sym_Either repr)
+type instance Error_of_Expr ast (Expr_Either lam root)      = No_Error_Expr
+
+instance Constraint_Type1 Functor (Type_Type1 (Proxy (Either l)) root) where
+       constraint_type1 _c Type_Type1{} = Just Dict
+instance Constraint_Type1 Functor (Type_Either root) where
+       constraint_type1 _c Type_Type2{} = Just Dict
+instance Constraint_Type1 Applicative (Type_Type1 (Proxy (Either l)) root)
+
+-- | Parsing utility to check that the given type is a 'Type_Either'
+-- or raise 'Error_Expr_Type_mismatch'.
+check_type_either
+ :: forall ast ex root ty h ret.
+ ( root ~ Root_of_Expr ex
+ , ty ~ Type_Root_of_Expr ex
+ , Lift_Type   Type_Either (Type_of_Expr root)
+ , Unlift_Type Type_Either (Type_of_Expr root)
+ , Lift_Error_Expr (Error_Expr (Error_of_Type ast ty) ty ast)
+                   (Error_of_Expr ast root)
+ )
+ => Proxy ex -> ast -> ty h
+ -> (Type_Either ty h -> Either (Error_of_Expr ast root) ret)
+ -> Either (Error_of_Expr ast root) ret
+check_type_either ex ast ty k =
+       case unlift_type $ unType_Root ty of
+        Just ty_e -> k ty_e
+        Nothing -> Left $
+               error_expr ex $
+               Error_Expr_Type_mismatch ast
+                (Exists_Type (type_either (type_var0 SZero) (type_var0 $ SSucc SZero)
+                :: Type_Root_of_Expr ex (Either Var0 Var0)))
+                (Exists_Type ty)
+
+-- | Parse 'left'.
+left_from
+ :: forall root lam ty ty_root ast hs ret.
+ ( ty ~ Type_Root_of_Expr (Expr_Either lam root)
+ , ty_root ~ Type_Root_of_Expr root
+ , Type_from ast ty_root
+ , Eq_Type (Type_Root_of_Expr root)
+ , Expr_from ast root
+ , Lift_Type   Type_Either (Type_of_Expr root)
+ , Unlift_Type Type_Either (Type_of_Expr root)
+ , Lift_Error_Expr (Error_Expr (Error_of_Type ast ty) ty ast)
+                   (Error_of_Expr ast root)
+ , Root_of_Expr root ~ root
+ ) => ast -> ast
+ -> Expr_From ast (Expr_Either lam root) hs ret
+left_from ast_l ast_ty_r ex ast ctx k =
+       expr_from (Proxy::Proxy root) ast_l ctx $
+        \(ty_l::Type_Root_of_Expr root h_l) (Forall_Repr_with_Context l) ->
+       either (\err -> Left $ error_expr ex $ Error_Expr_Type err ast) id $
+       type_from (Proxy::Proxy ty_root) ast_ty_r $ \ty_r -> Right $
+       k (type_either ty_l ty_r) $ Forall_Repr_with_Context $
+               \c -> left (l c)
+
+-- | Parse 'right'.
+right_from
+ :: forall root lam ty ty_root ast hs ret.
+ ( ty ~ Type_Root_of_Expr (Expr_Either lam root)
+ , ty_root ~ Type_Root_of_Expr root
+ , Type_from ast ty_root
+ , Eq_Type (Type_Root_of_Expr root)
+ , Expr_from ast root
+ , Lift_Type   Type_Either (Type_of_Expr root)
+ , Unlift_Type Type_Either (Type_of_Expr root)
+ , Lift_Error_Expr (Error_Expr (Error_of_Type ast ty) ty ast)
+                   (Error_of_Expr ast root)
+ , Root_of_Expr root ~ root
+ ) => ast -> ast
+ -> Expr_From ast (Expr_Either lam root) hs ret
+right_from ast_r ast_ty_l ex ast ctx k =
+       expr_from (Proxy::Proxy root) ast_r ctx $
+        \(ty_r::Type_Root_of_Expr root h_r) (Forall_Repr_with_Context r) ->
+       either (\err -> Left $ error_expr ex $ Error_Expr_Type err ast) id $
+       type_from (Proxy::Proxy ty_root) ast_ty_l $ \ty_l -> Right $
+       k (type_either ty_l ty_r) $ Forall_Repr_with_Context $
+               \c -> right (r c)
index b931e815abc68d369c9c2376659c16a136dab3b8..60fd2679d96447123265193a2e036b4ff33f0c84 100644 (file)
@@ -10,7 +10,9 @@ module Expr.Eq.Test where
 import Test.Tasty
 import Test.Tasty.HUnit
 
-import Control.Arrow (left)
+import qualified Control.Arrow as Arrow
+import qualified Control.Monad as Monad
+import Data.Functor.Identity
 import Data.Proxy (Proxy(..))
 import Data.Text (Text)
 import Data.Type.Equality ((:~:)(Refl))
@@ -41,26 +43,26 @@ ex_from = root_expr_from (Proxy::Proxy (Ex lam)) (Proxy::Proxy lam)
 (==>) ast expected =
        testCase (show ast) $
        case ex_from ast of
-        Left err -> Left err @?= snd `left` expected
+        Left err -> Left err @?= snd `Arrow.left` expected
         Right (Exists_Type_and_Repr ty (Forall_Repr r)) ->
                case expected of
                 Left (_, err) -> Right ("…"::String) @?= Left err
-                Right (ty_expected::Type_Root_of_Expr (Ex IO) h, _::h, _::Text) ->
-                       (>>= (@?= (\(_::Proxy h, err) -> err) `left` expected)) $
+                Right (ty_expected::Type_Root_of_Expr (Ex Identity) h, _::h, _::Text) ->
+                       (Monad.>>= (@?= (\(_::Proxy h, err) -> err) `Arrow.left` expected)) $
                        case ty `eq_type` ty_expected of
-                        Nothing -> return $ Left $
-                               error_expr (Proxy::Proxy (Ex IO)) $
+                        Nothing -> Monad.return $ Left $
+                               error_expr (Proxy::Proxy (Ex Identity)) $
                                Error_Expr_Type_mismatch ast
                                 (Exists_Type ty)
                                 (Exists_Type ty_expected)
                         Just Refl -> do
-                               h <- host_from_expr r
-                               return $
+                               let h = runIdentity $ host_from_expr r
+                               Monad.return $
                                        Right
                                         ( ty
                                         , h
                                         , text_from_expr r
-                                        -- , (text_from_expr :: Repr_String IO h -> Text) r
+                                        -- , (text_from_expr :: Repr_Text Identity h -> Text) r
                                         )
 
 tests :: TestTree
index cc84aa47d605b049a032ef48878371566579a98a..29b220045a87cdcb83370094afd2b282b10d733a 100644 (file)
@@ -238,7 +238,6 @@ check_type1
  ( root ~ Root_of_Expr ex
  , ty ~ Type_Root_of_Expr ex
  , Unlift_Type1 (Type_of_Expr root)
- , String_from_Type ty
  , Lift_Error_Expr (Error_Expr (Error_of_Type ast ty) ty ast)
                    (Error_of_Expr ast root)
  )
index 3d3424fd7a04ff8558aa6d63c097511dbefffcc9..9da1ab33eaa925a7752a206727eda995acce1e69 100644 (file)
@@ -64,7 +64,6 @@ instance Constraint_Type1 Functor (Type_Type2 px root)
 fmap_from
  :: forall root ty lam ast hs ret.
  ( ty ~ Type_Root_of_Expr (Expr_Functor lam root)
- , String_from_Type ty
  , Eq_Type (Type_Root_of_Expr root)
  , Expr_from ast root
  , Lift_Type   (Type_Fun lam) (Type_of_Expr root)
index db768ff968ca12416ab41fa3490778c2f075dc94..e079168f063eb4ed8acde92bcded9dad00875a41 100644 (file)
@@ -10,7 +10,8 @@ module Expr.Functor.Test where
 import Test.Tasty
 import Test.Tasty.HUnit
 
-import Control.Arrow (left)
+import qualified Control.Arrow as Arrow
+import qualified Control.Monad as Monad
 import Data.Functor.Identity
 import Data.Proxy (Proxy(..))
 import Data.Text (Text)
@@ -43,26 +44,26 @@ ex_from = root_expr_from (Proxy::Proxy (Ex lam)) (Proxy::Proxy lam)
 (==>) ast expected =
        testCase (show ast) $
        case ex_from ast of
-        Left err -> Left err @?= snd `left` expected
+        Left err -> Left err @?= snd `Arrow.left` expected
         Right (Exists_Type_and_Repr ty (Forall_Repr r)) ->
                case expected of
                 Left (_, err) -> Right ("…"::String) @?= Left err
                 Right (ty_expected::Type_Root_of_Expr (Ex Identity) h, _::h, _::Text) ->
-                       (>>= (@?= (\(_::Proxy h, err) -> err) `left` expected)) $
+                       (Monad.>>= (@?= (\(_::Proxy h, err) -> err) `Arrow.left` expected)) $
                        case ty `eq_type` ty_expected of
-                        Nothing -> return $ Left $
+                        Nothing -> Monad.return $ Left $
                                error_expr (Proxy::Proxy (Ex Identity)) $
                                Error_Expr_Type_mismatch ast
                                 (Exists_Type ty)
                                 (Exists_Type ty_expected)
                         Just Refl -> do
                                let h = runIdentity $ host_from_expr r
-                               return $
+                               Monad.return $
                                        Right
                                         ( ty
                                         , h
                                         , text_from_expr r
-                                        -- , (text_from_expr :: Repr_String Identity h -> Text) r
+                                        -- , (text_from_expr :: Repr_Text Identity h -> Text) r
                                         )
 
 tests :: TestTree
index 9b8c88279a4b48b065d3abd9b1f25d86d49af64c..e16d818a055aede5bb1900f0b6279f6fc380570e 100644 (file)
@@ -10,11 +10,13 @@ module Expr.If.Test where
 import Test.Tasty
 import Test.Tasty.HUnit
 
-import Control.Arrow (left)
+import qualified Control.Arrow as Arrow
+import qualified Control.Monad as Monad
+import Data.Functor.Identity
 import Data.Proxy (Proxy(..))
 import Data.Text (Text)
 import Data.Type.Equality ((:~:)(Refl))
-import Prelude hiding (maybe, not, (&&))
+import Prelude hiding (maybe, not, (&&), Monad(..))
 
 import Language.Symantic.Repr
 import Language.Symantic.Expr
@@ -38,26 +40,26 @@ ex_from = root_expr_from (Proxy::Proxy (Ex lam)) (Proxy::Proxy lam)
 (==>) ast expected =
        testCase (show ast) $
        case ex_from ast of
-        Left err -> Left err @?= snd `left` expected
+        Left err -> Left err @?= snd `Arrow.left` expected
         Right (Exists_Type_and_Repr ty (Forall_Repr r)) ->
                case expected of
                 Left (_, err) -> Right ("…"::String) @?= Left err
-                Right (ty_expected::Type_Root_of_Expr (Ex IO) h, _::h, _::Text) ->
-                       (>>= (@?= (\(_::Proxy h, err) -> err) `left` expected)) $
+                Right (ty_expected::Type_Root_of_Expr (Ex Identity) h, _::h, _::Text) ->
+                       (Monad.>>= (@?= (\(_::Proxy h, err) -> err) `Arrow.left` expected)) $
                        case ty `eq_type` ty_expected of
-                        Nothing -> return $ Left $
-                               error_expr (Proxy::Proxy (Ex IO)) $
+                        Nothing -> Monad.return $ Left $
+                               error_expr (Proxy::Proxy (Ex Identity)) $
                                Error_Expr_Type_mismatch ast
                                 (Exists_Type ty)
                                 (Exists_Type ty_expected)
                         Just Refl -> do
-                               h <- host_from_expr r
-                               return $
+                               let h = runIdentity $ host_from_expr r
+                               Monad.return $
                                        Right
                                         ( ty
                                         , h
                                         , text_from_expr r
-                                        -- , (text_from_expr :: Repr_String IO h -> String) r
+                                        -- , (text_from_expr :: Repr_Text Identity h -> Text) r
                                         )
 
 tests :: TestTree
index 06dda83efa1abe9a534fb7b531037f528c5460e4..bc62231c2c7c59385b709c89b49310da5cc55250 100644 (file)
@@ -10,8 +10,9 @@ module Expr.Int.Test where
 import Test.Tasty
 import Test.Tasty.HUnit
 
-import Control.Arrow (left)
--- import Data.Functor.Identity (Identity)
+import qualified Control.Arrow as Arrow
+import qualified Control.Monad as Monad
+import Data.Functor.Identity
 -- import Control.Monad.IO.Class (MonadIO(..))
 import Data.Proxy (Proxy(..))
 import Data.Text (Text)
@@ -58,26 +59,26 @@ ex_from = root_expr_from (Proxy::Proxy (Ex lam)) (Proxy::Proxy lam)
 (==>) ast expected =
        testCase (show ast) $
        case ex_from ast of
-        Left err -> Left err @?= snd `left` expected
+        Left err -> Left err @?= snd `Arrow.left` expected
         Right (Exists_Type_and_Repr ty (Forall_Repr r)) ->
                case expected of
                 Left (_, err) -> Right ("…"::String) @?= Left err
-                Right (ty_expected::Type_Root_of_Expr (Ex IO) h, _::h, _::Text) ->
-                       (>>= (@?= (\(_::Proxy h, err) -> err) `left` expected)) $
+                Right (ty_expected::Type_Root_of_Expr (Ex Identity) h, _::h, _::Text) ->
+                       (Monad.>>= (@?= (\(_::Proxy h, err) -> err) `Arrow.left` expected)) $
                        case ty `eq_type` ty_expected of
-                        Nothing -> return $ Left $
-                               error_expr (Proxy::Proxy (Ex IO)) $
+                        Nothing -> Monad.return $ Left $
+                               error_expr (Proxy::Proxy (Ex Identity)) $
                                Error_Expr_Type_mismatch ast
                                 (Exists_Type ty)
                                 (Exists_Type ty_expected)
                         Just Refl -> do
-                               h <- host_from_expr r
-                               return $
+                               let h = runIdentity $ host_from_expr r
+                               Monad.return $
                                        Right
                                         ( ty
                                         , h
                                         , text_from_expr r
-                                        -- , (text_from_expr :: Repr_Text IO h -> String) r
+                                        -- , (text_from_expr :: Repr_Text Identity h -> Text) r
                                         )
 
 tests :: TestTree
@@ -103,8 +104,8 @@ tests = testGroup "Int" $
        Error_Expr_Alt_Curr $
        Error_Expr_Type_mismatch ast
         (Exists_Type (type_var0 SZero `type_fun` type_var0 (SSucc SZero)
-        ::Type_Root (Type_Var0 :|: Type_Var1 :|: Type_Fun IO :|: Type_Int)
-                    (Lambda IO Var0 Var0)))
+        ::Type_Root (Type_Var0 :|: Type_Var1 :|: Type_Fun Identity :|: Type_Int)
+                    (Lambda Identity Var0 Var0)))
         (Exists_Type type_int))
  , AST "app"
         [ AST "val"
index 6bcc731aca7e14476ea8666522ce138759c63b51..2933317358f06b766cfa5cfb9166bc4fef834a9a 100644 (file)
@@ -39,6 +39,15 @@ import Language.Symantic.Trans.Common
 --
 -- Basically this means having sometimes to add a type annotation
 -- to the interpreter call to specify @lam@.
+--
+-- FIXME: 'lazy' requires @lam@ to contain 'IO',
+-- which is not escapable (there is no safe function: @IO a -> a@),
+-- this is problematic when writing 'Repr_Host' instances
+-- where 'app'lying an object-function on the elements of a 'Functor'
+-- return a value wrapped inside @lam@, which is then not unwrappable
+-- as needed to write the instance.
+-- I think the way to solve this is to require all object-values
+-- to be wrapped inside @lam@.
 type family Lambda_from_Repr (repr:: * -> *) :: {-lam-}(* -> *)
 
 -- * Class 'Sym_Lambda_App'
index f8ef1ec9b37dcb9aa9b53f2bd5307e3ba1bfa66f..9cf6adc1a1e78ebe315ea663d061ea15c407464c 100644 (file)
@@ -55,9 +55,11 @@ type instance Sym_of_Expr       (Expr_List lam root) repr = (Sym_List repr, Sym_
 type instance Error_of_Expr ast (Expr_List lam root)      = No_Error_Expr
 
 instance Constraint_Type1 Functor (Type_List root) where
-       constraint_type1 _c (Type_Type1 _ _) = Just Dict
+       constraint_type1 _c Type_Type1{} = Just Dict
 instance Constraint_Type1 Applicative (Type_List root) where
-       constraint_type1 _c (Type_Type1 _ _) = Just Dict
+       constraint_type1 _c Type_Type1{} = Just Dict
+instance Constraint_Type1 Traversable (Type_List root) where
+       constraint_type1 _c Type_Type1{} = Just Dict
 
 -- | Parsing utility to check that the given type is a 'Type_List'
 -- or raise 'Error_Expr_Type_mismatch'.
index 507b9e1f0afcb70fd044cc2cd36f2fdc4e43712b..53003047b38faf982508b7c05e4c49cfe892abd3 100644 (file)
@@ -4,6 +4,7 @@
 {-# LANGUAGE MultiParamTypeClasses #-}
 {-# LANGUAGE ScopedTypeVariables #-}
 {-# LANGUAGE TypeFamilies #-}
+{-# OPTIONS_GHC -fno-warn-orphans #-}
 -- | Expression for 'Map'.
 module Language.Symantic.Expr.Map where
 
@@ -129,3 +130,14 @@ type instance Root_of_Expr      (Expr_Map lam root)      = root
 type instance Type_of_Expr      (Expr_Map lam root)      = Type_Map
 type instance Sym_of_Expr       (Expr_Map lam root) repr = (Sym_Map repr, Sym_Map_Lam lam repr)
 type instance Error_of_Expr ast (Expr_Map lam root)      = No_Error_Expr
+
+instance Constraint_Type1 Functor (Type_Map root) where
+       constraint_type1 _c Type_Type2{} = Just Dict
+instance Constraint_Type1 Functor (Type_Type1 (Proxy (Map k)) root) where
+       constraint_type1 _c Type_Type1{} = Just Dict
+instance Constraint_Type1 Applicative (Type_Map root)
+instance Constraint_Type1 Applicative (Type_Type1 (Proxy (Map k)) root)
+instance Constraint_Type1 Traversable (Type_Map root) where
+       constraint_type1 _c Type_Type2{} = Just Dict
+instance Constraint_Type1 Traversable (Type_Type1 (Proxy (Map k)) root) where
+       constraint_type1 _c Type_Type1{} = Just Dict
index efdc8d0a5a081e53a4346d138ebd5b8f46159f51..5ce9485de7f892a7d6d56d825747b89c22ea5a8a 100644 (file)
@@ -58,6 +58,8 @@ instance Constraint_Type1 Functor (Type_Maybe root) where
        constraint_type1 _c (Type_Type1 _ _) = Just Dict
 instance Constraint_Type1 Applicative (Type_Maybe root) where
        constraint_type1 _c (Type_Type1 _ _) = Just Dict
+instance Constraint_Type1 Traversable (Type_Maybe root) where
+       constraint_type1 _c (Type_Type1 _ _) = Just Dict
 
 -- | Parsing utility to check that the given type is a 'Type_Maybe'
 -- or raise 'Error_Expr_Type_mismatch'.
index 4501be6b575350481c5a529a418a020b378f86a9..bc769f0fe4c9a6e5aa18e7c2590399b0a0197e9d 100644 (file)
@@ -10,7 +10,9 @@ module Expr.Maybe.Test where
 import Test.Tasty
 import Test.Tasty.HUnit
 
-import Control.Arrow (left)
+import qualified Control.Arrow as Arrow
+import qualified Control.Monad as Monad
+import Data.Functor.Identity
 import Data.Proxy (Proxy(..))
 import Data.Text (Text)
 import Data.Type.Equality ((:~:)(Refl))
@@ -37,26 +39,26 @@ ex_from = root_expr_from (Proxy::Proxy (Ex lam)) (Proxy::Proxy lam)
 (==>) ast expected =
        testCase (show ast) $
        case ex_from ast of
-        Left err -> Left err @?= snd `left` expected
+        Left err -> Left err @?= snd `Arrow.left` expected
         Right (Exists_Type_and_Repr ty (Forall_Repr r)) ->
                case expected of
                 Left (_, err) -> Right ("…"::String) @?= Left err
-                Right (ty_expected::Type_Root_of_Expr (Ex IO) h, _::h, _::Text) ->
-                       (>>= (@?= (\(_::Proxy h, err) -> err) `left` expected)) $
+                Right (ty_expected::Type_Root_of_Expr (Ex Identity) h, _::h, _::Text) ->
+                       (Monad.>>= (@?= (\(_::Proxy h, err) -> err) `Arrow.left` expected)) $
                        case ty `eq_type` ty_expected of
-                        Nothing -> return $ Left $
-                               error_expr (Proxy::Proxy (Ex IO)) $
+                        Nothing -> Monad.return $ Left $
+                               error_expr (Proxy::Proxy (Ex Identity)) $
                                Error_Expr_Type_mismatch ast
                                 (Exists_Type ty)
                                 (Exists_Type ty_expected)
                         Just Refl -> do
-                               h <- host_from_expr r
-                               return $
+                               let h = runIdentity $ host_from_expr r
+                               Monad.return $
                                        Right
                                         ( ty
                                         , h
                                         , text_from_expr r
-                                        -- , (text_from_expr :: Repr_String IO h -> Text) r
+                                        -- , (text_from_expr :: Repr_Text Identity h -> Text) r
                                         )
 
 tests :: TestTree
diff --git a/Language/Symantic/Expr/Monad.hs b/Language/Symantic/Expr/Monad.hs
new file mode 100644 (file)
index 0000000..219ddf1
--- /dev/null
@@ -0,0 +1,110 @@
+{-# LANGUAGE DefaultSignatures #-}
+{-# LANGUAGE GADTs #-}
+{-# LANGUAGE FlexibleContexts #-}
+{-# LANGUAGE FlexibleInstances #-}
+{-# LANGUAGE MultiParamTypeClasses #-}
+{-# LANGUAGE Rank2Types #-}
+{-# LANGUAGE ScopedTypeVariables #-}
+{-# LANGUAGE TypeFamilies #-}
+{-# LANGUAGE TypeOperators #-}
+{-# OPTIONS_GHC -fno-warn-orphans #-}
+-- | Expression for 'Monad'.
+module Language.Symantic.Expr.Monad where
+
+import Control.Monad (Monad)
+-- import qualified Control.Monad as Monad
+import Data.Proxy (Proxy(..))
+import Data.Type.Equality ((:~:)(Refl))
+import Prelude hiding ((<$>), Monad(..))
+
+import Language.Symantic.Type
+import Language.Symantic.Trans.Common
+import Language.Symantic.Expr.Root
+import Language.Symantic.Expr.Error
+import Language.Symantic.Expr.From
+import Language.Symantic.Expr.Lambda
+import Language.Symantic.Expr.Functor
+
+-- * Class 'Sym_Monad'
+-- | Symantic.
+class
+ ( Sym_Monad_Lam (Lambda_from_Repr repr) repr
+ ) => Sym_Monad repr where
+       return :: Monad f => repr a -> repr (f a)
+       default return :: (Trans t repr, Monad f) => t repr a -> t repr (f a)
+       return = trans_map1 return
+
+-- * Class 'Sym_Monad_Lam'
+-- | Symantic requiring a 'Lambda'.
+class Sym_Functor lam repr => Sym_Monad_Lam lam repr where
+       (>>=) :: Monad m => repr (m a) -> repr (Lambda lam a (m b)) -> repr (m b)
+       default (>>=) :: (Trans t repr, Monad m)
+        => t repr (m a) -> t repr (Lambda lam a (m b)) -> t repr (m b)
+       (>>=) = trans_map2 (>>=)
+infixl 1 >>=
+
+-- * Type 'Expr_Monad'
+-- | Expression.
+data Expr_Monad (lam:: * -> *) (root:: *)
+type instance Root_of_Expr      (Expr_Monad lam root)      = root
+type instance Type_of_Expr      (Expr_Monad lam root)      = No_Type
+type instance Sym_of_Expr       (Expr_Monad lam root) repr = (Sym_Monad repr, Sym_Monad_Lam lam repr)
+type instance Error_of_Expr ast (Expr_Monad lam root)      = No_Error_Expr
+instance Constraint_Type1 Monad (Type_Type0 px root)
+instance Constraint_Type1 Monad (Type_Var1 root)
+instance Constraint_Type1 Monad (Type_Type2 px root)
+
+return_from
+ :: forall root ty ty_root lam ast hs ret.
+ ( ty ~ Type_Root_of_Expr (Expr_Monad lam root)
+ , ty_root ~ Type_Root_of_Expr root
+ , Eq_Type (Type_Root_of_Expr root)
+ , Type1_from ast (Type_Root_of_Expr root)
+ , Expr_from ast root
+ , Lift_Error_Expr (Error_Expr (Error_of_Type ast ty) ty ast)
+                   (Error_of_Expr ast root)
+ , Root_of_Expr root ~ root
+ , Constraint_Type1 Monad ty
+ ) => ast -> ast
+ -> Expr_From ast (Expr_Monad lam root) hs ret
+return_from ast_f ast_a ex ast ctx k =
+ -- return :: Monad f => a -> f a
+       either (\err -> Left $ error_expr ex $ Error_Expr_Type err ast) id $
+       type1_from (Proxy::Proxy ty_root) ast_f $ \_f ty_f -> Right $
+               expr_from (Proxy::Proxy root) ast_a ctx $
+                \(ty_a::Type_Root_of_Expr root h_a) (Forall_Repr_with_Context a) ->
+               let ty_fa = ty_f ty_a in
+               check_constraint1_type ex (Proxy::Proxy Monad) ast ty_fa $ \Dict ->
+               k ty_fa $ Forall_Repr_with_Context $
+                \c -> return (a c)
+
+bind_from
+ :: forall root ty lam ast hs ret.
+ ( ty ~ Type_Root_of_Expr (Expr_Monad lam root)
+ , Eq_Type (Type_Root_of_Expr root)
+ , Eq_Type1 (Type_Root_of_Expr root)
+ , Expr_from ast root
+ , Lift_Type   (Type_Fun lam) (Type_of_Expr root)
+ , Unlift_Type (Type_Fun lam) (Type_of_Expr root)
+ , Unlift_Type1 (Type_of_Expr root)
+ , Lift_Error_Expr (Error_Expr (Error_of_Type ast ty) ty ast)
+                   (Error_of_Expr ast root)
+ , Root_of_Expr root ~ root
+ , Constraint_Type1 Monad ty
+ ) => ast -> ast
+ -> Expr_From ast (Expr_Monad lam root) hs ret
+bind_from ast_ma ast_f ex ast ctx k =
+ -- (>>=) :: Monad m => m a -> (a -> m b) -> m b
+       expr_from (Proxy::Proxy root) ast_ma ctx $
+        \(ty_ma::Type_Root_of_Expr root h_ma) (Forall_Repr_with_Context ma) ->
+       expr_from (Proxy::Proxy root) ast_f ctx $
+        \(ty_f::Type_Root_of_Expr root h_f) (Forall_Repr_with_Context f) ->
+       check_type1 ex ast ty_ma $ \(Type_Type1 m ty_m_a, Lift_Type1 ty_m) ->
+       check_type_fun ex ast ty_f $ \(Type_Type2 Proxy ty_f_a ty_f_mb
+        :: Type_Fun lam (Type_Root_of_Expr root) h_f) ->
+       check_eq_type ex ast ty_m_a ty_f_a $ \Refl ->
+       check_type1 ex ast ty_f_mb $ \(Type_Type1 _ ty_f_m_b, _) ->
+       check_eq_type1 ex ast ty_ma ty_f_mb $ \Refl ->
+       check_constraint1_type ex (Proxy::Proxy Monad) ast ty_ma $ \Dict ->
+       k (Type_Root $ ty_m $ Type_Type1 m ty_f_m_b) $ Forall_Repr_with_Context $
+        \c -> (>>=) (ma c) (f c)
diff --git a/Language/Symantic/Expr/Traversable.hs b/Language/Symantic/Expr/Traversable.hs
new file mode 100644 (file)
index 0000000..9d6931c
--- /dev/null
@@ -0,0 +1,88 @@
+{-# LANGUAGE DefaultSignatures #-}
+{-# LANGUAGE GADTs #-}
+{-# LANGUAGE FlexibleContexts #-}
+{-# LANGUAGE FlexibleInstances #-}
+{-# LANGUAGE MultiParamTypeClasses #-}
+{-# LANGUAGE Rank2Types #-}
+{-# LANGUAGE ScopedTypeVariables #-}
+{-# LANGUAGE TypeFamilies #-}
+{-# LANGUAGE TypeOperators #-}
+{-# LANGUAGE UndecidableInstances #-}
+{-# OPTIONS_GHC -fno-warn-orphans #-}
+-- | Expression for 'Traversable'.
+module Language.Symantic.Expr.Traversable where
+
+import Data.Traversable (Traversable)
+import Data.Proxy (Proxy(..))
+import Data.Type.Equality ((:~:)(Refl))
+import Prelude hiding (Traversable(..))
+
+import Language.Symantic.Type
+import Language.Symantic.Trans.Common
+import Language.Symantic.Expr.Root
+import Language.Symantic.Expr.Error
+import Language.Symantic.Expr.From
+import Language.Symantic.Expr.Lambda
+import Language.Symantic.Expr.Applicative
+
+-- * Class 'Sym_Traversable'
+-- | Symantic.
+class
+ ( Sym_Applicative repr
+ , Sym_Applicative_Lam lam repr
+ ) => Sym_Traversable lam repr where
+       traverse :: (Traversable t, Applicative f) => repr (Lambda lam a (f b)) -> repr (t a) -> repr (f (t b))
+       default traverse :: (Trans tr repr, Traversable t, Applicative f)
+        => tr repr (Lambda lam a (f b)) -> tr repr (t a) -> tr repr (f (t b))
+       traverse = trans_map2 traverse
+
+-- * Type 'Expr_Traversable'
+-- | Expression.
+data Expr_Traversable (lam:: * -> *) (root:: *)
+type instance Root_of_Expr      (Expr_Traversable lam root)      = root
+type instance Type_of_Expr      (Expr_Traversable lam root)      = No_Type
+type instance Sym_of_Expr       (Expr_Traversable lam root) repr = (Sym_Traversable lam repr)
+type instance Error_of_Expr ast (Expr_Traversable lam root)      = No_Error_Expr
+instance Constraint_Type1 Traversable (Type_Type0 px root)
+instance Constraint_Type1 Traversable (Type_Var1 root)
+instance Constraint_Type1 Traversable (Type_Type2 px root)
+
+traverse_from
+ :: forall root ty lam ast hs ret.
+ ( ty ~ Type_Root_of_Expr (Expr_Traversable lam root)
+ , Eq_Type (Type_Root_of_Expr root)
+ , Eq_Type1 (Type_Root_of_Expr root)
+ , Expr_from ast root
+ , Lift_Type   (Type_Fun lam) (Type_of_Expr root)
+ , Unlift_Type (Type_Fun lam) (Type_of_Expr root)
+ , Unlift_Type1 (Type_of_Expr root)
+ , Lift_Error_Expr (Error_Expr (Error_of_Type ast ty) ty ast)
+                   (Error_of_Expr ast root)
+ , Root_of_Expr root ~ root
+ , Constraint_Type1 Applicative ty
+ , Constraint_Type1 Traversable ty
+ ) => ast -> ast
+ -> Expr_From ast (Expr_Traversable lam root) hs ret
+traverse_from ast_g ast_ta ex ast ctx k =
+ -- traverse :: (Traversable t, Applicative f) => (a -> f b) -> t a -> f (t b)
+       expr_from (Proxy::Proxy root) ast_g ctx $
+        \(ty_g::Type_Root_of_Expr root h_g) (Forall_Repr_with_Context g) ->
+       expr_from (Proxy::Proxy root) ast_ta ctx $
+        \(ty_ta::Type_Root_of_Expr root h_ta) (Forall_Repr_with_Context ta) ->
+       
+       check_type_fun ex ast ty_g $ \(Type_Type2 Proxy ty_g_a ty_g_fb
+        :: Type_Fun lam (Type_Root_of_Expr root) h_g) ->
+       
+       check_type1 ex ast ty_g_fb $ \(Type_Type1 f (ty_g_fb_b::Type_Root_of_Expr root h_f), Lift_Type1 ty_f) ->
+       check_type1 ex ast ty_ta $ \(Type_Type1 t ty_ta_a, Lift_Type1 ty_t) ->
+       
+       check_constraint1_type ex (Proxy::Proxy Applicative) ast ty_g_fb $ \Dict ->
+       check_constraint1_type ex (Proxy::Proxy Traversable) ast ty_ta $ \Dict ->
+       
+       check_eq_type ex ast ty_g_a ty_ta_a $ \Refl ->
+       
+       k (
+               Type_Root $ ty_f $ Type_Type1 f $
+               Type_Root $ ty_t $ Type_Type1 t ty_g_fb_b
+        ) $ Forall_Repr_with_Context $
+        \c -> traverse (g c) (ta c)
diff --git a/Language/Symantic/Expr/Traversable/HLint.hs b/Language/Symantic/Expr/Traversable/HLint.hs
new file mode 120000 (symlink)
index 0000000..ab18269
--- /dev/null
@@ -0,0 +1 @@
+../HLint.hs
\ No newline at end of file
diff --git a/Language/Symantic/Expr/Traversable/Test.hs b/Language/Symantic/Expr/Traversable/Test.hs
new file mode 100644 (file)
index 0000000..1a94f42
--- /dev/null
@@ -0,0 +1,93 @@
+{-# LANGUAGE FlexibleContexts #-}
+{-# LANGUAGE FlexibleInstances #-}
+{-# LANGUAGE GADTs #-}
+{-# LANGUAGE NoMonomorphismRestriction #-}
+{-# LANGUAGE ScopedTypeVariables #-}
+{-# LANGUAGE TypeOperators #-}
+{-# OPTIONS_GHC -fno-warn-missing-signatures #-}
+
+module Expr.Applicative.Test where
+
+import Test.Tasty
+import Test.Tasty.HUnit
+
+import qualified Control.Arrow as Arrow
+import qualified Control.Monad as Monad
+import Data.Functor.Identity
+import Data.Proxy (Proxy(..))
+import Data.Text (Text)
+import Data.Type.Equality ((:~:)(Refl))
+import Prelude hiding ((&&), not, (||), (==), (<$>), (+), Applicative(..), (>>=))
+
+import Language.Symantic.Type
+import Language.Symantic.Expr as Expr
+import Language.Symantic.Repr
+
+import AST.Test
+
+-- * Expressions
+t = bool True
+f = bool False
+e1 = val (\x -> val $ \y -> x + y)
+ <$> just (int 1)
+ <*> just (int 2)
+
+-- * Tests
+type Ex lam = Expr_Root
+ (   Expr_Lambda_App lam
+ .|. Expr_Lambda_Val lam
+ .|. Expr_List lam
+ .|. Expr_Maybe lam
+ .|. Expr_Int
+ .|. Expr_Bool
+ .|. Expr_Functor lam
+ .|. Expr_Applicative lam
+ .|. Expr_Traversable lam
+ )
+ex_from = root_expr_from (Proxy::Proxy (Ex lam)) (Proxy::Proxy lam)
+
+(==>) ast expected =
+       testCase (show ast) $
+       case ex_from ast of
+        Left err -> Left err @?= snd `Arrow.left` expected
+        Right (Exists_Type_and_Repr ty (Forall_Repr r)) ->
+               case expected of
+                Left (_, err) -> Right ("…"::String) @?= Left err
+                Right (ty_expected::Type_Root_of_Expr (Ex Identity) h, _::h, _::Text) ->
+                       (Monad.>>= (@?= (\(_::Proxy h, err) -> err) `Arrow.left` expected)) $
+                       case ty `eq_type` ty_expected of
+                        Nothing -> Monad.return $ Left $
+                               error_expr (Proxy::Proxy (Ex Identity)) $
+                               Error_Expr_Type_mismatch ast
+                                (Exists_Type ty)
+                                (Exists_Type ty_expected)
+                        Just Refl -> do
+                               let h = runIdentity $ host_from_expr r
+                               Monad.return $
+                                       Right
+                                        ( ty
+                                        , h
+                                        , text_from_expr r
+                                        -- , (text_from_expr :: Repr_Text Identity h -> Text) r
+                                        )
+
+tests :: TestTree
+tests = testGroup "Traversable"
+ [ AST "<*>"
+        [ AST "<$>"
+                [ AST "val"
+                        [ AST "x" [], AST "Int" []
+                        , AST "val"
+                                [ AST "y" [], AST "Int" []
+                                , AST "+" [ AST "var" [AST "x" []]
+                                          , AST "var" [AST "y" []] ]
+                                ]
+                        ]
+                , AST "just" [ AST "int" [AST "1" []] ]
+                ]
+        , AST "just" [ AST "int" [AST "2" []] ]
+        ] ==> Right
+        ( type_maybe type_int
+        , Just 3
+        , "fmap (\\x0 -> (\\x1 -> x0 + x1)) (just 1) <*> just 2" )
+ ]
index da76797478899f470d65ce92f845dda807b42573..35ac2381ba129e99105d6ccc12f18cc33faf73c9 100644 (file)
@@ -5,7 +5,7 @@
 {-# LANGUAGE ScopedTypeVariables #-}
 {-# LANGUAGE TypeFamilies #-}
 {-# OPTIONS_GHC -fno-warn-orphans #-}
--- | Expression for 'Tuple'.
+-- | Expression for tuples.
 module Language.Symantic.Expr.Tuple where
 
 import Data.Proxy (Proxy(..))
@@ -24,9 +24,6 @@ class Sym_Tuple2 repr where
        default tuple2 :: Trans t repr => t repr a -> t repr b -> t repr (a, b)
        tuple2 = trans_map2 tuple2
 
-instance Constraint_Type1 Functor (Type_Type1 (Proxy ((,) fst)) root) where
-       constraint_type1 _c Type_Type1{} = Just Dict
-
 -- * Type 'Expr_Tuple2'
 -- | Expression.
 data Expr_Tuple2 (lam:: * -> *) (root:: *)
@@ -35,6 +32,12 @@ type instance Type_of_Expr      (Expr_Tuple2 lam root)      = Type_Tuple2
 type instance Sym_of_Expr       (Expr_Tuple2 lam root) repr = (Sym_Tuple2 repr)
 type instance Error_of_Expr ast (Expr_Tuple2 lam root)      = No_Error_Expr
 
+instance Constraint_Type1 Functor (Type_Type1 (Proxy ((,) fst)) root) where
+       constraint_type1 _c Type_Type1{} = Just Dict
+instance Constraint_Type1 Functor (Type_Tuple2 root) where
+       constraint_type1 _c Type_Type2{} = Just Dict
+instance Constraint_Type1 Applicative (Type_Type1 (Proxy ((,) fst)) root)
+
 -- | Parsing utility to check that the given type is a 'Type_Tuple2'
 -- or raise 'Error_Expr_Type_mismatch'.
 check_type_tuple2
@@ -51,7 +54,7 @@ check_type_tuple2
  -> Either (Error_of_Expr ast root) ret
 check_type_tuple2 ex ast ty k =
        case unlift_type $ unType_Root ty of
-        Just ty_l -> k ty_l
+        Just ty_t -> k ty_t
         Nothing -> Left $
                error_expr ex $
                Error_Expr_Type_mismatch ast
index 3453a43c549b1cd5d5cb337dfa25d58f4b1f7ee6..bd7721488f116d71f39e0924bc51bc61e65a3b8f 100644 (file)
@@ -1,7 +1,9 @@
+{-# LANGUAGE FlexibleContexts #-}
 {-# LANGUAGE FlexibleInstances #-}
 {-# LANGUAGE MultiParamTypeClasses #-}
 {-# LANGUAGE NoImplicitPrelude #-}
 {-# LANGUAGE TypeFamilies #-}
+{-# LANGUAGE UndecidableInstances #-}
 -- | Interpreter to duplicate the representation of an expression
 -- in order to evaluate it with different interpreters.
 --
@@ -135,3 +137,36 @@ instance -- Sym_Map_Lam
  ) => Sym_Map_Lam lam (Dup r1 r2) where
        map_map (f1 `Dup` f2) (m1 `Dup` m2) =
                map_map f1 m1 `Dup` map_map f2 m2
+instance -- Sym_Functor
+ ( Sym_Functor lam r1
+ , Sym_Functor lam r2
+ ) => Sym_Functor lam (Dup r1 r2) where
+       fmap (f1 `Dup` f2) (m1 `Dup` m2) =
+               fmap f1 m1 `Dup` fmap f2 m2
+instance -- Sym_Applicative
+ ( Sym_Applicative r1
+ , Sym_Applicative r2
+ , Sym_Applicative_Lam (Lambda_from_Repr r1) (Dup r1 r2)
+ ) => Sym_Applicative (Dup r1 r2) where
+       pure (a1 `Dup` a2) =
+               pure a1 `Dup` pure a2
+instance -- Sym_Applicative_Lam
+ ( Sym_Applicative_Lam lam r1
+ , Sym_Applicative_Lam lam r2
+ ) => Sym_Applicative_Lam lam (Dup r1 r2) where
+       (<*>) (f1 `Dup` f2) (m1 `Dup` m2) =
+               (<*>) f1 m1 `Dup` (<*>) f2 m2
+instance -- Sym_Traversable
+ ( Sym_Traversable lam r1
+ , Sym_Traversable lam r2
+ , lam ~ Lambda_from_Repr r1
+ ) => Sym_Traversable lam (Dup r1 r2) where
+       traverse (f1 `Dup` f2) (m1 `Dup` m2) =
+               traverse f1 m1 `Dup` traverse f2 m2
+instance -- Sym_Monad
+ ( Sym_Monad_Lam lam r1
+ , Sym_Monad_Lam lam r2
+ , lam ~ Lambda_from_Repr r1
+ ) => Sym_Monad_Lam lam (Dup r1 r2) where
+       (>>=) (m1 `Dup` m2) (f1 `Dup` f2) =
+               (>>=) m1 f1 `Dup` (>>=) m2 f2
index 400958f98c3b607fe43a3917c4093f362f2fd5b7..d448378942dc5f28760488538965ff5a444d1ef5 100644 (file)
@@ -1,23 +1,27 @@
+{-# LANGUAGE FlexibleContexts #-}
 {-# LANGUAGE FlexibleInstances #-}
 {-# LANGUAGE GeneralizedNewtypeDeriving #-}
 {-# LANGUAGE MultiParamTypeClasses #-}
 {-# LANGUAGE TypeFamilies #-}
+{-# LANGUAGE UndecidableInstances #-}
 -- | Interpreter to compute a host-term.
 module Language.Symantic.Repr.Host where
 
-import Data.Functor as Functor
-import Data.Functor.Identity
 import Control.Applicative as Applicative
 import Control.Monad as Monad
 import Control.Monad.IO.Class (MonadIO(..))
+import Data.Functor as Functor
+import Data.Functor.Identity
 import Data.IORef
+import Data.Traversable as Traversable
 import qualified Data.Bool as Bool
 import qualified Data.Maybe as Maybe
 import qualified Data.Map.Strict as Map
 
 import Language.Symantic.Lib.Control.Monad
 import Language.Symantic.Type
-import Language.Symantic.Expr
+import Language.Symantic.Expr hiding (Sym_Monad(..), Sym_Monad_Lam(..))
+import qualified Language.Symantic.Expr as Expr
 
 -- * Type 'Repr_Host'
 
@@ -99,7 +103,10 @@ instance Sym_Functor Identity (Repr_Host Identity) where
        {- NOTE: need Traversable
        fmap f = liftMJoin $ sequence . ((\a -> f `app` return a) Functor.<$>)
        -}
-instance Monad lam => Sym_Applicative (Repr_Host lam) where
+instance
+ ( Sym_Applicative_Lam lam (Repr_Host lam)
+ , Applicative lam
+ ) => Sym_Applicative (Repr_Host lam) where
        pure = Functor.fmap Applicative.pure
 instance Sym_Applicative_Lam Identity (Repr_Host Identity) where
        (<*>) = liftM2Join $ \fg ->
@@ -112,6 +119,20 @@ instance Sym_Applicative_Lam Identity (Repr_Host Identity) where
                 Applicative.<*>
                 (return Functor.<$> fa)
        -}
+instance Sym_Traversable Identity (Repr_Host Identity) where
+       traverse ra2fb = liftMJoin $
+               return . Traversable.traverse a2fb
+               where a2fb a = runIdentity $ unRepr_Host $ ra2fb `app` return a
+instance
+ ( Expr.Sym_Monad_Lam lam (Repr_Host lam)
+ , Monad lam
+ ) => Expr.Sym_Monad (Repr_Host lam) where
+       return = Functor.fmap Monad.return
+instance Expr.Sym_Monad_Lam Identity (Repr_Host Identity) where
+       (>>=) rma ra2mb = do
+               ma <- rma
+               return $ (Monad.>>=) ma a2mb
+               where a2mb a = runIdentity $ unRepr_Host $ ra2mb `app` return a
 
 -- | Helper to store arguments of 'lazy' into an 'IORef'.
 lazy_share :: MonadIO m => m a -> m (m a)
index b030daa414848f3db6184cf2ad123d5213679db9..bd7f16358e9b535753e8b6e153332720e647a59d 100644 (file)
@@ -15,7 +15,7 @@ import Test.Tasty.HUnit
 import Data.Functor.Identity
 import Data.Text (Text)
 import qualified Data.Text as Text
-import Prelude hiding (and, not, or)
+import Prelude hiding (and, not, or, Monad(..))
 
 import Language.Symantic.Repr
 import Language.Symantic.Expr
@@ -30,12 +30,13 @@ import qualified Expr.Applicative.Test as Applicative.Test
 tests :: TestTree
 tests = testGroup "Host" $
  [ testGroup "Bool" $
-       let (==>) (expr::forall repr. Sym_of_Expr (Expr_Root (  Expr_Lambda_App IO
-                                                           .|. Expr_Lambda_Val IO
+       let (==>) (expr::forall repr. Sym_of_Expr (Expr_Root (  Expr_Lambda_App Identity
+                                                           .|. Expr_Lambda_Val Identity
                                                            .|. Expr_Bool
                                                            )) repr => repr h) expected =
-               testCase (Text.unpack $ (text_from_expr :: Repr_Text IO _h -> Text) $ expr) $
-               (>>= (@?= expected)) $
+               testCase (Text.unpack $ (text_from_expr :: Repr_Text Identity _h -> Text) $ expr) $
+               -- (>>= (@?= expected)) $
+               (\(Identity a) -> (a @?= expected)) $
                host_from_expr expr in
         [ Bool.Test.e1 ==> False
         , Bool.Test.e2 ==> True
@@ -43,12 +44,12 @@ tests = testGroup "Host" $
         , Bool.Test.e4 ==> True
         ]
  , testGroup "Lambda" $
-       let (==>) (expr::forall repr. Sym_of_Expr (Expr_Root (  Expr_Lambda_App IO
-                                                           .|. Expr_Lambda_Val IO
+       let (==>) (expr::forall repr. Sym_of_Expr (Expr_Root (  Expr_Lambda_App Identity
+                                                           .|. Expr_Lambda_Val Identity
                                                            .|. Expr_Bool
                                                            )) repr => repr h) expected =
-               testCase (Text.unpack $ (text_from_expr :: Repr_Text IO _h -> Text) $ expr) $
-               (>>= (@?= expected)) $
+               testCase (Text.unpack $ (text_from_expr :: Repr_Text Identity _h -> Text) $ expr) $
+               (\(Identity a) -> (a @?= expected)) $
                host_from_expr expr in
         [ (Lambda.Test.e1 `app` bool True  `app` bool True)  ==> False
         , (Lambda.Test.e1 `app` bool True  `app` bool False) ==> True
@@ -73,35 +74,35 @@ tests = testGroup "Host" $
         , (Lambda.Test.e7 `app` val not) ==> False
         ]
  , testGroup "Maybe" $
-       let (==>) (expr::forall repr. Sym_of_Expr (Expr_Root (  Expr_Lambda_App IO
-                                                           .|. Expr_Lambda_Val IO
-                                                           .|. Expr_Maybe IO
+       let (==>) (expr::forall repr. Sym_of_Expr (Expr_Root (  Expr_Lambda_App Identity
+                                                           .|. Expr_Lambda_Val Identity
+                                                           .|. Expr_Maybe Identity
                                                            .|. Expr_Bool )) repr => repr h) expected =
-               testCase (Text.unpack $ (text_from_expr :: Repr_Text IO _h -> Text) $ expr) $
-               (>>= (@?= expected)) $
+               testCase (Text.unpack $ (text_from_expr :: Repr_Text Identity _h -> Text) $ expr) $
+               (\(Identity a) -> (a @?= expected)) $
                host_from_expr expr in
         [ Maybe.Test.e1 ==> False
         ]
  , testGroup "If" $
-       let (==>) (expr::forall repr. Sym_of_Expr (Expr_Root (  Expr_Lambda_App IO
-                                                           .|. Expr_Lambda_Val IO
+       let (==>) (expr::forall repr. Sym_of_Expr (Expr_Root (  Expr_Lambda_App Identity
+                                                           .|. Expr_Lambda_Val Identity
                                                            .|. Expr_If
                                                            .|. Expr_Bool )) repr => repr h) expected =
-               testCase (Text.unpack $ (text_from_expr :: Repr_Text IO _h -> Text) $ expr) $
-               (>>= (@?= expected)) $
+               testCase (Text.unpack $ (text_from_expr :: Repr_Text Identity _h -> Text) $ expr) $
+               (\(Identity a) -> (a @?= expected)) $
                host_from_expr expr in
         [ If.Test.e1 ==> False
         ]
  , testGroup "List" $
-       let (==>) (expr::forall repr. Sym_of_Expr (Expr_Root (  Expr_Lambda_App IO
-                                                           .|. Expr_Lambda_Val IO
-                                                           .|. Expr_List IO
+       let (==>) (expr::forall repr. Sym_of_Expr (Expr_Root (  Expr_Lambda_App Identity
+                                                           .|. Expr_Lambda_Val Identity
+                                                           .|. Expr_List Identity
                                                            .|. Expr_Bool
                                                            .|. Expr_Int
                                                            .|. Expr_If
                                                            .|. Expr_Eq )) repr => repr h) expected =
-               testCase (Text.unpack $ (text_from_expr :: Repr_Text IO _h -> Text) $ expr) $
-               (>>= (@?= expected)) $
+               testCase (Text.unpack $ (text_from_expr :: Repr_Text Identity _h -> Text) $ expr) $
+               (\(Identity a) -> (a @?= expected)) $
                host_from_expr expr in
         [ List.Test.e1 ==> [2::Int,4]
         ]
index a5ce615e467c92826bbf83d11ab8d5b00f243a59..bc962258ec0644e13049ef386555b7f2067e9c08 100644 (file)
@@ -1,8 +1,10 @@
+{-# LANGUAGE FlexibleContexts #-}
 {-# LANGUAGE FlexibleInstances #-}
 {-# LANGUAGE GeneralizedNewtypeDeriving #-}
 {-# LANGUAGE MultiParamTypeClasses #-}
 {-# LANGUAGE OverloadedStrings #-}
 {-# LANGUAGE TypeFamilies #-}
+{-# LANGUAGE UndecidableInstances #-}
 -- | Interpreter to serialize an expression into a 'Text'.
 module Language.Symantic.Repr.Text where
 
@@ -200,7 +202,7 @@ instance Monad lam => Sym_Functor lam (Repr_Text lam) where
                Repr_Text $ \_p v ->
                        let p' = precedence_App in
                        "fmap " <> f p' v <> " " <> m p' v
-instance Monad lam => Sym_Applicative (Repr_Text lam) where
+instance (Sym_Applicative_Lam lam (Repr_Text lam), Applicative lam) => Sym_Applicative (Repr_Text lam) where
        pure (Repr_Text a) =
                Repr_Text $ \_p v ->
                        let p' = precedence_App in
@@ -210,6 +212,11 @@ instance Monad lam => Sym_Applicative_Lam lam (Repr_Text lam) where
                Repr_Text $ \p v ->
                        let p' = precedence_LtStarGt in
                        paren p p' $ fg p' v <> " <*> " <> fa p' v
+instance Monad lam => Sym_Traversable lam (Repr_Text lam) where
+       traverse (Repr_Text g) (Repr_Text ta) =
+               Repr_Text $ \p v ->
+                       let p' = precedence_App in
+                       paren p p' $ "traverse " <> g p' v <> " " <> ta p' v
 
 -- ** Type 'Precedence'
 
index 84a40d3e9f6f8b24b83716a89d13e3b178b72410..b7ecf527cea0eb46ee2e72ed6070752825cb4098 100644 (file)
@@ -16,6 +16,7 @@ module Language.Symantic.Type
  , module Language.Symantic.Type.Ordering
  , module Language.Symantic.Type.Tuple
  , module Language.Symantic.Type.Map
+ , module Language.Symantic.Type.Either
  ) where
 
 import Language.Symantic.Type.Root
@@ -34,3 +35,4 @@ import Language.Symantic.Type.List
 import Language.Symantic.Type.Ordering
 import Language.Symantic.Type.Tuple
 import Language.Symantic.Type.Map
+import Language.Symantic.Type.Either
diff --git a/Language/Symantic/Type/Either.hs b/Language/Symantic/Type/Either.hs
new file mode 100644 (file)
index 0000000..266d668
--- /dev/null
@@ -0,0 +1,44 @@
+{-# LANGUAGE FlexibleContexts #-}
+{-# LANGUAGE FlexibleInstances #-}
+{-# LANGUAGE PatternSynonyms #-}
+{-# LANGUAGE ScopedTypeVariables #-}
+{-# LANGUAGE TypeFamilies #-}
+{-# OPTIONS_GHC -fno-warn-orphans #-}
+module Language.Symantic.Type.Either where
+
+import Data.Proxy
+import Language.Symantic.Type.Root
+import Language.Symantic.Type.Type0
+import Language.Symantic.Type.Type2
+
+-- * Type 'Type_Either'
+-- | The 'Either' type.
+type Type_Either = Type_Type2 (Proxy Either)
+
+type instance Constraint2_of (Proxy Either)
+ = Constraint2_Empty
+
+pattern Type_Either
+ :: root a -> root b
+ -> Type_Either root (Either a b)
+pattern Type_Either a b
+ = Type_Type2 Proxy a b
+
+instance -- String_from_Type
+ String_from_Type root =>
+ String_from_Type (Type_Either root) where
+       string_from_type (Type_Type2 _ a b) =
+               "Either"
+                ++ " (" ++ string_from_type a ++ ")"
+                ++ " (" ++ string_from_type b ++ ")"
+
+-- | Inject 'Type_Either' within a root type.
+type_either
+ :: forall root h_a h_b.
+ Lift_Type_Root Type_Either root
+ => root h_a
+ -> root h_b
+ -> root (Either h_a h_b)
+type_either a b = lift_type_root
+        (Type_Either a b
+        ::Type_Either root (Either h_a h_b))
index dda1d4f71b8c33207ea14ac22efc217f3f29131c..87f0af6d13a06c769ce1f97db99fd9522dfe3a07 100644 (file)
@@ -12,7 +12,7 @@ import Language.Symantic.Type.Type0
 import Language.Symantic.Type.Type2
 
 -- * Type 'Type_Tuple2'
--- | The (,) type.
+-- | The @(,)@ type.
 type Type_Tuple2 = Type_Type2 (Proxy (,))
 
 type instance Constraint2_of (Proxy (,))
index 0ecaea377381cc0fb4e862c280795b6dcdedcdc8..0f3f6c2141d1351f0d4c6ab4d6541c1e196e7564 100644 (file)
@@ -107,6 +107,7 @@ Library
     Language.Symantic.Expr.Alt
     Language.Symantic.Expr.Applicative
     Language.Symantic.Expr.Bool
+    Language.Symantic.Expr.Either
     Language.Symantic.Expr.Eq
     Language.Symantic.Expr.Error
     Language.Symantic.Expr.From
@@ -117,8 +118,10 @@ Library
     Language.Symantic.Expr.List
     Language.Symantic.Expr.Map
     Language.Symantic.Expr.Maybe
+    Language.Symantic.Expr.Monad
     Language.Symantic.Expr.Ord
     Language.Symantic.Expr.Root
+    Language.Symantic.Expr.Traversable
     Language.Symantic.Expr.Tuple
     Language.Symantic.Lib.Control.Monad
     Language.Symantic.Lib.Data.Bool
@@ -134,6 +137,7 @@ Library
     Language.Symantic.Type
     Language.Symantic.Type.Alt
     Language.Symantic.Type.Bool
+    Language.Symantic.Type.Either
     Language.Symantic.Type.Error
     Language.Symantic.Type.Fun
     Language.Symantic.Type.Int