Skip to content

Commit

Permalink
Checkpoint matrix expansion changes
Browse files Browse the repository at this point in the history
  • Loading branch information
armanbilge committed Feb 23, 2022
1 parent 17bd11e commit 5b86e38
Show file tree
Hide file tree
Showing 7 changed files with 123 additions and 37 deletions.
4 changes: 2 additions & 2 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -77,11 +77,11 @@ jobs:

- name: Make target directories
if: github.event_name != 'pull_request' && (startsWith(github.ref, 'refs/tags/v') || github.ref == 'refs/heads/series/0.4')
run: mkdir -p github/target github-actions/target kernel/target versioning/target ci-release/target target .js/target mdocs/target site/target ci-signing/target mima/target .jvm/target .native/target no-publish/target sonatype/target ci/target sonatype-ci-release/target core/target settings/target project/target
run: mkdir -p github/target github-actions/target kernel/target versioning/target ci-release/target target .js/target mdocs/target site/target ci-signing/target mergify/target mima/target .jvm/target .native/target no-publish/target sonatype/target ci/target sonatype-ci-release/target core/target settings/target project/target

- name: Compress target directories
if: github.event_name != 'pull_request' && (startsWith(github.ref, 'refs/tags/v') || github.ref == 'refs/heads/series/0.4')
run: tar cf targets.tar github/target github-actions/target kernel/target versioning/target ci-release/target target .js/target mdocs/target site/target ci-signing/target mima/target .jvm/target .native/target no-publish/target sonatype/target ci/target sonatype-ci-release/target core/target settings/target project/target
run: tar cf targets.tar github/target github-actions/target kernel/target versioning/target ci-release/target target .js/target mdocs/target site/target ci-signing/target mergify/target mima/target .jvm/target .native/target no-publish/target sonatype/target ci/target sonatype-ci-release/target core/target settings/target project/target

