From 181cb10e587b28c88047564e1b2eb9f13ca4cc10 Mon Sep 17 00:00:00 2001 From: Naftoli Gugenheim Date: Fri, 7 Jan 2022 00:58:46 -0500 Subject: [PATCH 1/2] Allow checking in an on-the-fly mode while still benefiting from other sbt settings --- .../tools/mima/plugin/MimaPlugin.scala | 88 +++++++++++++------ 1 file changed, 62 insertions(+), 26 deletions(-) diff --git a/sbtplugin/src/main/scala/com/typesafe/tools/mima/plugin/MimaPlugin.scala b/sbtplugin/src/main/scala/com/typesafe/tools/mima/plugin/MimaPlugin.scala index af76578a..65c6173f 100644 --- a/sbtplugin/src/main/scala/com/typesafe/tools/mima/plugin/MimaPlugin.scala +++ b/sbtplugin/src/main/scala/com/typesafe/tools/mima/plugin/MimaPlugin.scala @@ -1,14 +1,19 @@ package com.typesafe.tools.mima package plugin -import sbt._, Keys._ +import com.typesafe.tools.mima.core.Problem +import sbt.* +import sbt.Keys.* + /** MiMa's sbt plugin. */ object MimaPlugin extends AutoPlugin { override def trigger = allRequirements object autoImport extends MimaKeys - import autoImport._ + + import autoImport.* + override def globalSettings: Seq[Def.Setting[_]] = Seq( mimaPreviousArtifacts := NoPreviousArtifacts, @@ -20,6 +25,30 @@ object MimaPlugin extends AutoPlugin { mimaCheckDirection := "backward", ) + trait ArtifactsToClassfiles { + def apply(artifacts: Set[ModuleID]): Map[ModuleID, File] + } + + val artifactsToClassfiles = Def.task { + val depRes = mimaDependencyResolution.value + val taskStreams = streams.value + val smi = scalaModuleInfo.value + new ArtifactsToClassfiles { + override def apply(artifacts: Set[sbt.ModuleID]) = + artifacts match { + case _: NoPreviousArtifacts.type => NoPreviousClassfiles + case previousArtifacts => + previousArtifacts.iterator.map { m => + val moduleId = CrossVersion(m, smi) match { + case Some(f) => m.withName(f(m.name)).withCrossVersion(CrossVersion.disabled) + case None => m + } + moduleId -> SbtMima.getPreviousArtifact(moduleId, depRes, taskStreams) + }.toMap + } + } + } + override def projectSettings: Seq[Def.Setting[_]] = Seq( mimaReportBinaryIssues := { binaryIssuesIterator.value.foreach { case (moduleId, problems) => @@ -38,19 +67,7 @@ object MimaPlugin extends AutoPlugin { }, mimaDependencyResolution := dependencyResolution.value, mimaPreviousClassfiles := { - val depRes = mimaDependencyResolution.value - val taskStreams = streams.value - mimaPreviousArtifacts.value match { - case _: NoPreviousArtifacts.type => NoPreviousClassfiles - case previousArtifacts => - previousArtifacts.iterator.map { m => - val moduleId = CrossVersion(m, scalaModuleInfo.value) match { - case Some(f) => m.withName(f(m.name)).withCrossVersion(CrossVersion.disabled) - case None => m - } - moduleId -> SbtMima.getPreviousArtifact(moduleId, depRes, taskStreams) - }.toMap - } + artifactsToClassfiles.value.apply(mimaPreviousArtifacts.value) }, mimaCurrentClassfiles := (Compile / classDirectory).value, mimaFindBinaryIssues := binaryIssuesIterator.value.toMap, @@ -68,26 +85,45 @@ object MimaPlugin extends AutoPlugin { mimaBinaryIssueFilters.value ++ (if (mimaReportSignatureProblems.value) Nil else Seq(noSigs)) } + trait BinaryIssuesFinder { + def iterator(prevClassFiles: Map[ModuleID, File], + direction: String): Iterator[(ModuleID, (List[Problem], List[Problem]))] + } + // Allows reuse between mimaFindBinaryIssues and mimaReportBinaryIssues // without blowing up the Akka build's heap - private def binaryIssuesIterator = Def.task { + def binaryIssuesFinder = Def.task { val log = streams.value.log - val prevClassfiles = mimaPreviousClassfiles.value val currClassfiles = mimaCurrentClassfiles.value val cp = (mimaFindBinaryIssues / fullClasspath).value val sv = scalaVersion.value val excludeAnnots = mimaExcludeAnnotations.value.toList - - if (prevClassfiles eq NoPreviousClassfiles) { - val msg = "mimaPreviousArtifacts not set, not analyzing binary compatibility" - if (mimaFailOnNoPrevious.value) sys.error(msg) else log.info(s"${name.value}: $msg") - } else if (prevClassfiles.isEmpty) { - log.info(s"${name.value}: mimaPreviousArtifacts is empty, not analyzing binary compatibility.") + val requirePrevious = mimaFailOnNoPrevious.value + val projName = name.value + + new BinaryIssuesFinder { + override def iterator(prevClassfiles: Map[sbt.ModuleID, sbt.File], direction: String) = { + if (prevClassfiles eq NoPreviousClassfiles) { + val msg = "mimaPreviousArtifacts not set, not analyzing binary compatibility" + if (requirePrevious) + sys.error(msg) + else + log.info(s"$projName: $msg") + } else if (prevClassfiles.isEmpty) + log.info(s"$projName: mimaPreviousArtifacts is empty, not analyzing binary compatibility.") + + prevClassfiles.iterator.map { case (moduleId, prevClassfiles) => + moduleId -> SbtMima.runMima(prevClassfiles, currClassfiles, cp, direction, sv, log, excludeAnnots) + } + } } + } - prevClassfiles.iterator.map { case (moduleId, prevClassfiles) => - moduleId -> SbtMima.runMima(prevClassfiles, currClassfiles, cp, mimaCheckDirection.value, sv, log, excludeAnnots) - } + // Allows reuse between mimaFindBinaryIssues and mimaReportBinaryIssues + // without blowing up the Akka build's heap + private def binaryIssuesIterator = Def.task { + val it = binaryIssuesFinder.value + it.iterator(mimaPreviousClassfiles.value, mimaCheckDirection.value) } // Used to differentiate unset mimaPreviousArtifacts from empty mimaPreviousArtifacts From dab3a7be8f71c054f5d6b187e5c5ce2c3d13b387 Mon Sep 17 00:00:00 2001 From: Dale Wijnand Date: Thu, 7 Apr 2022 11:06:04 +0100 Subject: [PATCH 2/2] Minor reorder and rename new "on-the-fly" API points --- .../tools/mima/plugin/MimaPlugin.scala | 100 ++++++++---------- 1 file changed, 43 insertions(+), 57 deletions(-) diff --git a/sbtplugin/src/main/scala/com/typesafe/tools/mima/plugin/MimaPlugin.scala b/sbtplugin/src/main/scala/com/typesafe/tools/mima/plugin/MimaPlugin.scala index 65c6173f..2e95ebd2 100644 --- a/sbtplugin/src/main/scala/com/typesafe/tools/mima/plugin/MimaPlugin.scala +++ b/sbtplugin/src/main/scala/com/typesafe/tools/mima/plugin/MimaPlugin.scala @@ -1,20 +1,16 @@ package com.typesafe.tools.mima package plugin -import com.typesafe.tools.mima.core.Problem -import sbt.* -import sbt.Keys.* - +import sbt.*, Keys.* +import core.* /** MiMa's sbt plugin. */ object MimaPlugin extends AutoPlugin { override def trigger = allRequirements object autoImport extends MimaKeys - import autoImport.* - override def globalSettings: Seq[Def.Setting[_]] = Seq( mimaPreviousArtifacts := NoPreviousArtifacts, mimaExcludeAnnotations := Nil, @@ -25,30 +21,6 @@ object MimaPlugin extends AutoPlugin { mimaCheckDirection := "backward", ) - trait ArtifactsToClassfiles { - def apply(artifacts: Set[ModuleID]): Map[ModuleID, File] - } - - val artifactsToClassfiles = Def.task { - val depRes = mimaDependencyResolution.value - val taskStreams = streams.value - val smi = scalaModuleInfo.value - new ArtifactsToClassfiles { - override def apply(artifacts: Set[sbt.ModuleID]) = - artifacts match { - case _: NoPreviousArtifacts.type => NoPreviousClassfiles - case previousArtifacts => - previousArtifacts.iterator.map { m => - val moduleId = CrossVersion(m, smi) match { - case Some(f) => m.withName(f(m.name)).withCrossVersion(CrossVersion.disabled) - case None => m - } - moduleId -> SbtMima.getPreviousArtifact(moduleId, depRes, taskStreams) - }.toMap - } - } - } - override def projectSettings: Seq[Def.Setting[_]] = Seq( mimaReportBinaryIssues := { binaryIssuesIterator.value.foreach { case (moduleId, problems) => @@ -67,7 +39,7 @@ object MimaPlugin extends AutoPlugin { }, mimaDependencyResolution := dependencyResolution.value, mimaPreviousClassfiles := { - artifactsToClassfiles.value.apply(mimaPreviousArtifacts.value) + artifactsToClassfiles.value.toClassfiles(mimaPreviousArtifacts.value) }, mimaCurrentClassfiles := (Compile / classDirectory).value, mimaFindBinaryIssues := binaryIssuesIterator.value.toMap, @@ -80,50 +52,64 @@ object MimaPlugin extends AutoPlugin { @deprecated("Switch to enablePlugins(MimaPlugin)", "0.7.0") def mimaDefaultSettings: Seq[Setting[_]] = globalSettings ++ buildSettings ++ projectSettings - private def binaryIssueFilters = Def.task { - val noSigs = core.ProblemFilters.exclude[core.IncompatibleSignatureProblem]("*") - mimaBinaryIssueFilters.value ++ (if (mimaReportSignatureProblems.value) Nil else Seq(noSigs)) + trait ArtifactsToClassfiles { + def toClassfiles(previousArtifacts: Set[ModuleID]): Map[ModuleID, File] } trait BinaryIssuesFinder { - def iterator(prevClassFiles: Map[ModuleID, File], - direction: String): Iterator[(ModuleID, (List[Problem], List[Problem]))] + def runMima(prevClassFiles: Map[ModuleID, File], checkDirection: String) + : Iterator[(ModuleID, (List[Problem], List[Problem]))] } - // Allows reuse between mimaFindBinaryIssues and mimaReportBinaryIssues - // without blowing up the Akka build's heap - def binaryIssuesFinder = Def.task { + val artifactsToClassfiles: Def.Initialize[Task[ArtifactsToClassfiles]] = Def.task { + val depRes = mimaDependencyResolution.value + val taskStreams = streams.value + val smi = scalaModuleInfo.value + previousArtifacts => previousArtifacts match { + case _: NoPreviousArtifacts.type => NoPreviousClassfiles + case previousArtifacts => + previousArtifacts.iterator.map { m => + val moduleId = CrossVersion(m, smi) match { + case Some(f) => m.withName(f(m.name)).withCrossVersion(CrossVersion.disabled) + case None => m + } + moduleId -> SbtMima.getPreviousArtifact(moduleId, depRes, taskStreams) + }.toMap + } + } + + val binaryIssuesFinder: Def.Initialize[Task[BinaryIssuesFinder]] = Def.task { val log = streams.value.log val currClassfiles = mimaCurrentClassfiles.value val cp = (mimaFindBinaryIssues / fullClasspath).value val sv = scalaVersion.value val excludeAnnots = mimaExcludeAnnotations.value.toList - val requirePrevious = mimaFailOnNoPrevious.value + val failOnNoPrevious = mimaFailOnNoPrevious.value val projName = name.value - new BinaryIssuesFinder { - override def iterator(prevClassfiles: Map[sbt.ModuleID, sbt.File], direction: String) = { - if (prevClassfiles eq NoPreviousClassfiles) { - val msg = "mimaPreviousArtifacts not set, not analyzing binary compatibility" - if (requirePrevious) - sys.error(msg) - else - log.info(s"$projName: $msg") - } else if (prevClassfiles.isEmpty) - log.info(s"$projName: mimaPreviousArtifacts is empty, not analyzing binary compatibility.") - - prevClassfiles.iterator.map { case (moduleId, prevClassfiles) => - moduleId -> SbtMima.runMima(prevClassfiles, currClassfiles, cp, direction, sv, log, excludeAnnots) - } + (prevClassfiles, checkDirection) => { + if (prevClassfiles eq NoPreviousClassfiles) { + val msg = "mimaPreviousArtifacts not set, not analyzing binary compatibility" + if (failOnNoPrevious) sys.error(msg) else log.info(s"$projName: $msg") + } else if (prevClassfiles.isEmpty) { + log.info(s"$projName: mimaPreviousArtifacts is empty, not analyzing binary compatibility.") + } + + prevClassfiles.iterator.map { case (moduleId, prevClassfiles) => + moduleId -> SbtMima.runMima(prevClassfiles, currClassfiles, cp, checkDirection, sv, log, excludeAnnots) } } } + private val binaryIssueFilters = Def.task { + val noSigs = ProblemFilters.exclude[IncompatibleSignatureProblem]("*") + mimaBinaryIssueFilters.value ++ (if (mimaReportSignatureProblems.value) Nil else Seq(noSigs)) + } + // Allows reuse between mimaFindBinaryIssues and mimaReportBinaryIssues // without blowing up the Akka build's heap - private def binaryIssuesIterator = Def.task { - val it = binaryIssuesFinder.value - it.iterator(mimaPreviousClassfiles.value, mimaCheckDirection.value) + private val binaryIssuesIterator = Def.task { + binaryIssuesFinder.value.runMima(mimaPreviousClassfiles.value, mimaCheckDirection.value) } // Used to differentiate unset mimaPreviousArtifacts from empty mimaPreviousArtifacts