{-# language AllowAmbiguousTypes #-} {-# language DataKinds #-} {-# language DefaultSignatures #-} {-# language FlexibleContexts #-} {-# language FlexibleInstances #-} {-# language MultiParamTypeClasses #-} {-# language ScopedTypeVariables #-} {-# language StandaloneKindSignatures #-} {-# language TypeApplications #-} {-# language TypeFamilies #-} {-# language UndecidableInstances #-} module Rel8.Generic.Rel8able ( KRel8able, Rel8able , Algebra , GRep , GColumns, gfromColumns, gtoColumns , greify, gunreify , TUnreifyContext ) where -- base import Data.Kind ( Constraint, Type ) import Data.Proxy ( Proxy( Proxy ) ) import Data.Type.Equality ( (:~:)( Refl ) ) import GHC.Generics ( Generic, Rep, from, to ) import Prelude -- rel8 import Rel8.Kind.Context ( Reifiable ) import Rel8.FCF ( Eval, Exp ) import Rel8.Generic.Map ( GMap, GMappable, gmap, gunmap ) import Rel8.Generic.Record ( Record(..) ) import Rel8.Generic.Table ( GAlgebra ) import qualified Rel8.Generic.Table.Record as G import qualified Rel8.Kind.Algebra as K ( Algebra(..) ) import Rel8.Schema.Context ( Col ) import Rel8.Schema.Context.Label ( Labelable ) import Rel8.Schema.HTable ( HTable ) import qualified Rel8.Schema.Kind as K import Rel8.Schema.Reify ( Reify, UnwrapReify ) import Rel8.Schema.Result ( Result ) import Rel8.Table ( fromColumns, toColumns, reify, unreify , TTable, TColumns, TContext, TUnreify ) -- | The kind of 'Rel8able' types type KRel8able :: Type type KRel8able = K.Rel8able -- | This type class allows you to define custom 'Table's using higher-kinded -- data types. Higher-kinded data types are data types of the pattern: -- -- @ -- data MyType f = -- MyType { field1 :: Column f T1 OR HK1 f -- , field2 :: Column f T2 OR HK2 f -- , ... -- , fieldN :: Column f Tn OR HKn f -- } -- @ -- -- where @Tn@ is any Haskell type, and @HKn@ is any higher-kinded type. -- -- That is, higher-kinded data are records where all fields in the record are -- all either of the type @Column f T@ (for any @T@), or are themselves -- higher-kinded data: -- -- [Nested] -- -- @ -- data Nested f = -- Nested { nested1 :: MyType f -- , nested2 :: MyType f -- } -- @ -- -- The @Rel8able@ type class is used to give us a special mapping operation -- that lets us change the type parameter @f@. -- -- [Supplying @Rel8able@ instances] -- -- This type class should be derived generically for all table types in your -- project. To do this, enable the @DeriveAnyType@ and @DeriveGeneric@ language -- extensions: -- -- @ -- \{\-\# LANGUAGE DeriveAnyClass, DeriveGeneric #-\} -- -- data MyType f = MyType { fieldA :: Column f T } -- deriving ( GHC.Generics.Generic, Rel8able ) -- @ type Rel8able :: K.Rel8able -> Constraint class HTable (GColumns t) => Rel8able t where type GColumns t :: K.HTable gfromColumns :: (Labelable context, Reifiable context) => GColumns t (Col (Reify context)) -> t (Reify context) gtoColumns :: (Labelable context, Reifiable context) => t (Reify context) -> GColumns t (Col (Reify context)) greify :: (Labelable context, Reifiable context) => t context -> t (Reify context) gunreify :: (Labelable context, Reifiable context) => t (Reify context) -> t context type GColumns t = G.GColumns TColumns (GRep t (Reify Result)) default gfromColumns :: forall context. ( Generic (Record (t (Reify context))) , G.GTable (TTable (Reify context)) TColumns (Col (Reify context)) (GRep t (Reify context)) , G.GColumns TColumns (GRep t (Reify context)) ~ GColumns t ) => GColumns t (Col (Reify context)) -> t (Reify context) gfromColumns = unrecord . to . G.gfromColumns @(TTable (Reify context)) @TColumns fromColumns default gtoColumns :: forall context. ( Generic (Record (t (Reify context))) , G.GTable (TTable (Reify context)) TColumns (Col (Reify context)) (GRep t (Reify context)) , G.GColumns TColumns (GRep t (Reify context)) ~ GColumns t ) => t (Reify context) -> GColumns t (Col (Reify context)) gtoColumns = G.gtoColumns @(TTable (Reify context)) @TColumns toColumns . from . Record default greify :: forall context. ( Generic (Record (t context)) , Generic (Record (t (Reify context))) , GMappable (TTable (Reify context)) (GRep t (Reify context)) , GRep t context ~ GMap TUnreify (GRep t (Reify context)) ) => t context -> t (Reify context) greify = unrecord . to . gunmap @(TTable (Reify context)) (Proxy @TUnreify) (reify Refl) . from . Record default gunreify :: forall context. ( Generic (Record (t context)) , Generic (Record (t (Reify context))) , GMappable (TTable (Reify context)) (GRep t (Reify context)) , GRep t context ~ GMap TUnreify (GRep t (Reify context)) ) => t (Reify context) -> t context gunreify = unrecord . to . gmap @(TTable (Reify context)) (Proxy @TUnreify) (unreify Refl) . from . Record type Algebra :: K.Rel8able -> K.Algebra type Algebra t = GAlgebra (GRep t (Reify Result)) type GRep :: K.Rel8able -> K.Context -> Type -> Type type GRep t context = Rep (Record (t context)) data TUnreifyContext :: Type -> Exp K.Context type instance Eval (TUnreifyContext a) = UnwrapReify (Eval (TContext a))