diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 2501db8d..7c500d0b 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -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') diff --git a/github-actions/src/main/scala/org/typelevel/sbt/gha/GenerativePlugin.scala b/github-actions/src/main/scala/org/typelevel/sbt/gha/GenerativePlugin.scala index 988c59fd..9346535b 100644 --- a/github-actions/src/main/scala/org/typelevel/sbt/gha/GenerativePlugin.scala +++ b/github-actions/src/main/scala/org/typelevel/sbt/gha/GenerativePlugin.scala @@ -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 @@ -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) { @@ -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, _) = diff --git a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyAction.scala b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyAction.scala index fd113411..7940d823 100644 --- a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyAction.scala +++ b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyAction.scala @@ -16,7 +16,9 @@ package org.typelevel.sbt.mergify -sealed abstract class MergifyAction +sealed abstract class MergifyAction { + def name: String +} object MergifyAction { @@ -24,17 +26,20 @@ object MergifyAction { 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" + } } diff --git a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyCondition.scala b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyCondition.scala index 25320ef6..f85b43b0 100644 --- a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyCondition.scala +++ b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyCondition.scala @@ -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 } diff --git a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyPlugin.scala b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyPlugin.scala index 8918ca43..645ac832 100644 --- a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyPlugin.scala +++ b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyPlugin.scala @@ -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 + } + } + } diff --git a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyPrRule.scala b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyPrRule.scala index b052f112..4db01502 100644 --- a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyPrRule.scala +++ b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyPrRule.scala @@ -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] ) diff --git a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyStewardConfig.scala b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyStewardConfig.scala index deded328..5079fc99 100644 --- a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyStewardConfig.scala +++ b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyStewardConfig.scala @@ -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) + ) + } + +}