Skip to content

Commit

Permalink
Implement --matrix-extra for more matrix dimensions
Browse files Browse the repository at this point in the history
Sometimes you need to test your project along more dimensions than
just GHC version.  This is particularly important for
programs/libraries that use FFI to bind to libraries - they might
need to be tested against a range of library versions.

In general, you want to test all the combinations of GHC versions
and other properties, i.e. the cartesian product.  It is burdensome
for maintainers that need such a strategy to manually adjust the
matrix after every (re)generation of the CI script/spec.  Better
tool support for this scenario is warranted.

This commit implements a new --matrix-extra option, which adds
additional matrix dimensions.  The option value format is:

  --matrix-extra libfoo:2.6,3.0;libbar:0.1,0.2

haskell-ci adds all the combinations of GHC version and the
--matrix-extra fields to the matrix.  Corresponding build/test steps
can be introduced via --github-patches (or --travis-patches).

This commit implements this feature for GitHub actions only.  It can
be implemented for Travis in a subsequent commit, if desired.
  • Loading branch information
frasertweedale committed Jan 7, 2021
1 parent a4ea1ed commit c670e11
Show file tree
Hide file tree
Showing 3 changed files with 54 additions and 11 deletions.
26 changes: 26 additions & 0 deletions src/HaskellCI/Config.hs
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,7 @@ data Config = Config
, cfgPostgres :: !Bool
, cfgGoogleChrome :: !Bool
, cfgEnv :: M.Map Version String
, cfgMatrixExtra :: M.Map String (S.Set String)
, cfgAllowFailures :: !VersionRange
, cfgLastInSeries :: !Bool
, cfgOsx :: S.Set Version
Expand Down Expand Up @@ -133,6 +134,7 @@ emptyConfig = Config
, cfgPostgres = False
, cfgGoogleChrome = False
, cfgEnv = M.empty
, cfgMatrixExtra = M.empty
, cfgAllowFailures = noVersion
, cfgLastInSeries = False
, cfgOsx = S.empty
Expand Down Expand Up @@ -218,6 +220,8 @@ configGrammar = Config
^^^ help "Add google-chrome service"
<*> C.monoidalFieldAla "env" Env (field @"cfgEnv")
^^^ metahelp "ENV" "Environment variables per job (e.g. `8.0.2:HADDOCK=false`)"
<*> C.monoidalFieldAla "matrix-extra" MatrixExtra (field @"cfgMatrixExtra")
^^^ metahelp "MATRIX" "Extra matrix dimensions (e.g. `libfoo:2.6,3.0,git`)"
<*> C.optionalFieldDefAla "allow-failures" Range (field @"cfgAllowFailures") noVersion
^^^ metahelp "JOB" "Allow failures of particular GHC version"
<*> C.booleanFieldDef "last-in-series" (field @"cfgLastInSeries") False
Expand Down Expand Up @@ -290,6 +294,28 @@ instance C.Pretty Env where
pretty (Env m) = PP.fsep . PP.punctuate PP.comma . map p . M.toList $ m where
p (v, s) = C.pretty v PP.<> PP.colon PP.<> PP.text s


-------------------------------------------------------------------------------
-- MatrixExtra
-------------------------------------------------------------------------------

newtype MatrixExtra = MatrixExtra (M.Map String (S.Set String))
deriving anyclass (C.Newtype (M.Map String (S.Set String)))

instance C.Parsec MatrixExtra where
parsec = MatrixExtra . M.fromList . toList <$> C.sepByNonEmpty p (C.char ';')
where
p = do
k <- C.munch1 (/= ':')
_ <- C.char ':'
v <- foldMap S.singleton <$> C.sepByNonEmpty (C.munch1 (`notElem` [',', ';'])) (C.char ',')
pure (k, v)

instance C.Pretty MatrixExtra where
pretty (MatrixExtra m) = PP.fsep . PP.punctuate PP.semi . map p . M.toList $ m where
p (k, v) = PP.text k PP.<> PP.colon PP.<> PP.fsep (PP.punctuate PP.comma (map PP.text (toList v)))


-------------------------------------------------------------------------------
-- From Cabal
-------------------------------------------------------------------------------
Expand Down
34 changes: 25 additions & 9 deletions src/HaskellCI/GitHub.hs
Original file line number Diff line number Diff line change
Expand Up @@ -435,15 +435,7 @@ makeGitHub _argv config@Config {..} gitconfig prj jobs@JobVersions {..} = do
, ghjContinueOnError = Just "${{ matrix.allow-failure }}"
, ghjServices = mconcat
[ Map.singleton "postgres" postgresService | cfgPostgres ]
, ghjMatrix =
[ GitHubMatrixEntry
{ ghmeGhcVersion = v
, ghmeAllowFailure =
previewGHC cfgHeadHackage compiler
|| maybeGHC False (`C.withinRange` cfgAllowFailures) compiler
}
| compiler@(GHC v) <- reverse $ toList versions
]
, ghjMatrix = matrix
})
unless (null cfgIrcChannels) $
ircJob mainJobName projectName config gitconfig
Expand All @@ -453,6 +445,30 @@ makeGitHub _argv config@Config {..} gitconfig prj jobs@JobVersions {..} = do

Auxiliary {..} = auxiliary config prj jobs

-- extra matrix fields
matrixExtra :: [[(String, String)]]
matrixExtra =
sequence
$ (\(k, vs) -> fmap (\v -> (k, v)) (toList vs))
<$> Map.toList cfgMatrixExtra

mkMatrixEntries :: [(String, String)] -> [GitHubMatrixEntry]
mkMatrixEntries extra =
[ GitHubMatrixEntry
{ ghmeGhcVersion = v
, ghmeAllowFailure =
previewGHC cfgHeadHackage compiler
|| maybeGHC False (`C.withinRange` cfgAllowFailures) compiler
, ghmeMatrixExtra = extra
}
| compiler@(GHC v) <- reverse $ toList versions
]

matrix :: [GitHubMatrixEntry]
matrix = case matrixExtra of
[] -> mkMatrixEntries []
xs -> xs >>= mkMatrixEntries

-- step primitives
githubRun' :: String -> Map.Map String String -> ShM () -> ListBuilder (Either ShError GitHubStep) ()
githubRun' name env shm = item $ do
Expand Down
5 changes: 3 additions & 2 deletions src/HaskellCI/GitHub/Yaml.hs
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,7 @@ data GitHubJob = GitHubJob
data GitHubMatrixEntry = GitHubMatrixEntry
{ ghmeGhcVersion :: Version
, ghmeAllowFailure :: Bool
, ghmeMatrixExtra :: [(String, String)]
}
deriving (Show)

Expand Down Expand Up @@ -122,10 +123,10 @@ instance ToYaml GitHubJob where
item $ "steps" ~> ylistFilt [] (map toYaml $ filter notEmptyStep ghjSteps)

instance ToYaml GitHubMatrixEntry where
toYaml GitHubMatrixEntry {..} = ykeyValuesFilt []
toYaml GitHubMatrixEntry {..} = ykeyValuesFilt [] $
[ "ghc" ~> fromString (prettyShow ghmeGhcVersion)
, "allow-failure" ~> toYaml ghmeAllowFailure
]
] ++ fmap (\(k, v) -> k ~> fromString v) ghmeMatrixExtra

instance ToYaml GitHubStep where
toYaml GitHubStep {..} = ykeyValuesFilt [] $
Expand Down

0 comments on commit c670e11

Please sign in to comment.