{-# LANGUAGE CPP #-}
{-# LANGUAGE AllowAmbiguousTypes #-}
{-# LANGUAGE ConstraintKinds #-}
{-# LANGUAGE DataKinds #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE GADTs #-}
{-# LANGUAGE InstanceSigs #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE PolyKinds #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE TypeOperators #-}
{-# LANGUAGE UndecidableInstances #-}
{-# LANGUAGE UndecidableSuperClasses #-}
module Data.Row.Internal
(
Row(..)
, Label(..)
, KnownSymbol
, LT(..)
, Empty
, HideType(..)
, Extend, Modify, Rename
, type (.==), type (.!), type (.-), type (.\\)
, type (.+), type (.\/), type (.//)
, Lacks, type (.\), HasType
, Forall(..)
, BiForall(..)
, BiConstraint
, Unconstrained
, Unconstrained1
, Unconstrained2
, FrontExtends(..)
, FrontExtendsDict(..)
, WellBehaved, AllUniqueLabels
, Ap, ApSingle, Zip, Map, Subset, Disjoint
, Labels, labels, labels'
, show'
, toKey
, type (≈)
)
where
import Data.Bifunctor (Bifunctor(..))
import Data.Constraint
import Data.Functor.Const
import Data.Proxy
import Data.String (IsString (fromString))
import Data.Text (Text)
import qualified Data.Text as Text
import Data.Type.Equality (type (==))
import GHC.OverloadedLabels
import GHC.TypeLits
import qualified GHC.TypeLits as TL
newtype Row a = R [LT a]
data LT a = Symbol :-> a
data Label (s :: Symbol) = Label
deriving (Label s -> Label s -> Bool
(Label s -> Label s -> Bool)
-> (Label s -> Label s -> Bool) -> Eq (Label s)
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
forall (s :: Symbol). Label s -> Label s -> Bool
/= :: Label s -> Label s -> Bool
$c/= :: forall (s :: Symbol). Label s -> Label s -> Bool
== :: Label s -> Label s -> Bool
$c== :: forall (s :: Symbol). Label s -> Label s -> Bool
Eq)
instance KnownSymbol s => Show (Label s) where
show :: Label s -> String
show = Label s -> String
forall (n :: Symbol) (proxy :: Symbol -> *).
KnownSymbol n =>
proxy n -> String
symbolVal
instance x ≈ y => IsLabel x (Label y) where
#if __GLASGOW_HASKELL__ >= 802
fromLabel :: Label y
fromLabel = Label y
forall (s :: Symbol). Label s
Label
#else
fromLabel _ = Label
#endif
show' :: (IsString s, Show a) => a -> s
show' :: a -> s
show' = String -> s
forall a. IsString a => String -> a
fromString (String -> s) -> (a -> String) -> a -> s
forall b c a. (b -> c) -> (a -> b) -> a -> c
. a -> String
forall a. Show a => a -> String
show
toKey :: forall s. KnownSymbol s => Label s -> Text
toKey :: Label s -> Text
toKey = String -> Text
Text.pack (String -> Text) -> (Label s -> String) -> Label s -> Text
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Label s -> String
forall (n :: Symbol) (proxy :: Symbol -> *).
KnownSymbol n =>
proxy n -> String
symbolVal
type Empty = R '[]
data HideType where
HideType :: a -> HideType
infixl 4 .\
type family (r :: Row k) .\ (l :: Symbol) :: Constraint where
R '[] .\ l = Unconstrained
R r .\ l = LacksR l r r
type family Extend (l :: Symbol) (a :: k) (r :: Row k) :: Row k where
Extend l a (R '[]) = R (l :-> a ': '[])
Extend l a (R x) = R (Inject (l :-> a) x)
type family Modify (l :: Symbol) (a :: k) (r :: Row k) :: Row k where
Modify l a (R ρ) = R (ModifyR l a ρ)
type family Rename (l :: Symbol) (l' :: Symbol) (r :: Row k) :: Row k where
Rename l l' r = Extend l' (r .! l) (r .- l)
infixl 5 .!
type family (r :: Row k) .! (t :: Symbol) :: k where
R r .! l = Get l r
infixl 6 .-
type family (r :: Row k) .- (s :: Symbol) :: Row k where
R r .- l = R (Remove l r)
infixl 6 .\\
type family (l :: Row k) .\\ (r :: Row k) :: Row k where
R l .\\ R r = R (Diff l r)
infixl 6 .+
type family (l :: Row k) .+ (r :: Row k) :: Row k where
x .+ R '[] = x
R '[] .+ y = y
x .+ R '[l :-> a] = Extend l a x
R '[l :-> a] .+ y = Extend l a y
R l .+ R r = R (Merge l r)
infixl 6 .\/
type family (l :: Row k) .\/ (r :: Row k) where
x .\/ R '[] = x
R '[] .\/ y = y
R l .\/ R r = R (MinJoinR l r)
infixl 6 .//
type family (l :: Row k) .// (r :: Row k) where
x .// R '[] = x
R '[] .// y = y
R l .// R r = R (ConstUnionR l r)
class Lacks (l :: Symbol) (r :: Row *)
instance (r .\ l) => Lacks l r
class (r .! l ≈ a) => HasType l a r
instance (r .! l ≈ a) => HasType l a r
infix 7 .==
type (l :: Symbol) .== (a :: k) = Extend l a Empty
data FrontExtendsDict l t r = forall ρ. FrontExtendsDict (Dict (r ~ R ρ, R (l :-> t ': ρ) ≈ Extend l t (R ρ), AllUniqueLabelsR (l :-> t ': ρ)))
class FrontExtends l t r where
frontExtendsDict :: FrontExtendsDict l t r
instance (r ~ R ρ, R (l :-> t ': ρ) ≈ Extend l t (R ρ), AllUniqueLabelsR (l :-> t ': ρ)) => FrontExtends l t r where
frontExtendsDict :: FrontExtendsDict l t r
frontExtendsDict = Dict
(r ~ 'R ρ, 'R ((l ':-> t) : ρ) ≈ Extend l t ('R ρ),
AllUniqueLabelsR ((l ':-> t) : ρ))
-> FrontExtendsDict l t r
forall k (l :: Symbol) (t :: k) (r :: Row k) (ρ :: [LT k]).
Dict
(r ~ 'R ρ, 'R ((l ':-> t) : ρ) ≈ Extend l t ('R ρ),
AllUniqueLabelsR ((l ':-> t) : ρ))
-> FrontExtendsDict l t r
FrontExtendsDict Dict
(r ~ 'R ρ, 'R ((l ':-> t) : ρ) ≈ Extend l t ('R ρ),
AllUniqueLabelsR ((l ':-> t) : ρ))
forall (a :: Constraint). a => Dict a
Dict
class Forall (r :: Row k) (c :: k -> Constraint) where
metamorph :: forall (p :: * -> * -> *) (f :: Row k -> *) (g :: Row k -> *) (h :: k -> *). Bifunctor p
=> Proxy (Proxy h, Proxy p)
-> (f Empty -> g Empty)
-> (forall ℓ τ ρ. (KnownSymbol ℓ, c τ, HasType ℓ τ ρ)
=> Label ℓ -> f ρ -> p (f (ρ .- ℓ)) (h τ))
-> (forall ℓ τ ρ. (KnownSymbol ℓ, c τ, FrontExtends ℓ τ ρ, AllUniqueLabels (Extend ℓ τ ρ))
=> Label ℓ -> p (g ρ) (h τ) -> g (Extend ℓ τ ρ))
-> f r
-> g r
instance Forall (R '[]) c where
{-# INLINE metamorph #-}
metamorph :: Proxy (Proxy h, Proxy p)
-> (f Empty -> g Empty)
-> (forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, HasType ℓ τ ρ) =>
Label ℓ -> f ρ -> p (f (ρ .- ℓ)) (h τ))
-> (forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, FrontExtends ℓ τ ρ,
AllUniqueLabels (Extend ℓ τ ρ)) =>
Label ℓ -> p (g ρ) (h τ) -> g (Extend ℓ τ ρ))
-> f Empty
-> g Empty
metamorph Proxy (Proxy h, Proxy p)
_ f Empty -> g Empty
empty forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, HasType ℓ τ ρ) =>
Label ℓ -> f ρ -> p (f (ρ .- ℓ)) (h τ)
_ forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, FrontExtends ℓ τ ρ,
AllUniqueLabels (Extend ℓ τ ρ)) =>
Label ℓ -> p (g ρ) (h τ) -> g (Extend ℓ τ ρ)
_ = f Empty -> g Empty
empty
instance (KnownSymbol ℓ, c τ, Forall ('R ρ) c, FrontExtends ℓ τ ('R ρ), AllUniqueLabels (Extend ℓ τ ('R ρ))) => Forall ('R (ℓ :-> τ ': ρ) :: Row k) c where
{-# INLINE metamorph #-}
metamorph :: Proxy (Proxy h, Proxy p)
-> (f Empty -> g Empty)
-> (forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, HasType ℓ τ ρ) =>
Label ℓ -> f ρ -> p (f (ρ .- ℓ)) (h τ))
-> (forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, FrontExtends ℓ τ ρ,
AllUniqueLabels (Extend ℓ τ ρ)) =>
Label ℓ -> p (g ρ) (h τ) -> g (Extend ℓ τ ρ))
-> f ('R ((ℓ ':-> τ) : ρ))
-> g ('R ((ℓ ':-> τ) : ρ))
metamorph Proxy (Proxy h, Proxy p)
h f Empty -> g Empty
empty forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, HasType ℓ τ ρ) =>
Label ℓ -> f ρ -> p (f (ρ .- ℓ)) (h τ)
uncons forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, FrontExtends ℓ τ ρ,
AllUniqueLabels (Extend ℓ τ ρ)) =>
Label ℓ -> p (g ρ) (h τ) -> g (Extend ℓ τ ρ)
cons = case FrontExtends ℓ τ ('R ρ) => FrontExtendsDict ℓ τ ('R ρ)
forall k (l :: Symbol) (t :: k) (r :: Row k).
FrontExtends l t r =>
FrontExtendsDict l t r
frontExtendsDict @ℓ @τ @('R ρ) of
FrontExtendsDict Dict
('R ρ ~ 'R ρ, 'R ((ℓ ':-> τ) : ρ) ≈ Extend ℓ τ ('R ρ),
AllUniqueLabelsR ((ℓ ':-> τ) : ρ))
Dict ->
Label ℓ -> p (g ('R ρ)) (h τ) -> g (Extend ℓ τ ('R ρ))
forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, FrontExtends ℓ τ ρ,
AllUniqueLabels (Extend ℓ τ ρ)) =>
Label ℓ -> p (g ρ) (h τ) -> g (Extend ℓ τ ρ)
cons (Label ℓ
forall (s :: Symbol). Label s
Label @ℓ) (p (g ('R ρ)) (h τ) -> g ('R ((ℓ ':-> τ) : ρ)))
-> (f ('R ((ℓ ':-> τ) : ρ)) -> p (g ('R ρ)) (h τ))
-> f ('R ((ℓ ':-> τ) : ρ))
-> g ('R ((ℓ ':-> τ) : ρ))
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (f ('R ρ) -> g ('R ρ)) -> p (f ('R ρ)) (h τ) -> p (g ('R ρ)) (h τ)
forall (p :: * -> * -> *) a b c.
Bifunctor p =>
(a -> b) -> p a c -> p b c
first (Proxy (Proxy h, Proxy p)
-> (f Empty -> g Empty)
-> (forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, HasType ℓ τ ρ) =>
Label ℓ -> f ρ -> p (f (ρ .- ℓ)) (h τ))
-> (forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, FrontExtends ℓ τ ρ,
AllUniqueLabels (Extend ℓ τ ρ)) =>
Label ℓ -> p (g ρ) (h τ) -> g (Extend ℓ τ ρ))
-> f ('R ρ)
-> g ('R ρ)
forall k (r :: Row k) (c :: k -> Constraint) (p :: * -> * -> *)
(f :: Row k -> *) (g :: Row k -> *) (h :: k -> *).
(Forall r c, Bifunctor p) =>
Proxy (Proxy h, Proxy p)
-> (f Empty -> g Empty)
-> (forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, HasType ℓ τ ρ) =>
Label ℓ -> f ρ -> p (f (ρ .- ℓ)) (h τ))
-> (forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, FrontExtends ℓ τ ρ,
AllUniqueLabels (Extend ℓ τ ρ)) =>
Label ℓ -> p (g ρ) (h τ) -> g (Extend ℓ τ ρ))
-> f r
-> g r
metamorph @_ @('R ρ) @c Proxy (Proxy h, Proxy p)
h f Empty -> g Empty
empty forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, HasType ℓ τ ρ) =>
Label ℓ -> f ρ -> p (f (ρ .- ℓ)) (h τ)
uncons forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, FrontExtends ℓ τ ρ,
AllUniqueLabels (Extend ℓ τ ρ)) =>
Label ℓ -> p (g ρ) (h τ) -> g (Extend ℓ τ ρ)
cons) (p (f ('R ρ)) (h τ) -> p (g ('R ρ)) (h τ))
-> (f ('R ((ℓ ':-> τ) : ρ)) -> p (f ('R ρ)) (h τ))
-> f ('R ((ℓ ':-> τ) : ρ))
-> p (g ('R ρ)) (h τ)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Label ℓ
-> f ('R ((ℓ ':-> τ) : ρ))
-> p (f ('R ((ℓ ':-> τ) : ρ) .- ℓ)) (h τ)
forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, HasType ℓ τ ρ) =>
Label ℓ -> f ρ -> p (f (ρ .- ℓ)) (h τ)
uncons (Label ℓ
forall (s :: Symbol). Label s
Label @ℓ)
class BiForall (r1 :: Row k1) (r2 :: Row k2) (c :: k1 -> k2 -> Constraint) where
biMetamorph :: forall (p :: * -> * -> *) (f :: Row k1 -> Row k2 -> *) (g :: Row k1 -> Row k2 -> *)
(h :: k1 -> k2 -> *). Bifunctor p
=> Proxy (Proxy h, Proxy p)
-> (f Empty Empty -> g Empty Empty)
-> (forall ℓ τ1 τ2 ρ1 ρ2. (KnownSymbol ℓ, c τ1 τ2, HasType ℓ τ1 ρ1, HasType ℓ τ2 ρ2)
=> Label ℓ -> f ρ1 ρ2 -> p (f (ρ1 .- ℓ) (ρ2 .- ℓ)) (h τ1 τ2))
-> (forall ℓ τ1 τ2 ρ1 ρ2. (KnownSymbol ℓ, c τ1 τ2, FrontExtends ℓ τ1 ρ1, FrontExtends ℓ τ2 ρ2, AllUniqueLabels (Extend ℓ τ1 ρ1), AllUniqueLabels (Extend ℓ τ2 ρ2))
=> Label ℓ -> p (g ρ1 ρ2) (h τ1 τ2) -> g (Extend ℓ τ1 ρ1) (Extend ℓ τ2 ρ2))
-> f r1 r2 -> g r1 r2
instance BiForall (R '[]) (R '[]) c1 where
{-# INLINE biMetamorph #-}
biMetamorph :: Proxy (Proxy h, Proxy p)
-> (f Empty Empty -> g Empty Empty)
-> (forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c1 τ1 τ2, HasType ℓ τ1 ρ1, HasType ℓ τ2 ρ2) =>
Label ℓ -> f ρ1 ρ2 -> p (f (ρ1 .- ℓ) (ρ2 .- ℓ)) (h τ1 τ2))
-> (forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c1 τ1 τ2, FrontExtends ℓ τ1 ρ1,
FrontExtends ℓ τ2 ρ2, AllUniqueLabels (Extend ℓ τ1 ρ1),
AllUniqueLabels (Extend ℓ τ2 ρ2)) =>
Label ℓ
-> p (g ρ1 ρ2) (h τ1 τ2) -> g (Extend ℓ τ1 ρ1) (Extend ℓ τ2 ρ2))
-> f Empty Empty
-> g Empty Empty
biMetamorph Proxy (Proxy h, Proxy p)
_ f Empty Empty -> g Empty Empty
empty forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c1 τ1 τ2, HasType ℓ τ1 ρ1, HasType ℓ τ2 ρ2) =>
Label ℓ -> f ρ1 ρ2 -> p (f (ρ1 .- ℓ) (ρ2 .- ℓ)) (h τ1 τ2)
_ forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c1 τ1 τ2, FrontExtends ℓ τ1 ρ1,
FrontExtends ℓ τ2 ρ2, AllUniqueLabels (Extend ℓ τ1 ρ1),
AllUniqueLabels (Extend ℓ τ2 ρ2)) =>
Label ℓ
-> p (g ρ1 ρ2) (h τ1 τ2) -> g (Extend ℓ τ1 ρ1) (Extend ℓ τ2 ρ2)
_ = f Empty Empty -> g Empty Empty
empty
instance (KnownSymbol ℓ, c τ1 τ2, BiForall ('R ρ1) ('R ρ2) c, FrontExtends ℓ τ1 ('R ρ1), FrontExtends ℓ τ2 ('R ρ2), AllUniqueLabels (Extend ℓ τ1 ('R ρ1)), AllUniqueLabels (Extend ℓ τ2 ('R ρ2)))
=> BiForall ('R (ℓ :-> τ1 ': ρ1)) ('R (ℓ :-> τ2 ': ρ2)) c where
{-# INLINE biMetamorph #-}
biMetamorph :: Proxy (Proxy h, Proxy p)
-> (f Empty Empty -> g Empty Empty)
-> (forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c τ1 τ2, HasType ℓ τ1 ρ1, HasType ℓ τ2 ρ2) =>
Label ℓ -> f ρ1 ρ2 -> p (f (ρ1 .- ℓ) (ρ2 .- ℓ)) (h τ1 τ2))
-> (forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c τ1 τ2, FrontExtends ℓ τ1 ρ1,
FrontExtends ℓ τ2 ρ2, AllUniqueLabels (Extend ℓ τ1 ρ1),
AllUniqueLabels (Extend ℓ τ2 ρ2)) =>
Label ℓ
-> p (g ρ1 ρ2) (h τ1 τ2) -> g (Extend ℓ τ1 ρ1) (Extend ℓ τ2 ρ2))
-> f ('R ((ℓ ':-> τ1) : ρ1)) ('R ((ℓ ':-> τ2) : ρ2))
-> g ('R ((ℓ ':-> τ1) : ρ1)) ('R ((ℓ ':-> τ2) : ρ2))
biMetamorph Proxy (Proxy h, Proxy p)
h f Empty Empty -> g Empty Empty
empty forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c τ1 τ2, HasType ℓ τ1 ρ1, HasType ℓ τ2 ρ2) =>
Label ℓ -> f ρ1 ρ2 -> p (f (ρ1 .- ℓ) (ρ2 .- ℓ)) (h τ1 τ2)
uncons forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c τ1 τ2, FrontExtends ℓ τ1 ρ1,
FrontExtends ℓ τ2 ρ2, AllUniqueLabels (Extend ℓ τ1 ρ1),
AllUniqueLabels (Extend ℓ τ2 ρ2)) =>
Label ℓ
-> p (g ρ1 ρ2) (h τ1 τ2) -> g (Extend ℓ τ1 ρ1) (Extend ℓ τ2 ρ2)
cons = case (FrontExtends ℓ τ1 ('R ρ1) => FrontExtendsDict ℓ τ1 ('R ρ1)
forall k (l :: Symbol) (t :: k) (r :: Row k).
FrontExtends l t r =>
FrontExtendsDict l t r
frontExtendsDict @ℓ @τ1 @('R ρ1), FrontExtends ℓ τ2 ('R ρ2) => FrontExtendsDict ℓ τ2 ('R ρ2)
forall k (l :: Symbol) (t :: k) (r :: Row k).
FrontExtends l t r =>
FrontExtendsDict l t r
frontExtendsDict @ℓ @τ2 @('R ρ2)) of
(FrontExtendsDict Dict
('R ρ1 ~ 'R ρ, 'R ((ℓ ':-> τ1) : ρ) ≈ Extend ℓ τ1 ('R ρ),
AllUniqueLabelsR ((ℓ ':-> τ1) : ρ))
Dict, FrontExtendsDict Dict
('R ρ2 ~ 'R ρ, 'R ((ℓ ':-> τ2) : ρ) ≈ Extend ℓ τ2 ('R ρ),
AllUniqueLabelsR ((ℓ ':-> τ2) : ρ))
Dict) ->
Label ℓ
-> p (g ('R ρ1) ('R ρ2)) (h τ1 τ2)
-> g (Extend ℓ τ1 ('R ρ1)) (Extend ℓ τ2 ('R ρ2))
forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c τ1 τ2, FrontExtends ℓ τ1 ρ1,
FrontExtends ℓ τ2 ρ2, AllUniqueLabels (Extend ℓ τ1 ρ1),
AllUniqueLabels (Extend ℓ τ2 ρ2)) =>
Label ℓ
-> p (g ρ1 ρ2) (h τ1 τ2) -> g (Extend ℓ τ1 ρ1) (Extend ℓ τ2 ρ2)
cons (Label ℓ
forall (s :: Symbol). Label s
Label @ℓ) (p (g ('R ρ1) ('R ρ2)) (h τ1 τ2)
-> g ('R ((ℓ ':-> τ1) : ρ1)) ('R ((ℓ ':-> τ2) : ρ2)))
-> (f ('R ((ℓ ':-> τ1) : ρ1)) ('R ((ℓ ':-> τ2) : ρ2))
-> p (g ('R ρ1) ('R ρ2)) (h τ1 τ2))
-> f ('R ((ℓ ':-> τ1) : ρ1)) ('R ((ℓ ':-> τ2) : ρ2))
-> g ('R ((ℓ ':-> τ1) : ρ1)) ('R ((ℓ ':-> τ2) : ρ2))
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (f ('R ρ1) ('R ρ2) -> g ('R ρ1) ('R ρ2))
-> p (f ('R ρ1) ('R ρ2)) (h τ1 τ2)
-> p (g ('R ρ1) ('R ρ2)) (h τ1 τ2)
forall (p :: * -> * -> *) a b c.
Bifunctor p =>
(a -> b) -> p a c -> p b c
first (Proxy (Proxy h, Proxy p)
-> (f Empty Empty -> g Empty Empty)
-> (forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c τ1 τ2, HasType ℓ τ1 ρ1, HasType ℓ τ2 ρ2) =>
Label ℓ -> f ρ1 ρ2 -> p (f (ρ1 .- ℓ) (ρ2 .- ℓ)) (h τ1 τ2))
-> (forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c τ1 τ2, FrontExtends ℓ τ1 ρ1,
FrontExtends ℓ τ2 ρ2, AllUniqueLabels (Extend ℓ τ1 ρ1),
AllUniqueLabels (Extend ℓ τ2 ρ2)) =>
Label ℓ
-> p (g ρ1 ρ2) (h τ1 τ2) -> g (Extend ℓ τ1 ρ1) (Extend ℓ τ2 ρ2))
-> f ('R ρ1) ('R ρ2)
-> g ('R ρ1) ('R ρ2)
forall k1 k2 (r1 :: Row k1) (r2 :: Row k2)
(c :: k1 -> k2 -> Constraint) (p :: * -> * -> *)
(f :: Row k1 -> Row k2 -> *) (g :: Row k1 -> Row k2 -> *)
(h :: k1 -> k2 -> *).
(BiForall r1 r2 c, Bifunctor p) =>
Proxy (Proxy h, Proxy p)
-> (f Empty Empty -> g Empty Empty)
-> (forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c τ1 τ2, HasType ℓ τ1 ρ1, HasType ℓ τ2 ρ2) =>
Label ℓ -> f ρ1 ρ2 -> p (f (ρ1 .- ℓ) (ρ2 .- ℓ)) (h τ1 τ2))
-> (forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c τ1 τ2, FrontExtends ℓ τ1 ρ1,
FrontExtends ℓ τ2 ρ2, AllUniqueLabels (Extend ℓ τ1 ρ1),
AllUniqueLabels (Extend ℓ τ2 ρ2)) =>
Label ℓ
-> p (g ρ1 ρ2) (h τ1 τ2) -> g (Extend ℓ τ1 ρ1) (Extend ℓ τ2 ρ2))
-> f r1 r2
-> g r1 r2
biMetamorph @_ @_ @('R ρ1) @('R ρ2) @c Proxy (Proxy h, Proxy p)
h f Empty Empty -> g Empty Empty
empty forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c τ1 τ2, HasType ℓ τ1 ρ1, HasType ℓ τ2 ρ2) =>
Label ℓ -> f ρ1 ρ2 -> p (f (ρ1 .- ℓ) (ρ2 .- ℓ)) (h τ1 τ2)
uncons forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c τ1 τ2, FrontExtends ℓ τ1 ρ1,
FrontExtends ℓ τ2 ρ2, AllUniqueLabels (Extend ℓ τ1 ρ1),
AllUniqueLabels (Extend ℓ τ2 ρ2)) =>
Label ℓ
-> p (g ρ1 ρ2) (h τ1 τ2) -> g (Extend ℓ τ1 ρ1) (Extend ℓ τ2 ρ2)
cons) (p (f ('R ρ1) ('R ρ2)) (h τ1 τ2)
-> p (g ('R ρ1) ('R ρ2)) (h τ1 τ2))
-> (f ('R ((ℓ ':-> τ1) : ρ1)) ('R ((ℓ ':-> τ2) : ρ2))
-> p (f ('R ρ1) ('R ρ2)) (h τ1 τ2))
-> f ('R ((ℓ ':-> τ1) : ρ1)) ('R ((ℓ ':-> τ2) : ρ2))
-> p (g ('R ρ1) ('R ρ2)) (h τ1 τ2)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Label ℓ
-> f ('R ((ℓ ':-> τ1) : ρ1)) ('R ((ℓ ':-> τ2) : ρ2))
-> p (f ('R ((ℓ ':-> τ1) : ρ1) .- ℓ) ('R ((ℓ ':-> τ2) : ρ2) .- ℓ))
(h τ1 τ2)
forall (ℓ :: Symbol) (τ1 :: k1) (τ2 :: k2) (ρ1 :: Row k1)
(ρ2 :: Row k2).
(KnownSymbol ℓ, c τ1 τ2, HasType ℓ τ1 ρ1, HasType ℓ τ2 ρ2) =>
Label ℓ -> f ρ1 ρ2 -> p (f (ρ1 .- ℓ) (ρ2 .- ℓ)) (h τ1 τ2)
uncons (Label ℓ
forall (s :: Symbol). Label s
Label @ℓ)
class Unconstrained
instance Unconstrained
Unconstrained
class Unconstrained1 a
instance Unconstrained1 a
class Unconstrained2 a b
instance Unconstrained2 a b
class (c1 x, c2 y) => BiConstraint c1 c2 x y
instance (c1 x, c2 y) => BiConstraint c1 c2 x y
type family Labels (r :: Row a) where
Labels (R '[]) = '[]
Labels (R (l :-> a ': xs)) = l ': Labels (R xs)
labels :: forall ρ c s. (IsString s, Forall ρ c) => [s]
labels :: [s]
labels = Const [s] ρ -> [s]
forall a k (b :: k). Const a b -> a
getConst (Const [s] ρ -> [s]) -> Const [s] ρ -> [s]
forall a b. (a -> b) -> a -> b
$ Proxy (Proxy Proxy, Proxy Const)
-> (Const () Empty -> Const [s] Empty)
-> (forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, HasType ℓ τ ρ) =>
Label ℓ -> Const () ρ -> Const (Const () (ρ .- ℓ)) (Proxy τ))
-> (forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, FrontExtends ℓ τ ρ,
AllUniqueLabels (Extend ℓ τ ρ)) =>
Label ℓ
-> Const (Const [s] ρ) (Proxy τ) -> Const [s] (Extend ℓ τ ρ))
-> Const () ρ
-> Const [s] ρ
forall k (r :: Row k) (c :: k -> Constraint) (p :: * -> * -> *)
(f :: Row k -> *) (g :: Row k -> *) (h :: k -> *).
(Forall r c, Bifunctor p) =>
Proxy (Proxy h, Proxy p)
-> (f Empty -> g Empty)
-> (forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, HasType ℓ τ ρ) =>
Label ℓ -> f ρ -> p (f (ρ .- ℓ)) (h τ))
-> (forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, FrontExtends ℓ τ ρ,
AllUniqueLabels (Extend ℓ τ ρ)) =>
Label ℓ -> p (g ρ) (h τ) -> g (Extend ℓ τ ρ))
-> f r
-> g r
metamorph @_ @ρ @c @Const @(Const ()) @(Const [s]) @Proxy Proxy (Proxy Proxy, Proxy Const)
forall k (t :: k). Proxy t
Proxy (Const [s] Empty -> Const () Empty -> Const [s] Empty
forall a b. a -> b -> a
const (Const [s] Empty -> Const () Empty -> Const [s] Empty)
-> Const [s] Empty -> Const () Empty -> Const [s] Empty
forall a b. (a -> b) -> a -> b
$ [s] -> Const [s] Empty
forall k a (b :: k). a -> Const a b
Const []) forall k k p p (b :: k) (b :: k). p -> p -> Const (Const () b) b
forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, HasType ℓ τ ρ) =>
Label ℓ -> Const () ρ -> Const (Const () (ρ .- ℓ)) (Proxy τ)
doUncons forall k k k a a (b :: k) (b :: k) (b :: k).
(IsString a, Show a) =>
a -> Const (Const [a] b) b -> Const [a] b
forall (ℓ :: Symbol) (τ :: k) (ρ :: Row k).
(KnownSymbol ℓ, c τ, FrontExtends ℓ τ ρ,
AllUniqueLabels (Extend ℓ τ ρ)) =>
Label ℓ
-> Const (Const [s] ρ) (Proxy τ) -> Const [s] (Extend ℓ τ ρ)
doCons (() -> Const () ρ
forall k a (b :: k). a -> Const a b
Const ())
where doUncons :: p -> p -> Const (Const () b) b
doUncons p
_ p
_ = Const () b -> Const (Const () b) b
forall k a (b :: k). a -> Const a b
Const (Const () b -> Const (Const () b) b)
-> Const () b -> Const (Const () b) b
forall a b. (a -> b) -> a -> b
$ () -> Const () b
forall k a (b :: k). a -> Const a b
Const ()
doCons :: a -> Const (Const [a] b) b -> Const [a] b
doCons a
l (Const (Const [a]
c)) = [a] -> Const [a] b
forall k a (b :: k). a -> Const a b
Const ([a] -> Const [a] b) -> [a] -> Const [a] b
forall a b. (a -> b) -> a -> b
$ a -> a
forall s a. (IsString s, Show a) => a -> s
show' a
l a -> [a] -> [a]
forall a. a -> [a] -> [a]
: [a]
c
labels' :: forall ρ s. (IsString s, Forall ρ Unconstrained1) => [s]
labels' :: [s]
labels' = forall s. (IsString s, Forall ρ Unconstrained1) => [s]
forall k (ρ :: Row k) (c :: k -> Constraint) s.
(IsString s, Forall ρ c) =>
[s]
labels @ρ @Unconstrained1
type WellBehaved ρ = (Forall ρ Unconstrained1, AllUniqueLabels ρ)
type family AllUniqueLabels (r :: Row k) :: Constraint where
AllUniqueLabels (R r) = AllUniqueLabelsR r
type family AllUniqueLabelsR (r :: [LT k]) :: Constraint where
AllUniqueLabelsR '[] = Unconstrained
AllUniqueLabelsR '[l :-> a] = Unconstrained
AllUniqueLabelsR (l :-> a ': l :-> b ': _) = TypeError
(TL.Text "The label " :<>: ShowType l :<>: TL.Text " is not unique."
:$$: TL.Text "It is assigned to both " :<>: ShowType a :<>: TL.Text " and " :<>: ShowType b)
AllUniqueLabelsR (l :-> a ': l' :-> b ': r) = AllUniqueLabelsR (l' :-> b ': r)
type family Subset (r1 :: Row k) (r2 :: Row k) :: Constraint where
Subset ('R '[]) r = Unconstrained
Subset ('R (l ':-> a ': x)) r = (r .! l ≈ a, Subset ('R x) r)
type Disjoint l r = ( WellBehaved l
, WellBehaved r
, Subset l (l .+ r)
, Subset r (l .+ r)
, l .+ r .\\ l ≈ r
, l .+ r .\\ r ≈ l)
type family Map (f :: a -> b) (r :: Row a) :: Row b where
Map f (R r) = R (MapR f r)
type family MapR (f :: a -> b) (r :: [LT a]) :: [LT b] where
MapR f '[] = '[]
MapR f (l :-> v ': t) = l :-> f v ': MapR f t
type family Ap (fs :: Row (a -> b)) (r :: Row a) :: Row b where
Ap (R fs) (R r) = R (ApR fs r)
type family ApR (fs :: [LT (a -> b)]) (r :: [LT a]) :: [LT b] where
ApR '[] '[] = '[]
ApR (l :-> f ': tf) (l :-> v ': tv) = l :-> f v ': ApR tf tv
ApR _ _ = TypeError (TL.Text "Row types with different label sets cannot be App'd together.")
type family ApSingle (fs :: Row (a -> b)) (x :: a) :: Row b where
ApSingle (R fs) x = R (ApSingleR fs x)
type family ApSingleR (fs :: [LT (a -> b)]) (x :: a) :: [LT b] where
ApSingleR '[] _ = '[]
ApSingleR (l ':-> f ': fs) x = l ':-> f x ': ApSingleR fs x
type family Zip (r1 :: Row *) (r2 :: Row *) where
Zip (R r1) (R r2) = R (ZipR r1 r2)
type family ZipR (r1 :: [LT *]) (r2 :: [LT *]) where
ZipR '[] '[] = '[]
ZipR (l :-> t1 ': r1) (l :-> t2 ': r2) =
l :-> (t1, t2) ': ZipR r1 r2
ZipR (l :-> t1 ': r1) _ = TypeError (TL.Text "Row types with different label sets cannot be zipped"
:$$: TL.Text "For one, the label " :<>: ShowType l :<>: TL.Text " is not in both lists.")
ZipR '[] (l :-> t ': r) = TypeError (TL.Text "Row types with different label sets cannot be zipped"
:$$: TL.Text "For one, the label " :<>: ShowType l :<>: TL.Text " is not in both lists.")
type family Inject (l :: LT k) (r :: [LT k]) where
Inject (l :-> t) '[] = (l :-> t ': '[])
Inject (l :-> t) (l :-> t' ': x) = TypeError (TL.Text "Cannot inject a label into a row type that already has that label"
:$$: TL.Text "The label " :<>: ShowType l :<>: TL.Text " was already assigned the type "
:<>: ShowType t' :<>: TL.Text " and is now trying to be assigned the type "
:<>: ShowType t :<>: TL.Text ".")
Inject (l :-> t) (l' :-> t' ': x) =
Ifte (l <=.? l')
(l :-> t ': l' :-> t' ': x)
(l' :-> t' ': Inject (l :-> t) x)
type family ModifyR (l :: Symbol) (a :: k) (ρ :: [LT k]) :: [LT k] where
ModifyR l a (l :-> a' ': ρ) = l :-> a ': ρ
ModifyR l a (l' :-> a' ': ρ) = l' :-> a' ': ModifyR l a ρ
ModifyR l a '[] = TypeError (TL.Text "Tried to modify the label " :<>: ShowType l
:<>: TL.Text ", but it does not appear in the row-type.")
type family Ifte (c :: Bool) (t :: k) (f :: k) where
Ifte True t f = t
Ifte False t f = f
type family Get (l :: Symbol) (r :: [LT k]) where
Get l '[] = TypeError (TL.Text "No such field: " :<>: ShowType l)
Get l (l :-> t ': x) = t
Get l (l' :-> t ': x) = Get l x
type family Remove (l :: Symbol) (r :: [LT k]) where
Remove l r = RemoveT l r r
type family RemoveT (l :: Symbol) (r :: [LT k]) (r_orig :: [LT k]) where
RemoveT l (l :-> t ': x) _ = x
RemoveT l (l' :-> t ': x) r = l' :-> t ': RemoveT l x r
RemoveT l '[] r = TypeError (TL.Text "Cannot remove a label that does not occur in the row type."
:$$: TL.Text "The label " :<>: ShowType l :<>: TL.Text " is not in "
:<>: ShowRowType r)
type family LacksR (l :: Symbol) (r :: [LT k]) (r_orig :: [LT k]) :: Constraint where
LacksR l '[] _ = Unconstrained
LacksR l (l :-> t ': x) r = TypeError (TL.Text "The label " :<>: ShowType l
:<>: TL.Text " already exists in " :<>: ShowRowType r)
LacksR l (l' :-> _ ': x) r = Ifte (l <=.? l') Unconstrained (LacksR l x r)
type family Merge (l :: [LT k]) (r :: [LT k]) where
Merge '[] r = r
Merge l '[] = l
Merge (h :-> a ': tl) (h :-> a ': tr) =
TypeError (TL.Text "The label " :<>: ShowType h :<>: TL.Text " (of type "
:$$: ShowType a :<>: TL.Text ") has duplicate assignments.")
Merge (h :-> a ': tl) (h :-> b ': tr) =
TypeError (TL.Text "The label " :<>: ShowType h :<>: TL.Text " has conflicting assignments."
:$$: TL.Text "Its type is both " :<>: ShowType a :<>: TL.Text " and " :<>: ShowType b :<>: TL.Text ".")
Merge (hl :-> al ': tl) (hr :-> ar ': tr) =
Ifte (hl <=.? hr)
(hl :-> al ': Merge tl (hr :-> ar ': tr))
(hr :-> ar ': Merge (hl :-> al ': tl) tr)
type family MinJoinR (l :: [LT k]) (r :: [LT k]) where
MinJoinR '[] r = r
MinJoinR l '[] = l
MinJoinR (h :-> a ': tl) (h :-> a ': tr) =
(h :-> a ': MinJoinR tl tr)
MinJoinR (h :-> a ': tl) (h :-> b ': tr) =
TypeError (TL.Text "The label " :<>: ShowType h :<>: TL.Text " has conflicting assignments."
:$$: TL.Text "Its type is both " :<>: ShowType a :<>: TL.Text " and " :<>: ShowType b :<>: TL.Text ".")
MinJoinR (hl :-> al ': tl) (hr :-> ar ': tr) =
MinJoinRCase (CmpSymbol hl hr) hl al tl hr ar tr
type family MinJoinRCase (cmp :: Ordering) (hl :: Symbol) (al :: k) (tl :: [LT k])
(hr :: Symbol) (ar :: k) (tr :: [LT k]) where
MinJoinRCase 'LT hl al tl hr ar tr = hl :-> al ': MinJoinR tl (hr :-> ar ': tr)
MinJoinRCase _ hl al tl hr ar tr = hr :-> ar ': MinJoinR (hl :-> al ': tl) tr
type family ConstUnionR (l :: [LT k]) (r :: [LT k]) where
ConstUnionR '[] r = r
ConstUnionR l '[] = l
ConstUnionR (h :-> a ': tl) (h :-> b ': tr) =
(h :-> a ': ConstUnionR tl tr)
ConstUnionR (hl :-> al ': tl) (hr :-> ar ': tr) =
ConstUnionRCase (CmpSymbol hl hr) hl al tl hr ar tr
type family ConstUnionRCase (cmp :: Ordering) (hl :: Symbol) (al :: k) (tl :: [LT k])
(hr :: Symbol) (ar :: k) (tr :: [LT k]) where
ConstUnionRCase 'LT hl al tl hr ar tr = hl :-> al ': ConstUnionR tl (hr :-> ar ': tr)
ConstUnionRCase _ hl al tl hr ar tr = hr :-> ar ': ConstUnionR (hl :-> al ': tl) tr
type family Diff (l :: [LT k]) (r :: [LT k]) where
Diff '[] r = '[]
Diff l '[] = l
Diff (l :-> al ': tl) (l :-> al ': tr) = Diff tl tr
Diff (hl :-> al ': tl) (hr :-> ar ': tr) =
Ifte (hl <=.? hr)
(hl :-> al ': Diff tl (hr :-> ar ': tr))
(Diff (hl :-> al ': tl) tr)
type family ShowRowType (r :: [LT k]) :: ErrorMessage where
ShowRowType '[] = TL.Text "Empty"
ShowRowType '[l ':-> t] = TL.ShowType l TL.:<>: TL.Text " .== " TL.:<>: TL.ShowType t
ShowRowType ((l ':-> t) ': r) = TL.ShowType l TL.:<>: TL.Text " .== " TL.:<>: TL.ShowType t TL.:<>: TL.Text " .+ " TL.:<>: ShowRowType r
type a <=.? b = (CmpSymbol a b == 'LT)
infix 4 ≈
type a ≈ b = a ~ b