Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add Prometheus-readable metrics endpoint #194

Merged
merged 10 commits into from
Jan 3, 2023
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
152 changes: 90 additions & 62 deletions app/Main.hs
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,8 @@ import Control.Applicative ((<**>))
import Control.Monad (forM, unless, void)
import Control.Monad.IO.Class (liftIO)
import Control.Monad.Logger (runStdoutLoggingT)
import Data.List (zip4)
import Data.Maybe (maybeToList)
import Data.String (fromString)
import Data.Version (showVersion)
import System.Exit (die)
import System.IO (BufferMode (LineBuffering), hSetBuffering, stderr, stdout)
Expand All @@ -24,12 +25,16 @@ import qualified GitHub.Auth as Github3
import qualified System.Directory as FileSystem
import qualified Options.Applicative as Opts

import Configuration (Configuration)
import Configuration (Configuration, MetricsConfiguration (metricsPort, metricsHost))
import EventLoop (runGithubEventLoop, runLogicEventLoop)
import Project (ProjectState, emptyProjectState, loadProjectState, saveProjectState)
import Project (ProjectInfo (ProjectInfo), Owner)
import Server (buildServer)
diegodiv marked this conversation as resolved.
Show resolved Hide resolved

import qualified Metrics.Metrics as Metrics
import Metrics.Server (runMetricsServer, MetricsServerConfig(MetricsServerConfig,
metricsConfigPort, metricsConfigHost))

import qualified Paths_hoff (version)

import qualified Configuration as Config
Expand Down Expand Up @@ -92,6 +97,11 @@ initializeProjectState fname = do
main :: IO ()
main = Opts.execParser commandLineParser >>= runMain

getProjectInfo :: Config.ProjectConfiguration -> ProjectInfo
getProjectInfo pconfig = ProjectInfo owner repository
where owner = Config.owner pconfig
repository = Config.repository pconfig

runMain :: Options -> IO ()
runMain options = do
-- When the runtime detects that stdout is not connected to a console, it
Expand All @@ -103,8 +113,8 @@ runMain options = do
hSetBuffering stderr LineBuffering

putStrLn $ "Starting Hoff v" ++ version
putStrLn $ "Config file: " ++ (configFilePath options)
putStrLn $ "Read-only: " ++ (show $ readOnly options)
putStrLn $ "Config file: " ++ configFilePath options
putStrLn $ "Read-only: " ++ show (readOnly options)

-- Load configuration from the file specified as first program argument.
config <- loadConfigOrExit $ configFilePath options
Expand All @@ -125,11 +135,7 @@ runMain options = do
-- up, so the server will reject new events).
projectQueues <- forM (Config.projects config) $ \ pconfig -> do
projectQueue <- Logic.newEventQueue 10
let
owner = Config.owner pconfig
repository = Config.repository pconfig
projectInfo = ProjectInfo owner repository
return (projectInfo, projectQueue)
return (getProjectInfo pconfig, projectQueue)

-- Define a function that enqueues an event in the right project queue.
let
Expand All @@ -147,12 +153,7 @@ runMain options = do
-- Restore the previous state from disk if possible, or start clean.
projectStates <- forM (Config.projects config) $ \ pconfig -> do
projectState <- initializeProjectState (Config.stateFile pconfig)
let
-- TODO: DRY.
owner = Config.owner pconfig
repository = Config.repository pconfig
projectInfo = ProjectInfo owner repository
return (projectInfo, projectState)
return (getProjectInfo pconfig, projectState)

-- Keep track of the most recent state for every project, so the webinterface
-- can use it to serve a status page.
Expand All @@ -162,24 +163,81 @@ runMain options = do

-- Start a main event loop for every project.
let
-- TODO: This is very, very ugly. Get these per-project collections sorted
-- out.
zipped = zip4 (Config.projects config) projectQueues stateVars projectStates
tuples = map (\(cfg, (_, a), (_, b), (_, c)) -> (cfg, a, b, c)) zipped
projectThreads <- forM tuples $ \ (projectConfig, projectQueue, stateVar, projectState) -> do
zipped = zip3 (Config.projects config) projectQueues stateVars
projectsThreadData = map (\(cfg, (_, a), (_, b)) -> ProjectThreadData cfg a b) zipped
metrics <- Metrics.registerProjectMetrics
projectThreads <- forM projectsThreadData $ projectThread config options metrics

let
-- When the webhook server receives an event, enqueue it on the webhook
-- event queue if it is not full.
ghTryEnqueue = Github.tryEnqueueEvent ghQueue

-- Allow the webinterface to retrieve the latest project state per project.
getProjectState projectInfo = Logic.readStateVar <$> lookup projectInfo stateVars
getOwnerState :: Owner -> IO [(ProjectInfo, ProjectState)]
getOwnerState owner = do
let states = filter (\(projectInfo, _) -> Project.owner projectInfo == owner) stateVars
mapM (\(info, state) -> Logic.readStateVar state >>= \sVar -> pure (info, sVar)) states

