{-# LANGUAGE FlexibleInstances #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE RankNTypes #-} {-# LANGUAGE RecordWildCards #-} {-# LANGUAGE ScopedTypeVariables #-} {-# LANGUAGE TypeOperators #-} -- | -- Module: Salak -- Copyright: (c) 2019 Daniel YU -- License: BSD3 -- Maintainer: leptonyu@gmail.com -- Stability: experimental -- Portability: portable -- -- Configuration Loader for Production in Haskell. -- module Salak( -- * How to use this library -- $use -- * Salak Main Functions loadAndRunSalak , runSalak , runSalakWith , PropConfig(..) -- * Load Salak , LoadSalakT , loadSalak , loadSalakFile , loadCommandLine , ParseCommandLine , defaultParseCommandLine , loadEnv , loadMock -- ** Load Extensions , ExtLoad , loadByExt , HasLoad(..) , (:|:)(..) -- * Run Salak , RunSalakT , liftNT -- ** Get Static Properties , HasSourcePack(..) , fetch , require -- ** Dynamic Get Properties , ReloadResult(..) , exec , requireD -- ** Parse properties , SourcePack , Priority , Value(..) , Prop , FromProp(..) , FromEnumProp(..) , readPrimitive , PResult(..) , (.?=) , (.?:) ) where import Control.Monad (unless) import Control.Monad.IO.Class (MonadIO) import Control.Monad.Reader import Control.Monad.State import Data.Default import Data.Text (Text, cons) import Salak.Load.Dynamic import Salak.Load.Env import Salak.Prop import Salak.Types import Salak.Types.Value import System.Directory import System.FilePath (()) -- | Prop load configuration data PropConfig = PropConfig { configName :: Maybe String -- ^ Config name , configDirKey :: Text -- ^ Specify config dir , searchCurrent :: Bool -- ^ Search current directory, default true , searchHome :: Bool -- ^ Search home directory, default false. , commandLine :: ParseCommandLine -- ^ How to parse commandline , loadExt :: FilePath -> LoadSalakT IO () } instance Default PropConfig where def = PropConfig Nothing "salak.conf.dir" True False defaultParseCommandLine (\_ -> return ()) -- | Load and run salak `SourcePack` and fetch properties. loadAndRunSalak :: MonadIO m => LoadSalakT m () -- ^ Load properties monad. -> RunSalakT m a -- ^ Fetch properties monad. -> m a loadAndRunSalak spm a = do sp <- runLoadT Nothing spm let es = errs sp unless (null es) $ fail (head es) runT a sp -- | Load file by extension type ExtLoad = (String, FilePath -> LoadSalakT IO ()) class HasLoad a where loaders :: a -> [ExtLoad] data a :|: b = a :|: b infixr 3 :|: instance (HasLoad a, HasLoad b) => HasLoad (a :|: b) where loaders (a :|: b) = loaders a ++ loaders b loadByExt :: (HasLoad a, MonadIO m) => a -> FilePath -> LoadSalakT m () loadByExt xs f = mapM_ go (loaders xs) where go (ext, ly) = tryLoadFile (jump . ly) $ f ++ "." ++ ext -- | Default run salak. -- All these configuration sources has orders, from highest priority to lowest priority: -- -- > 1. loadCommandLine -- > 2. loadEnvironment -- > 3. loadConfFiles -- > 4. load file from folder `salak.conf.dir` if defined -- > 5. load file from current folder if enabled -- > 6. load file from home folder if enabled -- > 7. file extension matching, support yaml or toml or any other loader. -- loadSalak :: MonadIO m => PropConfig -> LoadSalakT m () loadSalak PropConfig{..} = do loadCommandLine commandLine loadEnv forM_ configName $ forM_ [ require configDirKey , ifS searchCurrent getCurrentDirectory , ifS searchHome getHomeDirectory ] . loadConf where ifS True gxd = Just <$> liftIO gxd ifS _ _ = return Nothing loadConf n mf = mf >>= mapM_ (jump . loadExt . ( n)) -- | Default run salak. -- All these configuration sources has orders, from highest priority to lowest priority: -- -- > 1. loadCommandLine -- > 2. loadEnvironment -- > 3. loadConfFiles -- > 4. load file from folder `salak.conf.dir` if defined -- > 5. load file from current folder if enabled -- > 6. load file from home folder if enabled -- > 7. file extension matching, support yaml or toml or any other loader. -- loadSalakFile :: (HasLoad file, MonadIO m) => String -> file -> LoadSalakT m () loadSalakFile name a = loadSalak def { configName = Just name, loadExt = loadByExt a} -- | Default run salak. -- All these configuration sources has orders, from highest priority to lowest priority: -- -- > 1. loadCommandLine -- > 2. loadEnvironment -- > 3. loadConfFiles -- > 4. load file from folder `salak.conf.dir` if defined -- > 5. load file from current folder if enabled -- > 6. load file from home folder if enabled -- > 7. file extension matching, support yaml or toml or any other loader. -- runSalak :: MonadIO m => PropConfig -> RunSalakT m a -> m a runSalak = loadAndRunSalak . loadSalak -- | Simplified run salak, should specified code name and file format. runSalakWith :: (HasLoad file, MonadIO m) => String -> file -> RunSalakT m a -> m a runSalakWith name a = loadAndRunSalak (loadSalakFile name a) -- | Monad that can fetch properties. class Monad m => HasSourcePack m where askSourcePack :: m SourcePack logSP :: Text -> m () logSP _ = return () readLogs :: m [Text] readLogs = return [] instance MonadIO m => HasSourcePack (RunSalakT m) where askSourcePack = askRSP logSP key = RunSalakT $ modify $ \rsp -> rsp { logs = key : logs rsp} readLogs = RunSalakT $ do rsp <- get let ls = logs rsp put rsp { logs = [] } return (reverse ls) instance Monad m => HasSourcePack (LoadSalakT m) where askSourcePack = LoadSalakT get -- | Try fetch properties from `SourcePack` fetch :: (HasSourcePack m, FromProp a) => Text -- ^ Properties key -> m (Either String a) fetch key = logSP key >> search key <$> askSourcePack -- | Fetch properties from `SourcePack`, or throw fail require :: (HasSourcePack m, FromProp a) => Text -- ^ Properties key -> m a require k = fetch k >>= either error return -- | Fetch dynamic properties from `SourcePack`, or throw fail requireD :: (MonadIO m, FromProp a) => Text -- ^ Properties key -> RunSalakT m (IO a) requireD k = logSP ('@' `cons` k) >> search' k >>= either error return -- $use -- -- | This library define a universal procedure to load configurations and parse properties, also supports reload configuration files. -- -- -- We can load configurations from command line, environment, configuration files such as yaml or toml etc, and we may want to have our own strategies to load configurations from multi sources and overwrite properties by orders of these sources. -- -- `PropConfig` defines a common loading strategy: -- -- > 1. loadCommandLine -- > 2. loadEnvironment -- > 3. loadConfFiles -- > 4. load file from folder `salak.conf.dir` if defined -- > 5. load file from current folder if enabled -- > 6. load file from home folder if enabled -- > 7. file extension matching, support yaml or toml or any other loader. -- -- Load earlier has higher priority, priorities cannot be changed. -- -- -- Usage: -- -- -- Environment: -- -- > export TEST_CONFIG_NAME=daniel -- -- Current Directory: salak.yaml -- -- > test.config: -- > name: noop -- > dir: ls -- -- Current Directory: salak.toml -- -- > [test.config] -- > ext=2 -- -- > data Config = Config -- > { name :: Text -- > , dir :: Maybe Text -- > , ext :: Int -- > } deriving (Eq, Show) -- > -- > instance FromProp Config where -- > fromProp = Config -- > <$> "user" ? pattern "[a-z]{5,16}" -- > <*> "pwd" -- > <*> "ext" .?= 1 -- > -- > main = runSalakWith "salak" (YAML :|: TOML) $ do -- > c :: Config <- require "test.config" -- > lift $ print c -- -- GHCi play -- -- > λ> import Salak -- > λ> import Salak.YAML -- > λ> import Salak.TOML -- > λ> :set -XTypeApplications -- > λ> instance FromProp Config where fromProp = Config <$> "user" <*> "dir" <*> "ext" .?= 1 -- > λ> f = runSalakWith "salak" (YAML :|: TOML) -- > λ> f (require "") >>= print @Config -- > Config {name = "daniel", dir = Just "ls", ext = 2} --