{-# LANGUAGE ApplicativeDo #-} {-# LANGUAGE DuplicateRecordFields #-} {-# LANGUAGE LambdaCase #-} {-# LANGUAGE NamedFieldPuns #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE ScopedTypeVariables #-} module Simplex.Chat.Options ( ChatOpts (..), getChatOpts, smpServersP, fullNetworkConfig, ) where import qualified Data.Attoparsec.ByteString.Char8 as A import qualified Data.ByteString.Char8 as B import Numeric.Natural (Natural) import Options.Applicative import Simplex.Chat.Controller (ChatLogLevel (..), updateStr, versionNumber, versionString) import Simplex.Messaging.Client (NetworkConfig (..), defaultNetworkConfig) import Simplex.Messaging.Encoding.String import Simplex.Messaging.Parsers (parseAll) import Simplex.Messaging.Protocol (SMPServerWithAuth) import Simplex.Messaging.Transport.Client (SocksProxy, defaultSocksProxy) import System.FilePath (combine) data ChatOpts = ChatOpts { dbFilePrefix :: String, dbKey :: String, smpServers :: [SMPServerWithAuth], networkConfig :: NetworkConfig, logLevel :: ChatLogLevel, logConnections :: Bool, logServerHosts :: Bool, logAgent :: Bool, tbqSize :: Natural, chatCmd :: String, chatCmdDelay :: Int, chatServerPort :: Maybe String, optFilesFolder :: Maybe FilePath, allowInstantFiles :: Bool, maintenance :: Bool } chatOpts :: FilePath -> FilePath -> Parser ChatOpts chatOpts appDir defaultDbFileName = do dbFilePrefix <- strOption ( long "database" <> short 'd' <> metavar "DB_FILE" <> help "Path prefix to chat and agent database files" <> value defaultDbFilePath <> showDefault ) dbKey <- strOption ( long "key" <> short 'k' <> metavar "KEY" <> help "Database encryption key/pass-phrase" <> value "" ) smpServers <- option parseSMPServers ( long "server" <> short 's' <> metavar "SERVER" <> help "Semicolon-separated list of SMP server(s) to use (each server can have more than one hostname)" <> value [] ) socksProxy <- flag' (Just defaultSocksProxy) (short 'x' <> help "Use local SOCKS5 proxy at :9050") <|> option parseSocksProxy ( long "socks-proxy" <> metavar "SOCKS5" <> help "Use SOCKS5 proxy at `ipv4:port` or `:port`" <> value Nothing ) t <- option auto ( long "tcp-timeout" <> metavar "TIMEOUT" <> help "TCP timeout, seconds (default: 5/10 without/with SOCKS5 proxy)" <> value 0 ) logLevel <- option parseLogLevel ( long "log-level" <> short 'l' <> metavar "LEVEL" <> help "Log level: debug, info, warn, error, important (default)" <> value CLLImportant ) logTLSErrors <- switch ( long "log-tls-errors" <> help "Log TLS errors (also enabled with `-l debug`)" ) logConnections <- switch ( long "connections" <> short 'c' <> help "Log every contact and group connection on start (also with `-l info`)" ) logServerHosts <- switch ( long "log-hosts" <> help "Log connections to servers (also with `-l info`)" ) logAgent <- switch ( long "log-agent" <> help "Enable logs from SMP agent (also with `-l debug`)" ) tbqSize <- option auto ( long "queue-size" <> short 'q' <> metavar "SIZE" <> help "Internal queue size" <> value 64 <> showDefault ) chatCmd <- strOption ( long "execute" <> short 'e' <> metavar "COMMAND" <> help "Execute chat command (received messages won't be logged) and exit" <> value "" ) chatCmdDelay <- option auto ( long "time" <> short 't' <> metavar "TIME" <> help "Time to wait after sending chat command before exiting, seconds" <> value 3 <> showDefault ) chatServerPort <- option parseServerPort ( long "chat-server-port" <> short 'p' <> metavar "PORT" <> help "Run chat server on specified port" <> value Nothing ) optFilesFolder <- optional $ strOption ( long "files-folder" <> metavar "FOLDER" <> help "Folder to use for sent and received files" ) allowInstantFiles <- switch ( long "allow-instant-files" <> short 'f' <> help "Send and receive instant files without acceptance" ) maintenance <- switch ( long "maintenance" <> short 'm' <> help "Run in maintenance mode (/_start to start chat)" ) pure ChatOpts { dbFilePrefix, dbKey, smpServers, networkConfig = fullNetworkConfig socksProxy (useTcpTimeout socksProxy t) (logTLSErrors || logLevel == CLLDebug), logLevel, logConnections = logConnections || logLevel <= CLLInfo, logServerHosts = logServerHosts || logLevel <= CLLInfo, logAgent = logAgent || logLevel == CLLDebug, tbqSize, chatCmd, chatCmdDelay, chatServerPort, optFilesFolder, allowInstantFiles, maintenance } where useTcpTimeout p t = 1000000 * if t > 0 then t else maybe 5 (const 10) p defaultDbFilePath = combine appDir defaultDbFileName fullNetworkConfig :: Maybe SocksProxy -> Int -> Bool -> NetworkConfig fullNetworkConfig socksProxy tcpTimeout logTLSErrors = let tcpConnectTimeout = (tcpTimeout * 3) `div` 2 in defaultNetworkConfig {socksProxy, tcpTimeout, tcpConnectTimeout, logTLSErrors} parseSMPServers :: ReadM [SMPServerWithAuth] parseSMPServers = eitherReader $ parseAll smpServersP . B.pack parseSocksProxy :: ReadM (Maybe SocksProxy) parseSocksProxy = eitherReader $ parseAll strP . B.pack parseServerPort :: ReadM (Maybe String) parseServerPort = eitherReader $ parseAll serverPortP . B.pack serverPortP :: A.Parser (Maybe String) serverPortP = Just . B.unpack <$> A.takeWhile A.isDigit smpServersP :: A.Parser [SMPServerWithAuth] smpServersP = strP `A.sepBy1` A.char ';' parseLogLevel :: ReadM ChatLogLevel parseLogLevel = eitherReader $ \case "debug" -> Right CLLDebug "info" -> Right CLLInfo "warn" -> Right CLLWarning "error" -> Right CLLError "important" -> Right CLLImportant _ -> Left "Invalid log level" getChatOpts :: FilePath -> FilePath -> IO ChatOpts getChatOpts appDir defaultDbFileName = execParser $ info (helper <*> versionOption <*> chatOpts appDir defaultDbFileName) (header versionStr <> fullDesc <> progDesc "Start chat with DB_FILE file and use SERVER as SMP server") where versionStr = versionString versionNumber versionOption = infoOption versionAndUpdate (long "version" <> short 'v' <> help "Show version") versionAndUpdate = versionStr <> "\n" <> updateStr