let
port = Config.port config
tlsConfig = Config.tls config
secret = Config.secret config
-- TODO: Do this in a cleaner way.
infos = getProjectInfo <$> Config.projects config
putStrLn $ "Listening for webhooks on port " ++ show port ++ "."
runServer <- fst <$> buildServer port tlsConfig infos secret ghTryEnqueue getProjectState getOwnerState
serverThread <- Async.async runServer
metricsThread <- runMetricsThread config

-- Note that a stop signal is never enqueued. The application just runs until
-- until it is killed, or until any of the threads stop due to an exception.
void $ Async.waitAny $ [serverThread, ghThread] ++ metricsThread ++ projectThreads

data ProjectThreadData = ProjectThreadData
{ projectThreadConfig :: Config.ProjectConfiguration
, projectThreadQueue :: Logic.EventQueue
, projectThreadStateVar :: Logic.StateVar
}

projectThread :: Configuration
diegodiv marked this conversation as resolved.
Show resolved Hide resolved
-> Options
-> Metrics.ProjectMetrics
-> ProjectThreadData
-> IO (Async.Async ())
projectThread config options metrics projectThreadData = do
-- At startup, enqueue a synchronize event. This will bring the state in
-- sync with the current state of GitHub, accounting for any webhooks that
-- we missed while not running, or just to fill the state initially after
-- setting up a new project.
liftIO $ Logic.enqueueEvent projectQueue Logic.Synchronize

let
projectThreadState <- Logic.readStateVar $ projectThreadStateVar projectThreadData
-- Start a worker thread to run the main event loop for the project.
Async.async
$ void
$ runStdoutLoggingT
$ Metrics.runLoggingMonitorT
$ runLogicEventLoop
(Config.trigger config)
projectConfig
(Config.mergeWindowExemption config)
runMetrics
runTime
runGit
runGithub
getNextEvent
publish
projectThreadState
where
-- When the event loop publishes the current project state, save it to
-- the configured file, and make the new state available to the
-- webinterface.
projectConfig = projectThreadConfig projectThreadData
projectQueue = projectThreadQueue projectThreadData
publish newState = do
liftIO $ saveProjectState (Config.stateFile projectConfig) newState
liftIO $ Logic.updateStateVar stateVar newState
liftIO $ Logic.updateStateVar (projectThreadStateVar projectThreadData) newState

-- When the event loop wants to get the next event, take one off the queue.
getNextEvent = liftIO $ Logic.dequeueEvent projectQueue
Expand All @@ -197,44 +255,14 @@ runMain options = do
then GithubApi.runGithubReadOnly auth projectInfo
else GithubApi.runGithub auth projectInfo
runTime = Time.runTime
-- Start a worker thread to run the main event loop for the project.
Async.async
$ void
$ runStdoutLoggingT
$ runLogicEventLoop
(Config.trigger config)
projectConfig
(Config.mergeWindowExemption config)
runTime
runGit
runGithub
getNextEvent
publish
projectState
runMetrics = Metrics.runMetrics metrics $ Config.repository projectConfig

let
-- When the webhook server receives an event, enqueue it on the webhook
-- event queue if it is not full.
ghTryEnqueue = Github.tryEnqueueEvent ghQueue

-- Allow the webinterface to retrieve the latest project state per project.
getProjectState projectInfo =
fmap Logic.readStateVar $ lookup projectInfo stateVars
getOwnerState :: Owner -> IO [(ProjectInfo, ProjectState)]
getOwnerState owner = do
let states = filter (\(projectInfo, _) -> Project.owner projectInfo == owner) stateVars
mapM (\(info, state) -> Logic.readStateVar state >>= \sVar -> pure (info, sVar)) states

let
port = Config.port config
tlsConfig = Config.tls config
secret = Config.secret config
-- TODO: Do this in a cleaner way.
infos = fmap (\ pc -> ProjectInfo (Config.owner pc) (Config.repository pc)) $ Config.projects config
putStrLn $ "Listening for webhooks on port " ++ (show port) ++ "."
runServer <- fmap fst $ buildServer port tlsConfig infos secret ghTryEnqueue getProjectState getOwnerState
serverThread <- Async.async runServer

