{-# LANGUAGE ApplicativeDo #-} module QBar.Cli where import QBar.Blocks import QBar.Blocks.Pipe import QBar.ControlSocket import QBar.Core import QBar.DefaultConfig import QBar.Server import QBar.Qubes.AdminAPI (printEvents, qubesVMStats, qubesEvents) import QBar.Theme import QBar.Time import Control.Monad (join) import Data.Maybe (fromMaybe) import qualified Data.Text.Lazy as T import Options.Applicative -- |Entry point. runQBar :: IO () runQBar = join parseMain parseMain :: IO (IO ()) parseMain = customExecParser parserPrefs parser where parser :: ParserInfo (IO ()) parser = info (mainParser <**> helper) (fullDesc <> header "qbar - queezles {i3,sway}bar infrastructure") parserPrefs :: ParserPrefs parserPrefs = prefs showHelpOnEmpty mainParser :: Parser (IO ()) mainParser = do verbose <- switch $ long "verbose" <> short 'v' <> help "Print more diagnostic output to stderr (including a copy of every bar update)." indicator <- switch $ long "indicator" <> short 'i' <> help "Show render indicator." socketLocation <- optional $ strOption $ long "socket" <> short 's' <> metavar "SOCKET" <> help "Control socket location. By default determined by WAYLAND_SOCKET location." barCommand <- barCommandParser return (barCommand MainOptions {verbose, indicator, socketLocation}) barCommandParser :: Parser (MainOptions -> IO ()) barCommandParser = hsubparser ( command "server" (info serverCommandParser (progDesc "Start a new server.")) <> command "mirror" (info mirrorCommandParser (progDesc "Mirror the output of a running server.")) <> command "pipe" (info pipeClientParser (progDesc "Redirects the stdin of this process to a running bar.")) <> command "theme" (info themeCommandParser (progDesc "Change the theme of the running qbar server.")) <> command "qubes" (info qubesCommandParser (progDesc "Display information about Qubes.")) ) serverCommandParser :: Parser (MainOptions -> IO ()) serverCommandParser = hsubparser ( command "swaybar" (info (runBarServer <$> barConfigurationParser) (progDesc "Start a new server. Should be called by swaybar.")) <> command "i3bar" (info (runBarServer <$> barConfigurationParser) (progDesc "Start a new server. Should be called by i3bar.")) <> command "send" (info (sendBlockStream <$> barConfigurationParser) (progDesc "Run blocks on this process but send them to another qbar server.")) <> command "send-stdio" (info (sendBlockStreamStdio <$> barConfigurationParser) (progDesc "Run blocks on this process but send them to another qbar server using stdin and stdout.")) ) where barConfigurationParser :: Parser (BarIO ()) barConfigurationParser = sequence_ <$> some blockParser mirrorCommandParser :: Parser (MainOptions -> IO ()) mirrorCommandParser = hsubparser ( command "swaybar" (info (runBarServerMirror <$> barConfigurationParser) (progDesc "Mirror the output of another server. Should be called by swaybar.")) <> command "i3bar" (info (runBarServerMirror <$> barConfigurationParser) (progDesc "Mirror the output of another server. Should be called by i3bar.")) ) where barConfigurationParser :: Parser (BarIO ()) barConfigurationParser = sequence_ <$> many blockParser themeCommandParser :: Parser (MainOptions -> IO ()) themeCommandParser = sendIpc . SetTheme <$> strArgument (metavar "THEME" <> completeWith (map T.unpack themeNames)) pipeClientParser :: Parser (MainOptions -> IO ()) pipeClientParser = do events <- switch $ long "events" <> short 'e' <> help "Also encode events to stdout. Every event will be a JSON-encoded line." pure $ runPipeClient events blockParser :: Parser (BarIO ()) blockParser = subparser ( commandGroup "Available presets:" <> metavar "CONFIG..." <> command "default" (info (pure defaultBarConfig) (progDesc "Load default set of blocks.")) ) <|> subparser ( commandGroup "Available blocks:" <> hidden <> command "date" (info (pure $ addBlock dateBlock) (progDesc "Load the date and time block.")) <> command "cpu" (info (pure $ addBlock $ cpuUsageBlock 1) (progDesc "Load the cpu usage block.")) <> command "battery" (info (pure $ addBlock $ batteryBlock) (progDesc "Load the battery block.")) <> command "disk" (info diskUsageBlockParser (progDesc "Load the disk usage block.")) <> command "networkmanager" (info (pure $ addBlock networkManagerBlock) (progDesc "Load the network-manager block.")) <> command "script" (info scriptBlockParser (progDesc "Display the output of an external script as a block.")) <> command "diskQubesPool" (info (pure $ addBlock diskUsageQubesBlock) (progDesc "Load a block that shows free space in Qubes' default pool.")) <> command "qubesProperty" (info qubesPropertyBlockParser (progDesc "Display the current value of a Qubes property.")) <> command "qubesCount" (info (pure $ addBlock qubesVMCountBlock) (progDesc "Display the number of running Qubes (VMs).")) ) diskUsageBlockParser :: Parser (BarIO ()) diskUsageBlockParser = do file <- strArgument (metavar "FILE" <> help "The FILE by which the file system is selected.") return $ addBlock $ diskUsageBlock file scriptBlockParser :: Parser (BarIO ()) scriptBlockParser = helper <*> do poll <- switch $ long "poll" <> short 'p' <> help "Run script in poll mode (at regular intervals)" -- HACK optparse-applicative does not support options of style --poll[=INTERVAL], -- so we add a second option to specify the interval explicitly instead -- https://github.com/pcapriotti/optparse-applicative/issues/243 pollInterval <- fromMaybe defaultInterval <$> (optional $ IntervalSeconds <$> option auto ( long "interval" <> short 'i' <> metavar "SECONDS" <> (help $ "Interval to use for --poll mode (default: " <> humanReadableInterval defaultInterval <> ")") )) clickEvents <- switch $ long "events" <> short 'e' <> help "Send click events to stdin of the script" script <- strArgument (metavar "SCRIPT" <> help "The script that will be executed with a shell.") return $ (if poll then addBlock . pollScriptBlock pollInterval else addBlock . scriptBlock clickEvents) script qubesPropertyBlockParser :: Parser (BarIO ()) qubesPropertyBlockParser = do name <- strArgument (metavar "NAME" <> help "The NAME of the property.") return $ addBlock $ qubesMonitorPropertyBlock name qubesCommandParser :: Parser (MainOptions -> IO ()) qubesCommandParser = hsubparser ( command "stats" (info (pure $ const $ printEvents qubesVMStats) (progDesc "Subscribe to VM stats and print them to stdout.")) <> command "events" (info (pure $ const $ printEvents qubesEvents) (progDesc "Subscribe to events and print them to stdout.")) )