module System.Commando (Options(..), options, commando) where
import Prelude hiding (FilePath)
import Control.Monad (void)
import System.Process (rawSystem, runCommand, runInteractiveCommand)
import System.FSNotify (startManager, watchTree, stopManager, Event(..))
import Filesystem.Path.CurrentOS (FilePath, fromText, toText)
import Data.Text (pack, unpack)
import System.IO (hPutStrLn, hGetContents, hSetBuffering, BufferMode(..))
import GHC.IO.Handle (hClose, hFlush)
import GHC.IO.Handle.Types (Handle)
import System.Process.Internals (ProcessHandle)
import Control.Applicative ((<$>), (<*>))
import Data.Monoid ((<>))
import Control.Concurrent (forkIO)
import Data.Maybe (fromMaybe, isJust, catMaybes)
import Control.Concurrent.Chan (newChan, writeChan, getChanContents, Chan)
import qualified Options.Applicative.Builder.Internal as X
import qualified Options.Applicative as O
type RunningProcess = (Handle, Handle, Handle, ProcessHandle)
data Options = Options {
command :: String
, quiet :: Bool
, consumer :: Bool
, stdin :: Bool
, persist :: Bool
, display :: Event -> String
, directory :: FilePath
}
commando :: Options -> IO [String]
commando o = do
c <- newChan
start c o
catMaybes . takeWhile isJust <$> getChanContents c
options :: O.Parser Options
options = Options
<$> defStr "echo event" ( O.metavar "COMMAND" <> O.help "Command run on events")
<*> O.switch ( O.short 'q' <> O.long "quiet" <> O.help "Hide non-essential output")
<*> O.switch ( O.short 'c' <> O.long "consumer" <> O.help "Pass events as argument to command")
<*> O.switch ( O.short 'i' <> O.long "stdin" <> O.help "Pipe events to command")
<*> O.switch ( O.short 'p' <> O.long "persist" <> O.help "Pipe events to persistent command")
<*> ((show <?> toFP)<$> ( O.switch ( O.short 'j' <> O.long "path-only" <> O.help "Only show the File-Path, not metadata")))
<*> (dir <$> defStr "." ( O.metavar "DIRECTORY" <> O.help "Directory to monitor" ))
defStr :: String -> X.Mod X.ArgumentFields String -> O.Parser String
defStr a = def a . O.argument O.str
def :: a -> O.Parser a -> O.Parser a
def a = fmap (fromMaybe a) . O.optional
dir :: String -> FilePath
dir = fromText . pack
start :: CH -> Options -> IO ()
start c o = do
man <- startManager
rc <- if persist o then Just <$> startPipe (command o)
else return Nothing
void $ forkIO $ whenM rc $ \(_,so,_,_) -> hGetContents so >>= putChan c
let cmd = command o
dsp = display o
void $ watchTree man (directory o) (const True)
$ case (consumer o, stdin o || persist o )
of (True , _ ) -> void . rawSystem cmd . return . dsp
(_ , True ) -> void . pipe c rc cmd . dsp
(_ , _ ) -> const $ void $ runCommand cmd
void $ getLine
void $ stopManager man
whenM rc pipeClose
closeChan c
whenM :: Monad m => Maybe a -> (a -> m ()) -> m ()
whenM m f = maybe (return ()) f m
startPipe :: String -> IO RunningProcess
startPipe cmd = do
rc@(hStdIn,_,_,_) <- runInteractiveCommand cmd
hSetBuffering hStdIn NoBuffering
return rc
pipe :: CH -> Maybe RunningProcess -> String -> String -> IO ()
pipe c Nothing cmd arg = startPipe cmd >>= pipeSend arg >>= pipeOutput c >>= pipeClose
pipe _ (Just rp) _ arg = void $ pipeSend arg rp
pipeOutput :: CH -> RunningProcess -> IO RunningProcess
pipeOutput c r@(_,so,_,_) = hGetContents so >>= putChan c >> return r
pipeClose :: RunningProcess -> IO ()
pipeClose (hStdIn, _, _, _) = hClose hStdIn
pipeSend :: String -> RunningProcess -> IO RunningProcess
pipeSend param rc@(hStdIn, _hStdOut, _stderr, _process) = do
hPutStrLn hStdIn param
hFlush hStdIn
return rc
(<?>) :: a -> a -> Bool -> a
x <?> y = \b -> if b then y else x
toFP :: Event -> String
toFP (Added fp _) = unpack (either id id (toText fp))
toFP (Modified fp _) = unpack (either id id (toText fp))
toFP (Removed fp _) = unpack (either id id (toText fp))
type CH = Chan (Maybe String)
putChan :: Chan (Maybe a) -> a -> IO ()
putChan c = writeChan c . Just
closeChan :: Chan (Maybe a) -> IO ()
closeChan c = writeChan c Nothing