{-# LANGUAGE CPP #-}
-- | Hiding existentials
module Hackage.Security.Util.Some (
    Some(..)
    -- ** Equality
  , DictEq(..)
  , SomeEq(..)
    -- ** Serialization
  , DictShow(..)
  , SomeShow(..)
    -- ** Pretty-printing
  , DictPretty(..)
  , SomePretty(..)
    -- ** Type checking
  , typecheckSome
#if !MIN_VERSION_base(4,7,0)
    -- ** Compatibility with base < 4.7
  , tyConSome
#endif
  ) where

import MyPrelude
#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

{-------------------------------------------------------------------------------
  Equality on Some types

  Note that we cannot really do something similar for ordering; what value
  should we return for

  > Some (f x) `compare` Some (f x')

  where @x :: a@, @x' :: a'@ and @a /= a'@? These are incomparable.
-------------------------------------------------------------------------------}

data DictEq a where
  DictEq :: Eq a => DictEq a

-- | Type @f@ satisfies @SomeEq f@ if @f a@ satisfies @Eq@ independent of @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 forall (f :: * -> *) typ typ'.
Unify f =>
f typ -> f typ' -> Maybe (typ :=: typ')
unify (forall (f :: * -> *) typ. Typed f => f typ -> TypeOf f typ
typeOf f a
x) (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 forall (f :: * -> *) a. SomeEq f => DictEq (f a)
someEq :: DictEq (f a) of DictEq (f a)
DictEq -> f a
x forall a. Eq a => a -> a -> Bool
== f a
y

{-------------------------------------------------------------------------------
  Showing Some types
-------------------------------------------------------------------------------}

data DictShow a where
  DictShow :: Show a => DictShow a

-- | Type @f@ satisfies @SomeShow f@ if @f a@ satisfies @Show@ independent of @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 forall (f :: * -> *) a. SomeShow f => DictShow (f a)
someShow :: DictShow (f a) of DictShow (f a)
DictShow -> forall a. Show a => a -> String
show f a
x

{-------------------------------------------------------------------------------
  Pretty-printing Some types
-------------------------------------------------------------------------------}

data DictPretty a where
  DictPretty :: Pretty a => DictPretty a

-- | Type @f@ satisfies @SomeShow f@ if @f a@ satisfies @Show@ independent of @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 forall (f :: * -> *) a. SomePretty f => DictPretty (f a)
somePretty :: DictPretty (f a) of DictPretty (f a)
DictPretty -> forall a. Pretty a => a -> String
pretty f a
x

{-------------------------------------------------------------------------------
  Typechecking Some types
-------------------------------------------------------------------------------}

typecheckSome :: Typed f => Some f -> Some (TypeOf f) -> Bool
typecheckSome :: forall (f :: * -> *). Typed f => Some f -> Some (TypeOf f) -> Bool
typecheckSome (Some f a
x) (Some TypeOf f a
typ) =
    case forall (f :: * -> *) typ typ'.
Unify f =>
f typ -> f typ' -> Maybe (typ :=: typ')
unify (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