diff --git a/cabal.project b/cabal.project index 6a3c0016742..604e576fd34 100644 --- a/cabal.project +++ b/cabal.project @@ -135,4 +135,3 @@ source-repository-package ouroboros-network-framework ouroboros-network-testing ouroboros-consensus-cardano-tools - diff --git a/cardano-api/src/Cardano/Api/SerialiseBech32.hs b/cardano-api/src/Cardano/Api/SerialiseBech32.hs index 66eb8b912e9..482607f30a0 100644 --- a/cardano-api/src/Cardano/Api/SerialiseBech32.hs +++ b/cardano-api/src/Cardano/Api/SerialiseBech32.hs @@ -25,7 +25,7 @@ import qualified Codec.Binary.Bech32 as Bech32 import Cardano.Api.Error import Cardano.Api.HasTypeProxy -import Cardano.Api.SerialiseRaw +import Cardano.Api.SerialiseRaw ( SerialiseAsRawBytes(..) ) import Cardano.Api.Utils diff --git a/cardano-cli/cardano-cli.cabal b/cardano-cli/cardano-cli.cabal index b1f2930556f..b6b19e32bb9 100644 --- a/cardano-cli/cardano-cli.cabal +++ b/cardano-cli/cardano-cli.cabal @@ -90,6 +90,9 @@ library Cardano.CLI.Shelley.Run.Read Cardano.CLI.Shelley.Run.Validate + Cardano.CLI.Ping + Cardano.CLI.Ping.Lib + Cardano.CLI.TopHandler other-modules: Paths_cardano_cli @@ -98,35 +101,38 @@ library , aeson-pretty >= 0.8.5 , ansi-terminal , attoparsec + , base16-bytestring >= 1.0 , bech32 >= 1.1.0 , binary , bytestring - , base16-bytestring >= 1.0 , canonical-json , cardano-api , cardano-binary - , cardano-git-rev , cardano-crypto , cardano-crypto-class ^>= 2.0 , cardano-crypto-wrapper ^>= 1.4 , cardano-data ^>= 0.1 + , cardano-git-rev , cardano-ledger-alonzo ^>= 0.1 , cardano-ledger-byron ^>= 0.1 , cardano-ledger-core ^>= 0.1 + , cardano-ledger-shelley ^>= 0.1 , cardano-ledger-shelley-ma ^>= 0.1 , cardano-prelude , cardano-protocol-tpraos ^>= 0.1 , cardano-slotting ^>= 0.1 - , vector-map ^>= 0.1 - , contra-tracer + , cardano-strict-containers ^>= 0.1 , cborg >= 0.2.4 && < 0.3 , containers + , contra-tracer , cryptonite , directory , filepath , formatting + , io-classes , iproute , network + , network-mux , optparse-applicative-fork , ouroboros-consensus , ouroboros-consensus-byron @@ -137,10 +143,10 @@ library , parsec , prettyprinter , random - , cardano-ledger-shelley ^>= 0.1 , set-algebra ^>= 0.1 , split - , cardano-strict-containers ^>= 0.1 + , strict-stm + , tdigest , text , time , transformers @@ -148,6 +154,7 @@ library , unliftio-core , utf8-string , vector + , vector-map ^>= 0.1 , yaml executable cardano-cli @@ -171,8 +178,8 @@ test-suite cardano-cli-test type: exitcode-stdio-1.0 build-depends: aeson - , bech32 >= 1.1.0 , base16-bytestring + , bech32 >= 1.1.0 , cardano-api , cardano-api:gen , cardano-cli diff --git a/cardano-cli/src/Cardano/CLI/Parsers.hs b/cardano-cli/src/Cardano/CLI/Parsers.hs index 97afcdf5d77..b71b2fb1d07 100644 --- a/cardano-cli/src/Cardano/CLI/Parsers.hs +++ b/cardano-cli/src/Cardano/CLI/Parsers.hs @@ -6,11 +6,12 @@ module Cardano.CLI.Parsers , pref ) where -import Cardano.Prelude import Cardano.CLI.Byron.Parsers (backwardsCompatibilityCommands, parseByronCommands) +import Cardano.CLI.Ping (parsePingCmd) import Cardano.CLI.Render (customRenderHelp) import Cardano.CLI.Run (ClientCommand (..)) import Cardano.CLI.Shelley.Parsers (parseShelleyCommands) +import Cardano.Prelude import Options.Applicative import Prelude (String) @@ -45,6 +46,7 @@ parseClientCommand = -- so we list it first. [ parseShelley , parseByron + , parsePing , parseDeprecatedShelleySubcommand , backwardsCompatibilityCommands , parseDisplayVersion opts @@ -62,6 +64,9 @@ parseByron = parseByronCommands ] +parsePing :: Parser ClientCommand +parsePing = CliPingCommand <$> parsePingCmd + -- | Parse Shelley-related commands at the top level of the CLI. parseShelley :: Parser ClientCommand parseShelley = ShelleyCommand <$> parseShelleyCommands diff --git a/cardano-cli/src/Cardano/CLI/Ping.hs b/cardano-cli/src/Cardano/CLI/Ping.hs new file mode 100644 index 00000000000..11daf31b3c6 --- /dev/null +++ b/cardano-cli/src/Cardano/CLI/Ping.hs @@ -0,0 +1,158 @@ +{-# LANGUAGE LambdaCase #-} +{-# LANGUAGE TypeApplications #-} + +module Cardano.CLI.Ping + ( PingCmd(..) + , PingClientCmdError(..) + , renderPingClientCmdError + , runPingCmd + , parsePingCmd + ) where + +import Cardano.CLI.Ping.Lib + +import Control.Applicative (Applicative (..), optional) +import Control.Exception (SomeException) +import Control.Monad (Monad (..), forM, mapM, mapM_, unless, when) +import Control.Monad.Class.MonadAsync (MonadAsync (async, wait, waitCatch)) +import Control.Concurrent.Class.MonadSTM.Strict +import Control.Monad.IO.Class (liftIO) +import Control.Monad.Trans.Except (ExceptT, throwE) +import Control.Tracer (Tracer (..)) +import Data.Bool ((&&)) +import Data.Either (Either (..)) +import Data.Function (($), (.)) +import Data.Functor ((<$>)) +import Data.List (foldl') +import Data.Maybe (Maybe (..), isNothing) +import Data.Monoid (mconcat) +import Data.Semigroup ((<>)) +import Data.String (String) +import Data.Text (Text) +import GHC.Enum (Bounded (..)) +import Network.Socket (AddrInfo) +import System.IO (IO) +import Text.Show (Show(..)) + +import qualified Data.List as L +import qualified Data.Text as T +import qualified Network.Socket as Socket +import qualified Options.Applicative as Opt +import qualified Prettyprinter as PP +import qualified System.Exit as IO +import qualified System.IO as IO + +data PingClientCmdError + = PingClientCmdErrorOfInvalidHostIp + | PingClientCmdErrorOfExceptions ![(AddrInfo, SomeException)] + +runPingCmd :: PingCmd -> ExceptT PingClientCmdError IO () +runPingCmd options = do + let hints = Socket.defaultHints { Socket.addrSocketType = Socket.Stream } + + msgQueue <- liftIO newEmptyTMVarIO + + when (isNothing (pingCmdHost options) && isNothing (pingCmdUnixSock options)) $ + throwE PingClientCmdErrorOfInvalidHostIp + + (addresses, versions) <- case pingCmdUnixSock options of + Nothing -> do + addrs <- liftIO $ Socket.getAddrInfo (Just hints) (pingCmdHost options) (Just (pingCmdPort options)) + return (addrs, supportedNodeToNodeVersions $ pingCmdMagic options) + Just fname -> + return + ( [ Socket.AddrInfo [] Socket.AF_UNIX Socket.Stream + Socket.defaultProtocol (Socket.SockAddrUnix fname) + Nothing + ] + , supportedNodeToClientVersions $ pingCmdMagic options + ) + + laid <- liftIO . async $ logger msgQueue $ pingCmdJson options + caids <- forM addresses $ liftIO . async . pingClient (Tracer $ doLog msgQueue) options versions + res <- L.zip addresses <$> mapM (liftIO . waitCatch) caids + liftIO $ doLog msgQueue LogEnd + liftIO $ wait laid + case foldl' partition ([],[]) res of + ([], _) -> liftIO IO.exitSuccess + (es, []) -> throwE $ PingClientCmdErrorOfExceptions es + (es, _) -> do + unless (pingCmdQuiet options) $ mapM_ (liftIO . IO.hPrint IO.stderr) es + liftIO IO.exitSuccess + + where + partition :: ([(AddrInfo, SomeException)], [AddrInfo]) + -> (AddrInfo, Either SomeException ()) + -> ([(AddrInfo, SomeException)], [AddrInfo]) + partition (es, as) (a, Left e) = ((a, e) : es, as) + partition (es, as) (a, Right _) = (es, a : as) + + doLog :: StrictTMVar IO LogMsg -> LogMsg -> IO () + doLog msgQueue msg = atomically $ putTMVar msgQueue msg + +renderPingClientCmdError :: PingClientCmdError -> Text +renderPingClientCmdError = \case + PingClientCmdErrorOfInvalidHostIp -> "Specify host/ip with '-h ' or a unix socket with -u " + PingClientCmdErrorOfExceptions es -> T.intercalate "\n" $ T.pack . show <$> es + +parsePingCmd :: Opt.Parser PingCmd +parsePingCmd = Opt.hsubparser $ mconcat + [ Opt.metavar "ping" + , Opt.command "ping" $ Opt.info pPing $ Opt.progDescDoc $ Just $ mconcat + [ PP.pretty @String "Ping a cardano node either using node-to-node or node-to-client protocol. " + , PP.pretty @String "It negotiates a handshake and keep sending keep alive messages." + ] + ] + +pPing :: Opt.Parser PingCmd +pPing = PingCmd + <$> Opt.option Opt.auto + ( Opt.long "count" + <> Opt.short 'c' + <> Opt.metavar "COUNT" + <> Opt.help + ( "Stop after sending count requests and receiving count responses. " + <> "If this option is not specified, ping will operate until interrupted. " + ) + <> Opt.value maxBound + ) + <*> optional + ( Opt.strOption + ( Opt.long "host" + <> Opt.short 'h' + <> Opt.metavar "HOST" + <> Opt.help "Hostname/IP, e.g. relay.iohk.example." + ) + ) + <*> optional + ( Opt.strOption + ( Opt.long "unixsock" + <> Opt.short 'u' + <> Opt.metavar "SOCKET" + <> Opt.help "Unix socket, e.g. file.socket." + ) + ) + <*> Opt.option Opt.auto + ( Opt.long "port" + <> Opt.short 'p' + <> Opt.metavar "PORT" + <> Opt.help "Port number, e.g. 1234." + <> Opt.value "3001" + ) + <*> Opt.option Opt.auto + ( Opt.long "magic" + <> Opt.short 'm' + <> Opt.metavar "MAGIC" + <> Opt.help "Network magic." + <> Opt.value mainnetMagic + ) + <*> Opt.switch + ( Opt.long "json" + <> Opt.short 'j' + <> Opt.help "JSON output flag." + ) + <*> Opt.switch + ( Opt.long "quiet" + <> Opt.short 'q' + <> Opt.help "Quiet flag, CSV/JSON only output" + ) diff --git a/cardano-cli/src/Cardano/CLI/Ping/Lib.hs b/cardano-cli/src/Cardano/CLI/Ping/Lib.hs new file mode 100644 index 00000000000..80bba8aea22 --- /dev/null +++ b/cardano-cli/src/Cardano/CLI/Ping/Lib.hs @@ -0,0 +1,463 @@ +{-# LANGUAGE BangPatterns #-} +{-# LANGUAGE DataKinds #-} +{-# LANGUAGE InstanceSigs #-} +{-# LANGUAGE NamedFieldPuns #-} +{-# LANGUAGE RankNTypes #-} +{-# LANGUAGE RecordWildCards #-} + +module Cardano.CLI.Ping.Lib + ( PingCmd(..) + , LogMsg(..) + , NodeVersion(..) + , HandshakeFailure(..) + , StatPoint(..) + , mainnetMagic + , pingClient + , logger + , supportedNodeToNodeVersions + , supportedNodeToClientVersions + ) where + +import Control.Exception (bracket) +import Control.Monad (Monad (..), replicateM, unless, when) +import Control.Concurrent.Class.MonadSTM.Strict +import Control.Monad.Class.MonadTime +import Control.Tracer (Tracer (..), nullTracer, traceWith) +import Data.Aeson hiding (Options, json) +import Data.Bits (clearBit, setBit, testBit) +import Data.Bool (Bool (..)) +import Data.ByteString.Lazy (ByteString) +import Data.Either (Either (..)) +import Data.Eq (Eq (..)) +import Data.Function (($), (.)) +import Data.Functor ((<$>)) +import Data.Int (Int) +import Data.Maybe (Maybe (..), fromMaybe,) +import Data.Monoid (mconcat) +import Data.Ord (Ord (..)) +import Data.Semigroup ((<>)) +import Data.String (String) +import Data.TDigest +import Data.Text (unpack) +import Data.Word (Word, Word16, Word32) +import GHC.Enum (Bounded (..)) +import GHC.Err (error) +import GHC.Float (Double) +import GHC.Num (Num (..)) +import GHC.Real (fromIntegral, realToFrac) +import Network.Mux.Bearer.Socket (socketAsMuxBearer) +import Network.Mux.Timeout (TimeoutFn, withTimeoutSerial) +import Network.Mux.Types +import Network.Socket (AddrInfo, StructLinger (..)) +import System.IO (IO) +import Text.Printf (printf) +import Text.Show (Show (..)) + +import qualified Codec.CBOR.Decoding as CBOR +import qualified Codec.CBOR.Encoding as CBOR +import qualified Codec.CBOR.Read as CBOR +import qualified Codec.CBOR.Write as CBOR +import qualified Control.Monad.Class.MonadTimer as MT +import qualified Data.ByteString.Char8 as BS.Char +import qualified Data.ByteString.Lazy as LBS +import qualified Data.ByteString.Lazy.Char8 as LBSC +import qualified Data.List as L +import qualified Network.Socket as Socket +import qualified System.IO as IO + +data PingCmd = PingCmd + { pingCmdCount :: !Word32 + , pingCmdHost :: !(Maybe String) + , pingCmdUnixSock :: !(Maybe String) + , pingCmdPort :: !String + , pingCmdMagic :: !Word32 + , pingCmdJson :: !Bool + , pingCmdQuiet :: !Bool + } deriving (Eq, Show) + +mainnetMagic :: Word32 +mainnetMagic = 764824073 + +handshakeNum :: MiniProtocolNum +handshakeNum = MiniProtocolNum 0 + +keepaliveNum :: MiniProtocolNum +keepaliveNum = MiniProtocolNum 8 + +nodeToClientVersionBit :: Int +nodeToClientVersionBit = 15 + +data LogMsg = LogMsg ByteString + | LogEnd + +logger :: StrictTMVar IO LogMsg -> Bool -> IO () +logger msgQueue json = go True + where + go first = do + msg <- atomically $ takeTMVar msgQueue + case msg of + LogMsg bs -> do + let bs' = case (json, first) of + (True, False) -> LBSC.pack ",\n" <> bs + (True, True) -> LBSC.pack "{ \"pongs\": [ " <> bs + (False, True) -> LBSC.pack " timestamp, host, cookie, sample, median, p90, mean, min, max, std\n" <> bs + (False, False) -> bs + + LBSC.putStr bs' + go False + LogEnd -> when json $ IO.putStrLn "] }" + +supportedNodeToNodeVersions :: Word32 -> [NodeVersion] +supportedNodeToNodeVersions magic = + [ NodeToNodeVersionV7 magic False + , NodeToNodeVersionV8 magic False + , NodeToNodeVersionV9 magic False + , NodeToNodeVersionV10 magic False + ] + +supportedNodeToClientVersions :: Word32 -> [NodeVersion] +supportedNodeToClientVersions magic = + [ NodeToClientVersionV9 magic + , NodeToClientVersionV10 magic + , NodeToClientVersionV11 magic + , NodeToClientVersionV12 magic + , NodeToClientVersionV13 magic + , NodeToClientVersionV14 magic + ] + +data NodeVersion + = NodeToClientVersionV9 !Word32 + | NodeToClientVersionV10 !Word32 + | NodeToClientVersionV11 !Word32 + | NodeToClientVersionV12 !Word32 + | NodeToClientVersionV13 !Word32 + | NodeToClientVersionV14 !Word32 + | NodeToNodeVersionV1 !Word32 + | NodeToNodeVersionV2 !Word32 + | NodeToNodeVersionV3 !Word32 + | NodeToNodeVersionV4 !Word32 !Bool + | NodeToNodeVersionV5 !Word32 !Bool + | NodeToNodeVersionV6 !Word32 !Bool + | NodeToNodeVersionV7 !Word32 !Bool + | NodeToNodeVersionV8 !Word32 !Bool + | NodeToNodeVersionV9 !Word32 !Bool + | NodeToNodeVersionV10 !Word32 !Bool + deriving (Eq, Ord, Show) + +keepAliveReqEnc :: NodeVersion -> Word16 -> CBOR.Encoding +keepAliveReqEnc v cookie | v >= NodeToNodeVersionV7 minBound minBound = + CBOR.encodeListLen 2 + <> CBOR.encodeWord 0 + <> CBOR.encodeWord16 cookie +keepAliveReqEnc _ cookie = + CBOR.encodeWord 0 + <> CBOR.encodeWord16 cookie + +keepAliveReq :: NodeVersion -> Word16 -> ByteString +keepAliveReq v c = CBOR.toLazyByteString $ keepAliveReqEnc v c + +keepAliveDone :: NodeVersion -> ByteString +keepAliveDone v | v >= NodeToNodeVersionV7 minBound minBound = + CBOR.toLazyByteString $ + CBOR.encodeListLen 1 + <> CBOR.encodeWord 2 +keepAliveDone _ = + CBOR.toLazyByteString $ + CBOR.encodeWord 2 + + +handshakeReqEnc :: [NodeVersion] -> CBOR.Encoding +handshakeReqEnc [] = error "null version list" +handshakeReqEnc versions = + CBOR.encodeListLen 2 + <> CBOR.encodeWord 0 + <> CBOR.encodeMapLen (fromIntegral $ L.length versions) + <> mconcat [ encodeVersion v + | v <- versions + ] + where + encodeVersion :: NodeVersion -> CBOR.Encoding + encodeVersion (NodeToClientVersionV9 magic) = + CBOR.encodeWord (9 `setBit` nodeToClientVersionBit) + <> CBOR.encodeInt (fromIntegral magic) + encodeVersion (NodeToClientVersionV10 magic) = + CBOR.encodeWord (10 `setBit` nodeToClientVersionBit) + <> CBOR.encodeInt (fromIntegral magic) + encodeVersion (NodeToClientVersionV11 magic) = + CBOR.encodeWord (11 `setBit` nodeToClientVersionBit) + <> CBOR.encodeInt (fromIntegral magic) + encodeVersion (NodeToClientVersionV12 magic) = + CBOR.encodeWord (12 `setBit` nodeToClientVersionBit) + <> CBOR.encodeInt (fromIntegral magic) + encodeVersion (NodeToClientVersionV13 magic) = + CBOR.encodeWord (13 `setBit` nodeToClientVersionBit) + <> CBOR.encodeInt (fromIntegral magic) + encodeVersion (NodeToClientVersionV14 magic) = + CBOR.encodeWord (14 `setBit` nodeToClientVersionBit) + <> CBOR.encodeInt (fromIntegral magic) + encodeVersion (NodeToNodeVersionV1 magic) = + CBOR.encodeWord 1 + <> CBOR.encodeInt (fromIntegral magic) + encodeVersion (NodeToNodeVersionV2 magic) = + CBOR.encodeWord 2 + <> CBOR.encodeInt (fromIntegral magic) + encodeVersion (NodeToNodeVersionV3 magic) = + CBOR.encodeWord 3 + <> CBOR.encodeInt (fromIntegral magic) + encodeVersion (NodeToNodeVersionV4 magic mode) = encodeWithMode 4 magic mode + encodeVersion (NodeToNodeVersionV5 magic mode) = encodeWithMode 5 magic mode + encodeVersion (NodeToNodeVersionV6 magic mode) = encodeWithMode 6 magic mode + encodeVersion (NodeToNodeVersionV7 magic mode) = encodeWithMode 7 magic mode + encodeVersion (NodeToNodeVersionV8 magic mode) = encodeWithMode 8 magic mode + encodeVersion (NodeToNodeVersionV9 magic mode) = encodeWithMode 9 magic mode + encodeVersion (NodeToNodeVersionV10 magic mode) = encodeWithMode 10 magic mode + + + encodeWithMode :: Word -> Word32 -> Bool -> CBOR.Encoding + encodeWithMode vn magic mode = + CBOR.encodeWord vn + <> CBOR.encodeListLen 2 + <> CBOR.encodeInt (fromIntegral magic) + <> CBOR.encodeBool mode + +handshakeReq :: [NodeVersion] -> ByteString +handshakeReq [] = LBS.empty +handshakeReq versions = CBOR.toLazyByteString $ handshakeReqEnc versions + +data HandshakeFailure + = UnknownVersionInRsp !Word + | UnknownKey !Word + | UnknownTag !Word + | VersionMissmath ![Word] + | DecodeError !Word !String + | Refused !Word !String + deriving Show + +newtype KeepAliveFailure = KeepAliveFailureKey Word deriving Show + +keepAliveRspDec :: NodeVersion + -> CBOR.Decoder s (Either KeepAliveFailure Word16) +keepAliveRspDec v | v >= NodeToNodeVersionV7 minBound minBound = do + len <- CBOR.decodeListLen + key <- CBOR.decodeWord + case (len, key) of + (2, 1) -> Right <$> CBOR.decodeWord16 + (_, k) -> return $ Left $ KeepAliveFailureKey k +keepAliveRspDec _ = do + key <- CBOR.decodeWord + case key of + 1 -> Right <$> CBOR.decodeWord16 + k -> return $ Left $ KeepAliveFailureKey k + +handshakeDec :: CBOR.Decoder s (Either HandshakeFailure NodeVersion) +handshakeDec = do + _ <- CBOR.decodeListLen + key <- CBOR.decodeWord + case key of + 1 -> decodeVersion + 2 -> do + _ <- CBOR.decodeListLen + tag <- CBOR.decodeWord + case tag of + 0 -> do -- VersionMismatch + len <- CBOR.decodeListLen + x <- replicateM len CBOR.decodeWord + return $ Left $ VersionMissmath x + 1 -> do -- HandshakeDecodeError + vn <- CBOR.decodeWord + msg <- unpack <$> CBOR.decodeString + return $ Left $ DecodeError vn msg + 2 -> do -- Refused + vn <- CBOR.decodeWord + msg <- unpack <$> CBOR.decodeString + return $ Left $ Refused vn msg + _ -> return $ Left $ UnknownTag tag + + k -> return $ Left $ UnknownKey k + where + decodeVersion :: CBOR.Decoder s (Either HandshakeFailure NodeVersion) + decodeVersion = do + version <- CBOR.decodeWord + let cb = version `clearBit` nodeToClientVersionBit + let tb = version `testBit` nodeToClientVersionBit + case (cb, tb) of + (7, False) -> decodeWithMode NodeToNodeVersionV7 + (8, False) -> decodeWithMode NodeToNodeVersionV8 + (9, False) -> decodeWithMode NodeToNodeVersionV9 + (10, False) -> decodeWithMode NodeToNodeVersionV10 + (9, True) -> Right . NodeToClientVersionV9 <$> CBOR.decodeWord32 + (10, True) -> Right . NodeToClientVersionV10 <$> CBOR.decodeWord32 + (11, True) -> Right . NodeToClientVersionV11 <$> CBOR.decodeWord32 + (12, True) -> Right . NodeToClientVersionV12 <$> CBOR.decodeWord32 + (13, True) -> Right . NodeToClientVersionV13 <$> CBOR.decodeWord32 + (14, True) -> Right . NodeToClientVersionV14 <$> CBOR.decodeWord32 + _ -> return $ Left $ UnknownVersionInRsp version + + decodeWithMode :: (Word32 -> Bool -> NodeVersion) -> CBOR.Decoder s (Either HandshakeFailure NodeVersion) + decodeWithMode vnFun = do + _ <- CBOR.decodeListLen + magic <- CBOR.decodeWord32 + Right . vnFun magic <$> CBOR.decodeBool + +wrap :: MiniProtocolNum -> MiniProtocolDir -> LBS.ByteString -> MuxSDU +wrap ptclNum ptclDir blob = MuxSDU + { msHeader = MuxSDUHeader + { mhTimestamp = RemoteClockModel 0 + , mhNum = ptclNum + , mhDir = ptclDir + , mhLength = fromIntegral $ LBS.length blob + } + , msBlob = blob + } + + +data StatPoint = StatPoint + { spTimestamp :: !UTCTime + , spHost :: !String + , spCookie :: !Word16 + , spSample :: !Double + , spMedian :: !Double + , spP90 :: !Double + , spMean :: !Double + , spMin :: !Double + , spMax :: !Double + , spStd :: !Double + } + +instance Show StatPoint where + show :: StatPoint -> String + show StatPoint {..} = + printf "%36s, %-28s, %7d, %7.3f, %7.3f, %7.3f, %7.3f, %7.3f, %7.3f, %7.3f" + (show spTimestamp) spHost spCookie spSample spMedian spP90 spMean spMin spMax spStd + +instance ToJSON StatPoint where + toJSON :: StatPoint -> Value + toJSON StatPoint {..} = + object + [ "timestamp" .= spTimestamp + , "host" .= spHost + , "cookie" .= spCookie + , "sample" .= spSample + , "median" .= spMedian + , "p90" .= spP90 + , "mean" .= spMean + , "min" .= spMin + , "max" .= spMax + ] + +toStatPoint :: UTCTime -> String -> Word16 -> Double -> TDigest 5 -> StatPoint +toStatPoint ts host cookie sample td = + StatPoint + { spTimestamp = ts + , spHost = host + , spCookie = cookie + , spSample = sample + , spMedian = quantile' 0.5 + , spP90 = quantile' 0.9 + , spMean = mean' + , spMin = minimumValue td + , spMax = maximumValue td + , spStd = stddev' + } + where + quantile' :: Double -> Double + quantile' q = fromMaybe 0 (quantile q td) + + mean' :: Double + mean' = fromMaybe 0 (mean td) + + stddev' :: Double + stddev' = fromMaybe 0 (stddev td) + +pingClient :: Tracer IO LogMsg -> PingCmd -> [NodeVersion] -> AddrInfo -> IO () +pingClient tracer PingCmd{pingCmdQuiet, pingCmdJson, pingCmdCount} versions peer = bracket + (Socket.socket (Socket.addrFamily peer) Socket.Stream Socket.defaultProtocol) + Socket.close + (\sd -> withTimeoutSerial $ \timeoutfn -> do + when (Socket.addrFamily peer /= Socket.AF_UNIX) $ do + Socket.setSocketOption sd Socket.NoDelay 1 + Socket.setSockOpt sd Socket.Linger $ StructLinger + { sl_onoff = 1 + , sl_linger = 0 + } + + !t0_s <- getMonotonicTime + Socket.connect sd (Socket.addrAddress peer) + !t0_e <- getMonotonicTime + peerStr <- peerString + unless pingCmdQuiet $ printf "%s network rtt: %.3f\n" peerStr $ toSample t0_e t0_s + + let timeout = 30 + bearer = socketAsMuxBearer timeout nullTracer sd + + !t1_s <- write bearer timeoutfn $ wrap handshakeNum InitiatorDir (handshakeReq versions) + (msg, !t1_e) <- nextMsg bearer timeoutfn handshakeNum + unless pingCmdQuiet $ printf "%s handshake rtt: %s\n" peerStr (show $ diffTime t1_e t1_s) + + case CBOR.deserialiseFromBytes handshakeDec msg of + Left err -> eprint $ printf "%s Decoding error %s\n" peerStr (show err) + Right (_, Left err) -> eprint $ printf "%s Protocol error %s\n" peerStr (show err) + Right (_, Right version) -> do + unless pingCmdQuiet $ printf "%s Negotiated version %s\n" peerStr (show version) + keepAlive bearer timeoutfn peerStr version (tdigest []) 0 + -- send terminating message + _ <- write bearer timeoutfn $ wrap keepaliveNum InitiatorDir (keepAliveDone version) + -- protocol idle timeout + MT.threadDelay 5 + + ) + where + + peerString :: IO String + peerString = + case Socket.addrFamily peer of + Socket.AF_UNIX -> return $ show $ Socket.addrAddress peer + _ -> do + (Just host, Just port) <- + Socket.getNameInfo + [Socket.NI_NUMERICHOST, Socket.NI_NUMERICSERV] + True True (Socket.addrAddress peer) + return $ host <> ":" <> port + + toSample :: Time -> Time -> Double + toSample t_e t_s = realToFrac $ diffTime t_e t_s + + eprint :: String -> IO () + eprint = BS.Char.hPutStr IO.stderr . BS.Char.pack + + nextMsg :: MuxBearer IO -> TimeoutFn IO -> MiniProtocolNum -> IO (LBS.ByteString, Time) + nextMsg bearer timeoutfn ptclNum = do + (sdu, t_e) <- Network.Mux.Types.read bearer timeoutfn + if mhNum (msHeader sdu) == ptclNum + then return (msBlob sdu, t_e) + else nextMsg bearer timeoutfn ptclNum + + keepAlive :: MuxBearer IO + -> TimeoutFn IO + -> String + -> NodeVersion + -> TDigest 5 + -> Word32 + -> IO () + keepAlive _ _ _ _ _ cookie | cookie == pingCmdCount = return () + keepAlive bearer timeoutfn peerStr version td !cookie = do + let cookie16 = fromIntegral cookie + !t_s <- write bearer timeoutfn $ wrap keepaliveNum InitiatorDir (keepAliveReq version cookie16) + (!msg, !t_e) <- nextMsg bearer timeoutfn keepaliveNum + let rtt = toSample t_e t_s + td' = insert rtt td + case CBOR.deserialiseFromBytes (keepAliveRspDec version) msg of + Left err -> eprint $ printf "%s keepalive decoding error %s\n" peerStr (show err) + Right (_, Left err) -> eprint $ printf "%s keepalive protocol error %s\n" peerStr (show err) + Right (_, Right cookie') -> do + when (cookie' /= cookie16) $ eprint $ printf "%s cookie missmatch %d /= %d\n" + peerStr cookie' cookie + + now <- getCurrentTime + let point = toStatPoint now peerStr cookie16 rtt td' + if pingCmdJson + then traceWith tracer $ LogMsg (encode point) + else traceWith tracer $ LogMsg $ LBSC.pack $ show point <> "\n" + IO.hFlush IO.stdout + MT.threadDelay 1 + keepAlive bearer timeoutfn peerStr version td' (cookie + 1) diff --git a/cardano-cli/src/Cardano/CLI/Run.hs b/cardano-cli/src/Cardano/CLI/Run.hs index c8be9b9d62b..d3951e793fa 100644 --- a/cardano-cli/src/Cardano/CLI/Run.hs +++ b/cardano-cli/src/Cardano/CLI/Run.hs @@ -18,6 +18,7 @@ import qualified Data.Text.IO as Text import Cardano.CLI.Byron.Commands (ByronCommand) import Cardano.CLI.Byron.Run (ByronClientCmdError, renderByronClientCmdError, runByronClientCommand) +import Cardano.CLI.Ping (PingCmd (..), PingClientCmdError (..), renderPingClientCmdError, runPingCmd) import Cardano.CLI.Shelley.Commands (ShelleyCommand) import Cardano.CLI.Shelley.Run (ShelleyClientCmdError, renderShelleyClientCmdError, runShelleyClientCommand) @@ -48,16 +49,20 @@ data ClientCommand = -- now-deprecated \"shelley\" subcommand. | DeprecatedShelleySubcommand ShelleyCommand + | CliPingCommand PingCmd + | forall a. Help ParserPrefs (ParserInfo a) | DisplayVersion data ClientCommandErrors = ByronClientError ByronClientCmdError | ShelleyClientError ShelleyCommand ShelleyClientCmdError + | PingClientError PingClientCmdError runClientCommand :: ClientCommand -> ExceptT ClientCommandErrors IO () runClientCommand (ByronCommand c) = firstExceptT ByronClientError $ runByronClientCommand c runClientCommand (ShelleyCommand c) = firstExceptT (ShelleyClientError c) $ runShelleyClientCommand c +runClientCommand (CliPingCommand c) = firstExceptT PingClientError $ runPingCmd c runClientCommand (DeprecatedShelleySubcommand c) = firstExceptT (ShelleyClientError c) $ runShelleyClientCommandWithDeprecationWarning @@ -70,6 +75,8 @@ renderClientCommandError (ByronClientError err) = renderByronClientCmdError err renderClientCommandError (ShelleyClientError cmd err) = renderShelleyClientCmdError cmd err +renderClientCommandError (PingClientError err) = + renderPingClientCmdError err -- | Combine an 'ExceptT' that will write a warning message to @stderr@ with -- the provided 'ExceptT'.