{-# LANGUAGE TemplateHaskell #-}
module AOC.Discover (
mkChallengeMap
, solutionList
, ChallengeMap
, ChallengeSpec(..)
, solSpec
, solSpecStr
, solSpecStr_
, charPart
, challengeName
, solverNFData
, deepInstance
) where
import AOC.Solver
import Advent
import Control.Applicative
import Control.DeepSeq
import Language.Haskell.TH.Datatype
import Control.Monad
import Control.Monad.Trans.Class
import Control.Monad.Trans.Maybe
import Data.Bifunctor
import Data.Data
import Data.Map (Map)
import Data.Maybe
import Data.Traversable
import Data.Void
import GHC.Exts
import Language.Haskell.Exts as E
import Language.Haskell.Names
import Language.Haskell.TH as TH
import Language.Haskell.TH.Syntax (TExp(..))
import Prelude
import System.Directory
import System.FilePath
import Text.Printf
import qualified Data.List.NonEmpty as NE
import qualified Data.Map as M
import qualified Hpack.Config as H
import qualified Text.Megaparsec as P
import qualified Text.Megaparsec.Char as P
import qualified Text.Megaparsec.Char.Lexer as PL
checkIfNFData :: Bool
checkIfNFData = True
data ChallengeSpec = CS { _csDay :: Day
, _csPart :: Part
}
deriving (Show, Eq, Ord)
type ChallengeMap = Map Day (Map Part SomeSolution)
solSpec :: TH.Name -> ChallengeSpec
solSpec n = solSpecStr_ (nameBase n)
solSpecStr :: String -> Either (P.ParseErrorBundle String Void) ChallengeSpec
solSpecStr = P.runParser challengeName ""
solSpecStr_ :: String -> ChallengeSpec
solSpecStr_ = either (error . P.errorBundlePretty) id . solSpecStr
instance IsString ChallengeSpec where
fromString = solSpecStr_
type Parser = P.Parsec Void String
solutionList :: FilePath -> Q (TExp [(Day, (Part, SomeSolution))])
solutionList dir = fmap (TExp . ListE)
. traverse (fmap unType . specExp)
=<< runIO (getChallengeSpecs dir)
mkChallengeMap :: [(Day, (Part, SomeSolution))] -> ChallengeMap
mkChallengeMap = M.unionsWith M.union
. map (uncurry M.singleton . second (uncurry M.singleton))
specExp :: ChallengeSpec -> Q (TExp (Day, (Part, SomeSolution)))
specExp s@(CS d p) = do
n <- lookupValueName (specName s)
con <- case n of
Nothing -> pure 'MkSomeSolWH
Just n' -> do
isNF <- solverNFData n'
pure $ if isNF
then 'MkSomeSolNF
else 'MkSomeSolWH
pure $ TExp $ TupE
[ VarE 'mkDay_ `AppE` LitE (IntegerL (dayInt d))
, TupE
[ ConE (partCon p)
, ConE con `AppE` VarE (mkName (specName s))
]
]
where
partCon Part1 = 'Part1
partCon Part2 = 'Part2
specName :: ChallengeSpec -> String
specName (CS d p) = printf "day%02d%c" (dayInt d) (partChar p)
getChallengeSpecs
:: FilePath
-> IO [ChallengeSpec]
getChallengeSpecs dir = do
exts <- defaultExtensions
files <- listDirectory dir
parsed <- forM files $ \f -> do
let mode = defaultParseMode { extensions = exts
, fixities = Just []
, parseFilename = f
}
res <- parseFileWithMode mode (dir </> f)
case res of
ParseOk x -> pure x
ParseFailed l e -> fail $ printf "Failed parsing %s at %s: %s" f (show l) e
pure $ moduleSolutions parsed
defaultExtensions :: IO [E.Extension]
defaultExtensions = do
Right H.DecodeResult{..} <- H.readPackageConfig H.defaultDecodeOptions
Just H.Section{..} <- pure $ H.packageLibrary decodeResultPackage
pure $ parseExtension <$> sectionDefaultExtensions
moduleSolutions :: (Data l, Eq l) => [Module l] -> [ChallengeSpec]
moduleSolutions = (foldMap . foldMap) (maybeToList . isSolution)
. flip resolve M.empty
isSolution :: Symbol -> Maybe ChallengeSpec
isSolution s = do
Value _ (Ident _ n) <- pure s
Right c <- pure $ P.runParser challengeName "" n
pure c
challengeName :: Parser ChallengeSpec
challengeName = do
_ <- P.string "day"
dInt <- PL.decimal
dFin <- maybe (fail $ "Day not in range: " ++ show dInt) pure $
mkDay dInt
c <- P.lowerChar
p <- maybe (fail $ printf "Part not parsed: %c" c) pure $
charPart c
pure $ CS dFin p
charPart :: Char -> Maybe Part
charPart 'a' = Just Part1
charPart 'b' = Just Part2
charPart _ = Nothing
solverNFData :: TH.Name -> Q Bool
solverNFData n
| checkIfNFData = reify n >>= \case
VarI _ (ConT c `AppT` a `AppT` _) _
| c == ''(:~>) -> deepInstance ''NFData a
_ -> pure False
| otherwise = pure False
deepInstance
:: TH.Name
-> TH.Type
-> Q Bool
deepInstance cn = fmap isJust . runMaybeT . deepInstance_ cn
deepInstance_
:: TH.Name
-> TH.Type
-> MaybeT Q ()
deepInstance_ cn t = do
insts <- maybe empty pure . NE.nonEmpty =<< lift (reifyInstances cn [t])
forM_ insts $ \case
InstanceD _ ctx instHead _ -> do
uni <- lift $ unifyTypes [ConT cn `AppT` t, instHead]
forM_ ctx $ \case
AppT (ConT c) v -> deepInstance_ c (applySubstitution uni v)
_ -> empty
_ -> empty