Skip to content

Commit

Permalink
Implement EKG statistics
Browse files Browse the repository at this point in the history
  • Loading branch information
Julian Ospald committed Nov 28, 2020
1 parent 8a847d8 commit c1d1f77
Show file tree
Hide file tree
Showing 9 changed files with 150 additions and 27 deletions.
1 change: 1 addition & 0 deletions cabal.project
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
-- Generated by stackage-to-hackage
--

index-state: 2020-11-11T15:40:10Z

Expand Down
1 change: 1 addition & 0 deletions lib/cli/cardano-wallet-cli.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@ library
aeson
, aeson-pretty
, ansi-terminal
, async
, base
, bytestring
, cardano-addresses
Expand Down
88 changes: 81 additions & 7 deletions lib/cli/src/Cardano/CLI.hs
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE NumericUnderscores #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeApplications #-}
Expand Down Expand Up @@ -94,6 +95,8 @@ module Cardano.CLI
, getDataDir
, setupDirectory
, waitForService
, getPrometheusURL
, getEKGURL
, WaitForServiceLog (..)
) where

Expand All @@ -104,8 +107,14 @@ import Cardano.BM.Backend.Switchboard
( Switchboard )
import Cardano.BM.Configuration.Static
( defaultConfigStdout )
import Cardano.BM.Counters
( readCounters )
import Cardano.BM.Data.Configuration
( Endpoint (..) )
import Cardano.BM.Data.Counter
( Counter (..), nameCounter )
import Cardano.BM.Data.LogItem
( LoggerName )
( LOContent (..), LoggerName, PrivacyAnnotation (..), mkLOMeta )
import Cardano.BM.Data.Output
( ScribeDefinition (..)
, ScribeFormat (..)
Expand All @@ -115,12 +124,14 @@ import Cardano.BM.Data.Output
)
import Cardano.BM.Data.Severity
( Severity (..) )
import Cardano.BM.Data.SubTrace
( SubTrace (..) )
import Cardano.BM.Data.Tracer
( HasPrivacyAnnotation (..), HasSeverityAnnotation (..) )
import Cardano.BM.Setup
( setupTrace_, shutdown )
import Cardano.BM.Trace
( Trace, appendName, logDebug )
( Trace, appendName, logDebug, traceNamedObject )
import Cardano.Mnemonic
( MkSomeMnemonic (..), SomeMnemonic (..) )
import Cardano.Wallet.Api.Client
Expand Down Expand Up @@ -185,10 +196,14 @@ import Control.Applicative
( optional, some, (<|>) )
import Control.Arrow
( first, left )
import Control.Concurrent
( threadDelay )
import Control.Exception
( bracket, catch )
import Control.Monad
( join, unless, void, when )
( forM_, forever, join, unless, void, when )
import Control.Monad.IO.Class
( MonadIO )
import Control.Tracer
( Tracer, traceWith )
import Data.Aeson
Expand Down Expand Up @@ -286,6 +301,8 @@ import System.Directory
, doesFileExist
, getXdgDirectory
)
import System.Environment
( lookupEnv )
import System.Exit
( exitFailure, exitSuccess )
import System.FilePath
Expand All @@ -309,8 +326,10 @@ import System.IO

import qualified Cardano.BM.Configuration.Model as CM
import qualified Cardano.BM.Data.BackendKind as CM
import qualified Cardano.BM.Data.Observable as Obs
import qualified Command.Key as Key
import qualified Command.RecoveryPhrase as RecoveryPhrase
import qualified Control.Concurrent.Async as Async
import qualified Data.Aeson as Aeson
import qualified Data.Aeson.Encode.Pretty as Aeson
import qualified Data.Aeson.Types as Aeson
Expand Down Expand Up @@ -1576,26 +1595,81 @@ mkScribeId :: LogOutput -> ScribeId
mkScribeId (LogToStdout _) = "StdoutSK::text"
mkScribeId (LogToFile file _) = T.pack $ "FileSK::" <> file

