{-# LANGUAGE ExistentialQuantification #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE RankNTypes #-}
module Imm.Boot (imm) where
import qualified Imm.Core as Core
import Imm.Database as Database
import Imm.Database.FeedTable as Database
import Imm.Dyre as Dyre
import Imm.Feed
import Imm.Hooks as Hooks
import Imm.HTTP as HTTP
import Imm.Logger as Logger
import Imm.Options as Options hiding (logLevel)
import Imm.Pretty
import Imm.XML as XML
import Control.Exception.Safe
import Data.Conduit.Combinators as Conduit (stdin)
import qualified Data.Map as Map
import Data.Text as Text hiding (length)
import Data.Text.IO as Text
import Relude.Unsafe (at)
import System.IO (hFlush)
imm :: Logger.Handle IO -> Database.Handle IO FeedTable -> HTTP.Handle IO -> Hooks.Handle IO -> XML.Handle IO -> IO ()
imm logger database httpClient hooks xmlParser = void $ do
options <- parseOptions
Dyre.wrap (optionDyreMode options) realMain (optionCommand options, optionLogLevel options, optionColorizeLogs options, logger, database, httpClient, hooks, xmlParser)
realMain :: (Command, LogLevel, Bool, Logger.Handle IO, Database.Handle IO FeedTable, HTTP.Handle IO, Hooks.Handle IO, XML.Handle IO) -> IO ()
realMain (command, logLevel, enableColors, logger, database, httpClient, hooks, xmlParser) = void $ do
setColorizeLogs logger enableColors
setLogLevel logger logLevel
log logger Debug . ("Dynamic reconfiguration settings:" <++>) . indent 2 =<< Dyre.describePaths
log logger Debug $ "Executing: " <> pretty command
log logger Debug . ("Using database:" <++>) . indent 2 =<< _describeDatabase database
handleAny (log logger Error . pretty . displayException) $ case command of
Check t -> Core.check logger database httpClient xmlParser =<< resolveTarget database ByPassConfirmation t
Help -> Text.putStrLn helpString
Import -> Core.importOPML logger database Conduit.stdin
Read t -> mapM_ (Database.markAsRead logger database) =<< resolveTarget database AskConfirmation t
Run t -> Core.run logger database httpClient hooks xmlParser =<< resolveTarget database ByPassConfirmation t
Show t -> Core.showFeed logger database =<< resolveTarget database ByPassConfirmation t
ShowVersion -> Core.printVersions
Subscribe u c -> Core.subscribe logger database u c
Unread t -> mapM_ (Database.markAsUnread logger database) =<< resolveTarget database AskConfirmation t
Unsubscribe t -> Database.deleteList logger database =<< resolveTarget database AskConfirmation t
_ -> return ()
Database.commit logger database
flushLogs logger
data SafeGuard = AskConfirmation | ByPassConfirmation
deriving(Eq, Read, Show)
data InterruptedException = InterruptedException deriving(Eq, Read, Show)
instance Exception InterruptedException where
displayException _ = "Process interrupted"
promptConfirm :: Text -> IO ()
promptConfirm s = do
Text.putStr $ s <> " Confirm [Y/n] "
hFlush stdout
x <- Text.getLine
unless (Text.null x || x == "Y") $ throwM InterruptedException
resolveTarget :: MonadIO m => MonadThrow m => Database.Handle m FeedTable -> SafeGuard -> Maybe Core.FeedRef -> m [FeedID]
resolveTarget database s Nothing = do
result <- Map.keys <$> Database.fetchAll database
when (s == AskConfirmation) $ liftIO $ promptConfirm $ "This will affect " <> show (length result) <> " feeds."
return result
resolveTarget database _ (Just (ByUID i)) = do
result <- fst . at (i-1) . Map.toList <$> Database.fetchAll database
return [result]
resolveTarget _ _ (Just (ByURI uri)) = return [FeedID uri]