{-# LANGUAGE DataKinds #-} {-# LANGUAGE DeriveGeneric #-} {-# LANGUAGE KindSignatures #-} {-# LANGUAGE LambdaCase #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE RecordWildCards #-} {-# LANGUAGE TemplateHaskell #-} {-# LANGUAGE TupleSections #-} {-# LANGUAGE TypeFamilies #-} module Config ( loadConfig , Config, port, verbose, template ) where import Universum import Data.List (isSuffixOf) import qualified Data.Map.Strict as M import Data.Tiled (LoadResult (Loaded), Tiledmap, loadTiledmap) import Lens.Micro.Platform (makeLenses, traverseOf) import System.Directory (listDirectory) import System.FilePath ((</>)) import Toml (TomlCodec, (.=)) import qualified Toml as T type family ConfigRes (b :: Bool) a where ConfigRes True a = a ConfigRes False a = FilePath -- | the server's configuration data Config (loaded :: Bool) = Config { _port :: Int , _verbose :: Bool , _template :: ConfigRes loaded (FilePath, Map Text Tiledmap) } deriving Generic makeLenses ''Config configCodec :: TomlCodec (Config False) configCodec = Config <$> T.int "port" .= _port <*> T.bool "verbose" .= _verbose <*> T.string "template" .= _template loadConfig :: FilePath -> IO (Config True) loadConfig path = do T.decodeFileEither configCodec path >>= \case Right c -> traverseOf template loadMaps c Left err -> error (show err) where loadMaps path = do maps <- listDirectory path <&> filter (".json" `isSuffixOf`) list <- forM maps $ \mapname -> loadTiledmap (path </> mapname) >>= \case Loaded tmap -> pure (toText mapname, tmap) err -> error (show err) pure (path, M.fromList list)