getPrometheusURL :: IO (Maybe (String, Port "Prometheus"))
getPrometheusURL = do
prometheus_port <- lookupEnv "CARDANO_WALLET_PROMETHEUS_PORT"
prometheus_host <- fromMaybe "localhost" <$> lookupEnv "CARDANO_WALLET_PROMETHEUS_HOST"
case (prometheus_host, prometheus_port) of
(host, Just port) ->
case fromText @(Port "Prometheus") $ T.pack port of
Right port' -> pure $ Just (host, port')
_ -> do
TIO.hPutStr stderr
"Port value for prometheus metrics invalid. Will be disabled."
pure Nothing
_ -> pure Nothing

getEKGURL :: IO (Maybe (String, Port "EKG"))
getEKGURL = do
ekg_port <- lookupEnv "CARDANO_WALLET_EKG_PORT"
ekg_host <- fromMaybe "localhost" <$> lookupEnv "CARDANO_WALLET_EKG_HOST"
case (ekg_host, ekg_port) of
(host, Just port) ->
case fromText @(Port "EKG") $ T.pack port of
Right port' -> pure $ Just (host, port')
_ -> do
TIO.hPutStr stderr
"Port value for EKB metrics invalid. Will be disabled."
pure Nothing
_ -> pure Nothing

-- | Initialize logging at the specified minimum 'Severity' level.
initTracer
:: LoggerName
-> [LogOutput]
-> IO (Switchboard Text, (CM.Configuration, Trace IO Text))
initTracer loggerName outputs = do
prometheusHP <- getPrometheusURL
ekgHP <- getEKGURL
cfg <- do
c <- defaultConfigStdout
CM.setSetupBackends c [CM.KatipBK, CM.AggregationBK]
CM.setSetupBackends c [CM.KatipBK, CM.AggregationBK, CM.EKGViewBK, CM.EditorBK]
CM.setDefaultBackends c [CM.KatipBK, CM.EKGViewBK]
CM.setSetupScribes c $ map mkScribe outputs
CM.setDefaultScribes c $ map mkScribeId outputs
forM_ ekgHP $ \(h, p) -> do
CM.setEKGBindAddr c $ Just (Endpoint (h, getPort p))
CM.setBackends c "cardano-wallet.metrics" (Just [CM.EKGViewBK])
forM_ prometheusHP $ \(h, p) ->
CM.setPrometheusBindAddr c $ Just (h, getPort p)
pure c
(tr, sb) <- setupTrace_ cfg loggerName
startCapturingMetrics tr
pure (sb, (cfg, tr))
where
-- https://github.com/input-output-hk/cardano-node/blob/f7d57e30c47028ba2aeb306a4f21b47bb41dec01/cardano-node/src/Cardano/Node/Configuration/Logging.hs#L224
startCapturingMetrics :: Trace IO Text -> IO ()
startCapturingMetrics trace0 = do
let trace = appendName "metrics" trace0
counters = [Obs.MemoryStats, Obs.ProcessStats
, Obs.NetStats, Obs.IOStats, Obs.GhcRtsStats, Obs.SysStats]
_ <- Async.async $ forever $ do
cts <- readCounters (ObservableTraceSelf counters)
traceCounters trace cts
threadDelay 30_000_000 -- 30 seconds
pure ()
where
traceCounters :: forall m a. MonadIO m => Trace m a -> [Counter] -> m ()
traceCounters _tr [] = return ()
traceCounters tr (c@(Counter _ct cn cv) : cs) = do
mle <- mkLOMeta Notice Confidential
traceNamedObject tr (mle, LogValue (nameCounter c <> "." <> cn) cv)
traceCounters tr cs

