diff --git a/.scalafix.conf b/.scalafix.conf new file mode 100644 index 0000000..c5c7c76 --- /dev/null +++ b/.scalafix.conf @@ -0,0 +1,7 @@ +rules = [ + OrganizeImports +] + +OrganizeImports { + preset = INTELLIJ_2020_3 +} \ No newline at end of file diff --git a/.scalafmt.conf b/.scalafmt.conf new file mode 100644 index 0000000..ae1c1e1 --- /dev/null +++ b/.scalafmt.conf @@ -0,0 +1,13 @@ +version = 3.4.3 + +runner.dialect = scala212 + +preset = defaultWithAlign + +maxColumn = 100 + +indent { + callSite = 2 + defnSite = 2 + extendSite = 2 +} \ No newline at end of file diff --git a/build.sbt b/build.sbt index 7ae494d..924a121 100644 --- a/build.sbt +++ b/build.sbt @@ -1,11 +1,11 @@ // Common settings -name := "sbt-tpolecat" -description := "scalac options for the enlightened" +name := "sbt-tpolecat" +description := "scalac options for the enlightened" organization := "io.github.davidgregory084" organizationName := "David Gregory" -startYear := Some(2022) +startYear := Some(2022) licenses += ("Apache-2.0", url("https://www.apache.org/licenses/LICENSE-2.0.html")) scmInfo := Some( ScmInfo( @@ -34,20 +34,27 @@ addSbtPlugin("org.lyranthe.sbt" % "partial-unification" % "1.1.2") // License headers Compile / headerCreate := { (Compile / headerCreate).triggeredBy(Compile / compile).value } -Test / headerCreate := { (Test / headerCreate).triggeredBy(Test / compile).value } +Test / headerCreate := { (Test / headerCreate).triggeredBy(Test / compile).value } + +scalacOptions += "-Xlint:unused" libraryDependencies ++= Seq( - "org.scalatest" %% "scalatest" % "3.2.11" % Test, - "org.scalacheck" %% "scalacheck" % "1.15.4" % Test, + "org.scalatest" %% "scalatest" % "3.2.11" % Test, + "org.scalacheck" %% "scalacheck" % "1.15.4" % Test, "org.scalatestplus" %% "scalacheck-1-15" % "3.2.11.0" % Test ) +ThisBuild / semanticdbEnabled := true +ThisBuild / semanticdbVersion := scalafixSemanticdb.revision +ThisBuild / scalafixDependencies += "com.github.liancheng" %% "organize-imports" % "0.6.0" + // Testing scriptedBufferLog := false scriptedLaunchOpts := scriptedLaunchOpts.value ++ Seq( - "-Xmx1024M", "-Dplugin.version=" + version.value + "-Xmx1024M", + "-Dplugin.version=" + version.value ) test := { diff --git a/project/plugins.sbt b/project/plugins.sbt index 1f55580..44aacbc 100644 --- a/project/plugins.sbt +++ b/project/plugins.sbt @@ -3,3 +3,7 @@ addSbtPlugin("de.heikoseeberger" % "sbt-header" % "5.6.5") addSbtPlugin("com.geirsson" % "sbt-ci-release" % "1.5.7") addSbtPlugin("com.timushev.sbt" % "sbt-updates" % "0.6.2") + +addSbtPlugin("org.scalameta" % "sbt-scalafmt" % "2.4.6") + +addSbtPlugin("ch.epfl.scala" % "sbt-scalafix" % "0.9.34") diff --git a/src/main/scala/io/github/davidgregory084/OptionsMode.scala b/src/main/scala/io/github/davidgregory084/OptionsMode.scala new file mode 100644 index 0000000..6fe9077 --- /dev/null +++ b/src/main/scala/io/github/davidgregory084/OptionsMode.scala @@ -0,0 +1,23 @@ +/* + * Copyright 2022 David Gregory + * + * 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 io.github.davidgregory084 + +sealed abstract class OptionsMode extends Product with Serializable + +case object DevMode extends OptionsMode +case object CiMode extends OptionsMode +case object ReleaseMode extends OptionsMode diff --git a/src/main/scala/io/github/davidgregory084/ScalaVersion.scala b/src/main/scala/io/github/davidgregory084/ScalaVersion.scala new file mode 100644 index 0000000..65c132a --- /dev/null +++ b/src/main/scala/io/github/davidgregory084/ScalaVersion.scala @@ -0,0 +1,38 @@ +/* + * Copyright 2022 David Gregory + * + * 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 io.github.davidgregory084 + +import scala.Ordering.Implicits._ + +case class ScalaVersion(major: Long, minor: Long, patch: Long) { + def isBetween(addedVersion: ScalaVersion, removedVersion: ScalaVersion) = + this >= addedVersion && this < removedVersion +} + +object ScalaVersion { + val V2_11_0 = ScalaVersion(2, 11, 0) + val V2_12_0 = ScalaVersion(2, 12, 0) + val V2_13_0 = ScalaVersion(2, 13, 0) + val V2_13_3 = ScalaVersion(2, 13, 3) + val V2_13_4 = ScalaVersion(2, 13, 4) + val V2_13_5 = ScalaVersion(2, 13, 5) + val V2_13_6 = ScalaVersion(2, 13, 6) + val V3_0_0 = ScalaVersion(3, 0, 0) + + implicit val scalaVersionOrdering: Ordering[ScalaVersion] = + Ordering.by(version => (version.major, version.minor, version.patch)) +} diff --git a/src/main/scala/io/github/davidgregory084/ScalacOption.scala b/src/main/scala/io/github/davidgregory084/ScalacOption.scala new file mode 100644 index 0000000..2af04de --- /dev/null +++ b/src/main/scala/io/github/davidgregory084/ScalacOption.scala @@ -0,0 +1,22 @@ +/* + * Copyright 2022 David Gregory + * + * 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 io.github.davidgregory084 + +case class ScalacOption( + tokens: List[String], + isSupported: ScalaVersion => Boolean = _ => true +) diff --git a/src/main/scala/io/github/davidgregory084/ScalacOptions.scala b/src/main/scala/io/github/davidgregory084/ScalacOptions.scala new file mode 100644 index 0000000..8490f51 --- /dev/null +++ b/src/main/scala/io/github/davidgregory084/ScalacOptions.scala @@ -0,0 +1,358 @@ +/* + * Copyright 2022 David Gregory + * + * 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 io.github.davidgregory084 + +import scala.Ordering.Implicits._ +import scala.collection.immutable.ListSet + +trait ScalacOptions { + import ScalaVersion._ + + def encoding(enc: String) = ScalacOption(List("-encoding", enc)) + + val deprecation = ScalacOption( + List("-deprecation"), + version => version < V2_13_0 || version >= V3_0_0 + ) + + val feature = ScalacOption(List("-feature")) + + val languageExistentials = ScalacOption( + List("-language:existentials"), + version => version < V3_0_0 + ) + + val languageExperimentalMacros = ScalacOption( + List("-language:experimental.macros") + ) + + val languageHigherKinds = ScalacOption( + List("-language:higherKinds") + ) + + val languageImplicitConversions = ScalacOption( + List("-language:implicitConversions") + ) + + val languageFeatures: Set[ScalacOption] = ListSet( + languageExistentials, + languageExperimentalMacros, + languageHigherKinds, + languageImplicitConversions + ) + + val unchecked = ScalacOption(List("-unchecked")) + + def advancedOption(name: String, isSupported: ScalaVersion => Boolean = _ => true): ScalacOption = + ScalacOption(List(s"-X$name"), isSupported) + + val checkInit = + advancedOption("checkinit", version => version < V3_0_0) + + val fatalWarnings = + advancedOption("fatal-warnings", version => version < V2_13_0 || version >= V3_0_0) + + val lint = + advancedOption("lint", version => version < V2_11_0) + + def lintOption(name: String, isSupported: ScalaVersion => Boolean = _ => true) = + advancedOption(s"lint:$name", isSupported) + + def disableLintOption(name: String, isSupported: ScalaVersion => Boolean = _ => true) = + advancedOption(s"lint:-$name", isSupported) + + val lintAdaptedArgs = + lintOption("adapted-args", version => version.isBetween(V2_11_0, V3_0_0)) + + val lintByNameRightAssociative = + lintOption("by-name-right-associative", version => version.isBetween(V2_11_0, V2_13_0)) + + val lintConstant = + lintOption("constant", version => version.isBetween(V2_12_0, V3_0_0)) + + val lintDelayedInitSelect = + lintOption("delayedinit-select", version => version.isBetween(V2_11_0, V3_0_0)) + + val lintDeprecation = + lintOption("deprecation", version => version.isBetween(V2_13_0, V3_0_0)) + + val lintDocDetached = + lintOption("doc-detached", version => version.isBetween(V2_11_0, V3_0_0)) + + val lintImplicitRecursion = + lintOption("implicit-recursion", version => version.isBetween(V2_13_3, V3_0_0)) + + val lintImplicitNotFound = + lintOption("implicit-not-found", version => version.isBetween(V2_13_0, V3_0_0)) + + val lintInaccessible = + lintOption("inaccessible", version => version.isBetween(V2_11_0, V3_0_0)) + + val lintInferAny = + lintOption("infer-any", version => version.isBetween(V2_11_0, V3_0_0)) + + val lintMissingInterpolator = + lintOption("missing-interpolator", version => version.isBetween(V2_11_0, V3_0_0)) + + val lintNullaryOverride = + lintOption("nullary-override", version => version.isBetween(V2_11_0, V2_13_0)) + + val lintNullaryUnit = + lintOption("nullary-unit", version => version.isBetween(V2_11_0, V3_0_0)) + + val lintOptionImplicit = + lintOption("option-implicit", version => version.isBetween(V2_11_0, V3_0_0)) + + val lintPackageObjectClasses = + lintOption("package-object-classes", version => version.isBetween(V2_11_0, V3_0_0)) + + val lintPolyImplicitOverload = + lintOption("poly-implicit-overload", version => version.isBetween(V2_11_0, V3_0_0)) + + val lintPrivateShadow = + lintOption("private-shadow", version => version.isBetween(V2_11_0, V3_0_0)) + + val lintStarsAlign = + lintOption("stars-align", version => version.isBetween(V2_11_0, V3_0_0)) + + val lintStrictUnsealedPatmat = + lintOption("strict-unsealed-patmat", version => version.isBetween(V2_13_4, V3_0_0)) + + val lintTypeParameterShadow = + lintOption("type-parameter-shadow", version => version.isBetween(V2_11_0, V3_0_0)) + + val lintUnsoundMatch = + lintOption("unsound-match", version => version.isBetween(V2_11_0, V2_13_0)) + + val disableLintBynameImplicit = + disableLintOption("byname-implicit", version => version.isBetween(V2_13_3, V3_0_0)) + + val lintOptions: Set[ScalacOption] = ListSet( + lintAdaptedArgs, + lintByNameRightAssociative, + lintConstant, + lintDelayedInitSelect, + lintDeprecation, + lintDocDetached, + lintImplicitRecursion, + lintImplicitNotFound, + lintInaccessible, + lintInferAny, + lintMissingInterpolator, + lintNullaryOverride, + lintNullaryUnit, + lintOptionImplicit, + lintPackageObjectClasses, + lintPolyImplicitOverload, + lintPrivateShadow, + lintStarsAlign, + lintStrictUnsealedPatmat, + lintTypeParameterShadow, + lintUnsoundMatch, + disableLintBynameImplicit + ) + + val advancedOptions: Set[ScalacOption] = ListSet( + checkInit, + lint + ) ++ lintOptions + + def privateOption(name: String, isSupported: ScalaVersion => Boolean = _ => true) = + ScalacOption(List(s"-Y$name"), isSupported) + + val privateNoAdaptedArgs = + privateOption("no-adapted-args", version => version < V2_13_0) + + val privateKindProjector = + privateOption("kind-projector", version => version >= V3_0_0) + + def privateWarnOption(name: String, isSupported: ScalaVersion => Boolean = _ => true) = + privateOption(s"warn-$name", isSupported) + + val privateWarnDeadCode = + privateWarnOption("dead-code", version => version < V2_13_0) + + val privateWarnExtraImplicit = + privateWarnOption("extra-implicit", version => version.isBetween(V2_12_0, V2_13_0)) + + val privateWarnInaccessible = + privateWarnOption("inaccessible", version => version < V2_11_0) + + val privateWarnNullaryOverride = + privateWarnOption("nullary-override", version => version < V2_13_0) + + val privateWarnNullaryUnit = + privateWarnOption("nullary-unit", version => version < V2_13_0) + + val privateWarnNumericWiden = + privateWarnOption("numeric-widen", version => version < V2_13_0) + + val privateWarnUnused = + privateWarnOption("unused", version => version.isBetween(V2_11_0, V2_12_0)) + + val privateWarnUnusedImport = + privateWarnOption("unused-import", version => version.isBetween(V2_11_0, V2_12_0)) + + val privateWarnValueDiscard = + privateWarnOption("value-discard", version => version < V2_13_0) + + def privateWarnUnusedOption(name: String, isSupported: ScalaVersion => Boolean = _ => true) = + privateWarnOption(s"unused:$name", isSupported) + + val privateWarnUnusedImplicits = + privateWarnUnusedOption("implicits", version => version.isBetween(V2_12_0, V2_13_0)) + + val privateWarnUnusedImports = + privateWarnUnusedOption("imports", version => version.isBetween(V2_12_0, V2_13_0)) + + val privateWarnUnusedLocals = + privateWarnUnusedOption("locals", version => version.isBetween(V2_12_0, V2_13_0)) + + val privateWarnUnusedParams = + privateWarnUnusedOption("params", version => version.isBetween(V2_12_0, V2_13_0)) + + val privateWarnUnusedPatVars = + privateWarnUnusedOption("patvars", version => version.isBetween(V2_12_0, V2_13_0)) + + val privateWarnUnusedPrivates = + privateWarnUnusedOption("privates", version => version.isBetween(V2_12_0, V2_13_0)) + + val privateWarnUnusedOptions: Set[ScalacOption] = ListSet( + privateWarnUnusedImplicits, + privateWarnUnusedImports, + privateWarnUnusedLocals, + privateWarnUnusedParams, + privateWarnUnusedPatVars, + privateWarnUnusedPrivates + ) + + val privateWarnOptions: Set[ScalacOption] = ListSet( + privateWarnDeadCode, + privateWarnExtraImplicit, + privateWarnInaccessible, + privateWarnNullaryOverride, + privateWarnNullaryUnit, + privateWarnNumericWiden, + privateWarnUnused, + privateWarnUnusedImport, + privateWarnValueDiscard + ) ++ privateWarnUnusedOptions + + val privateOptions: Set[ScalacOption] = ListSet( + privateNoAdaptedArgs, + privateKindProjector + ) ++ privateWarnOptions + + def warnOption(name: String, isSupported: ScalaVersion => Boolean = _ => true) = + ScalacOption(List(s"-W$name"), isSupported) + + val warnDeadCode = + warnOption("dead-code", version => version.isBetween(V2_13_0, V3_0_0)) + + val warnExtraImplicit = + warnOption("extra-implicit", version => version.isBetween(V2_13_0, V3_0_0)) + + val warnNumericWiden = + warnOption("numeric-widen", version => version.isBetween(V2_13_0, V3_0_0)) + + val warnValueDiscard = + warnOption("value-discard", version => version.isBetween(V2_13_0, V3_0_0)) + + val warnError = + warnOption("error", version => version.isBetween(V2_13_0, V3_0_0)) + + def warnUnusedOption(name: String, isSupported: ScalaVersion => Boolean = _ => true) = + ScalacOption(List(s"-Wunused:$name"), isSupported) + + val warnUnusedNoWarn = + warnUnusedOption("nowarn", version => version.isBetween(V2_13_0, V3_0_0)) + + val warnUnusedImplicits = + warnUnusedOption("implicits", version => version.isBetween(V2_13_0, V3_0_0)) + + val warnUnusedExplicits = + warnUnusedOption("explicits", version => version.isBetween(V2_13_0, V3_0_0)) + + val warnUnusedImports = + warnUnusedOption("imports", version => version.isBetween(V2_13_0, V3_0_0)) + + val warnUnusedLocals = + warnUnusedOption("locals", version => version.isBetween(V2_13_0, V3_0_0)) + + val warnUnusedParams = + warnUnusedOption("params", version => version.isBetween(V2_13_0, V3_0_0)) + + val warnUnusedPatVars = + warnUnusedOption("patvars", version => version.isBetween(V2_13_0, V3_0_0)) + + val warnUnusedPrivates = + warnUnusedOption("privates", version => version.isBetween(V2_13_0, V3_0_0)) + + val warnUnusedOptions: Set[ScalacOption] = ListSet( + warnUnusedNoWarn, + warnUnusedImplicits, + warnUnusedExplicits, + warnUnusedImports, + warnUnusedLocals, + warnUnusedParams, + warnUnusedPatVars, + warnUnusedPrivates + ) + + val warnOptions: Set[ScalacOption] = ListSet( + warnDeadCode, + warnExtraImplicit, + warnNumericWiden, + warnValueDiscard + ) ++ warnUnusedOptions + + val fatalWarningOptions: Set[ScalacOption] = ListSet( + fatalWarnings, + warnError + ) + + val default: Set[ScalacOption] = ListSet( + encoding("utf8"), + deprecation, + feature, + unchecked + ) ++ languageFeatures ++ advancedOptions ++ privateOptions ++ warnOptions + + def optimizerOption(name: String, isSupported: ScalaVersion => Boolean = _ => true) = + ScalacOption(List(s"-opt$name"), isSupported) + + val optimizerMethodLocal = + optimizerOption(":l:method", version => version.isBetween(V2_12_0, V3_0_0)) + + val optimizerInline = + optimizerOption(":l:inline", version => version.isBetween(V2_12_0, V3_0_0)) + + val optimizerWarnings = + optimizerOption("-warnings", version => version.isBetween(V2_12_0, V3_0_0)) + + def optimizerInlineFrom(inlineFromPackages: String*) = + optimizerOption( + s"-inline-from:${inlineFromPackages.mkString(":")}", + version => version.isBetween(V2_12_0, V3_0_0) + ) + + def optimizerOptions(inlineFromPackages: String*): Set[ScalacOption] = ListSet( + optimizerMethodLocal, + optimizerInline, + optimizerInlineFrom(inlineFromPackages: _*) + ) +} diff --git a/src/main/scala/io/github/davidgregory084/TpolecatPlugin.scala b/src/main/scala/io/github/davidgregory084/TpolecatPlugin.scala index 611f875..13c4cd8 100644 --- a/src/main/scala/io/github/davidgregory084/TpolecatPlugin.scala +++ b/src/main/scala/io/github/davidgregory084/TpolecatPlugin.scala @@ -16,153 +16,127 @@ package io.github.davidgregory084 -import sbt._ import sbt.Keys._ +import sbt._ + import scala.util.Try object TpolecatPlugin extends AutoPlugin { override def trigger: PluginTrigger = allRequirements - case class Version(major: Long, minor: Long, patch: Long) - object Version { - val V2_11_0 = Version(2, 11, 0) - val V2_12_0 = Version(2, 12, 0) - val V2_13_0 = Version(2, 13, 0) - val V2_13_3 = Version(2, 13, 3) - val V2_13_4 = Version(2, 13, 4) - val V2_13_5 = Version(2, 13, 5) - val V2_13_6 = Version(2, 13, 6) - val V3_0_0 = Version(3, 0, 0) - - implicit val versionOrdering: Ordering[Version] = - Ordering.by(version => (version.major, version.minor, version.patch)) - } - - case class ScalacOption( - name: String, - addedIn: Option[Version] = None, - removedIn: Option[Version] = None - ) - - import Version._ - - val allScalacOptions = List( - ScalacOption("-deprecation", removedIn = Some(V2_13_0)), // Emit warning and location for usages of deprecated APIs. Not really removed but deprecated in 2.13. - ScalacOption("-deprecation", addedIn = Some(V3_0_0)), // Emit warning and location for usages of deprecated APIs. - ScalacOption("-feature"), // Emit warning and location for usages of features that should be imported explicitly. - ScalacOption("-language:existentials", removedIn = Some(V3_0_0)), // Existential types (besides wildcard types) can be written and inferred - ScalacOption("-language:experimental.macros"), // Allow macro definition (besides implementation and application) - ScalacOption("-language:higherKinds"), // Allow higher-kinded types - ScalacOption("-language:implicitConversions"), // Allow definition of implicit functions called views - ScalacOption("-unchecked"), // Enable additional warnings where generated code depends on assumptions. - ScalacOption("-Xcheckinit", removedIn = Some(V3_0_0)), // Wrap field accessors to throw an exception on uninitialized access. - ScalacOption("-Xfatal-warnings"), // Fail the compilation if there are any warnings. - ScalacOption("-Xlint", removedIn = Some(V2_11_0)), // Used to mean enable all linting options but now the syntax for that is different (-Xlint:_ I think) - ScalacOption("-Xlint:adapted-args", addedIn = Some(V2_11_0), removedIn = Some(V3_0_0)), // Warn if an argument list is modified to match the receiver. - ScalacOption("-Xlint:by-name-right-associative", addedIn = Some(V2_11_0), removedIn = Some(V2_13_0)), // By-name parameter of right associative operator. - ScalacOption("-Xlint:constant", addedIn = Some(V2_12_0), removedIn = Some(V3_0_0)), // Evaluation of a constant arithmetic expression results in an error. - ScalacOption("-Xlint:delayedinit-select", addedIn = Some(V2_11_0), removedIn = Some(V3_0_0)), // Selecting member of DelayedInit. - ScalacOption("-Xlint:deprecation", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // Emit warning and location for usages of deprecated APIs. - ScalacOption("-Xlint:doc-detached", addedIn = Some(V2_11_0), removedIn = Some(V3_0_0)), // A Scaladoc comment appears to be detached from its element. - ScalacOption("-Xlint:implicit-recursion", addedIn = Some(V2_13_3), removedIn = Some(V3_0_0)), // Warn when an implicit resolves to an enclosing self-definition - ScalacOption("-Xlint:implicit-not-found", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // Warn when an @implicitNotFound or @implicitAmbigous annotation references an invalid type parameter. - ScalacOption("-Xlint:inaccessible", addedIn = Some(V2_11_0), removedIn = Some(V3_0_0)), // Warn about inaccessible types in method signatures. - ScalacOption("-Xlint:infer-any", addedIn = Some(V2_11_0), removedIn = Some(V3_0_0)), // Warn when a type argument is inferred to be `Any`. - ScalacOption("-Xlint:missing-interpolator", addedIn = Some(V2_11_0), removedIn = Some(V3_0_0)), // A string literal appears to be missing an interpolator id. - ScalacOption("-Xlint:nullary-override", addedIn = Some(V2_11_0), removedIn = Some(V2_13_0)), // Warn when non-nullary `def f()' overrides nullary `def f'. - ScalacOption("-Xlint:nullary-unit", addedIn = Some(V2_11_0), removedIn = Some(V3_0_0)), // Warn when nullary methods return Unit. - ScalacOption("-Xlint:option-implicit", addedIn = Some(V2_11_0), removedIn = Some(V3_0_0)), // Option.apply used implicit view. - ScalacOption("-Xlint:package-object-classes", addedIn = Some(V2_11_0), removedIn = Some(V3_0_0)), // Class or object defined in package object. - ScalacOption("-Xlint:poly-implicit-overload", addedIn = Some(V2_11_0), removedIn = Some(V3_0_0)), // Parameterized overloaded implicit methods are not visible as view bounds. - ScalacOption("-Xlint:private-shadow", addedIn = Some(V2_11_0), removedIn = Some(V3_0_0)), // A private field (or class parameter) shadows a superclass field. - ScalacOption("-Xlint:stars-align", addedIn = Some(V2_11_0), removedIn = Some(V3_0_0)), // Pattern sequence wildcard must align with sequence component. - ScalacOption("-Xlint:strict-unsealed-patmat", addedIn = Some(V2_13_4), removedIn = Some(V3_0_0)), // Warn when a pattern match on an unsealed type may not be exhaustive - ScalacOption("-Xlint:type-parameter-shadow", addedIn = Some(V2_11_0), removedIn = Some(V3_0_0)), // A local type parameter shadows a type already in scope. - ScalacOption("-Xlint:unsound-match", addedIn = Some(V2_11_0), removedIn = Some(V2_13_0)), // Pattern match may not be typesafe. - ScalacOption("-Xlint:-byname-implicit", addedIn = Some(V2_13_3), removedIn = Some(V3_0_0)), // This conflicts with generic derivation used via shapeless - ScalacOption("-Wunused:nowarn", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // Ensure that a `@nowarn` annotation actually suppresses a warning. - ScalacOption("-Yno-adapted-args", removedIn = Some(V2_13_0)), // Do not adapt an argument list (either by inserting () or creating a tuple) to match the receiver. - ScalacOption("-Ywarn-dead-code", removedIn = Some(V2_13_0)), // Warn when dead code is identified. - ScalacOption("-Wdead-code", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // ^ Replaces the above - ScalacOption("-Ywarn-extra-implicit", addedIn = Some(V2_12_0), removedIn = Some(V2_13_0)), // Warn when more than one implicit parameter section is defined. - ScalacOption("-Wextra-implicit", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // ^ Replaces the above - ScalacOption("-Ywarn-inaccessible", removedIn = Some(V2_11_0)), // Warn about inaccessible types in method signatures. Alias for -Xlint:inaccessible so can be removed as of 2.11. - ScalacOption("-Ywarn-nullary-override", removedIn = Some(V2_13_0)), // Warn when non-nullary `def f()' overrides nullary `def f'. - ScalacOption("-Ywarn-nullary-unit", removedIn = Some(V2_13_0)), // Warn when nullary methods return Unit. - ScalacOption("-Ywarn-numeric-widen", removedIn = Some(V2_13_0)), // Warn when numerics are widened. - ScalacOption("-Wnumeric-widen", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // ^ Replaces the above - ScalacOption("-Ywarn-unused", addedIn = Some(V2_11_0), removedIn = Some(V2_12_0)), // Warn when local and private vals, vars, defs, and types are unused. - ScalacOption("-Ywarn-unused-import", addedIn = Some(V2_11_0), removedIn = Some(V2_12_0)), // Warn if an import selector is not referenced. - ScalacOption("-Ywarn-unused:implicits", addedIn = Some(V2_12_0), removedIn = Some(V2_13_0)), // Warn if an implicit parameter is unused. - ScalacOption("-Wunused:implicits", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // ^ Replaces the above - ScalacOption("-Wunused:explicits", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // Warn if an explicit parameter is unused. - ScalacOption("-Ywarn-unused:imports", addedIn = Some(V2_12_0), removedIn = Some(V2_13_0)), // Warn if an import selector is not referenced. - ScalacOption("-Wunused:imports", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // ^ Replaces the above - ScalacOption("-Ywarn-unused:locals", addedIn = Some(V2_12_0), removedIn = Some(V2_13_0)), // Warn if a local definition is unused. - ScalacOption("-Wunused:locals", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // ^ Replaces the above - ScalacOption("-Ywarn-unused:params", addedIn = Some(V2_12_0), removedIn = Some(V2_13_0)), // Warn if a value parameter is unused. - ScalacOption("-Wunused:params", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // ^ Replaces the above - ScalacOption("-Ywarn-unused:patvars", addedIn = Some(V2_12_0), removedIn = Some(V2_13_0)), // Warn if a variable bound in a pattern is unused. - ScalacOption("-Wunused:patvars", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // ^ Replaces the above - ScalacOption("-Ywarn-unused:privates", addedIn = Some(V2_12_0), removedIn = Some(V2_13_0)), // Warn if a private member is unused. - ScalacOption("-Wunused:privates", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // ^ Replaces the above - ScalacOption("-Ywarn-value-discard", removedIn = Some(V2_13_0)), // Warn when non-Unit expression results are unused. - ScalacOption("-Wvalue-discard", addedIn = Some(V2_13_0), removedIn = Some(V3_0_0)), // ^ Replaces the above - ScalacOption("-Ykind-projector", addedIn = Some(V3_0_0)), // Enables a subset of kind-projector syntax (see https://github.com/lampepfl/dotty/pull/7775) - ) + import ScalaVersion._ object autoImport { - def validFor(currentVersion: Version)(opt: ScalacOption)(implicit ord: Ordering[Version]) = { - val addedPriorTo = opt.addedIn - .map(addedVersion => ord.gteq(currentVersion, addedVersion)) - .getOrElse(true) - val notYetRemoved = opt.removedIn - .map(removedVersion => ord.lt(currentVersion, removedVersion)) - .getOrElse(true) - - addedPriorTo && notYetRemoved + object ScalacOptions extends ScalacOptions + + def scalacOptionsFor( + version: String, + modeScalacOptions: Set[ScalacOption] + ): Seq[String] = { + val supportedOptions = (CrossVersion.partialVersion(version), version.split('.')) match { + case (Some((0, min)), _) => // dotty prereleases use 0 as major version + modeScalacOptions + .filter(_.isSupported(V3_0_0)) // treat dotty prereleases as 3.0.0 + case (Some((maj, min)), Array(maj2, min2, patch)) + if maj.toString == maj2 && min.toString == min2 => + modeScalacOptions + .filter(_.isSupported(ScalaVersion(maj, min, Try(patch.toLong).getOrElse(0)))) + case (Some((maj, min)), _) => + modeScalacOptions + .filter(_.isSupported(ScalaVersion(maj, min, 0))) + case (None, _) => + Nil + } + + supportedOptions.toList.flatMap(_.tokens) } - def scalacOptionsFor(version: String): Seq[String] = - List( - "-encoding", "utf8" // Specify character encoding used by source files. - ) ++ { - - val flags = (CrossVersion.partialVersion(version), version.split('.')) match { - case (Some((0, min)), _) => // dotty prereleases use 0 as major version - allScalacOptions - .filter(validFor(V3_0_0)) // treat dotty prereleases as 3.0.0 - case (Some((maj, min)), Array(maj2, min2, patch)) if maj.toString == maj2 && min.toString == min2 => - allScalacOptions - .filter(validFor(Version(maj, min, Try(patch.toLong).getOrElse(0)))) - case (Some((maj, min)), _) => - allScalacOptions - .filter(validFor(Version(maj, min, 0))) - case (None, _) => - Nil - } - - flags.map(_.name) -} - val filterConsoleScalacOptions = { options: Seq[String] => - options.filterNot(Set( - "-Werror", - "-Wdead-code", - "-Wunused:imports", - "-Ywarn-unused", - "-Ywarn-unused:imports", - "-Ywarn-unused-import", - "-Ywarn-dead-code", - "-Xfatal-warnings" - )) + val tpolecatConsoleOptionsFilter = { options: Set[ScalacOption] => + options.filterNot( + ScalacOptions.privateWarnUnusedOptions ++ + ScalacOptions.warnUnusedOptions ++ + ScalacOptions.fatalWarningOptions + + ScalacOptions.privateWarnDeadCode + + ScalacOptions.warnDeadCode + ) } + + val tpolecatDefaultOptionsMode = settingKey[OptionsMode]( + "The default mode to use for configuring scalac options via the sbt-tpolecat plugin." + ) + + val tpolecatOptionsMode = settingKey[OptionsMode]( + "The mode to use for configuring scalac options via the sbt-tpolecat plugin." + ) + + val tpolecatDevModeEnvVar = settingKey[String]( + "The environment variable to use to enable the sbt-tpolecat development mode." + ) + + val tpolecatCiModeEnvVar = settingKey[String]( + "The environment variable to use to enable the sbt-tpolecat continuous integration mode." + ) + + val tpolecatReleaseModeEnvVar = settingKey[String]( + "The environment variable to use to enable the sbt-tpolecat release mode." + ) + + val tpolecatDevModeOptions = settingKey[Set[ScalacOption]]( + "The set of scalac options that will be applied by the sbt-tpolecat plugin in the development mode." + ) + + val tpolecatCiModeOptions = settingKey[Set[ScalacOption]]( + "The set of scalac options that will be applied by the sbt-tpolecat plugin in the continuous integration mode." + ) + + val tpolecatReleaseModeOptions = settingKey[Set[ScalacOption]]( + "The set of scalac options that will be applied by the sbt-tpolecat plugin in the release mode." + ) + + val tpolecatScalacOptions = settingKey[Set[ScalacOption]]( + "The set of scalac options that will be applied by the sbt-tpolecat plugin." + ) } import autoImport._ + val commandAliases = + addCommandAlias( + "tpolecatDevMode", + "set tpolecatOptionsMode := _root_.io.github.davidgregory084.DevMode" + ) ++ + addCommandAlias( + "tpolecatCiMode", + "set tpolecatOptionsMode := _root_.io.github.davidgregory084.CiMode" + ) ++ + addCommandAlias( + "tpolecatReleaseMode", + "set tpolecatOptionsMode := _root_.io.github.davidgregory084.ReleaseMode" + ) + override def projectSettings: Seq[Setting[_]] = Seq( - scalacOptions ++= scalacOptionsFor(scalaVersion.value), - Compile / console / scalacOptions ~= filterConsoleScalacOptions, - Test / console / scalacOptions ~= filterConsoleScalacOptions - ) + scalacOptions := scalacOptionsFor(scalaVersion.value, tpolecatScalacOptions.value), + tpolecatDefaultOptionsMode := DevMode, + tpolecatDevModeEnvVar := "SBT_TPOLECAT_DEV", + tpolecatCiModeEnvVar := "SBT_TPOLECAT_CI", + tpolecatReleaseModeEnvVar := "SBT_TPOLECAT_RELEASE", + tpolecatDevModeOptions := ScalacOptions.default, + tpolecatCiModeOptions := ScalacOptions.default ++ ScalacOptions.fatalWarningOptions, + tpolecatReleaseModeOptions := tpolecatCiModeOptions.value + ScalacOptions.optimizerMethodLocal, + tpolecatOptionsMode := { + if (sys.env.get(tpolecatReleaseModeEnvVar.value).nonEmpty) ReleaseMode + else if (sys.env.get(tpolecatCiModeEnvVar.value).nonEmpty) CiMode + else if (sys.env.get(tpolecatDevModeEnvVar.value).nonEmpty) DevMode + else tpolecatDefaultOptionsMode.value + }, + tpolecatScalacOptions := { + tpolecatOptionsMode.value match { + case DevMode => tpolecatDevModeOptions.value + case CiMode => tpolecatCiModeOptions.value + case ReleaseMode => tpolecatReleaseModeOptions.value + } + }, + Compile / console / tpolecatScalacOptions ~= tpolecatConsoleOptionsFilter, + Test / console / tpolecatScalacOptions ~= tpolecatConsoleOptionsFilter + ) ++ commandAliases } diff --git a/src/sbt-test/sbt-tpolecat/scalacOptions/build.sbt b/src/sbt-test/sbt-tpolecat/scalacOptions/build.sbt index 372e677..ed4c23b 100644 --- a/src/sbt-test/sbt-tpolecat/scalacOptions/build.sbt +++ b/src/sbt-test/sbt-tpolecat/scalacOptions/build.sbt @@ -1,47 +1,235 @@ import scala.util.Try +import munit.Assertions._ -val scala2Versions = Seq( - "2.11.12", - "2.12.15", - "2.13.8" -) +val Scala211 = "2.11.12" +val Scala212 = "2.12.15" +val Scala213 = "2.13.8" +val Scala30 = "3.0.2" +val Scala31 = "3.1.1" -val scala3Versions = Seq( - "3.0.2", - "3.1.1" +crossScalaVersions := Seq( + Scala211, + Scala212, + Scala213, + Scala30, + Scala31 ) -crossScalaVersions := { - object ExtractVersion { - val sbtVersionMatch = raw"(\d+)\.(\d+)\.(\d+)".r - def unapply(sbtVersion: String): Option[(Long, Long, Long)] = { - sbtVersion match { - case sbtVersionMatch(major, minor, patch) => - for { - maj <- Try(major.toLong).toOption - min <- Try(minor.toLong).toOption - p <- Try(patch.toLong).toOption - } yield (maj, min, p) - - case _ => - None - } - } +tpolecatReleaseModeOptions ++= ScalacOptions.optimizerOptions("**") + +val Scala211Options = + Seq( + "-encoding", + "utf8", + "-deprecation", + "-feature", + "-unchecked", + "-language:existentials", + "-language:experimental.macros", + "-language:higherKinds", + "-language:implicitConversions", + "-Xcheckinit", + "-Xlint:adapted-args", + "-Xlint:by-name-right-associative", + "-Xlint:delayedinit-select", + "-Xlint:doc-detached", + "-Xlint:inaccessible", + "-Xlint:infer-any", + "-Xlint:missing-interpolator", + "-Xlint:nullary-override", + "-Xlint:nullary-unit", + "-Xlint:option-implicit", + "-Xlint:package-object-classes", + "-Xlint:poly-implicit-overload", + "-Xlint:private-shadow", + "-Xlint:stars-align", + "-Xlint:type-parameter-shadow", + "-Xlint:unsound-match", + "-Yno-adapted-args", + "-Ywarn-dead-code", + "-Ywarn-nullary-override", + "-Ywarn-nullary-unit", + "-Ywarn-numeric-widen", + "-Ywarn-unused", + "-Ywarn-unused-import", + "-Ywarn-value-discard" + ) + +val Scala212Options = + Seq( + "-encoding", + "utf8", + "-deprecation", + "-feature", + "-unchecked", + "-language:existentials", + "-language:experimental.macros", + "-language:higherKinds", + "-language:implicitConversions", + "-Xcheckinit", + "-Xlint:adapted-args", + "-Xlint:by-name-right-associative", + "-Xlint:constant", + "-Xlint:delayedinit-select", + "-Xlint:doc-detached", + "-Xlint:inaccessible", + "-Xlint:infer-any", + "-Xlint:missing-interpolator", + "-Xlint:nullary-override", + "-Xlint:nullary-unit", + "-Xlint:option-implicit", + "-Xlint:package-object-classes", + "-Xlint:poly-implicit-overload", + "-Xlint:private-shadow", + "-Xlint:stars-align", + "-Xlint:type-parameter-shadow", + "-Xlint:unsound-match", + "-Yno-adapted-args", + "-Ywarn-dead-code", + "-Ywarn-extra-implicit", + "-Ywarn-nullary-override", + "-Ywarn-nullary-unit", + "-Ywarn-numeric-widen", + "-Ywarn-value-discard", + "-Ywarn-unused:implicits", + "-Ywarn-unused:imports", + "-Ywarn-unused:locals", + "-Ywarn-unused:params", + "-Ywarn-unused:patvars", + "-Ywarn-unused:privates" + ) + +val Scala213Options = + Seq( + "-encoding", + "utf8", + "-feature", + "-unchecked", + "-language:existentials", + "-language:experimental.macros", + "-language:higherKinds", + "-language:implicitConversions", + "-Xcheckinit", + "-Xlint:adapted-args", + "-Xlint:constant", + "-Xlint:delayedinit-select", + "-Xlint:deprecation", + "-Xlint:doc-detached", + "-Xlint:implicit-recursion", + "-Xlint:implicit-not-found", + "-Xlint:inaccessible", + "-Xlint:infer-any", + "-Xlint:missing-interpolator", + "-Xlint:nullary-unit", + "-Xlint:option-implicit", + "-Xlint:package-object-classes", + "-Xlint:poly-implicit-overload", + "-Xlint:private-shadow", + "-Xlint:stars-align", + "-Xlint:strict-unsealed-patmat", + "-Xlint:type-parameter-shadow", + "-Xlint:-byname-implicit", + "-Wdead-code", + "-Wextra-implicit", + "-Wnumeric-widen", + "-Wvalue-discard", + "-Wunused:nowarn", + "-Wunused:implicits", + "-Wunused:explicits", + "-Wunused:imports", + "-Wunused:locals", + "-Wunused:params", + "-Wunused:patvars", + "-Wunused:privates" + ) + +val Scala30Options = + Seq( + "-encoding", + "utf8", + "-deprecation", + "-feature", + "-unchecked", + "-language:experimental.macros", + "-language:higherKinds", + "-language:implicitConversions", + "-Ykind-projector" + ) + +val Scala31Options = + Seq( + "-encoding", + "utf8", + "-deprecation", + "-feature", + "-unchecked", + "-language:experimental.macros", + "-language:higherKinds", + "-language:implicitConversions", + "-Ykind-projector" + ) + +TaskKey[Unit]("checkDevMode") := { + val scalaV = scalaVersion.value + + val expectedOptions = scalaV match { + case Scala211 => Scala211Options ++ Seq("-Ypartial-unification") + case Scala212 => Scala212Options ++ Seq("-Ypartial-unification") + case Scala213 => Scala213Options + case Scala30 => Scala30Options + case Scala31 => Scala31Options + } + + val actualOptions = scalacOptions.value + + assertEquals(actualOptions, expectedOptions) +} + +TaskKey[Unit]("checkCiMode") := { + val scalaV = scalaVersion.value + + val expectedOptions = scalaV match { + case Scala211 => Scala211Options ++ Seq("-Xfatal-warnings", "-Ypartial-unification") + case Scala212 => Scala212Options ++ Seq("-Xfatal-warnings", "-Ypartial-unification") + case Scala213 => Scala213Options ++ Seq("-Werror") + case Scala30 => Scala30Options ++ Seq("-Xfatal-warnings") + case Scala31 => Scala31Options ++ Seq("-Xfatal-warnings") } - def supportedSbtVersions(major: Long, minor: Long, patch: Long): Boolean = - (major, minor, patch) match { - case (1, 5, patch) if patch >= 2 => true - case (1, minor, _) if minor > 5 => true - case (major, _, _) if major > 1 => true - case _ => false - } + val actualOptions = scalacOptions.value - sbtVersion.value match { - case ExtractVersion(major, minor, patch) if supportedSbtVersions(major, minor, patch) => - scala2Versions ++ scala3Versions + assertEquals(actualOptions, expectedOptions) +} + +TaskKey[Unit]("checkReleaseMode") := { + val scalaV = scalaVersion.value - case _ => - scala2Versions + val expectedOptions = scalaV match { + case Scala211 => + Scala211Options ++ Seq( + "-Xfatal-warnings", + "-Ypartial-unification" + ) + case Scala212 => + Scala212Options ++ Seq( + "-Xfatal-warnings", + "-opt:l:method", + "-opt:l:inline", + "-opt-inline-from:**", + "-Ypartial-unification" + ) + case Scala213 => + Scala213Options ++ Seq( + "-Werror", + "-opt:l:method", + "-opt:l:inline", + "-opt-inline-from:**" + ) + case Scala30 => Scala30Options ++ Seq("-Xfatal-warnings") + case Scala31 => Scala31Options ++ Seq("-Xfatal-warnings") } + + val actualOptions = scalacOptions.value + + assertEquals(actualOptions, expectedOptions) } diff --git a/src/sbt-test/sbt-tpolecat/scalacOptions/project/build.sbt b/src/sbt-test/sbt-tpolecat/scalacOptions/project/build.sbt new file mode 100644 index 0000000..bbffd4b --- /dev/null +++ b/src/sbt-test/sbt-tpolecat/scalacOptions/project/build.sbt @@ -0,0 +1 @@ +libraryDependencies += "org.scalameta" %% "munit" % "0.7.29" diff --git a/src/sbt-test/sbt-tpolecat/scalacOptions/test b/src/sbt-test/sbt-tpolecat/scalacOptions/test index 6e9376c..a70e1a0 100644 --- a/src/sbt-test/sbt-tpolecat/scalacOptions/test +++ b/src/sbt-test/sbt-tpolecat/scalacOptions/test @@ -1,2 +1,15 @@ -> +show scalacOptions -> +compile \ No newline at end of file +# Dev mode is the default mode +> +checkDevMode +> +compile +# Check dev mode options +> +tpolecatDevMode +> +checkDevMode +> +compile +# Check CI mode options +> +tpolecatCiMode +> +checkCiMode +> +compile +# Check release mode options +> +tpolecatReleaseMode +> +checkReleaseMode +> +compile diff --git a/src/test/scala/io/github/davidgregory084/TpolecatPluginSuite.scala b/src/test/scala/io/github/davidgregory084/TpolecatPluginSuite.scala index ec041d5..b1a9073 100644 --- a/src/test/scala/io/github/davidgregory084/TpolecatPluginSuite.scala +++ b/src/test/scala/io/github/davidgregory084/TpolecatPluginSuite.scala @@ -16,191 +16,204 @@ package io.github.davidgregory084 +import org.scalacheck.Gen import org.scalatest.funsuite.AnyFunSuite import org.scalatestplus.scalacheck.ScalaCheckDrivenPropertyChecks -import org.scalacheck.Gen -import org.scalacheck.Prop._ -import TpolecatPlugin._ -import TpolecatPlugin.autoImport._ +import scala.Ordering.Implicits._ class TpolecatPluginSuite extends AnyFunSuite with ScalaCheckDrivenPropertyChecks { val versionGen = Gen.chooseNum(0L, 20L) - test("valid when neither addedIn nor removedIn") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val scalacOption = ScalacOption("-some-opt", None, None) - assert( - validFor(currentVersion)(scalacOption), - "Should be valid when neither addedIn nor removedIn" - ) + test("valid when no version predicate") { + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val scalacOption = ScalacOption(List("-some-opt")) + assert( + scalacOption.isSupported(currentVersion), + "Should be valid when neither addedIn nor removedIn" + ) } } test("valid when added in past major release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val addedVersion = Version(currentMaj - 1, 0, 0) - val scalacOption = ScalacOption("-some-opt", addedIn = Some(addedVersion)) - assert( - validFor(currentVersion)(scalacOption), - "Should be valid when addedIn matches past major release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val addedVersion = ScalaVersion(currentMaj - 1, 0, 0) + val scalacOption = ScalacOption(List("-some-opt"), version => version >= addedVersion) + assert( + scalacOption.isSupported(currentVersion), + "Should be valid when addedIn matches past major release" + ) } } test("valid when added in past minor release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val addedVersion = Version(currentMaj, currentMin - 1, 0) - val scalacOption = ScalacOption("-some-opt", addedIn = Some(addedVersion)) - assert( - validFor(currentVersion)(scalacOption), - "Should be valid when addedIn matches past minor release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val addedVersion = ScalaVersion(currentMaj, currentMin - 1, 0) + val scalacOption = ScalacOption(List("-some-opt"), version => version >= addedVersion) + assert( + scalacOption.isSupported(currentVersion), + "Should be valid when addedIn matches past minor release" + ) } } test("valid when added in past patch release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val addedVersion = Version(currentMaj, currentMin, currentPatch - 1) - val scalacOption = ScalacOption("-some-opt", addedIn = Some(addedVersion)) - assert( - validFor(currentVersion)(scalacOption), - "Should be valid when addedIn matches past patch release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val addedVersion = ScalaVersion(currentMaj, currentMin, currentPatch - 1) + val scalacOption = ScalacOption(List("-some-opt"), version => version >= addedVersion) + assert( + scalacOption.isSupported(currentVersion), + "Should be valid when addedIn matches past patch release" + ) } } test("valid when added in this minor/patch release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val scalacOption = ScalacOption("-some-opt", addedIn = Some(currentVersion)) - assert( - validFor(currentVersion)(scalacOption), - "Should be valid when addedIn matches this minor/patch release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val scalacOption = ScalacOption(List("-some-opt"), version => version >= currentVersion) + assert( + scalacOption.isSupported(currentVersion), + "Should be valid when addedIn matches this minor/patch release" + ) } } test("not valid when added in a future major release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val addedVersion = Version(currentMaj + 1, currentMin, currentPatch) - val scalacOption = ScalacOption("-some-opt", addedIn = Some(addedVersion)) - assert( - !validFor(currentVersion)(scalacOption), - "Should not be valid when addedIn matches a future major release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val addedVersion = ScalaVersion(currentMaj + 1, currentMin, currentPatch) + val scalacOption = ScalacOption(List("-some-opt"), version => version >= addedVersion) + assert( + !scalacOption.isSupported(currentVersion), + "Should not be valid when addedIn matches a future major release" + ) } } test("not valid when added in a future minor release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val addedVersion = Version(currentMaj, currentMin + 1, currentPatch) - val scalacOption = ScalacOption("-some-opt", addedIn = Some(addedVersion)) - assert( - !validFor(currentVersion)(scalacOption), - "Should not be valid when addedIn matches a future minor release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val addedVersion = ScalaVersion(currentMaj, currentMin + 1, currentPatch) + val scalacOption = ScalacOption(List("-some-opt"), version => version >= addedVersion) + assert( + !scalacOption.isSupported(currentVersion), + "Should not be valid when addedIn matches a future minor release" + ) } } test("not valid when added in a future patch release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val addedVersion = Version(currentMaj, currentMin, currentPatch + 1) - val scalacOption = ScalacOption("-some-opt", addedIn = Some(addedVersion)) - assert( - !validFor(currentVersion)(scalacOption), - "Should not be valid when addedIn matches a future patch release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val addedVersion = ScalaVersion(currentMaj, currentMin, currentPatch + 1) + val scalacOption = ScalacOption(List("-some-opt"), version => version >= addedVersion) + assert( + !scalacOption.isSupported(currentVersion), + "Should not be valid when addedIn matches a future patch release" + ) } } test("valid when removed in next major release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val removedVersion = Version(currentMaj + 1, 0, 0) - val scalacOption = ScalacOption("-some-opt", None, removedIn = Some(removedVersion)) - assert( - validFor(currentVersion)(scalacOption), - "Should be valid when removedIn matches next major release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val removedVersion = ScalaVersion(currentMaj + 1, 0, 0) + val scalacOption = ScalacOption(List("-some-opt"), version => version < removedVersion) + assert( + scalacOption.isSupported(currentVersion), + "Should be valid when removedIn matches next major release" + ) } } test("valid when removed in next minor release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val removedVersion = Version(currentMaj, currentMin + 1, currentPatch) - val scalacOption = ScalacOption("-some-opt", None, removedIn = Some(removedVersion)) - assert( - validFor(currentVersion)(scalacOption), - "Should be valid when removedIn matches next minor release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val removedVersion = ScalaVersion(currentMaj, currentMin + 1, currentPatch) + val scalacOption = ScalacOption(List("-some-opt"), version => version < removedVersion) + assert( + scalacOption.isSupported(currentVersion), + "Should be valid when removedIn matches next minor release" + ) } } test("valid when removed in next patch release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val removedVersion = Version(currentMaj, currentMin, currentPatch + 1) - val scalacOption = ScalacOption("-some-opt", None, removedIn = Some(removedVersion)) - assert( - validFor(currentVersion)(scalacOption), - "Should be valid when removedIn matches next patch release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val removedVersion = ScalaVersion(currentMaj, currentMin, currentPatch + 1) + val scalacOption = ScalacOption(List("-some-opt"), version => version < removedVersion) + assert( + scalacOption.isSupported(currentVersion), + "Should be valid when removedIn matches next patch release" + ) } } test("not valid when removed in this minor/patch release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val scalacOption = ScalacOption("-some-opt", None, removedIn = Some(currentVersion)) - assert( - !validFor(currentVersion)(scalacOption), - "Should not be valid when removedIn matches this minor/patch release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val scalacOption = ScalacOption(List("-some-opt"), version => version < currentVersion) + assert( + !scalacOption.isSupported(currentVersion), + "Should not be valid when removedIn matches this minor/patch release" + ) } } test("not valid when removed in an old major release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val removedVersion = Version(currentMaj - 1, currentMin, currentPatch) - val scalacOption = ScalacOption("-some-opt", None, removedIn = Some(removedVersion)) - assert( - !validFor(currentVersion)(scalacOption), - "Should not be valid when removedIn matches an old major release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val removedVersion = ScalaVersion(currentMaj - 1, currentMin, currentPatch) + val scalacOption = ScalacOption(List("-some-opt"), version => version < removedVersion) + assert( + !scalacOption.isSupported(currentVersion), + "Should not be valid when removedIn matches an old major release" + ) } } test("not valid when removed in an old minor release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val removedVersion = Version(currentMaj, currentMin - 1, currentPatch) - val scalacOption = ScalacOption("-some-opt", None, removedIn = Some(removedVersion)) - assert( - !validFor(currentVersion)(scalacOption), - "Should not be valid when removedIn matches an old minor release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val removedVersion = ScalaVersion(currentMaj, currentMin - 1, currentPatch) + val scalacOption = ScalacOption(List("-some-opt"), version => version < removedVersion) + assert( + !scalacOption.isSupported(currentVersion), + "Should not be valid when removedIn matches an old minor release" + ) } } test("not valid when removed in an old patch release") { - forAll(versionGen, versionGen, versionGen) { (currentMaj: Long, currentMin: Long, currentPatch: Long) => - val currentVersion = Version(currentMaj, currentMin, currentPatch) - val removedVersion = Version(currentMaj, currentMin, currentPatch - 1) - val scalacOption = ScalacOption("-some-opt", None, removedIn = Some(removedVersion)) - assert( - !validFor(currentVersion)(scalacOption), - "Should not be valid when removedIn matches an old patch release" - ) + forAll(versionGen, versionGen, versionGen) { + (currentMaj: Long, currentMin: Long, currentPatch: Long) => + val currentVersion = ScalaVersion(currentMaj, currentMin, currentPatch) + val removedVersion = ScalaVersion(currentMaj, currentMin, currentPatch - 1) + val scalacOption = ScalacOption(List("-some-opt"), version => version < removedVersion) + assert( + !scalacOption.isSupported(currentVersion), + "Should not be valid when removedIn matches an old patch release" + ) } } }