-- Note that a stop signal is never enqueued. The application just runs until
-- until it is killed, or until any of the threads stop due to an exception.
void $ Async.waitAny $ serverThread : ghThread : projectThreads
runMetricsThread :: Configuration -> IO [Async.Async ()]
runMetricsThread configuration =
forM (maybeToList $ Config.metricsConfig configuration) $
\metricsConf -> do
let servConfig = MetricsServerConfig
{ metricsConfigPort = metricsPort metricsConf
, metricsConfigHost = fromString $ metricsHost metricsConf }
Async.async $ runMetricsServer servConfig
1 change: 1 addition & 0 deletions default.nix
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ in
pkgs.dpkg
pkgs.git
pkgs.haskellPackages.haskell-language-server
pkgs.haskellPackages.stylish-haskell
pkgs.niv
pkgs.shellcheck
pkgs.stack
Expand Down
4 changes: 4 additions & 0 deletions doc/example-dev-config.json
Original file line number Diff line number Diff line change
Expand Up @@ -19,5 +19,9 @@
"mergeWindowExemption": ["hoffbot"],
"trigger": {
"commentPrefix": "@hoffbot"
},
"metrics": {
"metricsPort": 3333,
"metricsHost": "*"
}
}
4 changes: 4 additions & 0 deletions hoff.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,8 @@ library
, Logic
, Project
, Server
, Metrics.Server
, Metrics.Metrics
, Time
, Types
, WebInterface
Expand All @@ -53,13 +55,15 @@ library
, monad-logger
, process
, process-extras
, prometheus-client
, scotty
, stm
, text
, text-format
, time
, vector
, wai
, wai-middleware-prometheus
, warp
, warp-tls
other-modules: Paths_hoff
Expand Down
2 changes: 2 additions & 0 deletions nix/haskell-dependencies.nix
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ haskellPackages: with haskellPackages; [
optparse-applicative
process
process-extras
prometheus
scotty
stm
text
Expand All @@ -37,6 +38,7 @@ haskellPackages: with haskellPackages; [
uuid
vector
wai
wai-middleware-prometheus
warp
warp-tls
]
15 changes: 14 additions & 1 deletion src/Configuration.hs
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ module Configuration
TriggerConfiguration (..),
UserConfiguration (..),
MergeWindowExemptionConfiguration (..),
MetricsConfiguration (..),
loadConfiguration
)
where
Expand All @@ -24,6 +25,7 @@ import Data.ByteString (readFile)
import Data.Text (Text)
import GHC.Generics
import Prelude hiding (readFile)
import qualified Network.Wai.Handler.Warp as Warp

data ProjectConfiguration = ProjectConfiguration
{
Expand Down Expand Up @@ -62,6 +64,13 @@ data TlsConfiguration = TlsConfiguration
}
deriving (Generic, Show)

data MetricsConfiguration = MetricsConfiguration
{
metricsPort :: Warp.Port,
metricsHost :: String
diegodiv marked this conversation as resolved.
Show resolved Hide resolved
}
deriving (Generic, Show)

newtype MergeWindowExemptionConfiguration = MergeWindowExemptionConfiguration [Text]
deriving (Generic, Show)

Expand Down Expand Up @@ -95,7 +104,10 @@ data Configuration = Configuration

-- List of users that are exempted from the merge window. This is useful for
-- bots that automatically merge low impact changes.
mergeWindowExemption :: MergeWindowExemptionConfiguration
mergeWindowExemption :: MergeWindowExemptionConfiguration,

-- Configuration for the Prometheus metrics server.
metricsConfig :: Maybe MetricsConfiguration
}
deriving (Generic)

Expand All @@ -105,6 +117,7 @@ instance FromJSON TlsConfiguration
instance FromJSON TriggerConfiguration
instance FromJSON UserConfiguration
instance FromJSON MergeWindowExemptionConfiguration
instance FromJSON MetricsConfiguration

-- Reads and parses the configuration. Returns Nothing if parsing failed, but
-- crashes if the file could not be read.
Expand Down
8 changes: 6 additions & 2 deletions src/EventLoop.hs
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ import Control.Monad.STM (atomically)
import Control.Monad.Free (foldFree)
import Data.Foldable (traverse_)
import Data.Functor.Sum (Sum (InL, InR))
import Prometheus (MonadMonitor)

import Data.Text (Text)
import qualified Data.Text as Text
Expand All @@ -42,6 +43,7 @@ import qualified Github
import qualified GithubApi
import qualified Logic
import qualified Project
import qualified Metrics.Metrics as Metrics

eventFromPullRequestPayload :: PullRequestPayload -> Logic.Event
eventFromPullRequestPayload payload =
Expand Down Expand Up @@ -128,10 +130,12 @@ runSum runF runG = go
runLogicEventLoop
:: MonadIO m
=> MonadLogger m
=> MonadMonitor m
=> TriggerConfiguration
-> ProjectConfiguration
-> MergeWindowExemptionConfiguration
-- Interpreters for Git and GitHub actions.
-> (forall a. Metrics.MetricsOperationFree a -> m a)
-> (forall a. Time.TimeOperationFree a -> m a)
-> (forall a. Git.GitOperationFree a -> m a)
-> (forall a. GithubApi.GithubOperationFree a -> m a)
Expand All @@ -145,10 +149,10 @@ runLogicEventLoop
-> m ProjectState
runLogicEventLoop
triggerConfig projectConfig mergeWindowExemptionConfig
runTime runGit runGithub
runMetrics runTime runGit runGithub
getNextEvent publish initialState =
let
runAll = foldFree (runSum runTime (runSum runGit runGithub))
runAll = foldFree (runSum (runSum runMetrics runTime) (runSum runGit runGithub))
runAction = Logic.runAction projectConfig

handleAndContinue state0 event = do
Expand Down
Loading