- name: Upload target directories
if: github.event_name != 'pull_request' && (startsWith(github.ref, 'refs/tags/v') || github.ref == 'refs/heads/series/0.4')
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -606,7 +606,7 @@ ${indent(jobs.map(compileJob(_, sbt)).mkString("\n\n"), 1)}
githubWorkflowGeneratedDownloadSteps := {
val extraKeys = githubWorkflowArtifactDownloadExtraKeys.value
val additions = githubWorkflowBuildMatrixAdditions.value
val matrix = additions.map {
val matrixAdds = additions.map {
case (key, values) =>
if (extraKeys(key))
key -> values // we want to iterate over all values
Expand All @@ -615,25 +615,23 @@ ${indent(jobs.map(compileJob(_, sbt)).mkString("\n\n"), 1)}
}

val keys = "scala" :: additions.keys.toList.sorted
val oses = githubWorkflowOSes.value.toList
val scalas = githubWorkflowScalaVersions.value.toList
val exclusions = githubWorkflowBuildMatrixExclusions.value
val javas = githubWorkflowJavaVersions.value.toList
val exclusions = githubWorkflowBuildMatrixExclusions.value.toList

// we build the list of artifacts, by iterating over all combinations of keys
val artifacts = matrix
.toList
.sortBy(_._1)
.map(_._2)
.foldLeft(scalas.map(List(_))) { (artifacts, values) =>
for {
artifact <- artifacts
value <- values
} yield artifact :+ value
} // then, we filter artifacts for keys that are excluded from the matrix
.filterNot { artifact =>
val job = keys.zip(artifact).toMap
exclusions.exists { // there is an exclude that matches the current job
case MatrixExclude(matching) => matching.toSet.subsetOf(job.toSet)
}
val artifacts =
expandMatrix(
oses,
scalas,
javas,
matrixAdds,
Nil,
exclusions
).map {
case _ :: scala :: _ :: tail => scala :: tail
case _ => sys.error("Bug generating artifact download steps") // shouldn't happen
}

if (githubWorkflowArtifactUpload.value) {
Expand Down Expand Up @@ -862,7 +860,37 @@ ${indent(jobs.map(compileJob(_, sbt)).mkString("\n\n"), 1)}
}
)

private[gha] def diff(expected: String, actual: String): String = {
private[sbt] def expandMatrix(
oses: List[String],
scalas: List[String],
javas: List[JavaSpec],
matrixAdds: Map[String, List[String]],
includes: List[MatrixInclude],
excludes: List[MatrixExclude]
): List[List[String]] = {
val keys = "os" :: "scala" :: "java" :: matrixAdds.keys.toList.sorted
val matrix =
matrixAdds + ("os" -> oses) + ("scala" -> scalas) + ("java" -> javas.map(_.render))

// expand the matrix
keys
.foldLeft(List(List.empty[String])) { (cells, key) =>
val values = matrix.getOrElse(key, Nil)
cells.flatMap { cell => values.map(v => cell ::: v :: Nil) }
}
.filterNot { cell => // remove the excludes
val job = keys.zip(cell).toMap
excludes.exists { // there is an exclude that matches the current job
case MatrixExclude(matching) => matching.toSet.subsetOf(job.toSet)
}
} ::: includes.map { // add the includes
case MatrixInclude(matching, additions) =>
// yoloing here, but let's wait for the bug report
keys.map(matching) ::: additions.values.toList
}
}

private[sbt] def diff(expected: String, actual: String): String = {
val expectedLines = expected.split("\n", -1)
val actualLines = actual.split("\n", -1)
val (lines, _) =
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,25 +16,30 @@

package org.typelevel.sbt.mergify

sealed abstract class MergifyAction
sealed abstract class MergifyAction {
def name: String
}

object MergifyAction {

final case class Merge(
method: Option[String] = None,
rebaseFallback: Option[String] = None,
commitMessageTemplate: Option[String] = None
) extends MergifyAction
) extends MergifyAction {
def name = "merge"
}

final case class Label(
add: List[String] = Nil,
remove: List[String] = Nil,
removeAll: Option[Boolean] = None
) extends MergifyAction
) extends MergifyAction {
def name = "label"
}

// this should prevent exhaustivity checking,
// so pattern matches always have to include a default case
// this lets us add more cases without breaking backwards compat
private[this] object Dummy extends MergifyAction
private[this] object Dummy extends MergifyAction { // break exhaustivity checking
def name = "dummy"
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,8 @@ package org.typelevel.sbt.mergify
sealed abstract class MergifyCondition

object MergifyCondition {
final case class And(conditions: List[MergifyCondition])
final case class Or(conditions: List[MergifyCondition])
final case class Custom(condition: String) extends MergifyCondition
final case class And(conditions: List[MergifyCondition]) extends MergifyCondition
final case class Or(conditions: List[MergifyCondition]) extends MergifyCondition
private[this] final object Dummy extends MergifyCondition // break exhaustivity checking
}
Original file line number Diff line number Diff line change
Expand Up @@ -23,16 +23,48 @@ object MergifyPlugin extends AutoPlugin {

object autoImport {
lazy val mergifyPrRules = settingKey[Seq[MergifyPrRule]]("The mergify pull request rules")
lazy val mergifyEnableSteward = settingKey[Boolean](
"Whether to generate an automerge rule for Scala Steward PRs (default: true)")

lazy val mergifyStewardConfig = settingKey[Option[MergifyStewardConfig]](
"Config for the automerge rule for Scala Steward PRs, set to None to disable.")

lazy val mergifyRequiredJobs =
settingKey[Seq[String]]("Ids for jobs that must succeed for merging (default: [build])")
lazy val mergifySuccessConditions = settingKey[List[String]](

lazy val mergifySuccessConditions = settingKey[Seq[MergifyCondition]](
"Success conditions for merging (default: auto-generated from `mergifyRequiredJobs` setting)")
}

override def requires = GenerativePlugin
override def trigger: PluginTrigger = allRequirements

import autoImport._
import GenerativePlugin.autoImport._

override def buildSettings: Seq[Setting[_]] = Seq(
mergifyRequiredJobs := Seq("build"),
mergifySuccessConditions := jobSuccessConditions.value,
mergifyPrRules := {
mergifyStewardConfig.value.map(_.toPrRule(mergifySuccessConditions.value.toList)).toList
}
)

private lazy val jobSuccessConditions = Def.setting {
githubWorkflowGeneratedCI.value.flatMap {
case job if mergifyRequiredJobs.value.contains(job.id) =>
GenerativePlugin
.expandMatrix(
job.oses,
job.scalas,
job.javas,
job.matrixAdds,
job.matrixIncs,
job.matrixExcs
)
.map { cell =>
MergifyCondition.Custom(s"status-success=${job.name} (${cell.mkString(", ")})")
}
case _ => Nil
}
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,6 @@ package org.typelevel.sbt.mergify

final case class MergifyPrRule(
name: String,
conditions: List[String],
actions: Map[String, MergifyAction]
conditions: List[MergifyCondition],
actions: List[MergifyAction]
)
Original file line number Diff line number Diff line change
Expand Up @@ -17,8 +17,27 @@
package org.typelevel.sbt.mergify

final case class MergifyStewardConfig(
name: String = "merge scala-steward's PRs",
author: String = "scala-steward",
minor: Boolean = false,
patch: Boolean = true,
labels: List[String]
)
merge: MergifyAction.Merge = MergifyAction.Merge()
) {

private[mergify] def toPrRule(buildConditions: List[MergifyCondition]): MergifyPrRule = {
val authorCond = MergifyCondition.Custom(s"author=$author")

val bodyCond = {
val patchCond = MergifyCondition.Custom("body~=labels:.*early-semver-spec-patch")
val minorCond = MergifyCondition.Custom("body~=labels:.*early-semver-minor")
if (minor) MergifyCondition.Or(List(patchCond, minorCond))
else patchCond
}

MergifyPrRule(
name,
authorCond :: bodyCond :: buildConditions,
List(merge)
)
}

}

0 comments on commit 5b86e38

Please sign in to comment.