sing.hs with TH
This commit is contained in:
parent
91443f21a4
commit
0269c06f9f
2 changed files with 51 additions and 17 deletions
32
Dispatchable.hs
Normal file
32
Dispatchable.hs
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
{-# language GHC2024, AllowAmbiguousTypes, TemplateHaskell, TypeAbstractions #-}
|
||||||
|
|
||||||
|
module Dispatchable where
|
||||||
|
|
||||||
|
import GHC.TypeLits
|
||||||
|
import Language.Haskell.TH
|
||||||
|
|
||||||
|
|
||||||
|
data Dict c = c => Dict
|
||||||
|
|
||||||
|
-- since TH needs to run code during compilation it introduces order-dependence.
|
||||||
|
-- declaration splices split the file up into chunks that are checked
|
||||||
|
-- separately, and `reify` (for example) can only see chunks that have already
|
||||||
|
-- been done. so that `pure []` is to cause `Dispatchable` to be processed and
|
||||||
|
-- be visible
|
||||||
|
|
||||||
|
class Dispatchable d where dispatch :: String
|
||||||
|
instance Dispatchable "a" where dispatch = "A"
|
||||||
|
instance Dispatchable "b" where dispatch = "B"
|
||||||
|
|
||||||
|
pure []
|
||||||
|
|
||||||
|
data DispatchableInstance = forall s. DI (SSymbol s) (Dict (Dispatchable s))
|
||||||
|
|
||||||
|
instances :: Q Exp
|
||||||
|
instances = do
|
||||||
|
ClassI _ insts <- reify ''Dispatchable
|
||||||
|
listE $ map fromInstance insts
|
||||||
|
where
|
||||||
|
fromInstance :: Dec -> Q Exp
|
||||||
|
fromInstance (InstanceD _ _ (AppT _ str) _) =
|
||||||
|
[| DI (SSymbol @($(pure str))) Dict |]
|
36
sing.hs
36
sing.hs
|
@ -1,18 +1,27 @@
|
||||||
{-# language GHC2024, AllowAmbiguousTypes, TypeAbstractions #-}
|
{-# language GHC2024, AllowAmbiguousTypes, TemplateHaskell, TypeAbstractions #-}
|
||||||
|
|
||||||
|
import Dispatchable
|
||||||
|
-- due to TH's staging restriction (cannot splice definitions from the same
|
||||||
|
-- file), the dispatchable stuff has to be separated out
|
||||||
|
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import Data.Type.Equality
|
import Data.Type.Equality
|
||||||
import GHC.TypeLits
|
import GHC.TypeLits
|
||||||
import System.Environment
|
import System.Environment
|
||||||
|
|
||||||
main = do
|
|
||||||
[s] <- getArgs
|
|
||||||
putStrLn $ fromMaybe "some default value" $ dispatchDyn s
|
|
||||||
|
|
||||||
|
stringDispatchableWith :: KnownSymbol m =>
|
||||||
|
[DispatchableInstance] -> Maybe (Dict (Dispatchable m))
|
||||||
|
stringDispatchableWith @_ [] = Nothing
|
||||||
|
stringDispatchableWith @m (DI (SSymbol @s) Dict : insts)
|
||||||
|
| Just Refl <- is @s @m = Just Dict
|
||||||
|
| otherwise = stringDispatchableWith @m insts
|
||||||
|
where is :: (KnownSymbol a, KnownSymbol b) => Maybe (a :~: b)
|
||||||
|
is @a @b = sameSymbol (SSymbol @a) (SSymbol @b)
|
||||||
|
|
||||||
|
stringDispatchable :: KnownSymbol m => Maybe (Dict (Dispatchable m))
|
||||||
|
stringDispatchable = stringDispatchableWith $instances
|
||||||
|
|
||||||
class Dispatchable d where dispatch :: String
|
|
||||||
instance Dispatchable "a" where dispatch = "A"
|
|
||||||
instance Dispatchable "b" where dispatch = "B"
|
|
||||||
|
|
||||||
dispatchDyn :: String -> Maybe String
|
dispatchDyn :: String -> Maybe String
|
||||||
dispatchDyn m = withSomeSSymbol m $ \(SSymbol @m) ->
|
dispatchDyn m = withSomeSSymbol m $ \(SSymbol @m) ->
|
||||||
|
@ -21,13 +30,6 @@ dispatchDyn m = withSomeSSymbol m $ \(SSymbol @m) ->
|
||||||
Nothing -> Nothing
|
Nothing -> Nothing
|
||||||
|
|
||||||
|
|
||||||
stringDispatchable :: KnownSymbol m => Maybe (Dict (Dispatchable m))
|
main = do
|
||||||
stringDispatchable @m
|
[s] <- getArgs
|
||||||
| Just Refl <- is @"a" @m = Just Dict
|
putStrLn $ fromMaybe "some default value" $ dispatchDyn s
|
||||||
| Just Refl <- is @"b" @m = Just Dict
|
|
||||||
| otherwise = Nothing
|
|
||||||
where is :: (KnownSymbol a, KnownSymbol b) => Maybe (a :~: b)
|
|
||||||
is @a @b = sameSymbol (SSymbol @a) (SSymbol @b)
|
|
||||||
|
|
||||||
|
|
||||||
data Dict c = c => Dict
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue