mirror of
https://github.com/urbit/shrub.git
synced 2025-01-02 01:25:55 +03:00
270 lines
9.1 KiB
Haskell
270 lines
9.1 KiB
Haskell
{-|
|
|
Template Haskell Code to Generate FromNoun and ToNoun Instances
|
|
-}
|
|
module Urbit.Noun.TH (deriveNoun, deriveToNoun, deriveFromNoun) where
|
|
|
|
import ClassyPrelude hiding (fromList)
|
|
import Language.Haskell.TH
|
|
import Language.Haskell.TH.Syntax
|
|
import Urbit.Noun.Convert
|
|
|
|
import Urbit.Noun.Core (textToUtf8Atom)
|
|
|
|
import qualified Data.Char as C
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
type ConInfo = (Name, [Type])
|
|
|
|
data Shape
|
|
= Vod
|
|
| Tup ConInfo
|
|
| Sum [(String, Name)] [(String, ConInfo)]
|
|
deriving (Eq, Ord, Show)
|
|
|
|
typeShape :: Name -> Q ([TyVarBndr], Shape)
|
|
typeShape tyName = do
|
|
(vars, cs) <-
|
|
reify tyName >>= \case
|
|
TyConI (DataD _ nm vars _ cs _) -> pure (vars, unpackCon <$> cs)
|
|
TyConI (NewtypeD _ nm vars _ c _) -> pure (vars, [unpackCon c])
|
|
TyConI _ -> fail badSynonym
|
|
_ -> fail "not type"
|
|
|
|
let prefix = getPrefix (nameStr . fst <$> cs)
|
|
splits = splitFn ([], []) cs
|
|
splitFn (l, r) = \case
|
|
[] -> (l, r)
|
|
(n,[]) : cs -> splitFn (tagName prefix n:l, r) cs
|
|
conInf : cs -> splitFn (l, tagConInfo prefix conInf:r) cs
|
|
|
|
pure $ (vars,) $ case cs of
|
|
[] -> Vod
|
|
[c] -> Tup c
|
|
cs -> uncurry Sum splits
|
|
|
|
where
|
|
badSynonym = "deriveFunctor: tyCon may not be a type synonym."
|
|
|
|
tagConInfo :: Int -> ConInfo -> (String, ConInfo)
|
|
tagConInfo pre ci@(nm, _) = (tagString pre nm, ci)
|
|
|
|
tagName :: Int -> Name -> (String, Name)
|
|
tagName pre n = (tagString pre n, n)
|
|
|
|
tyStr = nameStr tyName
|
|
tyAbbrv = filter C.isUpper tyStr
|
|
|
|
typePrefixed = (tyStr `isPrefixOf`)
|
|
abbrvPrefixed = (tyAbbrv `isPrefixOf`)
|
|
|
|
getPrefix :: [String] -> Int
|
|
getPrefix cs | all typePrefixed cs = length tyStr
|
|
getPrefix cs | all abbrvPrefixed cs = length tyAbbrv
|
|
getPrefix _ = 0
|
|
|
|
unpackCon :: Con -> ConInfo
|
|
unpackCon = \case
|
|
NormalC nm bangTypes -> (nm, snd <$> bangTypes)
|
|
RecC nm varBangTypes -> (nm, varBangTypes <&> (\(_, _, t) -> t))
|
|
InfixC bangType1 nm bangType2 -> error "Infix Cnstrs are not supported"
|
|
ForallC tyVarBndrs ctx con -> error "Polymorphic tys are not supported"
|
|
GadtC nm bangTypes ty -> error "GADTs are not supported"
|
|
RecGadtC nm varBangTypes ty -> error "GADTs are not supported"
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
deriveNoun :: Name -> Q [Dec]
|
|
deriveNoun n = (<>) <$> deriveToNoun n <*> deriveFromNoun n
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
deriveToNoun :: Name -> Q [Dec]
|
|
deriveToNoun tyName = do
|
|
(params, shape) <- typeShape tyName
|
|
|
|
let exp = case shape of Vod -> vodToNoun
|
|
Tup con -> tupToNoun con
|
|
-- Enu cons -> enumToAtom cons
|
|
Sum atoms cells -> sumToNoun atoms cells
|
|
|
|
params <- pure $ zip ['a' ..] params <&> \(n,_) -> mkName (singleton n)
|
|
|
|
let ty = foldl' (\acc v -> AppT acc (VarT v)) (ConT tyName) params
|
|
|
|
let overlap = Nothing
|
|
body = NormalB exp
|
|
ctx = params <&> \t -> AppT (ConT ''ToNoun) (VarT t)
|
|
inst = AppT (ConT ''ToNoun) ty
|
|
|
|
pure [InstanceD overlap ctx inst [ValD (VarP 'toNoun) body []]]
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
addErrTag :: String -> Exp -> Exp
|
|
addErrTag tag exp =
|
|
InfixE (Just $ AppE (VarE 'named) str) (VarE (mkName ".")) (Just exp)
|
|
where
|
|
str = LitE $ StringL tag
|
|
|
|
deriveFromNoun :: Name -> Q [Dec]
|
|
deriveFromNoun tyName = do
|
|
(params, shape) <- typeShape tyName
|
|
|
|
let exp = case shape of Vod -> vodFromNoun
|
|
Tup con -> tupFromNoun con
|
|
-- Enu cons -> enumFromAtom cons
|
|
Sum atoms cells -> sumFromNoun atoms cells
|
|
|
|
params <- pure $ zip ['a' ..] params <&> \(n,_) -> mkName (singleton n)
|
|
|
|
let ty = foldl' (\acc v -> AppT acc (VarT v)) (ConT tyName) params
|
|
|
|
let overlap = Nothing
|
|
body = NormalB (addErrTag (nameStr tyName) exp)
|
|
ctx = params <&> \t -> AppT (ConT ''FromNoun) (VarT t)
|
|
inst = AppT (ConT ''FromNoun) ty
|
|
|
|
pure [InstanceD overlap ctx inst [ValD (VarP 'parseNoun) body []]]
|
|
|
|
sumFromNoun :: [(String, Name)] -> [(String, ConInfo)] -> Exp
|
|
sumFromNoun [] cl = taggedFromNoun cl
|
|
sumFromNoun at [] = enumFromAtom at
|
|
sumFromNoun at cl = eitherParser (taggedFromNoun cl) (enumFromAtom at)
|
|
where
|
|
eitherParser :: Exp -> Exp -> Exp
|
|
eitherParser x y =
|
|
LamE [VarP n] $
|
|
InfixE (Just xCase) (VarE (mkName "<|>")) (Just yCase)
|
|
where
|
|
xCase = AppE x (VarE n)
|
|
yCase = AppE y (VarE n)
|
|
n = mkName "atomOrCell"
|
|
|
|
enumFromAtom :: [(String, Name)] -> Exp
|
|
enumFromAtom cons = LamE [VarP x] body
|
|
where
|
|
(x, c) = (mkName "x", mkName "c")
|
|
getTag = BindS (VarP c) $ AppE (VarE 'parseNounUtf8Atom) (VarE x)
|
|
examine = NoBindS $ CaseE (VarE c) (matches ++ [fallback])
|
|
matches = mkMatch <$> cons
|
|
fallback = Match WildP (NormalB $ AppE (VarE 'fail) matchFail) []
|
|
body = DoE [getTag, examine]
|
|
matchFail = LitE $ StringL ("Expected one of: " <> possible)
|
|
possible = intercalate " " (('%':) . fst <$> cons)
|
|
mkMatch = \(tag, nm) ->
|
|
Match (SigP (LitP $ StringL tag) (ConT ''Text))
|
|
(NormalB $ AppE (VarE 'pure) (ConE nm))
|
|
[]
|
|
|
|
applyE :: Exp -> [Exp] -> Exp
|
|
applyE e [] = e
|
|
applyE e (a:as) = applyE (AppE e a) as
|
|
|
|
vodFromNoun :: Exp
|
|
vodFromNoun = LamE [WildP] body
|
|
where
|
|
body = AppE (VarE 'fail)
|
|
$ LitE $ StringL "Can't FromNoun on uninhabited data type"
|
|
|
|
tupFromNoun :: ConInfo -> Exp
|
|
tupFromNoun (n, tys) = LamE [VarP x] body
|
|
where
|
|
x = mkName "x"
|
|
vars = mkName . singleton . fst <$> zip ['a'..] tys
|
|
body = DoE [getTup, convert]
|
|
convert = NoBindS $ AppE (VarE 'pure) $ applyE (ConE n) (VarE <$> vars)
|
|
getTup = BindS (TupP $ VarP <$> vars) $ AppE (VarE 'parseNoun) (VarE x)
|
|
|
|
unexpectedTag :: [String] -> Exp -> Exp
|
|
unexpectedTag expected got =
|
|
applyE (VarE 'mappend) [LitE (StringL prefix), AppE (VarE 'unpack) got]
|
|
where
|
|
possible = intercalate " " (('%':) <$> expected)
|
|
prefix = "Expected one of: " <> possible <> " but got %"
|
|
|
|
taggedFromNoun :: [(String, ConInfo)] -> Exp
|
|
taggedFromNoun cons = LamE [VarP n] (DoE [getHead, getTag, examine])
|
|
where
|
|
(n, h, t, c) = (mkName "noun", mkName "hed", mkName "tel", mkName "tag")
|
|
|
|
getHead = BindS (TupP [VarP h, VarP t])
|
|
$ AppE (VarE 'parseNoun) (VarE n)
|
|
|
|
getTag = BindS (SigP (VarP c) (ConT ''Text))
|
|
$ AppE (VarE 'parseNounUtf8Atom) (VarE h)
|
|
|
|
examine = NoBindS
|
|
$ CaseE (VarE c) (matches ++ [fallback])
|
|
|
|
matches = mkMatch <$> cons
|
|
mkMatch = \(tag, (n, tys)) ->
|
|
let body = AppE (addErrTag ('%':tag) (tupFromNoun (n, tys)))
|
|
(VarE t)
|
|
in Match (LitP $ StringL tag) (NormalB body) []
|
|
|
|
fallback = Match WildP (NormalB $ AppE (VarE 'fail) matchFail) []
|
|
matchFail = unexpectedTag (fst <$> cons) (VarE c)
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
tagString :: Int -> Name -> String
|
|
tagString prefix = hsToHoon . drop prefix . nameStr
|
|
|
|
nameStr :: Name -> String
|
|
nameStr (Name (OccName n) _) = n
|
|
|
|
tagNoun :: String -> Exp
|
|
tagNoun = AppE (VarE 'textToUtf8Atom)
|
|
. LitE
|
|
. StringL
|
|
|
|
tagTup :: String -> [Name] -> Exp
|
|
tagTup tag args = AppE (VarE 'toNoun) $ TupE (tagNoun tag : fmap VarE args)
|
|
|
|
tup :: [Name] -> Exp
|
|
tup = AppE (VarE 'toNoun) . TupE . fmap VarE
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
vodToNoun :: Exp
|
|
vodToNoun = LamCaseE []
|
|
|
|
tupToNoun :: ConInfo -> Exp
|
|
tupToNoun cons = LamCaseE [mkMatch cons]
|
|
where
|
|
mkMatch :: ConInfo -> Match
|
|
mkMatch (nm, tys) = Match (ConP nm params) (NormalB body) []
|
|
where vars = (zip tys ['a'..]) <&> (mkName . singleton . snd)
|
|
params = VarP <$> vars
|
|
body = tup vars
|
|
|
|
sumToNoun :: [(String, Name)] -> [(String, ConInfo)] -> Exp
|
|
sumToNoun a c =
|
|
LamCaseE (mixed <&> uncurry mkMatch)
|
|
where
|
|
mixed = mconcat [ a <&> \(x,y) -> (x, Left y)
|
|
, c <&> \(x,y) -> (x, Right y)
|
|
]
|
|
|
|
mkMatch :: String -> Either Name ConInfo -> Match
|
|
mkMatch tag = \case
|
|
Left nm -> Match (ConP nm []) (NormalB $ tagNoun tag) []
|
|
Right (nm, tys) -> Match (ConP nm params) (NormalB body) []
|
|
where vars = (zip tys ['a'..]) <&> (mkName . singleton . snd)
|
|
params = VarP <$> vars
|
|
body = tagTup tag vars
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
hsToHoon :: String -> String
|
|
hsToHoon = go []
|
|
where
|
|
go acc [] = intercalate "-" $ reverse acc
|
|
go acc (c:cs) = go (elem:acc) remain
|
|
where
|
|
head = C.toLower c
|
|
(tail, remain) = break C.isUpper cs
|
|
elem = head:tail
|