{-# LANGUAGE CPP #-}
module Hackage.Security.Util.Some (
Some(..)
, DictEq(..)
, SomeEq(..)
, DictShow(..)
, SomeShow(..)
, DictPretty(..)
, SomePretty(..)
, typecheckSome
#if !MIN_VERSION_base(4,7,0)
, tyConSome
#endif
) where
#if MIN_VERSION_base(4,7,0)
import Data.Typeable (Typeable)
#else
import qualified Data.Typeable as Typeable
#endif
import Hackage.Security.Util.TypedEmbedded
import Hackage.Security.Util.Pretty
data Some f = forall a. Some (f a)
#if MIN_VERSION_base(4,7,0)
deriving instance Typeable Some
#else
tyConSome :: Typeable.TyCon
tyConSome = Typeable.mkTyCon3 "hackage-security" "Hackage.Security.Util.Some" "Some"
#endif
data DictEq a where
DictEq :: Eq a => DictEq a
class SomeEq f where
someEq :: DictEq (f a)
instance (Typed f, SomeEq f) => Eq (Some f) where
Some (f a
x :: f a) == :: Some f -> Some f -> Bool
== Some (f a
y :: f a') =
case TypeOf f a -> TypeOf f a -> Maybe (a :=: a)
forall (f :: * -> *) typ typ'.
Unify f =>
f typ -> f typ' -> Maybe (typ :=: typ')
unify (f a -> TypeOf f a
forall (f :: * -> *) typ. Typed f => f typ -> TypeOf f typ
typeOf f a
x) (f a -> TypeOf f a
forall (f :: * -> *) typ. Typed f => f typ -> TypeOf f typ
typeOf f a
y) of
Maybe (a :=: a)
Nothing -> Bool
False
Just a :=: a
Refl -> case DictEq (f a)
forall (f :: * -> *) a. SomeEq f => DictEq (f a)
someEq :: DictEq (f a) of DictEq (f a)
DictEq -> f a
x f a -> f a -> Bool
forall a. Eq a => a -> a -> Bool
== f a
f a
y
data DictShow a where
DictShow :: Show a => DictShow a
class SomeShow f where
someShow :: DictShow (f a)
instance SomeShow f => Show (Some f) where
show :: Some f -> String
show (Some (f a
x :: f a)) =
case DictShow (f a)
forall (f :: * -> *) a. SomeShow f => DictShow (f a)
someShow :: DictShow (f a) of DictShow (f a)
DictShow -> f a -> String
forall a. Show a => a -> String
show f a
x
data DictPretty a where
DictPretty :: Pretty a => DictPretty a
class SomePretty f where
somePretty :: DictPretty (f a)
instance SomePretty f => Pretty (Some f) where
pretty :: Some f -> String
pretty (Some (f a
x :: f a)) =
case DictPretty (f a)
forall (f :: * -> *) a. SomePretty f => DictPretty (f a)
somePretty :: DictPretty (f a) of DictPretty (f a)
DictPretty -> f a -> String
forall a. Pretty a => a -> String
pretty f a
x
typecheckSome :: Typed f => Some f -> Some (TypeOf f) -> Bool
typecheckSome :: Some f -> Some (TypeOf f) -> Bool
typecheckSome (Some f a
x) (Some TypeOf f a
typ) =
case TypeOf f a -> TypeOf f a -> Maybe (a :=: a)
forall (f :: * -> *) typ typ'.
Unify f =>
f typ -> f typ' -> Maybe (typ :=: typ')
unify (f a -> TypeOf f a
forall (f :: * -> *) typ. Typed f => f typ -> TypeOf f typ
typeOf f a
x) TypeOf f a
typ of
Just a :=: a
Refl -> Bool
True
Maybe (a :=: a)
Nothing -> Bool
False