-- TODO QuickCheck: -- -- * `takeSegs`, etc. -- * `stepState` -- TODO changelog -- TODO check dependency versions module Main (main) where import Control.Concurrent (forkIO, killThread, threadDelay) import Control.Concurrent.STM.TVar (TVar, newTVarIO, readTVar, writeTVar) import Control.Monad (forever, forM_, void, when) import Control.Monad.STM (atomically) import Data.Maybe (fromMaybe) import Data.Text (Text) import Data.Version (showVersion) import Data.Time.Clock (diffUTCTime, getCurrentTime) import qualified Data.Text as Text import qualified Data.Text.Lazy as LText import GHC.Generics (Generic) import System.Exit (exitSuccess) import System.IO (BufferMode (..)) import qualified System.Process.ListLike as Process import qualified System.Process.Text as Text import System.FSNotify (eventTime, withManager) import qualified System.FSNotify as FSNotify import Options.Generic (ParseRecord (..), type () (..), getWithHelp, lispCaseModifiers, parseRecordWithModifiers, shortNameModifier) import Brick.BChan (BChan, newBChan, writeBChan) import qualified Paths_trackit as Trackit import Options import TUI type LText = LText.Text data CmdOptions = CmdOptions { _watchDir :: [FilePath] "Directory to watch for changes in (not sub-directories). Cannot be used together with '--watch-tree'." , _watchTree :: [FilePath] "Directory tree to watch for changes in (including sub-directories). Cannot be used together with '--watch-dir'." , _command :: Maybe String "Command to run" , _followTail :: Bool "Follow the tail of the generated output." , _showRunning :: Bool "Display a message while the command is running." , _incremental :: Bool "Allow output to be updated incrementally. Redraws the buffer for every output line, so should only be used for \ \slow outputs. Implies '--show-running'." , _stabilization :: Maybe Int "Minimal time (milliseconds) between any file event and the next command update (default: 200)" , _version :: Bool "Print the version number" , _help :: Bool , _debug :: Bool "Show debug information in the lower right corner" } deriving (Show, Generic) -- `--show-running` is not on by default because it causes a quick flickering -- for fast commands. shortName :: String -> Maybe Char shortName "_watchDir" = Just 'd' shortName "_watchTree" = Just 't' shortName "_showRunning" = Just 'r' shortName "_debug" = Just 'g' shortName (_:c:_) = Just c shortName _ = Nothing instance ParseRecord CmdOptions where parseRecord = parseRecordWithModifiers lispCaseModifiers {shortNameModifier = shortName} getOptions :: IO Options getOptions = do (CmdOptions {..}, showHelp) <- getWithHelp "trackit" if | _help -> showHelp >> exitSuccess | unHelpful _version -> do putStrLn $ showVersion Trackit.version exitSuccess | otherwise -> return $ let watchDirs = map (, Single) (unHelpful _watchDir) ++ map (, Recursive) (unHelpful _watchTree) command = unHelpful _command followTail = unHelpful _followTail showRunning = unHelpful _showRunning incremental = unHelpful _incremental stabPerMs = fromMaybe 200 $ unHelpful _stabilization stabilization = fromIntegral stabPerMs / 1000 debug = unHelpful _debug in Options {..} helpText :: [Text] helpText = [ "No command provided. Run 'trackit --help' for help.\n\n" , "Press 'q' to exit this window." ] concatChunks :: [Process.Chunk LText] -> LText concatChunks cs = LText.concat [c | Process.Stdout c <- cs] -- I initially thought that there would be one chunk per line given the -- `LineBuffering` setting, but this doesn't seem to be the case. Hence -- `Text.lines` is needed on the result. -- -- There is an instance of `readCreateProcessLazy` that can return -- `(ExitCode, LText, LText)` directly without any chunks. However, that -- instance doesn't seem to have the desired laziness. -- | Run the command provided by the user, or print a helpful text if no command -- was given runCMD :: Options -> IO [Text] runCMD Options {..} = case command of Nothing -> return helpText Just cmd -> do (_, o, _) <- Text.readCreateProcessWithExitCode (Process.shell cmd) "" return $ Text.lines o -- | Run the command provided by the user, or print a helpful text if no command -- was given -- -- The output is returned as a lazy list of lines. runLazyCMD :: Options -> IO [Text] runLazyCMD Options {..} = case command of Nothing -> return helpText Just cmd -> map LText.toStrict . LText.lines . concatChunks <$> Process.readCreateProcessLazy (Process.shell cmd, LineBuffering, LineBuffering) "" -- | Explicit request to run the command and update output data UpdateRequest = UpdateRequest deriving (Eq, Show) -- | A loop that continuously looks for events in the two variables and runs the -- given action in response. The variables are emptied whenever the action runs. -- -- Events in the second variable immediately trigger the action. For events in -- the second variable, the action only happens if the event occurred more than -- 'stabilization' seconds ago. If the event occurred less time ago, it will -- remain in the variable and processed in a later iteration (unless emptied by -- another event in the meantime). worker :: Options -> TVar (Maybe FSNotify.Event) -- ^ Variable holding the last file event that has not yet been processed -> TVar (Maybe UpdateRequest) -- ^ Variable holding a potential update request -> IO () -- ^ Action to perform when the file event has stabilized -> IO () worker Options {..} lastFSEv updReq action = forever $ do threadDelay loopPeriod t <- getCurrentTime act <- atomically $ do mupd <- readTVar updReq case mupd of Just UpdateRequest -> resetEvents >> return True Nothing -> do mfsEv <- readTVar lastFSEv case mfsEv of Nothing -> return False Just fsEv -> do let stable = diffUTCTime t (eventTime fsEv) >= stabilization when stable resetEvents return stable when act action where resetEvents = do writeTVar lastFSEv Nothing writeTVar updReq Nothing loopPeriod = max 10000 $ round (stabilization * 1e6 / 5) -- Cap at 10 ms to avoid making the loop too busy when the stabilization -- period is small. -- | Run the command and feed the output lines to the GUI updater :: Options -> BChan TrackitEvent -> IO () updater opts@Options {..} updEv | incremental = do writeBChan updEv Start ls <- runLazyCMD opts mapM_ (writeBChan updEv . AddLine) ls writeBChan updEv Done | otherwise = do when showRunning $ writeBChan updEv Running ls <- runCMD opts writeBChan updEv $ UpdateBuffer $ reverse ls main = do opts@Options {..} <- getOptions -- Channel holding the last file event lastFSEv <- newTVarIO Nothing -- Channel holding user update requests (set to request an initial update) updReq <- newTVarIO (Just UpdateRequest) -- Channel for GUI update events updEv <- newBChan 1 let mkUpdReq = atomically $ writeTVar updReq (Just UpdateRequest) setFsEvent ev = atomically $ writeTVar lastFSEv $ Just ev tid <- forkIO $ worker opts lastFSEv updReq (updater opts updEv) void $ withManager $ \m -> do forM_ watchDirs $ \(path, depth) -> void $ case depth of Single -> FSNotify.watchDir m path (const True) setFsEvent Recursive -> FSNotify.watchTree m path (const True) setFsEvent appMain opts mkUpdReq updEv killThread tid -- Note: The "debouncing" option of fsnotify makes it so that only the *first* -- in a tight series of events is reported. However, this is problematic since -- it means that the GUI may miss file events. This can happen if a Git command -- performs multiple file system operations (which is usually the case) and the -- command take more time than updating the GUI (e.g. due to the repository -- being large). It can of course also happen if a Git command is issued just -- after another one. -- -- In contrast, the approach taken here is to react to the *last* in a tight -- sequence of events. A tight sequence is defined as a sequence in which each -- consecutive pair of events has a time distance of less than `stabilization` -- seconds. And since `worker` runs continuously, there's never a risk that an -- event will be missed.