-- | See 'withLoggingNamed'
withLogging
:: [LogOutput]
-> ((CM.Configuration, Trace IO Text) -> IO a)
-> ((Switchboard Text, (CM.Configuration, Trace IO Text)) -> IO a)
-> IO a
withLogging =
withLoggingNamed "cardano-wallet"
Expand All @@ -1605,10 +1679,10 @@ withLogging =
withLoggingNamed
:: LoggerName
-> [LogOutput]
-> ((CM.Configuration, Trace IO Text) -> IO a)
-> ((Switchboard Text, (CM.Configuration, Trace IO Text)) -> IO a)
-- ^ The action to run with logging configured.
-> IO a
withLoggingNamed loggerName outputs action = bracket before after (action . snd)
withLoggingNamed loggerName outputs = bracket before after
where
before = initTracer loggerName outputs
after (sb, (_, tr)) = do
Expand Down
3 changes: 3 additions & 0 deletions lib/shelley/cardano-wallet.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -117,6 +117,7 @@ executable cardano-wallet
, cardano-wallet
, contra-tracer
, iohk-monitoring
, lobemo-backend-ekg
, network
, optparse-applicative
, text
Expand Down Expand Up @@ -147,6 +148,7 @@ executable shelley-test-cluster
, cardano-wallet
, contra-tracer
, iohk-monitoring
, lobemo-backend-ekg
, text
, text-class
hs-source-dirs:
Expand Down Expand Up @@ -233,6 +235,7 @@ test-suite integration
, hspec
, http-client
, iohk-monitoring
, lobemo-backend-ekg
, text
, text-class
build-tools:
Expand Down
12 changes: 10 additions & 2 deletions lib/shelley/exe/cardano-wallet.hs
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,8 @@ import Prelude

import Cardano.BM.Data.Severity
( Severity (..) )
import Cardano.BM.Plugin
( loadPlugin )
import Cardano.BM.Trace
( Trace, appendName, logDebug, logError, logInfo, logNotice )
import Cardano.CLI
Expand All @@ -44,6 +46,7 @@ import Cardano.CLI
, cmdWalletCreate
, databaseOption
, enableWindowsANSI
, getEKGURL
, helperTracing
, hostPreferenceOption
, listenOption
Expand Down Expand Up @@ -102,7 +105,7 @@ import Cardano.Wallet.Version
import Control.Applicative
( Const (..), optional )
import Control.Monad
( void )
( forM_, void )
import Control.Monad.Trans.Except
( runExceptT )
import Control.Tracer
Expand Down Expand Up @@ -134,6 +137,7 @@ import System.Environment
import System.Exit
( ExitCode (..), exitWith )

import qualified Cardano.BM.Backend.EKGView as EKG
import qualified Data.Text as T

