{-# LANGUAGE TemplateHaskell, OverloadedStrings, PackageImports, ScopedTypeVariables #-} {-| Embedded documentation files in various formats, and helpers for viewing them. |-} module Hledger.Cli.DocFiles ( Topic ,printHelpForTopic ,runManForTopic ,runInfoForTopic ,runPagerForTopic ,runTldrForPage ) where import Control.Exception import Data.ByteString (ByteString) import qualified Data.ByteString.Char8 as BC import Data.Maybe (fromMaybe) import Data.String import System.Environment (setEnv) import System.IO import System.IO.Temp import System.Process import Hledger.Utils (first3, second3, third3, embedFileRelative, error') import Text.Printf (printf) import System.Environment (lookupEnv) import Hledger.Utils.Debug -- The name of any hledger executable. type Tool = String -- Any heading in the hledger user manual (and perhaps later the hledger-ui/hledger-web manuals). type Topic = String -- Any name of a hledger tldr page (hledger, hledger-ui, hledger-print etc.) type TldrPage = String -- | All hledger-related pages from the tldr-pages project. -- All are symlinked into the hledger package directory to allow embeddeding. tldrs :: [(TldrPage, ByteString)] tldrs = [ ("hledger-accounts", $(embedFileRelative "embeddedfiles/hledger-accounts.md")) ,("hledger-add", $(embedFileRelative "embeddedfiles/hledger-add.md")) ,("hledger-aregister", $(embedFileRelative "embeddedfiles/hledger-aregister.md")) ,("hledger-balance", $(embedFileRelative "embeddedfiles/hledger-balance.md")) ,("hledger-balancesheet", $(embedFileRelative "embeddedfiles/hledger-balancesheet.md")) ,("hledger-import", $(embedFileRelative "embeddedfiles/hledger-import.md")) ,("hledger-incomestatement", $(embedFileRelative "embeddedfiles/hledger-incomestatement.md")) ,("hledger-print", $(embedFileRelative "embeddedfiles/hledger-print.md")) ,("hledger-ui", $(embedFileRelative "embeddedfiles/hledger-ui.md")) ,("hledger-web", $(embedFileRelative "embeddedfiles/hledger-web.md")) ,("hledger", $(embedFileRelative "embeddedfiles/hledger.md")) ] -- | The main hledger manuals as source for man, info and as plain text. -- All are symlinked into the hledger package directory to allow embeddeding. manuals :: [(Tool, (ByteString, ByteString, ByteString))] manuals = [ ("hledger", ($(embedFileRelative "embeddedfiles/hledger.1") ,$(embedFileRelative "embeddedfiles/hledger.txt") ,$(embedFileRelative "embeddedfiles/hledger.info") )) ,("hledger-ui", ($(embedFileRelative "embeddedfiles/hledger-ui.1") ,$(embedFileRelative "embeddedfiles/hledger-ui.txt") ,$(embedFileRelative "embeddedfiles/hledger-ui.info") )) ,("hledger-web", ($(embedFileRelative "embeddedfiles/hledger-web.1") ,$(embedFileRelative "embeddedfiles/hledger-web.txt") ,$(embedFileRelative "embeddedfiles/hledger-web.info") )) ] -- | Get the manual as plain text for this tool, or a not found message. manualTxt :: Tool -> ByteString manualTxt name = maybe (fromString $ "No text manual found for tool: "++name) second3 $ lookup name manuals -- | Get the manual as man source (nroff) for this tool, or a not found message. manualMan :: Tool -> ByteString manualMan name = maybe (fromString $ "No man page found for tool: "++name) first3 $ lookup name manuals -- | Get the manual as info source (texinfo) for this tool, or a not found message. manualInfo :: Tool -> ByteString manualInfo name = maybe (fromString $ "No info manual found for tool: "++name) third3 $ lookup name manuals -- | Print plain text help for this tool. -- Takes an optional topic argument for convenience but it is currently ignored. printHelpForTopic :: Tool -> Maybe Topic -> IO () printHelpForTopic tool _mtopic = BC.putStr (manualTxt tool) -- | Display an info manual for this topic, opened at the given topic if provided, -- using the "info" executable in $PATH. -- Topic can be an exact heading or a heading prefix; info will favour an exact match. runInfoForTopic :: Tool -> Maybe Topic -> IO () runInfoForTopic tool mtopic = withSystemTempFile ("hledger-"++tool++".info") $ \f h -> do BC.hPutStrLn h $ manualInfo tool hClose h callCommand $ dbg1 "info command" $ "info -f " ++ f ++ maybe "" (printf " -n '%s'") mtopic -- less with any vertical whitespace squashed, case-insensitive searching, the $ regex metacharacter accessible as \$. less = "less -s -i --use-backslash" -- | Display plain text help for this tool, scrolled to the given topic if any, using the users $PAGER or "less". -- When a topic is provided we always use less, ignoring $PAGER. -- -- This is less robust than the newer Hledger.Utils.IO.runPager, -- but that one doesn't yet support scrolling to a topic. runPagerForTopic :: Tool -> Maybe Topic -> IO () runPagerForTopic tool mtopic = do withSystemTempFile ("hledger-"++tool++".txt") $ \f h -> do BC.hPutStrLn h $ manualTxt tool hClose h envpager <- fromMaybe less <$> lookupEnv "PAGER" let exactmatch = True (pager, searcharg) = case mtopic of Nothing -> (envpager, "") Just t -> (less, "-p'^( )?" ++ t ++ if exactmatch then "\\$'" else "") callCommand $ dbg1 "pager command" $ unwords [pager, searcharg, f] -- | Display a man page for this tool, scrolled to the given topic if provided, using "man". -- When a topic is provided we force man to use "less", ignoring $MANPAGER and $PAGER. runManForTopic :: Tool -> Maybe Topic -> IO () runManForTopic tool mtopic = -- This temp file path should have a slash in it, man requires at least one. withSystemTempFile ("hledger-"++tool++".1") $ \f h -> do BC.hPutStrLn h $ manualMan tool hClose h let exactmatch = True pagerarg = case mtopic of Nothing -> "" Just t -> "-P \"" ++ less ++ " -p'^( )?" ++ t ++ (if exactmatch then "\\\\$" else "") ++ "'\"" callCommand $ dbg1 "man command" $ unwords ["man", pagerarg, f] -- | Get the named tldr page's source, if we know it. tldr :: TldrPage -> Maybe ByteString tldr name = lookup name tldrs -- | Display one of the hledger tldr pages, using "tldr". runTldrForPage :: TldrPage -> IO () runTldrForPage name = case tldr name of Nothing -> error' $ "sorry, there's no " <> name <> " tldr page yet" Just b -> (do withSystemTempFile (name++".md") $ \f h -> do BC.hPutStrLn h b hClose h -- tldr clients tend to auto-update their data, try to discourage that here -- tealdeer - doesn't auto-update by default -- tlrc - ? -- tldr-node-client - undocumented env var suggested in output setEnv "TLDR_AUTO_UPDATE_DISABLED" "1" callCommand $ dbg1 "tldr command" $ "tldr --render " <> f ) `catch` (\(_e::IOException) -> do hPutStrLn stderr $ "Warning: could not run tldr --render, using fallback viewer instead.\n" BC.putStrLn b )