diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 5963d5d5..9121286b 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -86,11 +86,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/.mergify.yml b/.mergify.yml new file mode 100644 index 00000000..5d985a70 --- /dev/null +++ b/.mergify.yml @@ -0,0 +1,154 @@ +# This file was automatically generated by sbt-typelevel-mergify using the +# mergifyGenerate task. You should add and commit this file to +# your git repository. It goes without saying that you shouldn't edit +# this file by hand! Instead, if you wish to make changes, you should +# change your sbt build configuration to revise the mergify configuration +# to meet your needs, then regenerate this file. + +pull_request_rules: +- name: merge scala-steward's PRs + conditions: + - author=scala-steward + - or: + - body~=labels:.*early-semver-patch + - body~=labels:.*early-semver-minor + - status-success=Build and Test (ubuntu-latest, 2.12.15, temurin@8, rootJVM) + - '#approved-reviews-by>=1' + actions: + merge: {} +- name: Label ci PRs + conditions: + - files~=^ci/ + actions: + label: + add: + - ci + remove: [] +- name: Label ci-release PRs + conditions: + - files~=^ci-release/ + actions: + label: + add: + - ci-release + remove: [] +- name: Label ci-signing PRs + conditions: + - files~=^ci-signing/ + actions: + label: + add: + - ci-signing + remove: [] +- name: Label core PRs + conditions: + - files~=^core/ + actions: + label: + add: + - core + remove: [] +- name: Label docs PRs + conditions: + - files~=^docs/ + actions: + label: + add: + - docs + remove: [] +- name: Label github PRs + conditions: + - files~=^github/ + actions: + label: + add: + - github + remove: [] +- name: Label github-actions PRs + conditions: + - files~=^github-actions/ + actions: + label: + add: + - github-actions + remove: [] +- name: Label kernel PRs + conditions: + - files~=^kernel/ + actions: + label: + add: + - kernel + remove: [] +- name: Label mdocs PRs + conditions: + - files~=^mdocs/ + actions: + label: + add: + - mdocs + remove: [] +- name: Label mergify PRs + conditions: + - files~=^mergify/ + actions: + label: + add: + - mergify + remove: [] +- name: Label mima PRs + conditions: + - files~=^mima/ + actions: + label: + add: + - mima + remove: [] +- name: Label no-publish PRs + conditions: + - files~=^no-publish/ + actions: + label: + add: + - no-publish + remove: [] +- name: Label settings PRs + conditions: + - files~=^settings/ + actions: + label: + add: + - settings + remove: [] +- name: Label site PRs + conditions: + - files~=^site/ + actions: + label: + add: + - site + remove: [] +- name: Label sonatype PRs + conditions: + - files~=^sonatype/ + actions: + label: + add: + - sonatype + remove: [] +- name: Label sonatype-ci-release PRs + conditions: + - files~=^sonatype-ci-release/ + actions: + label: + add: + - sonatype-ci-release + remove: [] +- name: Label versioning PRs + conditions: + - files~=^versioning/ + actions: + label: + add: + - versioning + remove: [] diff --git a/build.sbt b/build.sbt index 6ad3c07a..a01ad0be 100644 --- a/build.sbt +++ b/build.sbt @@ -11,12 +11,17 @@ ThisBuild / developers := List( tlGitHubDev("djspiewak", "Daniel Spiewak") ) +ThisBuild / mergifyStewardConfig ~= { _.map(_.copy(mergeMinors = true)) } +ThisBuild / mergifySuccessConditions += MergifyCondition.Custom("#approved-reviews-by>=1") +ThisBuild / mergifyLabelPaths += { "docs" -> file("docs") } + lazy val root = tlCrossRootProject.aggregate( kernel, noPublish, settings, github, githubActions, + mergify, versioning, mima, sonatype, @@ -66,6 +71,15 @@ lazy val githubActions = project name := "sbt-typelevel-github-actions" ) +lazy val mergify = project + .in(file("mergify")) + .enablePlugins(SbtPlugin) + .settings( + name := "sbt-typelevel-mergify", + tlVersionIntroduced := Map("2.12" -> "0.4.6") + ) + .dependsOn(githubActions) + lazy val versioning = project .in(file("versioning")) .enablePlugins(SbtPlugin) 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/build.sbt b/mergify/build.sbt new file mode 100644 index 00000000..ec725236 --- /dev/null +++ b/mergify/build.sbt @@ -0,0 +1 @@ +libraryDependencies += "io.circe" %% "circe-yaml" % "0.14.1" diff --git a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyAction.scala b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyAction.scala new file mode 100644 index 00000000..9c96203e --- /dev/null +++ b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyAction.scala @@ -0,0 +1,62 @@ +/* + * Copyright 2022 Typelevel + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.typelevel.sbt.mergify + +import io.circe.Encoder +import io.circe.syntax._ + +sealed abstract class MergifyAction { + private[mergify] def name = getClass.getSimpleName.toLowerCase +} + +object MergifyAction { + + implicit def encoder: Encoder[MergifyAction] = Encoder.instance { + case merge: Merge => merge.asJson + case label: Label => label.asJson + case _ => sys.error("should not happen") + } + + final case class Merge( + method: Option[String] = None, + rebaseFallback: Option[String] = None, + commitMessageTemplate: Option[String] = None + ) extends MergifyAction + + object Merge { + implicit def encoder: Encoder[Merge] = + Encoder.forProduct3("method", "rebase_fallback", "commit_message_template") { + (m: Merge) => (m.method, m.rebaseFallback, m.commitMessageTemplate) + } + } + + final case class Label( + add: List[String] = Nil, + remove: List[String] = Nil, + removeAll: Option[Boolean] = None + ) extends MergifyAction + + object Label { + implicit def encoder: Encoder[Label] = + Encoder.forProduct3("add", "remove", "remove_all") { (l: Label) => + (l.add, l.remove, l.removeAll) + } + } + + private[this] object Dummy extends MergifyAction + +} diff --git a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyCondition.scala b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyCondition.scala new file mode 100644 index 00000000..ea6f8c29 --- /dev/null +++ b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyCondition.scala @@ -0,0 +1,48 @@ +/* + * Copyright 2022 Typelevel + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.typelevel.sbt.mergify + +import io.circe.Encoder +import io.circe.syntax._ + +sealed abstract class MergifyCondition + +object MergifyCondition { + implicit def encoder: Encoder[MergifyCondition] = Encoder.instance { + case custom: Custom => custom.asJson + case and: And => and.asJson + case or: Or => or.asJson + case _ => sys.error("shouldn't happen") + } + + final case class Custom(condition: String) extends MergifyCondition + object Custom { + implicit def encoder: Encoder[Custom] = Encoder.encodeString.contramap(_.condition) + } + + final case class And(conditions: List[MergifyCondition]) extends MergifyCondition + object And { + implicit def encoder: Encoder[And] = Encoder.forProduct1("and")(_.conditions) + } + + final case class Or(conditions: List[MergifyCondition]) extends MergifyCondition + object Or { + implicit def encoder: Encoder[Or] = Encoder.forProduct1("or")(_.conditions) + } + + 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 new file mode 100644 index 00000000..91706851 --- /dev/null +++ b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyPlugin.scala @@ -0,0 +1,193 @@ +/* + * Copyright 2022 Typelevel + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.typelevel.sbt.mergify + +import sbt._, Keys._ +import org.typelevel.sbt.gha._ + +import java.nio.file.Path + +object MergifyPlugin extends AutoPlugin { + + object autoImport { + lazy val mergifyGenerate = taskKey[Unit]( + "Generates (and overwrites if extant) a .mergify.yml according to configuration") + lazy val mergifyCheck = taskKey[Unit]( + "Checks to see if the .mergify.yml files are equivalent to what would be generated and errors if otherwise") + + lazy val mergifyPrRules = settingKey[Seq[MergifyPrRule]]("The mergify pull request rules") + + 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[Seq[MergifyCondition]]( + "Success conditions for merging (default: auto-generated from `mergifyRequiredJobs` setting)") + + lazy val mergifyLabelPaths = settingKey[Map[String, File]]( + "A map from label to file path (default: auto-populated for every subproject in your build)") + + type MergifyAction = org.typelevel.sbt.mergify.MergifyAction + val MergifyAction = org.typelevel.sbt.mergify.MergifyAction + type MergifyCondition = org.typelevel.sbt.mergify.MergifyCondition + val MergifyCondition = org.typelevel.sbt.mergify.MergifyCondition + type MergifyPrRule = org.typelevel.sbt.mergify.MergifyPrRule + val MergifyPrRule = org.typelevel.sbt.mergify.MergifyPrRule + type MergifyStewardConfig = org.typelevel.sbt.mergify.MergifyStewardConfig + val MergifyStewardConfig = org.typelevel.sbt.mergify.MergifyStewardConfig + } + + override def requires = GenerativePlugin + override def trigger: PluginTrigger = allRequirements + + import autoImport._ + import GenerativePlugin.autoImport._ + + override def buildSettings: Seq[Setting[_]] = Seq( + mergifyStewardConfig := Some(MergifyStewardConfig()), + mergifyRequiredJobs := Seq("build"), + mergifyLabelPaths := Map.empty, + mergifySuccessConditions := jobSuccessConditions.value, + mergifyPrRules := { + val baseDir = (LocalRootProject / baseDirectory).value.toPath + val stewardRule = + mergifyStewardConfig.value.map(_.toPrRule(mergifySuccessConditions.value.toList)).toList + val labelRules = + mergifyLabelPaths.value.toList.sorted.map { + case (label, file) => + val relPath = baseDir.relativize(file.toPath.toAbsolutePath.normalize) + val suffix = if (file.isDirectory) "/" else "" + MergifyPrRule( + s"Label ${label} PRs", + List(MergifyCondition.Custom(s"files~=^${relPath}${suffix}")), + List(MergifyAction.Label(add = List(label))) + ) + } + stewardRule ++ labelRules + }, + mergifyGenerate := { + IO.write(mergifyYaml.value, generateMergifyContents.value) + }, + mergifyCheck := { + val log = state.value.log + + def reportMismatch(file: File, expected: String, actual: String): Unit = { + log.error(s"Expected:\n$expected") + log.error(s"Actual:\n${GenerativePlugin.diff(expected, actual)}") + sys.error( + s"${file.getName} does not contain contents that would have been generated by sbt-typelevel-mergify; try running mergifyGenerate") + } + + def compare(file: File, expected: String): Unit = { + val actual = IO.read(file) + if (expected != actual) { + reportMismatch(file, expected, actual) + } + } + + compare(mergifyYaml.value, generateMergifyContents.value) + } + ) + + override def projectSettings: Seq[Setting[_]] = Seq( + mergifyGenerate / aggregate := false, + mergifyCheck / aggregate := false, + githubWorkflowGenerate := githubWorkflowGenerate + .dependsOn((ThisBuild / mergifyGenerate)) + .value, + githubWorkflowCheck := githubWorkflowCheck.dependsOn((ThisBuild / mergifyCheck)).value, + ThisBuild / mergifyLabelPaths := { + val labelPaths = (ThisBuild / mergifyLabelPaths).value + projectLabel.value.fold(labelPaths) { + case (label, path) => + val add = labelPaths.get(label) match { + case Some(f) => label -> commonAncestor(f.toPath, path) + case None => label -> path + } + labelPaths + (add._1 -> add._2.toFile) + } + } + ) + + 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 + } + } + + private lazy val projectLabel = Def.setting { + val path = (Compile / sourceDirectories) + .? + .value + .getOrElse(Seq.empty) + .map(_.toPath) + .foldLeft(baseDirectory.value.toPath)(commonAncestor(_, _)) + + val label = path.getFileName.toString + + def isRoot = path == (LocalRootProject / baseDirectory).value.toPath + if (label.startsWith(".") || isRoot) // don't label this project + None + else Some(label -> path) + } + + // x and y should be absolute/normalized + private def commonAncestor(x: Path, y: Path): Path = { + val n = math.min(x.getNameCount, y.getNameCount) + (0 until n) + .takeWhile(i => x.getName(i) == y.getName(i)) + .map(x.getName(_)) + .foldLeft(java.nio.file.Paths.get("/"))(_.resolve(_)) + } + + private lazy val mergifyYaml = Def.setting { + (ThisBuild / baseDirectory).value / ".mergify.yml" + } + + private lazy val generateMergifyContents = Def.task { + import _root_.io.circe.syntax._ + import _root_.io.circe.yaml.Printer + + val contents = Map("pull_request_rules" -> mergifyPrRules.value.toList) + val printer = Printer.spaces2.copy(dropNullKeys = true) + + s"""|# This file was automatically generated by sbt-typelevel-mergify using the + |# mergifyGenerate task. You should add and commit this file to + |# your git repository. It goes without saying that you shouldn't edit + |# this file by hand! Instead, if you wish to make changes, you should + |# change your sbt build configuration to revise the mergify configuration + |# to meet your needs, then regenerate this file. + | + |${printer.pretty(contents.asJson)}""".stripMargin + } + +} diff --git a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyPrRule.scala b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyPrRule.scala new file mode 100644 index 00000000..a9421bb5 --- /dev/null +++ b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyPrRule.scala @@ -0,0 +1,32 @@ +/* + * Copyright 2022 Typelevel + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.typelevel.sbt.mergify + +import io.circe.Encoder + +final case class MergifyPrRule( + name: String, + conditions: List[MergifyCondition], + actions: List[MergifyAction] +) + +object MergifyPrRule { + implicit def encoder: Encoder[MergifyPrRule] = + Encoder.forProduct3("name", "conditions", "actions") { m => + (m.name, m.conditions, m.actions.map(a => a.name -> a).toMap) + } +} diff --git a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyStewardConfig.scala b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyStewardConfig.scala new file mode 100644 index 00000000..84be682e --- /dev/null +++ b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyStewardConfig.scala @@ -0,0 +1,43 @@ +/* + * Copyright 2022 Typelevel + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.typelevel.sbt.mergify + +final case class MergifyStewardConfig( + name: String = "merge scala-steward's PRs", + author: String = "scala-steward", + mergeMinors: Boolean = false, + action: 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-patch") + val minorCond = MergifyCondition.Custom("body~=labels:.*early-semver-minor") + if (mergeMinors) MergifyCondition.Or(List(patchCond, minorCond)) + else patchCond + } + + MergifyPrRule( + name, + authorCond :: bodyCond :: buildConditions, + List(action) + ) + } + +} diff --git a/project/build.sbt b/project/build.sbt index 27c448e3..673bf53b 100644 --- a/project/build.sbt +++ b/project/build.sbt @@ -6,6 +6,7 @@ val modules = List( "github", "github-actions", "kernel", + "mergify", "mima", "no-publish", "settings", diff --git a/project/mergify.sbt b/project/mergify.sbt new file mode 120000 index 00000000..9a4cbd2c --- /dev/null +++ b/project/mergify.sbt @@ -0,0 +1 @@ +../mergify/build.sbt \ No newline at end of file