{-# OPTIONS -fno-warn-orphans #-} {-# LANGUAGE TemplateHaskell , OverloadedStrings , FlexibleInstances #-} module Data.Label.Derive ( mkLabels , mkLabelsNoTypes ) where import Control.Arrow import Control.Category import Control.Monad import Data.Char import Data.Function (on) import Data.Label.Abstract import Data.List import Data.Ord import Data.String import Language.Haskell.TH import Language.Haskell.TH.Syntax import Prelude hiding ((.), id) -- Throw a fclabels specific error. fclError :: String -> a fclError err = error ("Data.Label.Derive: " ++ err) -- | Derive lenses including type signatures for all the record selectors in a -- datatype. mkLabels :: [Name] -> Q [Dec] mkLabels = liftM concat . mapM (derive1 True) -- | Derive lenses without type signatures for all the record selectors in a -- datatype. mkLabelsNoTypes :: [Name] -> Q [Dec] mkLabelsNoTypes = liftM concat . mapM (derive1 False) -- Helpers to generate all labels. derive1 :: Bool -> Name -> Q [Dec] derive1 signatures datatype = do i <- reify datatype let -- Only process data and newtype declarations, filter out all -- constructors and the type variables. (tyname, cons, vars) = case i of TyConI (DataD _ n vs cs _) -> (n, cs, vs) TyConI (NewtypeD _ n vs c _) -> (n, [c], vs) _ -> fclError "Can only derive labels for datatypes and newtypes." -- We are only interested in lenses of record constructors. recordOnly = groupByCtor [ (f, n) | RecC n fs <- cons, f <- fs ] concat `liftM` mapM (derive signatures tyname vars (length cons)) recordOnly where groupByCtor = map (\xs -> (fst (head xs), map snd xs)) . groupBy ((==) `on` (fst3 . fst)) . sortBy (comparing (fst3 . fst)) where fst3 (a, _, _) = a -- Generate the code for the labels. derive :: Bool -> Name -> [TyVarBndr] -> Int -> (VarStrictType, [Name]) -> Q [Dec] derive signatures tyname vars total ((field, _, fieldtyp), ctors) = do (sign, body) <- if length ctors == total then function derivePureLabel else function deriveMaybeLabel return $ if signatures then [sign, inline, body] else [inline, body] where -- Generate an inline declaration for the label. inline = PragmaD (InlineP labelName (InlineSpec True True (Just (True, 0)))) -- Build a single record label definition for labels that might fail. deriveMaybeLabel = (sign, body) where sign = forallT vars (return []) [t| (ArrowChoice (~>), ArrowZero (~>)) => Lens (~>) $(inputType) $(return fieldtyp) |] body = [| let c = zeroArrow ||| returnA in lens (c . $(getter)) (c . $(setter)) |] where getter = [| arr (\ p -> $(caseE [|p|] (cases (bodyG [|p|] ) ++ wild))) |] setter = [| arr (\(v, p) -> $(caseE [|p|] (cases (bodyS [|p|] [|v|]) ++ wild))) |] cases b = map (\ctor -> match (recP ctor []) (normalB b) []) ctors wild = [match wildP (normalB [| Left () |]) []] bodyS p v = [| Right $( record p fieldName v ) |] bodyG p = [| Right $( fromString fieldName `appE` p ) |] -- Build a single record label definition for labels that cannot fail. derivePureLabel = (sign, body) where sign = forallT vars (return []) [t| Arrow (~>) => Lens (~>) $(inputType) $(return fieldtyp) |] body = [| lens $(getter) $(setter) |] where getter = [| arr $(fromString fieldName) |] setter = [| arr (\(v, p) -> $(record [| p |] fieldName [| v |])) |] -- Generate a name for the label. If the original selector starts with an -- underscore, remove it and make the next character lowercase. Otherwise, -- add 'l', and make the next character uppercase. fieldName = nameBase field labelName = mkName $ case nameBase field of '_' : c : rest -> toLower c : rest f : rest -> 'l' : toUpper f : rest n -> fclError ("Cannot derive label for record selector with name: " ++ n) -- Compute the type (including type variables of the record datatype. inputType = return $ foldr (flip AppT) (ConT tyname) (map tvToVarT (reverse vars)) -- Convert a type variable binder to a regular type variable. tvToVarT (PlainTV tv) = VarT tv tvToVarT _ = fclError "No support for special-kinded type variables." -- Q style record updating. record rec fld val = val >>= \v -> recUpdE rec [return (mkName fld, v)] -- Build a function declaration with both a type signature and body. function (s, b) = liftM2 (,) (sigD labelName s) (funD labelName [ clause [] (normalB b) [] ]) -- IsString instances for TH types. instance IsString Exp where fromString = VarE . mkName instance IsString (Q Pat) where fromString = varP . mkName instance IsString (Q Exp) where fromString = varE . mkName