{-------------------------------------------------------------------------------
Expand Down Expand Up @@ -293,14 +297,18 @@ withTracers
-> (Trace IO MainLog -> Tracers IO -> IO a)
-> IO a
withTracers logOpt action =
withLogging [LogToStdout (loggingMinSeverity logOpt)] $ \(_, tr) -> do
withLogging [LogToStdout (loggingMinSeverity logOpt)] $ \(sb, (cfg, tr)) -> do
ekgHP <- getEKGURL
forM_ ekgHP $ \_ ->
EKG.plugin cfg tr sb >>= loadPlugin sb
let trMain = appendName "main" (transformTextTrace tr)
let tracers = setupTracers (loggingTracers logOpt) tr
logInfo trMain $ MsgVersion version gitRevision
logInfo trMain =<< MsgCmdLine <$> getExecutablePath <*> getArgs
action trMain tracers



{-------------------------------------------------------------------------------
Options
-------------------------------------------------------------------------------}
Expand Down
40 changes: 29 additions & 11 deletions lib/shelley/exe/shelley-test-cluster.hs
Original file line number Diff line number Diff line change
Expand Up @@ -14,8 +14,10 @@ import Cardano.BM.Data.Severity
( Severity (..) )
import Cardano.BM.Data.Tracer
( HasPrivacyAnnotation (..), HasSeverityAnnotation (..) )
import Cardano.BM.Plugin
( loadPlugin )
import Cardano.CLI
( LogOutput (..), withLoggingNamed )
( LogOutput (..), Port, getEKGURL, getPrometheusURL, withLoggingNamed )
import Cardano.Startup
( withUtf8Encoding )
import Cardano.Wallet.Api.Types
Expand Down Expand Up @@ -54,7 +56,7 @@ import Cardano.Wallet.Shelley.Launch
import Control.Arrow
( first )
import Control.Monad
( void )
( forM_, void )
import Control.Tracer
( contramap, traceWith )
import Data.Proxy
Expand All @@ -68,6 +70,7 @@ import System.IO
import Test.Integration.Faucet
( genRewardAccounts, mirMnemonics, shelleyIntegrationTestFunds )

import qualified Cardano.BM.Backend.EKGView as EKG
import qualified Data.Text as T

-- |
Expand Down Expand Up @@ -208,8 +211,8 @@ main = do
poolConfigs <- poolConfigsFromEnv
withUtf8Encoding
$ withLoggingNamed "cardano-wallet" walletLogs
$ \(_, trWallet) -> withLoggingNamed "test-cluster" clusterLogs
$ \(_, trCluster) -> withSystemTempDir (trMessageText trCluster) "testCluster"
$ \(sb, (cfg, trWallet)) -> withLoggingNamed "test-cluster" clusterLogs
$ \(_, (_, trCluster)) -> withSystemTempDir (trMessageText trCluster) "testCluster"
$ \dir -> withTempDir (trMessageText trCluster) dir "wallets"
$ \db -> withCluster
(contramap MsgCluster $ trMessageText trCluster)
Expand All @@ -219,7 +222,7 @@ main = do
Nothing
whenByron
(whenShelley dir (trMessageText trCluster))
(whenReady trWallet (trMessageText trCluster) db)
(whenReady sb cfg trWallet (trMessageText trCluster) db)
where
whenByron _ = pure ()

Expand All @@ -233,9 +236,20 @@ main = do
sendFaucetFundsTo trCluster' dir addresses
moveInstantaneousRewardsTo trCluster' dir rewards

whenReady tr trCluster db (RunningNode socketPath block0 (gp, vData)) = do
whenReady sb cfg tr trCluster db (RunningNode socketPath block0 (gp, vData)) = do
ekgHP <- getEKGURL
forM_ ekgHP $ \_ ->
EKG.plugin cfg tr sb >>= loadPlugin sb
let tracers = setupTracers (tracerSeverities (Just Info)) tr
listen <- walletListenFromEnv
prometheusUrl <- (maybe "none"
(\(h, p) -> T.pack h <> ":" <> toText @(Port "Prometheus") p)
)
<$> getPrometheusURL
ekgUrl <- (maybe "none"
(\(h, p) -> T.pack h <> ":" <> toText @(Port "EKG") p)
)
<$> getEKGURL
void $ serveWallet @(IO Shelley)
(SomeNetworkDiscriminant $ Proxy @'Mainnet)
tracers
Expand All @@ -249,26 +263,30 @@ main = do
socketPath
block0
(gp, vData)
(traceWith trCluster . MsgBaseUrl . T.pack . show)
(\u -> traceWith trCluster $ MsgBaseUrl (T.pack . show $ u)
ekgUrl prometheusUrl)

-- Logging

data TestsLog
= MsgBaseUrl Text
= MsgBaseUrl Text Text Text -- wallet url, ekg url, prometheus url
| MsgSettingUpFaucet
| MsgCluster ClusterLog
deriving (Show)

instance ToText TestsLog where
toText = \case
MsgBaseUrl addr ->
"Wallet backend server listening on " <> T.pack (show addr)
MsgBaseUrl walletUrl ekgUrl prometheusUrl -> mconcat
[ "Wallet url: " , walletUrl
, ", EKG url: " , ekgUrl
, ", Prometheus url:", prometheusUrl
]
MsgSettingUpFaucet -> "Setting up faucet..."
MsgCluster msg -> toText msg

instance HasPrivacyAnnotation TestsLog
instance HasSeverityAnnotation TestsLog where
getSeverityAnnotation = \case
MsgSettingUpFaucet -> Notice
MsgBaseUrl _ -> Notice
MsgBaseUrl {} -> Notice
MsgCluster msg -> getSeverityAnnotation msg
Loading

0 comments on commit c1d1f77

Please sign in to comment.