2021-05-04 16:16:00 +00:00
#! / usr / bin / env nix - shell
#! nix - shell - p " haskellPackages.ghcWithPackages (p: [p.aeson p.req]) "
2022-05-20 19:27:31 +00:00
#! nix - shell - p hydra - unstable
2021-05-04 16:16:00 +00:00
#! nix - shell - i runhaskell
{-
The purpose of this script is
1 ) download the state of the nixpkgs / haskell - updates job from hydra ( with get - report )
2 ) print a summary of the state suitable for pasting into a github comment ( with ping - maintainers )
3 ) print a list of broken packages suitable for pasting into configuration - hackage2nix . yaml
2021-05-08 05:15:20 +00:00
Because step 1 ) is quite expensive and takes roughly ~ 5 minutes the result is cached in a json file in XDG_CACHE .
2021-05-04 16:16:00 +00:00
- }
{- # LANGUAGE BlockArguments # -}
{- # LANGUAGE DeriveAnyClass # -}
{- # LANGUAGE DeriveGeneric # -}
2021-05-15 06:53:19 +00:00
{- # LANGUAGE DerivingStrategies # -}
2021-05-04 16:16:00 +00:00
{- # LANGUAGE DuplicateRecordFields # -}
{- # LANGUAGE LambdaCase # -}
{- # LANGUAGE NamedFieldPuns # -}
{- # LANGUAGE OverloadedStrings # -}
{- # LANGUAGE ScopedTypeVariables # -}
{- # LANGUAGE TupleSections # -}
2021-09-09 22:55:12 +00:00
{- # LANGUAGE ViewPatterns # -}
2022-05-20 19:41:30 +00:00
{- # OPTIONS_GHC - Wall # -}
2023-02-19 23:20:42 +00:00
{- # LANGUAGE DataKinds # -}
2021-05-04 16:16:00 +00:00
2021-05-08 22:37:05 +00:00
import Control.Monad ( forM_ , ( <=< ) )
2021-05-04 16:16:00 +00:00
import Control.Monad.Trans ( MonadIO ( liftIO ) )
import Data.Aeson (
FromJSON ,
ToJSON ,
decodeFileStrict' ,
eitherDecodeStrict' ,
encodeFile ,
)
2021-05-08 22:37:05 +00:00
import Data.Foldable ( Foldable ( toList ) , foldl' )
2021-05-04 16:16:00 +00:00
import Data.List.NonEmpty ( NonEmpty , nonEmpty )
import qualified Data.List.NonEmpty as NonEmpty
import Data.Map.Strict ( Map )
import qualified Data.Map.Strict as Map
2021-09-18 17:39:26 +00:00
import Data.Maybe ( fromMaybe , mapMaybe , isNothing )
2021-05-04 16:16:00 +00:00
import Data.Monoid ( Sum ( Sum , getSum ) )
import Data.Sequence ( Seq )
import qualified Data.Sequence as Seq
import Data.Set ( Set )
import qualified Data.Set as Set
import Data.Text ( Text )
import qualified Data.Text as Text
import Data.Text.Encoding ( encodeUtf8 )
import Data.Time ( defaultTimeLocale , formatTime , getCurrentTime )
import Data.Time.Clock ( UTCTime )
import GHC.Generics ( Generic )
import Network.HTTP.Req (
2023-02-19 23:20:42 +00:00
GET ( GET ) ,
HttpResponse ( HttpResponseBody ) ,
NoReqBody ( NoReqBody ) ,
Option ,
Req ,
Scheme ( Https ) ,
bsResponse ,
defaultHttpConfig ,
header ,
https ,
jsonResponse ,
req ,
responseBody ,
responseTimeout ,
runReq ,
( /: ) ,
2021-05-04 16:16:00 +00:00
)
import System.Directory ( XdgDirectory ( XdgCache ) , getXdgDirectory )
import System.Environment ( getArgs )
import System.Process ( readProcess )
2021-05-08 22:37:05 +00:00
import Prelude hiding ( id )
2021-09-09 22:55:12 +00:00
import Data.List ( sortOn )
import Control.Concurrent.Async ( concurrently )
import Control.Exception ( evaluate )
import qualified Data.IntMap.Strict as IntMap
import qualified Data.IntSet as IntSet
import Data.Bifunctor ( second )
2023-02-19 23:20:42 +00:00
import Data.Data ( Proxy )
import Data.ByteString ( ByteString )
import qualified Data.ByteString.Char8 as ByteString
import Distribution.Simple.Utils ( safeLast , fromUTF8BS )
2021-05-04 16:16:00 +00:00
newtype JobsetEvals = JobsetEvals
{ evals :: Seq Eval
}
deriving ( Generic , ToJSON , FromJSON , Show )
newtype Nixpkgs = Nixpkgs { revision :: Text }
deriving ( Generic , ToJSON , FromJSON , Show )
newtype JobsetEvalInputs = JobsetEvalInputs { nixpkgs :: Nixpkgs }
deriving ( Generic , ToJSON , FromJSON , Show )
data Eval = Eval
2021-05-08 05:15:20 +00:00
{ id :: Int
2021-05-04 16:16:00 +00:00
, jobsetevalinputs :: JobsetEvalInputs
}
deriving ( Generic , ToJSON , FromJSON , Show )
data Build = Build
{ job :: Text
2021-05-08 05:13:20 +00:00
, buildstatus :: Maybe Int
2021-05-04 16:16:00 +00:00
, finished :: Int
, id :: Int
, nixname :: Text
, system :: Text
, jobsetevals :: Seq Int
}
deriving ( Generic , ToJSON , FromJSON , Show )
main :: IO ()
main = do
args <- getArgs
case args of
[ " get-report " ] -> getBuildReports
[ " ping-maintainers " ] -> printMaintainerPing
[ " mark-broken-list " ] -> printMarkBrokenList
2023-01-09 23:34:51 +00:00
[ " eval-info " ] -> printEvalInfo
_ -> putStrLn " Usage: get-report | ping-maintainers | mark-broken-list | eval-info "
2021-05-04 16:16:00 +00:00
reportFileName :: IO FilePath
reportFileName = getXdgDirectory XdgCache " haskell-updates-build-report.json "
showT :: Show a => a -> Text
showT = Text . pack . show
getBuildReports :: IO ()
2021-05-08 05:15:20 +00:00
getBuildReports = runReq defaultHttpConfig do
2023-02-19 23:20:42 +00:00
evalMay <- Seq . lookup 0 . evals <$> hydraJSONQuery mempty [ " jobset " , " nixpkgs " , " haskell-updates " , " evals " ]
2021-05-08 05:15:20 +00:00
eval @ Eval { id } <- maybe ( liftIO $ fail " No Evalution found " ) pure evalMay
liftIO . putStrLn $ " Fetching evaluation " <> show id <> " from Hydra. This might take a few minutes... "
2023-02-19 23:20:42 +00:00
buildReports :: Seq Build <- hydraJSONQuery ( responseTimeout 600000000 ) [ " eval " , showT id , " builds " ]
2021-05-08 05:15:20 +00:00
liftIO do
fileName <- reportFileName
putStrLn $ " Finished fetching all builds from Hydra, saving report as " <> fileName
now <- getCurrentTime
encodeFile fileName ( eval , now , buildReports )
2023-02-19 23:20:42 +00:00
hydraQuery :: HttpResponse a => Proxy a -> Option 'Https -> [ Text ] -> Req ( HttpResponseBody a )
hydraQuery responseType option query =
responseBody
<$> req
GET
( foldl' ( /: ) ( https " hydra.nixos.org " ) query )
NoReqBody
responseType
( header " User-Agent " " hydra-report.hs/v1 (nixpkgs;maintainers/scripts/haskell) " <> option )
hydraJSONQuery :: FromJSON a => Option 'Https -> [ Text ] -> Req a
hydraJSONQuery = hydraQuery jsonResponse
hydraPlainQuery :: [ Text ] -> Req ByteString
hydraPlainQuery = hydraQuery bsResponse mempty
2021-05-04 16:16:00 +00:00
2021-05-08 22:37:05 +00:00
hydraEvalCommand :: FilePath
2021-05-04 16:16:00 +00:00
hydraEvalCommand = " hydra-eval-jobs "
2021-05-15 06:53:19 +00:00
2021-05-08 22:37:05 +00:00
hydraEvalParams :: [ String ]
2021-05-04 16:16:00 +00:00
hydraEvalParams = [ " -I " , " . " , " pkgs/top-level/release-haskell.nix " ]
2021-05-15 06:53:19 +00:00
2021-09-09 22:55:12 +00:00
nixExprCommand :: FilePath
nixExprCommand = " nix-instantiate "
2021-05-15 06:53:19 +00:00
2021-09-09 22:55:12 +00:00
nixExprParams :: [ String ]
nixExprParams = [ " --eval " , " --strict " , " --json " ]
2021-05-04 16:16:00 +00:00
2021-05-16 01:43:39 +00:00
-- | This newtype is used to parse a Hydra job output from @hydra-eval-jobs@.
-- The only field we are interested in is @maintainers@, which is why this
-- is just a newtype.
--
2021-09-09 22:55:12 +00:00
-- Note that there are occasionally jobs that don't have a maintainers
2021-05-16 01:43:39 +00:00
-- field, which is why this has to be @Maybe Text@.
newtype Maintainers = Maintainers { maintainers :: Maybe Text }
deriving stock ( Generic , Show )
2021-05-15 06:53:19 +00:00
deriving anyclass ( FromJSON , ToJSON )
-- | This is a 'Map' from Hydra job name to maintainer email addresses.
--
-- It has values similar to the following:
--
-- @@
-- fromList
2021-05-16 01:43:39 +00:00
-- [ ("arion.aarch64-linux", Maintainers (Just "robert@example.com"))
-- , ("bench.x86_64-linux", Maintainers (Just ""))
-- , ("conduit.x86_64-linux", Maintainers (Just "snoy@man.com, web@ber.com"))
-- , ("lens.x86_64-darwin", Maintainers (Just "ek@category.com"))
2021-05-15 06:53:19 +00:00
-- ]
-- @@
--
-- Note that Hydra jobs without maintainers will have an empty string for the
-- maintainer list.
2021-05-04 16:16:00 +00:00
type HydraJobs = Map Text Maintainers
2021-05-15 06:53:19 +00:00
-- | Map of email addresses to GitHub handles.
-- This is built from the file @../../maintainer-list.nix@.
--
-- It has values similar to the following:
--
-- @@
-- fromList
-- [ ("robert@example.com", "rob22")
-- , ("ek@category.com", "edkm")
-- ]
-- @@
type EmailToGitHubHandles = Map Text Text
-- | Map of Hydra jobs to maintainer GitHub handles.
--
-- It has values similar to the following:
--
-- @@
-- fromList
-- [ ("arion.aarch64-linux", ["rob22"])
-- , ("conduit.x86_64-darwin", ["snoyb", "webber"])
-- ]
-- @@
2021-05-04 16:16:00 +00:00
type MaintainerMap = Map Text ( NonEmpty Text )
2021-09-09 22:55:12 +00:00
-- | Information about a package which lists its dependencies and whether the
-- package is marked broken.
data DepInfo = DepInfo {
deps :: Set Text ,
broken :: Bool
}
deriving stock ( Generic , Show )
deriving anyclass ( FromJSON , ToJSON )
-- | Map from package names to their DepInfo. This is the data we get out of a
-- nix call.
type DependencyMap = Map Text DepInfo
-- | Map from package names to its broken state, number of reverse dependencies (fst) and
-- unbroken reverse dependencies (snd).
type ReverseDependencyMap = Map Text ( Int , Int )
-- | Calculate the (unbroken) reverse dependencies of a package by transitively
-- going through all packages if it’ s a dependency of them.
calculateReverseDependencies :: DependencyMap -> ReverseDependencyMap
calculateReverseDependencies depMap = Map . fromDistinctAscList $ zip keys ( zip ( rdepMap False ) ( rdepMap True ) )
where
-- This code tries to efficiently invert the dependency map and calculate
-- it’ s transitive closure by internally identifying every pkg with it’ s index
-- in the package list and then using memoization.
keys = Map . keys depMap
pkgToIndexMap = Map . fromDistinctAscList ( zip keys [ 0 .. ] )
intDeps = zip [ 0 .. ] $ ( \ DepInfo { broken , deps } -> ( broken , mapMaybe ( ` Map . lookup ` pkgToIndexMap ) $ Set . toList deps ) ) <$> Map . elems depMap
rdepMap onlyUnbroken = IntSet . size <$> resultList
where
resultList = go <$> [ 0 .. ]
oneStepMap = IntMap . fromListWith IntSet . union $ ( \ ( key , ( _ , deps ) ) -> ( , IntSet . singleton key ) <$> deps ) <=< filter ( \ ( _ , ( broken , _ ) ) -> not ( broken && onlyUnbroken ) ) $ intDeps
go pkg = IntSet . unions ( oneStep : ( ( resultList !! ) <$> IntSet . toList oneStep ) )
where oneStep = IntMap . findWithDefault mempty pkg oneStepMap
-- | Generate a mapping of Hydra job names to maintainer GitHub handles. Calls
-- hydra-eval-jobs and the nix script ./maintainer-handles.nix.
2021-05-04 16:16:00 +00:00
getMaintainerMap :: IO MaintainerMap
getMaintainerMap = do
2021-05-15 06:53:19 +00:00
hydraJobs :: HydraJobs <-
2021-09-09 22:55:12 +00:00
readJSONProcess hydraEvalCommand hydraEvalParams " Failed to decode hydra-eval-jobs output: "
2021-05-15 06:53:19 +00:00
handlesMap :: EmailToGitHubHandles <-
2021-09-09 22:55:12 +00:00
readJSONProcess nixExprCommand ( " maintainers/scripts/haskell/maintainer-handles.nix " : nixExprParams ) " Failed to decode nix output for lookup of github handles: "
2021-05-16 01:43:39 +00:00
pure $ Map . mapMaybe ( splitMaintainersToGitHubHandles handlesMap ) hydraJobs
2023-03-24 11:26:01 +00:00
where
2021-05-16 01:43:39 +00:00
-- Split a comma-spearated string of Maintainers into a NonEmpty list of
-- GitHub handles.
splitMaintainersToGitHubHandles
:: EmailToGitHubHandles -> Maintainers -> Maybe ( NonEmpty Text )
splitMaintainersToGitHubHandles handlesMap ( Maintainers maint ) =
nonEmpty . mapMaybe ( ` Map . lookup ` handlesMap ) . Text . splitOn " , " $ fromMaybe " " maint
2021-05-15 06:53:19 +00:00
2021-09-09 22:55:12 +00:00
-- | Get the a map of all dependencies of every package by calling the nix
-- script ./dependencies.nix.
getDependencyMap :: IO DependencyMap
getDependencyMap =
readJSONProcess nixExprCommand ( " maintainers/scripts/haskell/dependencies.nix " : nixExprParams ) " Failed to decode nix output for lookup of dependencies: "
2021-05-15 06:53:19 +00:00
-- | Run a process that produces JSON on stdout and and decode the JSON to a
-- data type.
--
-- If the JSON-decoding fails, throw the JSON-decoding error.
readJSONProcess
:: FromJSON a
=> FilePath -- ^ Filename of executable.
-> [ String ] -- ^ Arguments
-> String -- ^ String to prefix to JSON-decode error.
-> IO a
2021-09-09 22:55:12 +00:00
readJSONProcess exe args err = do
output <- readProcess exe args " "
2021-05-15 06:53:19 +00:00
let eitherDecodedOutput = eitherDecodeStrict' . encodeUtf8 . Text . pack $ output
case eitherDecodedOutput of
Left decodeErr -> error $ err <> decodeErr <> " \ n Raw: ' " <> take 1000 output <> " ' "
Right decodedOutput -> pure decodedOutput
2021-05-04 16:16:00 +00:00
-- BuildStates are sorted by subjective importance/concerningness
2021-05-15 06:53:19 +00:00
data BuildState
= Failed
| DependencyFailed
| OutputLimitExceeded
| Unknown ( Maybe Int )
| TimedOut
| Canceled
| HydraFailure
| Unfinished
| Success
deriving stock ( Show , Eq , Ord )
2021-05-04 16:16:00 +00:00
icon :: BuildState -> Text
icon = \ case
Failed -> " :x: "
DependencyFailed -> " :heavy_exclamation_mark: "
OutputLimitExceeded -> " :warning: "
Unknown x -> " unknown code " <> showT x
2021-05-09 00:50:19 +00:00
TimedOut -> " :hourglass::no_entry_sign: "
Canceled -> " :no_entry_sign: "
2021-05-08 22:37:05 +00:00
Unfinished -> " :hourglass_flowing_sand: "
2021-05-11 00:27:49 +00:00
HydraFailure -> " :construction: "
2021-05-04 16:16:00 +00:00
Success -> " :heavy_check_mark: "
platformIcon :: Platform -> Text
platformIcon ( Platform x ) = case x of
" x86_64-linux " -> " :penguin: "
" aarch64-linux " -> " :iphone: "
" x86_64-darwin " -> " :apple: "
2023-03-10 13:43:00 +00:00
" aarch64-darwin " -> " :green_apple: "
2021-05-04 16:16:00 +00:00
_ -> x
data BuildResult = BuildResult { state :: BuildState , id :: Int } deriving ( Show , Eq , Ord )
newtype Platform = Platform { platform :: Text } deriving ( Show , Eq , Ord )
newtype Table row col a = Table ( Map ( row , col ) a )
2021-09-09 22:55:12 +00:00
data SummaryEntry = SummaryEntry {
summaryBuilds :: Table Text Platform BuildResult ,
summaryMaintainers :: Set Text ,
summaryReverseDeps :: Int ,
summaryUnbrokenReverseDeps :: Int
}
type StatusSummary = Map Text SummaryEntry
2021-05-04 16:16:00 +00:00
2023-03-24 11:55:52 +00:00
tableSingleton :: row -> col -> a -> Table row col a
tableSingleton row col a = Table ( Map . singleton ( row , col ) a )
2021-05-04 16:16:00 +00:00
instance ( Ord row , Ord col , Semigroup a ) => Semigroup ( Table row col a ) where
Table l <> Table r = Table ( Map . unionWith ( <> ) l r )
instance ( Ord row , Ord col , Semigroup a ) => Monoid ( Table row col a ) where
mempty = Table Map . empty
instance Functor ( Table row col ) where
fmap f ( Table a ) = Table ( fmap f a )
instance Foldable ( Table row col ) where
foldMap f ( Table a ) = foldMap f a
2023-02-19 23:20:42 +00:00
getBuildState :: Build -> BuildState
getBuildState Build { finished , buildstatus } = case ( finished , buildstatus ) of
( 0 , _ ) -> Unfinished
( _ , Just 0 ) -> Success
( _ , Just 1 ) -> Failed
( _ , Just 2 ) -> DependencyFailed
( _ , Just 3 ) -> HydraFailure
( _ , Just 4 ) -> Canceled
( _ , Just 7 ) -> TimedOut
( _ , Just 11 ) -> OutputLimitExceeded
( _ , i ) -> Unknown i
2021-09-09 22:55:12 +00:00
buildSummary :: MaintainerMap -> ReverseDependencyMap -> Seq Build -> StatusSummary
2023-03-24 11:55:52 +00:00
buildSummary maintainerMap reverseDependencyMap =
foldl ( Map . unionWith unionSummary ) Map . empty . fmap toSummary
2021-05-04 16:16:00 +00:00
where
2023-03-24 11:55:52 +00:00
unionSummary :: SummaryEntry -> SummaryEntry -> SummaryEntry
unionSummary ( SummaryEntry ( Table lb ) lm lr lu ) ( SummaryEntry ( Table rb ) rm rr ru ) =
SummaryEntry ( Table $ Map . union lb rb ) ( lm <> rm ) ( max lr rr ) ( max lu ru )
toSummary :: Build -> StatusSummary
toSummary build @ Build { job , id , system } = Map . singleton name summaryEntry
2021-05-04 16:16:00 +00:00
where
2023-03-24 11:55:52 +00:00
packageName :: Text
2021-05-04 16:16:00 +00:00
packageName = fromMaybe job ( Text . stripSuffix ( " . " <> system ) job )
2023-03-24 11:55:52 +00:00
splitted :: Maybe ( NonEmpty Text )
2021-05-04 16:16:00 +00:00
splitted = nonEmpty $ Text . splitOn " . " packageName
2023-03-24 11:55:52 +00:00
name :: Text
2021-05-04 16:16:00 +00:00
name = maybe packageName NonEmpty . last splitted
2023-03-24 11:55:52 +00:00
set :: Text
2021-05-04 16:16:00 +00:00
set = maybe " " ( Text . intercalate " . " . NonEmpty . init ) splitted
2023-03-24 11:55:52 +00:00
maintainers :: Set Text
2021-05-04 16:16:00 +00:00
maintainers = maybe mempty ( Set . fromList . toList ) ( Map . lookup job maintainerMap )
2023-03-24 11:55:52 +00:00
2021-09-09 22:55:12 +00:00
( reverseDeps , unbrokenReverseDeps ) = Map . findWithDefault ( 0 , 0 ) name reverseDependencyMap
2021-05-04 16:16:00 +00:00
2023-03-24 11:55:52 +00:00
buildTable :: Table Text Platform BuildResult
buildTable =
tableSingleton set ( Platform system ) ( BuildResult ( getBuildState build ) id )
summaryEntry = SummaryEntry buildTable maintainers reverseDeps unbrokenReverseDeps
2021-05-04 16:16:00 +00:00
readBuildReports :: IO ( Eval , UTCTime , Seq Build )
readBuildReports = do
file <- reportFileName
fromMaybe ( error $ " Could not decode " <> file ) <$> decodeFileStrict' file
sep :: Text
sep = " | "
joinTable :: [ Text ] -> Text
joinTable t = sep <> Text . intercalate sep t <> sep
type NumSummary = Table Platform BuildState Int
printTable :: ( Ord rows , Ord cols ) => Text -> ( rows -> Text ) -> ( cols -> Text ) -> ( entries -> Text ) -> Table rows cols entries -> [ Text ]
printTable name showR showC showE ( Table mapping ) = joinTable <$> ( name : map showC cols ) : replicate ( length cols + sepsInName + 1 ) " --- " : map printRow rows
where
sepsInName = Text . count " | " name
printRow row = showR row : map ( \ col -> maybe " " showE ( Map . lookup ( row , col ) mapping ) ) cols
rows = toList $ Set . fromList ( fst <$> Map . keys mapping )
cols = toList $ Set . fromList ( snd <$> Map . keys mapping )
2021-05-09 13:13:07 +00:00
printJob :: Int -> Text -> ( Table Text Platform BuildResult , Text ) -> [ Text ]
printJob evalId name ( Table mapping , maintainers ) =
2021-05-04 16:16:00 +00:00
if length sets <= 1
then map printSingleRow sets
2021-05-09 13:13:07 +00:00
else [ " - [ ] " <> makeJobSearchLink " " name <> " " <> maintainers ] <> map printRow sets
2021-05-04 16:16:00 +00:00
where
2021-05-09 13:13:07 +00:00
printRow set = " - " <> printState set <> " " <> makeJobSearchLink set ( if Text . null set then " toplevel " else set )
printSingleRow set = " - [ ] " <> printState set <> " " <> makeJobSearchLink set ( makePkgName set ) <> " " <> maintainers
makePkgName set = ( if Text . null set then " " else set <> " . " ) <> name
2021-05-04 16:16:00 +00:00
printState set = Text . intercalate " " $ map ( \ pf -> maybe " " ( label pf ) $ Map . lookup ( set , pf ) mapping ) platforms
2021-05-16 18:20:27 +00:00
makeJobSearchLink set linkLabel = makeSearchLink evalId linkLabel ( makePkgName set )
2021-05-04 16:16:00 +00:00
sets = toList $ Set . fromList ( fst <$> Map . keys mapping )
platforms = toList $ Set . fromList ( snd <$> Map . keys mapping )
label pf ( BuildResult s i ) = " [[ " <> platformIcon pf <> icon s <> " ]](https://hydra.nixos.org/build/ " <> showT i <> " ) "
2021-05-09 13:13:07 +00:00
makeSearchLink :: Int -> Text -> Text -> Text
makeSearchLink evalId linkLabel query = " [ " <> linkLabel <> " ]( " <> " https://hydra.nixos.org/eval/ " <> showT evalId <> " ?filter= " <> query <> " ) "
2021-05-04 16:16:00 +00:00
statusToNumSummary :: StatusSummary -> NumSummary
statusToNumSummary = fmap getSum . foldMap ( fmap Sum . jobTotals )
2021-09-09 22:55:12 +00:00
jobTotals :: SummaryEntry -> Table Platform BuildState Int
jobTotals ( summaryBuilds -> Table mapping ) = getSum <$> Table ( Map . foldMapWithKey ( \ ( _ , platform ) ( BuildResult buildstate _ ) -> Map . singleton ( platform , buildstate ) ( Sum 1 ) ) mapping )
2021-05-04 16:16:00 +00:00
details :: Text -> [ Text ] -> [ Text ]
details summary content = [ " <details><summary> " <> summary <> " </summary> " , " " ] <> content <> [ " </details> " , " " ]
2023-01-09 23:34:51 +00:00
evalLine :: Eval -> UTCTime -> Text
evalLine Eval { id , jobsetevalinputs = JobsetEvalInputs { nixpkgs = Nixpkgs { revision } } } fetchTime =
2023-03-24 11:26:01 +00:00
" *evaluation [ "
<> showT id
<> " ](https://hydra.nixos.org/eval/ "
<> showT id
<> " ) of nixpkgs commit [ "
<> Text . take 7 revision
<> " ](https://github.com/NixOS/nixpkgs/commits/ "
<> revision
<> " ) as of "
<> Text . pack ( formatTime defaultTimeLocale " %Y-%m-%d %H:%M UTC " fetchTime )
<> " * "
2023-01-09 23:34:51 +00:00
2021-09-09 22:55:12 +00:00
printBuildSummary :: Eval -> UTCTime -> StatusSummary -> [ ( Text , Int ) ] -> Text
2023-01-09 23:34:51 +00:00
printBuildSummary eval @ Eval { id } fetchTime summary topBrokenRdeps =
2023-03-24 11:26:01 +00:00
Text . unlines $
headline <> [ " " ] <> tldr <> ( ( " * " <> ) <$> ( errors <> warnings ) ) <> [ " " ]
<> totals
<> optionalList " #### Maintained packages with build failure " ( maintainedList fails )
<> optionalList " #### Maintained packages with failed dependency " ( maintainedList failedDeps )
<> optionalList " #### Maintained packages with unknown error " ( maintainedList unknownErr )
<> optionalHideableList " #### Unmaintained packages with build failure " ( unmaintainedList fails )
<> optionalHideableList " #### Unmaintained packages with failed dependency " ( unmaintainedList failedDeps )
<> optionalHideableList " #### Unmaintained packages with unknown error " ( unmaintainedList unknownErr )
<> optionalHideableList " #### Top 50 broken packages, sorted by number of reverse dependencies " ( brokenLine <$> topBrokenRdeps )
<> [ " " , " *:arrow_heading_up:: The number of packages that depend (directly or indirectly) on this package (if any). If two numbers are shown the first (lower) number considers only packages which currently have enabled hydra jobs, i.e. are not marked broken. The second (higher) number considers all packages.* " , " " ]
<> footer
where
footer = [ " *Report generated with [maintainers/scripts/haskell/hydra-report.hs](https://github.com/NixOS/nixpkgs/blob/haskell-updates/maintainers/scripts/haskell/hydra-report.hs)* " ]
headline =
[ " ### [haskell-updates build report from hydra](https://hydra.nixos.org/jobset/nixpkgs/haskell-updates) "
, evalLine eval fetchTime ]
totals =
[ " #### Build summary "
, " "
]
<> printTable " Platform " ( \ x -> makeSearchLink id ( platform x <> " " <> platformIcon x ) ( " . " <> platform x ) ) ( \ x -> showT x <> " " <> icon x ) showT numSummary
brokenLine ( name , rdeps ) = " [ " <> name <> " ](https://packdeps.haskellers.com/reverse/ " <> name <> " ) :arrow_heading_up: " <> Text . pack ( show rdeps ) <> " "
numSummary = statusToNumSummary summary
2023-03-24 11:55:52 +00:00
jobsByState :: ( BuildState -> Bool ) -> Map Text SummaryEntry
2023-03-24 11:26:01 +00:00
jobsByState predicate = Map . filter ( predicate . worstState ) summary
2023-03-24 11:55:52 +00:00
worstState :: SummaryEntry -> BuildState
2023-03-24 11:26:01 +00:00
worstState = foldl' min Success . fmap state . summaryBuilds
2023-03-24 11:55:52 +00:00
fails :: Map Text SummaryEntry
2023-03-24 11:26:01 +00:00
fails = jobsByState ( == Failed )
2023-03-24 11:55:52 +00:00
2023-03-24 11:26:01 +00:00
failedDeps = jobsByState ( == DependencyFailed )
unknownErr = jobsByState ( \ x -> x > DependencyFailed && x < TimedOut )
withMaintainer = Map . mapMaybe ( \ e -> ( summaryBuilds e , ) <$> nonEmpty ( Set . toList ( summaryMaintainers e ) ) )
withoutMaintainer = Map . mapMaybe ( \ e -> if Set . null ( summaryMaintainers e ) then Just e else Nothing )
optionalList heading list = if null list then mempty else [ heading ] <> list
optionalHideableList heading list = if null list then mempty else [ heading ] <> details ( showT ( length list ) <> " job(s) " ) list
maintainedList = showMaintainedBuild <=< Map . toList . withMaintainer
unmaintainedList = showBuild <=< sortOn ( \ ( snd -> x ) -> ( negate ( summaryUnbrokenReverseDeps x ) , negate ( summaryReverseDeps x ) ) ) . Map . toList . withoutMaintainer
showBuild ( name , entry ) = printJob id name ( summaryBuilds entry , Text . pack ( if summaryReverseDeps entry > 0 then " :arrow_heading_up: " <> show ( summaryUnbrokenReverseDeps entry ) <> " | " <> show ( summaryReverseDeps entry ) else " " ) )
showMaintainedBuild ( name , ( table , maintainers ) ) = printJob id name ( table , Text . intercalate " " ( fmap ( " @ " <> ) ( toList maintainers ) ) )
tldr = case ( errors , warnings ) of
( [] , [] ) -> [ " :green_circle: **Ready to merge** (if there are no [evaluation errors](https://hydra.nixos.org/jobset/nixpkgs/haskell-updates)) " ]
( [] , _ ) -> [ " :yellow_circle: **Potential issues** (and possibly [evaluation errors](https://hydra.nixos.org/jobset/nixpkgs/haskell-updates)) " ]
_ -> [ " :red_circle: **Branch not mergeable** " ]
warnings =
if' ( Unfinished > maybe Success worstState maintainedJob ) " `maintained` jobset failed. " <>
if' ( Unfinished == maybe Success worstState mergeableJob ) " `mergeable` jobset is not finished. " <>
if' ( Unfinished == maybe Success worstState maintainedJob ) " `maintained` jobset is not finished. "
errors =
if' ( isNothing mergeableJob ) " No `mergeable` job found. " <>
if' ( isNothing maintainedJob ) " No `maintained` job found. " <>
if' ( Unfinished > maybe Success worstState mergeableJob ) " `mergeable` jobset failed. " <>
if' ( outstandingJobs ( Platform " x86_64-linux " ) > 100 ) " Too many outstanding jobs on x86_64-linux. " <>
if' ( outstandingJobs ( Platform " aarch64-linux " ) > 100 ) " Too many outstanding jobs on aarch64-linux. " <>
if' ( outstandingJobs ( Platform " aarch64-darwin " ) > 100 ) " Too many outstanding jobs on aarch64-darwin. "
if' p e = if p then [ e ] else mempty
outstandingJobs platform | Table m <- numSummary = Map . findWithDefault 0 ( platform , Unfinished ) m
maintainedJob = Map . lookup " maintained " summary
mergeableJob = Map . lookup " mergeable " summary
2021-05-04 16:16:00 +00:00
2023-01-09 23:34:51 +00:00
printEvalInfo :: IO ()
printEvalInfo = do
( eval , fetchTime , _ ) <- readBuildReports
putStrLn ( Text . unpack $ evalLine eval fetchTime )
2021-05-04 16:16:00 +00:00
printMaintainerPing :: IO ()
printMaintainerPing = do
2021-09-09 22:55:12 +00:00
( maintainerMap , ( reverseDependencyMap , topBrokenRdeps ) ) <- concurrently getMaintainerMap do
depMap <- getDependencyMap
rdepMap <- evaluate . calculateReverseDependencies $ depMap
let tops = take 50 . sortOn ( negate . snd ) . fmap ( second fst ) . filter ( \ x -> maybe False broken $ Map . lookup ( fst x ) depMap ) . Map . toList $ rdepMap
pure ( rdepMap , tops )
2021-05-04 16:16:00 +00:00
( eval , fetchTime , buildReport ) <- readBuildReports
2021-09-09 22:55:12 +00:00
putStrLn ( Text . unpack ( printBuildSummary eval fetchTime ( buildSummary maintainerMap reverseDependencyMap buildReport ) topBrokenRdeps ) )
2021-05-04 16:16:00 +00:00
printMarkBrokenList :: IO ()
printMarkBrokenList = do
2023-02-19 23:20:42 +00:00
( _ , fetchTime , buildReport ) <- readBuildReports
runReq defaultHttpConfig $ forM_ buildReport \ build @ Build { job , id } ->
case ( getBuildState build , Text . splitOn " . " job ) of
( Failed , [ " haskellPackages " , name , " x86_64-linux " ] ) -> do
-- Fetch build log from hydra to figure out the cause of the error.
build_log <- ByteString . lines <$> hydraPlainQuery [ " build " , showT id , " nixlog " , " 1 " , " raw " ]
-- We use the last probable error cause found in the build log file.
let error_message = fromMaybe " failure " $ safeLast $ mapMaybe probableErrorCause build_log
liftIO $ putStrLn $ " - " <> Text . unpack name <> " # " <> error_message <> " in job https://hydra.nixos.org/build/ " <> show id <> " at " <> formatTime defaultTimeLocale " %Y-%m-%d " fetchTime
2021-05-04 16:16:00 +00:00
_ -> pure ()
2023-02-19 23:20:42 +00:00
{- | This function receives a line from a Nix Haskell builder build log and returns a possible error cause.
| We might need to add other causes in the future if errors happen in unusual parts of the builder .
- }
probableErrorCause :: ByteString -> Maybe String
probableErrorCause " Setup: Encountered missing or private dependencies: " = Just " dependency missing "
probableErrorCause " running tests " = Just " test failure "
probableErrorCause build_line | ByteString . isPrefixOf " Building " build_line = Just ( " failure building " <> fromUTF8BS ( fst $ ByteString . breakSubstring " for " $ ByteString . drop 9 build_line ) )
probableErrorCause build_line | ByteString . isSuffixOf " Phase " build_line = Just ( " failure in " <> fromUTF8BS build_line )
probableErrorCause _ = Nothing