diff --git a/.editorconfig b/.editorconfig new file mode 100644 index 00000000..0e7d8d87 --- /dev/null +++ b/.editorconfig @@ -0,0 +1,11 @@ +root = true + +[*] +charset = utf-8 +end_of_line = lf +insert_final_newline = true +trim_trailing_whitespace = true + +[*.{scala,sbt}] +indent_style = space +indent_size = 2 diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 05a46e3e..867690f1 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -15,11 +15,6 @@ on: tags: [v*] env: - PGP_PASSPHRASE: ${{ secrets.PGP_PASSPHRASE }} - SONATYPE_PASSWORD: ${{ secrets.SONATYPE_PASSWORD }} - SONATYPE_CREDENTIAL_HOST: ${{ secrets.SONATYPE_CREDENTIAL_HOST }} - SONATYPE_USERNAME: ${{ secrets.SONATYPE_USERNAME }} - PGP_SECRET: ${{ secrets.PGP_SECRET }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} jobs: @@ -30,25 +25,25 @@ jobs: os: [ubuntu-latest] scala: [2.12.17] java: [temurin@8, temurin@17] - project: [rootJVM] + project: [sbt-typelevelJVM] runs-on: ${{ matrix.os }} steps: - name: Checkout current branch (full) - uses: actions/checkout@v2 + uses: actions/checkout@v3 with: fetch-depth: 0 - name: Download Java (temurin@8) id: download-java-temurin-8 if: matrix.java == 'temurin@8' - uses: typelevel/download-java@v1 + uses: typelevel/download-java@v2 with: distribution: temurin java-version: 8 - name: Setup Java (temurin@8) if: matrix.java == 'temurin@8' - uses: actions/setup-java@v2 + uses: actions/setup-java@v3 with: distribution: jdkfile java-version: 8 @@ -57,21 +52,21 @@ jobs: - name: Download Java (temurin@17) id: download-java-temurin-17 if: matrix.java == 'temurin@17' - uses: typelevel/download-java@v1 + uses: typelevel/download-java@v2 with: distribution: temurin java-version: 17 - name: Setup Java (temurin@17) if: matrix.java == 'temurin@17' - uses: actions/setup-java@v2 + uses: actions/setup-java@v3 with: distribution: jdkfile java-version: 17 jdkFile: ${{ steps.download-java-temurin-17.outputs.jdkFile }} - name: Cache sbt - uses: actions/cache@v2 + uses: actions/cache@v3 with: path: | ~/.sbt @@ -105,16 +100,16 @@ jobs: run: sbt 'project ${{ matrix.project }}' '++${{ matrix.scala }}' doc - 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 scalafix/target target .js/target mdocs/target site/target ci-signing/target mergify/target unidoc/target mima/target .jvm/target .native/target no-publish/target sonatype/target ci/target sonatype-ci-release/target core/target settings/target project/target + if: github.event_name != 'pull_request' && (startsWith(github.ref, 'refs/tags/v') || github.ref == 'refs/heads/main') + run: mkdir -p github/target github-actions/target kernel/target versioning/target ci-release/target scalafix/target .jvm/target mdocs/target site/target ci-signing/target mergify/target unidoc/target .native/target mima/target no-publish/target sonatype/target ci/target sonatype-ci-release/target core/target settings/target target .js/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 scalafix/target target .js/target mdocs/target site/target ci-signing/target mergify/target unidoc/target mima/target .jvm/target .native/target no-publish/target sonatype/target ci/target sonatype-ci-release/target core/target settings/target project/target + if: github.event_name != 'pull_request' && (startsWith(github.ref, 'refs/tags/v') || github.ref == 'refs/heads/main') + run: tar cf targets.tar github/target github-actions/target kernel/target versioning/target ci-release/target scalafix/target .jvm/target mdocs/target site/target ci-signing/target mergify/target unidoc/target .native/target mima/target no-publish/target sonatype/target ci/target sonatype-ci-release/target core/target settings/target target .js/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') - uses: actions/upload-artifact@v2 + if: github.event_name != 'pull_request' && (startsWith(github.ref, 'refs/tags/v') || github.ref == 'refs/heads/main') + uses: actions/upload-artifact@v3 with: name: target-${{ matrix.os }}-${{ matrix.java }}-${{ matrix.scala }}-${{ matrix.project }} path: targets.tar @@ -122,7 +117,7 @@ jobs: publish: name: Publish Artifacts needs: [build] - if: github.event_name != 'pull_request' && (startsWith(github.ref, 'refs/tags/v') || github.ref == 'refs/heads/series/0.4') + if: github.event_name != 'pull_request' && (startsWith(github.ref, 'refs/tags/v') || github.ref == 'refs/heads/main') strategy: matrix: os: [ubuntu-latest] @@ -131,21 +126,21 @@ jobs: runs-on: ${{ matrix.os }} steps: - name: Checkout current branch (full) - uses: actions/checkout@v2 + uses: actions/checkout@v3 with: fetch-depth: 0 - name: Download Java (temurin@8) id: download-java-temurin-8 if: matrix.java == 'temurin@8' - uses: typelevel/download-java@v1 + uses: typelevel/download-java@v2 with: distribution: temurin java-version: 8 - name: Setup Java (temurin@8) if: matrix.java == 'temurin@8' - uses: actions/setup-java@v2 + uses: actions/setup-java@v3 with: distribution: jdkfile java-version: 8 @@ -154,21 +149,21 @@ jobs: - name: Download Java (temurin@17) id: download-java-temurin-17 if: matrix.java == 'temurin@17' - uses: typelevel/download-java@v1 + uses: typelevel/download-java@v2 with: distribution: temurin java-version: 17 - name: Setup Java (temurin@17) if: matrix.java == 'temurin@17' - uses: actions/setup-java@v2 + uses: actions/setup-java@v3 with: distribution: jdkfile java-version: 17 jdkFile: ${{ steps.download-java-temurin-17.outputs.jdkFile }} - name: Cache sbt - uses: actions/cache@v2 + uses: actions/cache@v3 with: path: | ~/.sbt @@ -179,30 +174,102 @@ jobs: ~/Library/Caches/Coursier/v1 key: ${{ runner.os }}-sbt-cache-v2-${{ hashFiles('**/*.sbt') }}-${{ hashFiles('project/build.properties') }} - - name: Download target directories (2.12.17, rootJVM) - uses: actions/download-artifact@v2 + - name: Download target directories (2.12.17, sbt-typelevelJVM) + uses: actions/download-artifact@v3 with: - name: target-${{ matrix.os }}-${{ matrix.java }}-2.12.17-rootJVM + name: target-${{ matrix.os }}-${{ matrix.java }}-2.12.17-sbt-typelevelJVM - - name: Inflate target directories (2.12.17, rootJVM) + - name: Inflate target directories (2.12.17, sbt-typelevelJVM) run: | tar xf targets.tar rm targets.tar - name: Import signing key if: env.PGP_SECRET != '' && env.PGP_PASSPHRASE == '' + env: + PGP_SECRET: ${{ secrets.PGP_SECRET }} + PGP_PASSPHRASE: ${{ secrets.PGP_PASSPHRASE }} run: echo $PGP_SECRET | base64 -di | gpg --import - name: Import signing key and strip passphrase if: env.PGP_SECRET != '' && env.PGP_PASSPHRASE != '' + env: + PGP_SECRET: ${{ secrets.PGP_SECRET }} + PGP_PASSPHRASE: ${{ secrets.PGP_PASSPHRASE }} run: | echo "$PGP_SECRET" | base64 -di > /tmp/signing-key.gpg echo "$PGP_PASSPHRASE" | gpg --pinentry-mode loopback --passphrase-fd 0 --import /tmp/signing-key.gpg (echo "$PGP_PASSPHRASE"; echo; echo) | gpg --command-fd 0 --pinentry-mode loopback --change-passphrase $(gpg --list-secret-keys --with-colons 2> /dev/null | grep '^sec:' | cut --delimiter ':' --fields 5 | tail -n 1) - name: Publish + env: + SONATYPE_USERNAME: ${{ secrets.SONATYPE_USERNAME }} + SONATYPE_PASSWORD: ${{ secrets.SONATYPE_PASSWORD }} + SONATYPE_CREDENTIAL_HOST: ${{ secrets.SONATYPE_CREDENTIAL_HOST }} run: sbt '++${{ matrix.scala }}' tlRelease + dependency-submission: + name: Submit Dependencies + if: github.event_name != 'pull_request' + strategy: + matrix: + os: [ubuntu-latest] + scala: [2.12.17] + java: [temurin@8] + runs-on: ${{ matrix.os }} + steps: + - name: Checkout current branch (full) + uses: actions/checkout@v3 + with: + fetch-depth: 0 + + - name: Download Java (temurin@8) + id: download-java-temurin-8 + if: matrix.java == 'temurin@8' + uses: typelevel/download-java@v2 + with: + distribution: temurin + java-version: 8 + + - name: Setup Java (temurin@8) + if: matrix.java == 'temurin@8' + uses: actions/setup-java@v3 + with: + distribution: jdkfile + java-version: 8 + jdkFile: ${{ steps.download-java-temurin-8.outputs.jdkFile }} + + - name: Download Java (temurin@17) + id: download-java-temurin-17 + if: matrix.java == 'temurin@17' + uses: typelevel/download-java@v2 + with: + distribution: temurin + java-version: 17 + + - name: Setup Java (temurin@17) + if: matrix.java == 'temurin@17' + uses: actions/setup-java@v3 + with: + distribution: jdkfile + java-version: 17 + jdkFile: ${{ steps.download-java-temurin-17.outputs.jdkFile }} + + - name: Cache sbt + uses: actions/cache@v3 + with: + path: | + ~/.sbt + ~/.ivy2/cache + ~/.coursier/cache/v1 + ~/.cache/coursier/v1 + ~/AppData/Local/Coursier/Cache/v1 + ~/Library/Caches/Coursier/v1 + key: ${{ runner.os }}-sbt-cache-v2-${{ hashFiles('**/*.sbt') }}-${{ hashFiles('project/build.properties') }} + + - name: Submit Dependencies + uses: scalacenter/sbt-dependency-submission@v2 + site: name: Generate Site strategy: @@ -213,21 +280,21 @@ jobs: runs-on: ${{ matrix.os }} steps: - name: Checkout current branch (full) - uses: actions/checkout@v2 + uses: actions/checkout@v3 with: fetch-depth: 0 - name: Download Java (temurin@8) id: download-java-temurin-8 if: matrix.java == 'temurin@8' - uses: typelevel/download-java@v1 + uses: typelevel/download-java@v2 with: distribution: temurin java-version: 8 - name: Setup Java (temurin@8) if: matrix.java == 'temurin@8' - uses: actions/setup-java@v2 + uses: actions/setup-java@v3 with: distribution: jdkfile java-version: 8 @@ -236,21 +303,21 @@ jobs: - name: Download Java (temurin@17) id: download-java-temurin-17 if: matrix.java == 'temurin@17' - uses: typelevel/download-java@v1 + uses: typelevel/download-java@v2 with: distribution: temurin java-version: 17 - name: Setup Java (temurin@17) if: matrix.java == 'temurin@17' - uses: actions/setup-java@v2 + uses: actions/setup-java@v3 with: distribution: jdkfile java-version: 17 jdkFile: ${{ steps.download-java-temurin-17.outputs.jdkFile }} - name: Cache sbt - uses: actions/cache@v2 + uses: actions/cache@v3 with: path: | ~/.sbt diff --git a/.mergify.yml b/.mergify.yml index 6f1f6ce2..d5ed22eb 100644 --- a/.mergify.yml +++ b/.mergify.yml @@ -12,8 +12,8 @@ pull_request_rules: - or: - body~=labels:.*early-semver-patch - body~=labels:.*early-semver-minor - - status-success=Build and Test (ubuntu-latest, 2.12.17, temurin@8, rootJVM) - - status-success=Build and Test (ubuntu-latest, 2.12.17, temurin@17, rootJVM) + - status-success=Build and Test (ubuntu-latest, 2.12.17, temurin@8, sbt-typelevelJVM) + - status-success=Build and Test (ubuntu-latest, 2.12.17, temurin@17, sbt-typelevelJVM) - '#approved-reviews-by>=1' actions: merge: {} diff --git a/build.sbt b/build.sbt index 7c5ce872..86edf9fb 100644 --- a/build.sbt +++ b/build.sbt @@ -1,10 +1,9 @@ name := "sbt-typelevel" -ThisBuild / tlBaseVersion := "0.4" -ThisBuild / tlCiReleaseBranches := Seq("series/0.4") +ThisBuild / tlBaseVersion := "0.5" ThisBuild / tlSitePublishBranch := Some("series/0.4") ThisBuild / crossScalaVersions := Seq("2.12.17") -ThisBuild / developers := List( +ThisBuild / developers ++= List( tlGitHubDev("armanbilge", "Arman Bilge"), tlGitHubDev("rossabaker", "Ross A. Baker"), tlGitHubDev("ChristopherDavenport", "Christopher Davenport"), @@ -30,7 +29,9 @@ ThisBuild / scalafixDependencies ++= Seq( "com.github.liancheng" %% "organize-imports" % "0.6.0" ) -lazy val root = tlCrossRootProject.aggregate( +val MunitVersion = "0.7.29" + +lazy val `sbt-typelevel` = tlCrossRootProject.aggregate( kernel, noPublish, settings, @@ -55,7 +56,8 @@ lazy val kernel = project .in(file("kernel")) .enablePlugins(SbtPlugin) .settings( - name := "sbt-typelevel-kernel" + name := "sbt-typelevel-kernel", + libraryDependencies += "org.scalameta" %% "munit" % MunitVersion % Test ) lazy val noPublish = project @@ -210,6 +212,7 @@ lazy val docs = project "Laika" -> url("https://planet42.github.io/Laika/"), "sbt-unidoc" -> url("https://github.com/sbt/sbt-unidoc") ), + tlSiteIsTypelevelProject := true, mdocVariables ++= { import coursier.complete.Complete import java.time._ diff --git a/ci-signing/src/main/scala/org/typelevel/sbt/TypelevelCiSigningPlugin.scala b/ci-signing/src/main/scala/org/typelevel/sbt/TypelevelCiSigningPlugin.scala index 848352f1..bee1f008 100644 --- a/ci-signing/src/main/scala/org/typelevel/sbt/TypelevelCiSigningPlugin.scala +++ b/ci-signing/src/main/scala/org/typelevel/sbt/TypelevelCiSigningPlugin.scala @@ -31,15 +31,12 @@ object TypelevelCiSigningPlugin extends AutoPlugin { override def trigger = allRequirements override def buildSettings = Seq( - githubWorkflowEnv ++= Map( - "PGP_SECRET" -> s"$${{ secrets.PGP_SECRET }}", - "PGP_PASSPHRASE" -> s"$${{ secrets.PGP_PASSPHRASE }}" - ), githubWorkflowPublishPreamble := Seq( WorkflowStep.Run( // if your key is not passphrase-protected List("echo $PGP_SECRET | base64 -di | gpg --import"), name = Some("Import signing key"), - cond = Some("env.PGP_SECRET != '' && env.PGP_PASSPHRASE == ''") + cond = Some("env.PGP_SECRET != '' && env.PGP_PASSPHRASE == ''"), + env = env ), WorkflowStep.Run( // if your key is passphrase-protected List( @@ -48,7 +45,8 @@ object TypelevelCiSigningPlugin extends AutoPlugin { "(echo \"$PGP_PASSPHRASE\"; echo; echo) | gpg --command-fd 0 --pinentry-mode loopback --change-passphrase $(gpg --list-secret-keys --with-colons 2> /dev/null | grep '^sec:' | cut --delimiter ':' --fields 5 | tail -n 1)" ), name = Some("Import signing key and strip passphrase"), - cond = Some("env.PGP_SECRET != '' && env.PGP_PASSPHRASE != ''") + cond = Some("env.PGP_SECRET != '' && env.PGP_PASSPHRASE != ''"), + env = env ) ) ) @@ -59,4 +57,9 @@ object TypelevelCiSigningPlugin extends AutoPlugin { gpgWarnOnFailure := isSnapshot.value ) + private val env = Map( + "PGP_SECRET" -> s"$${{ secrets.PGP_SECRET }}", + "PGP_PASSPHRASE" -> s"$${{ secrets.PGP_PASSPHRASE }}" + ) + } diff --git a/ci/src/main/scala/org/typelevel/sbt/CrossRootProject.scala b/ci/src/main/scala/org/typelevel/sbt/CrossRootProject.scala index 1fbaed70..98223bf0 100644 --- a/ci/src/main/scala/org/typelevel/sbt/CrossRootProject.scala +++ b/ci/src/main/scala/org/typelevel/sbt/CrossRootProject.scala @@ -20,8 +20,8 @@ import org.typelevel.sbt.gha.GenerativePlugin.autoImport._ import sbt._ /** - * Simultaneously creates a `root`, `rootJVM`, `rootJS`, and `rootNative` project, and - * automatically enables the `NoPublishPlugin`. + * Simultaneously creates a root project, a Scala JVM aggregate project, a Scala.js aggregate + * project, a Scala Native aggregate project and automatically enables the `NoPublishPlugin`. */ final class CrossRootProject private ( val all: Project, @@ -96,17 +96,26 @@ final class CrossRootProject private ( } object CrossRootProject { - def unapply(root: CrossRootProject): Some[(Project, Project, Project, Project)] = - Some((root.all, root.jvm, root.js, root.native)) - - private[sbt] def apply(): CrossRootProject = new CrossRootProject( - Project("root", file(".")), - Project("rootJVM", file(".jvm")), - Project("rootJS", file(".js")), - Project("rootNative", file(".native")) + def unapply(rootProject: CrossRootProject): Some[(Project, Project, Project, Project)] = + Some((rootProject.all, rootProject.jvm, rootProject.js, rootProject.native)) + + def apply(id: String): CrossRootProject = new CrossRootProject( + Project(id, file(".")), + Project(s"${id}JVM", file(".jvm")), + Project(s"${id}JS", file(".js")), + Project(s"${id}Native", file(".native")) ).enablePlugins(NoPublishPlugin, TypelevelCiCrossPlugin) } +/** + * This trait provides an anonymous setting giving access to the local root project ID. + */ +private[sbt] trait RootProjectId { + protected def rootProjectId = Def.setting { + (LocalRootProject / Keys.thisProject).value.id + } +} + /** * This plugin is used internally by CrossRootProject. */ @@ -122,36 +131,39 @@ object TypelevelCiCrossPlugin extends AutoPlugin { // The following plugins are used internally to support CrossRootProject. -object TypelevelCiJVMPlugin extends AutoPlugin { +object TypelevelCiJVMPlugin extends AutoPlugin with RootProjectId { override def requires = TypelevelCiCrossPlugin override def buildSettings: Seq[Setting[_]] = Seq( - githubWorkflowBuildMatrixAdditions ~= { matrix => - matrix.updated("project", matrix("project") ::: "rootJVM" :: Nil) + githubWorkflowBuildMatrixAdditions := { + val matrix = githubWorkflowBuildMatrixAdditions.value + matrix.updated("project", matrix("project") ::: s"${rootProjectId.value}JVM" :: Nil) } ) } -object TypelevelCiJSPlugin extends AutoPlugin { +object TypelevelCiJSPlugin extends AutoPlugin with RootProjectId { override def requires = TypelevelCiCrossPlugin override def buildSettings: Seq[Setting[_]] = Seq( - githubWorkflowBuildMatrixAdditions ~= { matrix => - matrix.updated("project", matrix("project") ::: "rootJS" :: Nil) + githubWorkflowBuildMatrixAdditions := { + val matrix = githubWorkflowBuildMatrixAdditions.value + matrix.updated("project", matrix("project") ::: s"${rootProjectId.value}JS" :: Nil) }, githubWorkflowBuildMatrixExclusions ++= { githubWorkflowJavaVersions .value .tail - .map(java => MatrixExclude(Map("project" -> "rootJS", "java" -> java.render))) + .map(java => + MatrixExclude(Map("project" -> s"${rootProjectId.value}JS", "java" -> java.render))) }, - githubWorkflowBuild ~= { steps => - steps.flatMap { + githubWorkflowBuild := { + githubWorkflowBuild.value.flatMap { case testStep @ WorkflowStep.Sbt(List("test"), _, _, _, _, _) => val fastOptStep = WorkflowStep.Sbt( List("Test/scalaJSLinkerResult"), name = Some("scalaJSLink"), - cond = Some("matrix.project == 'rootJS'") + cond = Some(s"matrix.project == '${rootProjectId.value}JS'") ) List(fastOptStep, testStep) case step => List(step) @@ -161,26 +173,29 @@ object TypelevelCiJSPlugin extends AutoPlugin { } -object TypelevelCiNativePlugin extends AutoPlugin { +object TypelevelCiNativePlugin extends AutoPlugin with RootProjectId { override def requires = TypelevelCiCrossPlugin override def buildSettings: Seq[Setting[_]] = Seq( - githubWorkflowBuildMatrixAdditions ~= { matrix => - matrix.updated("project", matrix("project") ::: "rootNative" :: Nil) + githubWorkflowBuildMatrixAdditions := { + val matrix = githubWorkflowBuildMatrixAdditions.value + matrix.updated("project", matrix("project") ::: s"${rootProjectId.value}Native" :: Nil) }, githubWorkflowBuildMatrixExclusions ++= { githubWorkflowJavaVersions .value .tail - .map(java => MatrixExclude(Map("project" -> "rootNative", "java" -> java.render))) + .map(java => + MatrixExclude( + Map("project" -> s"${rootProjectId.value}Native", "java" -> java.render))) }, - githubWorkflowBuild ~= { steps => - steps.flatMap { + githubWorkflowBuild := { + githubWorkflowBuild.value.flatMap { case testStep @ WorkflowStep.Sbt(List("test"), _, _, _, _, _) => val nativeLinkStep = WorkflowStep.Sbt( List("Test/nativeLink"), name = Some("nativeLink"), - cond = Some("matrix.project == 'rootNative'") + cond = Some(s"matrix.project == '${rootProjectId.value}Native'") ) List(nativeLinkStep, testStep) case step => List(step) diff --git a/ci/src/main/scala/org/typelevel/sbt/CrossRootProjectMacros.scala b/ci/src/main/scala/org/typelevel/sbt/CrossRootProjectMacros.scala new file mode 100644 index 00000000..5e819ad8 --- /dev/null +++ b/ci/src/main/scala/org/typelevel/sbt/CrossRootProjectMacros.scala @@ -0,0 +1,68 @@ +/* + * 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 + +import scala.annotation.tailrec +import scala.reflect.macros.blackbox + +private[sbt] object CrossRootProjectMacros { + + // Copied from sbt.std.KeyMacro + def definingValName(c: blackbox.Context, invalidEnclosingTree: String => String): String = { + import c.universe.{Apply => ApplyTree, _} + val methodName = c.macroApplication.symbol.name + def processName(n: Name): String = + n.decodedName + .toString + .trim // trim is not strictly correct, but macros don't expose the API necessary + @tailrec def enclosingVal(trees: List[c.Tree]): String = { + trees match { + case ValDef(_, name, _, _) :: _ => processName(name) + case (_: ApplyTree | _: Select | _: TypeApply) :: xs => enclosingVal(xs) + // lazy val x: X = has this form for some reason (only when the explicit type is present, though) + case Block(_, _) :: DefDef(mods, name, _, _, _, _) :: _ if mods.hasFlag(Flag.LAZY) => + processName(name) + case _ => + c.error(c.enclosingPosition, invalidEnclosingTree(methodName.decodedName.toString)) + "" + } + } + enclosingVal(enclosingTrees(c).toList) + } + + // Copied from sbt.std.KeyMacro + def enclosingTrees(c: blackbox.Context): Seq[c.Tree] = + c.asInstanceOf[reflect.macros.runtime.Context] + .callsiteTyper + .context + .enclosingContextChain + .map(_.tree.asInstanceOf[c.Tree]) + + def crossRootProjectImpl(c: blackbox.Context): c.Expr[CrossRootProject] = { + import c.universe._ + + val enclosingValName = definingValName( + c, + methodName => + s"""$methodName must be directly assigned to a val, such as `val x = $methodName`. Alternatively, you can use `org.typelevel.sbt.CrossRootProject.apply`""" + ) + + val name = c.Expr[String](Literal(Constant(enclosingValName))) + + reify { CrossRootProject(name.splice) } + } +} diff --git a/ci/src/main/scala/org/typelevel/sbt/TypelevelCiPlugin.scala b/ci/src/main/scala/org/typelevel/sbt/TypelevelCiPlugin.scala index ddb1f24e..cd129d9a 100644 --- a/ci/src/main/scala/org/typelevel/sbt/TypelevelCiPlugin.scala +++ b/ci/src/main/scala/org/typelevel/sbt/TypelevelCiPlugin.scala @@ -20,15 +20,20 @@ import com.typesafe.tools.mima.plugin.MimaPlugin import org.typelevel.sbt.gha.GenerativePlugin import org.typelevel.sbt.gha.GenerativePlugin.autoImport._ import org.typelevel.sbt.gha.GitHubActionsPlugin +import org.typelevel.sbt.gha.WorkflowStep import sbt._ +import scala.language.experimental.macros + +import Keys._ + object TypelevelCiPlugin extends AutoPlugin { override def requires = GitHubActionsPlugin && GenerativePlugin && MimaPlugin override def trigger = allRequirements object autoImport { - def tlCrossRootProject: CrossRootProject = CrossRootProject() + def tlCrossRootProject: CrossRootProject = macro CrossRootProjectMacros.crossRootProjectImpl lazy val tlCiHeaderCheck = settingKey[Boolean]("Whether to do header check in CI (default: false)") @@ -40,6 +45,9 @@ object TypelevelCiPlugin extends AutoPlugin { settingKey[Boolean]("Whether to do MiMa binary issues check in CI (default: true)") lazy val tlCiDocCheck = settingKey[Boolean]("Whether to build API docs in CI (default: true)") + + lazy val tlCiDependencyGraphJob = + settingKey[Boolean]("Whether to add a job to submit dependencies to GH (default: true)") } import autoImport._ @@ -50,6 +58,7 @@ object TypelevelCiPlugin extends AutoPlugin { tlCiScalafixCheck := false, tlCiMimaBinaryIssueCheck := true, tlCiDocCheck := true, + tlCiDependencyGraphJob := true, githubWorkflowTargetBranches ++= Seq( "!update/**", // ignore steward branches "!pr/**" // escape-hatch to disable ci on a branch @@ -123,7 +132,24 @@ object TypelevelCiPlugin extends AutoPlugin { style ++ test ++ scalafix ++ mima ++ doc }, - githubWorkflowJavaVersions := Seq(JavaSpec.temurin("8")) + githubWorkflowJavaVersions := Seq(JavaSpec.temurin("8")), + githubWorkflowAddedJobs ++= { + val dependencySubmission = + if (tlCiDependencyGraphJob.value) + List( + WorkflowJob( + "dependency-submission", + "Submit Dependencies", + scalas = List(scalaVersion.value), + javas = List(githubWorkflowJavaVersions.value.head), + steps = githubWorkflowJobSetup.value.toList :+ + WorkflowStep.DependencySubmission, + cond = Some("github.event_name != 'pull_request'") + )) + else Nil + + dependencySubmission + } ) private val primaryJavaCond = Def.setting { diff --git a/core/build.sbt b/core/build.sbt index 7856f1a8..8cb03c98 100644 --- a/core/build.sbt +++ b/core/build.sbt @@ -1,3 +1,3 @@ addSbtPlugin("org.scalameta" % "sbt-scalafmt" % "2.4.6") -addSbtPlugin("de.heikoseeberger" % "sbt-header" % "5.6.0") +addSbtPlugin("de.heikoseeberger" % "sbt-header" % "5.6.5") addSbtPlugin("org.portable-scala" % "sbt-scalajs-crossproject" % "1.2.0") diff --git a/core/src/main/scala/org/typelevel/sbt/TypelevelPlugin.scala b/core/src/main/scala/org/typelevel/sbt/TypelevelPlugin.scala index 720aaed4..e339be97 100644 --- a/core/src/main/scala/org/typelevel/sbt/TypelevelPlugin.scala +++ b/core/src/main/scala/org/typelevel/sbt/TypelevelPlugin.scala @@ -114,10 +114,4 @@ object TypelevelPlugin extends AutoPlugin { // override for bincompat override def projectSettings = immutable.Seq.empty - - private val primaryJavaCond = Def.setting { - val java = githubWorkflowJavaVersions.value.head - s"matrix.java == '${java.render}'" - } - } diff --git a/docs/site.md b/docs/site.md index eedb7c40..1d64c901 100644 --- a/docs/site.md +++ b/docs/site.md @@ -1,6 +1,9 @@ # sbt-typelevel-site -**sbt-typelevel-site** is an optional plugin for generating websites with [mdoc](https://scalameta.org/mdoc/) and [Laika](https://planet42.github.io/Laika/) and deploying to GitHub Pages from CI. You can add it to your build alongside either the **sbt-typelevel** or **sbt-typelevel-ci-release** plugin or also use it stand-alone. +**sbt-typelevel-site** is an optional plugin for generating websites with [mdoc](https://scalameta.org/mdoc/) +and [Laika](https://planet42.github.io/Laika/) and deploying to GitHub Pages from CI. +You can add it to your build alongside either the **sbt-typelevel** or **sbt-typelevel-ci-release** plugin +or also use it stand-alone. ## Quick start @@ -19,9 +22,11 @@ ThisBuild / tlSitePublishBranch := Some("main") lazy val docs = project.in(file("site")).enablePlugins(TypelevelSitePlugin) ``` -Place your `.md` files in the `docs/` directory of your project. To preview locally, run `docs/tlSitePreview`. This will start a preview server at http://localhost:4242. +Place your `.md` files in the `docs/` directory of your project. To preview locally, run `docs/tlSitePreview`. +This will start a preview server at http://localhost:4242. -The site is generated using [mdoc](https://scalameta.org/mdoc/) and [Laika](https://planet42.github.io/Laika/) and published to the `gh-pages` branch on every push to the specified branch. +The site is generated using [mdoc](https://scalameta.org/mdoc/) and [Laika](https://planet42.github.io/Laika/) +and published to the `gh-pages` branch on every push to the specified branch. You will also need to configure your repository settings: @@ -30,15 +35,109 @@ You will also need to configure your repository settings: 2. Set the GitHub pages source to the `/` (root) directory on the `gh-pages` branch. `https://github.com/{user}/{repo}/settings/pages` + +## Configuration + +If you run the plugin with its defaults it will generate a site that will look like this documentation. + +Below we'll describe how the default settings differ from Laika standalone +as well as a few pointers for the most relevant customization options. + + +### Site Default Settings + +Whereas Laika standalone is a general purpose tool, this plugin's domain is project documentation +which allows us to make a few additional assumptions and add useful defaults based on those. + +On top of the defaults of Laika standalone, sbt-typelevel-site adds: + +* GitHubFlavor for Markdown is enabled by default (e.g. for fenced code blocks). +* Laika's builtin syntax highlighting is enabled by default (which does not require JavaScript highlighters). +* Metadata is pre-populated based on sbt settings (e.g. title, author, version). +* A link to the GitHub repository is inserted into the top navigation bar based on the output of `git ls-remote --get-url`. +* If you define the `tlSiteApiUrl` setting, a link to the API documentation is inserted into the top navigation bar + and a redirect for `/api/` to that URL will be added to your site. + + +#### Additional Defaults for Typelevel Projects + +If the generated documentation is for a Typelevel project, you can optionally enable a set of additional defaults +on top of the generic defaults listed in the previous section: + +```scala +tlSiteIsTypelevelProject := true +``` + +With the flag above (which defaults to `false`) these additional settings apply: + +* The home link in the top navigation bar carries the Typelevel logo and points to the Typelevel site. +* Links to the Typelevel Discord and Typelevel Twitter are inserted into the top navigation bar. +* The Typelevel favicons are used for the generared site. +* A default footer for Typelevel projects is added to the bottom of each page. +* Theme support for the browser's dark mode is disabled. + + +### Customization + +All customization options are based on Laika's configuration APIs for which we refer you to the comprehensive [Laika manual][Laika] +and specifically the [`laikaTheme` setting](https://planet42.github.io/Laika/0.18/02-running-laika/01-sbt-plugin.html#laikatheme-setting). + +@:callout(warning) +For all code examples in the Laika manual you need to replace `Helium.defaults` with `tlSiteHelium.value` +**unless** you explicitly want to remove all additional defaults listed above. +Everything else should be identical with using Laika standalone. +@:@ + +Some of the customization options which are most likely of interest in the context of project documentation: + +* **Versioned Documentation** - Laika can generate versioned documentation that works well for a standard workflow + where maintenance branches update only the pages specific to that version. + It inserts a dynamic version switcher into the top navigation bar that is driven by configuration + (meaning older versions can see newer versions without re-publishing them). + Examples for existing versioned sites are [Laika] or [http4s]. + See [Versioned Documentation] for details. + +* **Landing Page** - Laika comes with a default look & feel for a landing page, but it is disabled by default, + as it needs to be populated with your content (text, links, logo, etc.). + Example sites with a standard landing page are again [Laika] or [http4s]. + See [Website Landing Page] for details. + +* **Theme Colors and Fonts** - The color theme (including syntax highlighting) and font choices can be adjusted + without the need for handwritten CSS. + An example of a site with a different color scheme is [http4s]. + See [Colors] or [Fonts] for details. + +* **Additional Links** - Both the main/left navigation panel and the top navigation bar can be populated + with additional icon links, text links or menus. + See [Navigation, Links & Favicons][laika-nav] for details. + +For a complete list of customization options please see the full [Laika] documentation. + + +[Laika]: https://planet42.github.io/Laika/index.html +[http4s]: https://http4s.org/ +[Versioned Documentation]: https://planet42.github.io/Laika/0.18/03-preparing-content/01-directory-structure.html#versioned-documentation +[Website Landing Page]: https://planet42.github.io/Laika/0.18/03-preparing-content/03-theme-settings.html#website-landing-page +[Colors]: https://planet42.github.io/Laika/0.18/03-preparing-content/03-theme-settings.html#colors +[Fonts]: https://planet42.github.io/Laika/0.18/03-preparing-content/03-theme-settings.html#fonts +[laika-nav]: https://planet42.github.io/Laika/0.18/03-preparing-content/03-theme-settings.html#navigation-links-favicons + + +## FAQ + ### How can I include my project version on the website? -**sbt-typelevel-site** automatically adds `VERSION` and `SNAPSHOT_VERSION` to the `mdocVariables` setting which can be used with [variable injection](https://scalameta.org/mdoc/docs/why.html#variable-injection). +**sbt-typelevel-site** automatically adds `VERSION` and `SNAPSHOT_VERSION` to the `mdocVariables` setting +which can be used with [variable injection](https://scalameta.org/mdoc/docs/why.html#variable-injection). For example, the sbt-typelevel `VERSION` is `@VERSION@` and `SNAPSHOT_VERSION` is `@SNAPSHOT_VERSION@`. ### How can I publish "unidoc" API docs? -If you generate your API documentation with [sbt-unidoc](https://github.com/sbt/sbt-unidoc), you can use the `TypelevelUnidocPlugin` to publish a Scaladoc-only artifact to Sonatype/Maven alongside your library artifacts. This makes it possible to browse your unidocs at [javadoc.io](https://www.javadoc.io/); for example, the sbt-typelevel [API docs](@API_URL@) are published like this. +If you generate your API documentation with [sbt-unidoc](https://github.com/sbt/sbt-unidoc), +you can use the `TypelevelUnidocPlugin` to publish a Scaladoc-only artifact to Sonatype/Maven alongside your library artifacts. +This makes it possible to browse your unidocs at [javadoc.io](https://www.javadoc.io/); +for example, the sbt-typelevel [API docs](@API_URL@) are published like this. ```scala // Make sure to add to your root aggregate so it gets published! @@ -50,7 +149,3 @@ lazy val unidocs = project ScalaUnidoc / unidoc / unidocProjectFilter := inProjects(core.jvm, heffalump) ) ``` - -### How can I customize my website's appearance? - -We refer you to the comprehensive [Laika manual](https://planet42.github.io/Laika/index.html) and specifically the [`laikaTheme` setting](https://planet42.github.io/Laika/0.18/02-running-laika/01-sbt-plugin.html#laikatheme-setting). diff --git a/github-actions/build.sbt b/github-actions/build.sbt index d2f41845..a8eb76b9 100644 --- a/github-actions/build.sbt +++ b/github-actions/build.sbt @@ -1 +1 @@ -libraryDependencies += "org.yaml" % "snakeyaml" % "1.30" +libraryDependencies += "org.yaml" % "snakeyaml" % "1.32" 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 cfb25a69..a95182e0 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 @@ -592,7 +592,7 @@ ${indent(jobs.map(compileJob(_, sbt)).mkString("\n\n"), 1)} (List("os", "java", "scala") ::: keys).map(k => s"$${{ matrix.$k }}").mkString("-") val upload = WorkflowStep.Use( - UseRef.Public("actions", "upload-artifact", "v2"), + UseRef.Public("actions", "upload-artifact", "v3"), name = Some(s"Upload target directories"), params = Map("name" -> s"target-$artifactId", "path" -> "targets.tar"), cond = Some(publicationCond.value) @@ -614,7 +614,6 @@ ${indent(jobs.map(compileJob(_, sbt)).mkString("\n\n"), 1)} key -> values.take(1) // we only want the primary value } - val keys = "scala" :: additions.keys.toList.sorted val oses = githubWorkflowOSes.value.toList.take(1) val scalas = githubWorkflowScalaVersions.value.toList val javas = githubWorkflowJavaVersions.value.toList.take(1) @@ -639,7 +638,7 @@ ${indent(jobs.map(compileJob(_, sbt)).mkString("\n\n"), 1)} val pretty = v.mkString(", ") val download = WorkflowStep.Use( - UseRef.Public("actions", "download-artifact", "v2"), + UseRef.Public("actions", "download-artifact", "v3"), name = Some(s"Download target directories ($pretty)"), params = Map("name" -> s"target-$${{ matrix.os }}-$${{ matrix.java }}-${v.mkString("-")}") @@ -662,7 +661,7 @@ ${indent(jobs.map(compileJob(_, sbt)).mkString("\n\n"), 1)} Seq( WorkflowStep.Use( - UseRef.Public("actions", "cache", "v2"), + UseRef.Public("actions", "cache", "v3"), name = Some("Cache sbt"), params = Map( "path" -> Seq( diff --git a/github-actions/src/main/scala/org/typelevel/sbt/gha/WorkflowStep.scala b/github-actions/src/main/scala/org/typelevel/sbt/gha/WorkflowStep.scala index ef874389..243f6d62 100644 --- a/github-actions/src/main/scala/org/typelevel/sbt/gha/WorkflowStep.scala +++ b/github-actions/src/main/scala/org/typelevel/sbt/gha/WorkflowStep.scala @@ -21,17 +21,22 @@ sealed trait WorkflowStep extends Product with Serializable { def name: Option[String] def cond: Option[String] def env: Map[String, String] + + def withId(id: Option[String]): WorkflowStep + def withName(name: Option[String]): WorkflowStep + def withCond(cond: Option[String]): WorkflowStep + def withEnv(env: Map[String, String]): WorkflowStep } object WorkflowStep { val CheckoutFull: WorkflowStep = Use( - UseRef.Public("actions", "checkout", "v2"), + UseRef.Public("actions", "checkout", "v3"), name = Some("Checkout current branch (full)"), params = Map("fetch-depth" -> "0")) val Checkout: WorkflowStep = Use( - UseRef.Public("actions", "checkout", "v2"), + UseRef.Public("actions", "checkout", "v3"), name = Some("Checkout current branch (fast)")) def SetupJava(versions: List[JavaSpec]): List[WorkflowStep] = @@ -49,14 +54,14 @@ object WorkflowStep { val id = s"download-java-${dist.rendering}-$version" List( WorkflowStep.Use( - UseRef.Public("typelevel", "download-java", "v1"), + UseRef.Public("typelevel", "download-java", "v2"), name = Some(s"Download Java (${jv.render})"), id = Some(id), cond = cond, params = Map("distribution" -> dist.rendering, "java-version" -> version) ), WorkflowStep.Use( - UseRef.Public("actions", "setup-java", "v2"), + UseRef.Public("actions", "setup-java", "v3"), name = Some(s"Setup Java (${jv.render})"), cond = cond, params = Map( @@ -69,7 +74,7 @@ object WorkflowStep { case jv @ JavaSpec(dist, version) => WorkflowStep.Use( - UseRef.Public("actions", "setup-java", "v2"), + UseRef.Public("actions", "setup-java", "v3"), name = Some(s"Setup Java (${jv.render})"), cond = Some(s"matrix.java == '${jv.render}'"), params = Map("distribution" -> dist.rendering, "java-version" -> version) @@ -77,7 +82,13 @@ object WorkflowStep { } val Tmate: WorkflowStep = - Use(UseRef.Public("mxschmitt", "action-tmate", "v2"), name = Some("Setup tmate session")) + Use(UseRef.Public("mxschmitt", "action-tmate", "v3"), name = Some("Setup tmate session")) + + val DependencySubmission: WorkflowStep = + Use( + UseRef.Public("scalacenter", "sbt-dependency-submission", "v2"), + name = Some("Submit Dependencies") + ) def ComputeVar(name: String, cmd: String): WorkflowStep = Run( @@ -96,7 +107,13 @@ object WorkflowStep { cond: Option[String] = None, env: Map[String, String] = Map(), params: Map[String, String] = Map()) - extends WorkflowStep + extends WorkflowStep { + def withId(id: Option[String]) = copy(id = id) + def withName(name: Option[String]) = copy(name = name) + def withCond(cond: Option[String]) = copy(cond = cond) + def withEnv(env: Map[String, String]) = copy(env = env) + } + final case class Sbt( commands: List[String], id: Option[String] = None, @@ -104,7 +121,13 @@ object WorkflowStep { cond: Option[String] = None, env: Map[String, String] = Map(), params: Map[String, String] = Map()) - extends WorkflowStep + extends WorkflowStep { + def withId(id: Option[String]) = copy(id = id) + def withName(name: Option[String]) = copy(name = name) + def withCond(cond: Option[String]) = copy(cond = cond) + def withEnv(env: Map[String, String]) = copy(env = env) + } + final case class Use( ref: UseRef, params: Map[String, String] = Map(), @@ -112,5 +135,10 @@ object WorkflowStep { name: Option[String] = None, cond: Option[String] = None, env: Map[String, String] = Map()) - extends WorkflowStep + extends WorkflowStep { + def withId(id: Option[String]) = copy(id = id) + def withName(name: Option[String]) = copy(name = name) + def withCond(cond: Option[String]) = copy(cond = cond) + def withEnv(env: Map[String, String]) = copy(env = env) + } } diff --git a/github/build.sbt b/github/build.sbt index 389f9956..5d1244e6 100644 --- a/github/build.sbt +++ b/github/build.sbt @@ -1,2 +1,2 @@ addSbtPlugin("org.scala-js" % "sbt-scalajs" % "1.7.1") -addSbtPlugin("com.typesafe.sbt" % "sbt-git" % "1.0.2") +addSbtPlugin("com.github.sbt" % "sbt-git" % "2.0.0") diff --git a/github/src/main/scala/org/typelevel/sbt/TypelevelGitHubPlugin.scala b/github/src/main/scala/org/typelevel/sbt/TypelevelGitHubPlugin.scala index 7e87f379..47365114 100644 --- a/github/src/main/scala/org/typelevel/sbt/TypelevelGitHubPlugin.scala +++ b/github/src/main/scala/org/typelevel/sbt/TypelevelGitHubPlugin.scala @@ -16,7 +16,7 @@ package org.typelevel.sbt -import com.typesafe.sbt.SbtGit.git +import com.github.sbt.git.SbtGit.git import org.typelevel.sbt.kernel.GitHelper import sbt._ @@ -51,7 +51,21 @@ object TypelevelGitHubPlugin extends AutoPlugin { sLog.value.info(s"set scmInfo to https://github.com/$user/$repo") gitHubScmInfo(user, repo) }, - homepage := homepage.value.orElse(scmInfo.value.map(_.browseUrl)) + homepage := homepage.value.orElse(scmInfo.value.map(_.browseUrl)), + developers := { + gitHubUserRepo + .value + .toList + .map { + case (user, repo) => + Developer( + user, + s"$repo contributors", + s"@$user", + url(s"https://github.com/$user/$repo/contributors") + ) + } + } ) override def projectSettings: Seq[Setting[_]] = Seq( diff --git a/github/src/main/scala/org/typelevel/sbt/TypelevelScalaJSGitHubPlugin.scala b/github/src/main/scala/org/typelevel/sbt/TypelevelScalaJSGitHubPlugin.scala index 8f53882c..ba3ca46c 100644 --- a/github/src/main/scala/org/typelevel/sbt/TypelevelScalaJSGitHubPlugin.scala +++ b/github/src/main/scala/org/typelevel/sbt/TypelevelScalaJSGitHubPlugin.scala @@ -16,7 +16,7 @@ package org.typelevel.sbt -import com.typesafe.sbt.SbtGit.git +import com.github.sbt.git.SbtGit.git import org.scalajs.sbtplugin.ScalaJSPlugin import org.typelevel.sbt.kernel.GitHelper import sbt._ diff --git a/kernel/src/main/scala/org/typelevel/sbt/kernel/V.scala b/kernel/src/main/scala/org/typelevel/sbt/kernel/V.scala index 48ebc6f8..5d7d0217 100644 --- a/kernel/src/main/scala/org/typelevel/sbt/kernel/V.scala +++ b/kernel/src/main/scala/org/typelevel/sbt/kernel/V.scala @@ -34,7 +34,8 @@ private[sbt] final case class V( this.major == that.major && this.minor == that.minor def mustBeBinCompatWith(that: V): Boolean = - this >= that && !that.isPrerelease && this.major == that.major && (major > 0 || this.minor == that.minor) + this >= that && !that.isPrerelease && this.major == that.major && + (major > 0 || (this.minor == that.minor && minor > 0)) def compare(that: V): Int = { val x = this.major.compare(that.major) @@ -43,8 +44,8 @@ private[sbt] final case class V( if (y != 0) return y (this.patch, that.patch) match { case (None, None) => 0 - case (None, Some(patch)) => 1 - case (Some(patch), None) => -1 + case (None, Some(_)) => 1 + case (Some(_), None) => -1 case (Some(thisPatch), Some(thatPatch)) => val z = thisPatch.compare(thatPatch) if (z != 0) return z diff --git a/kernel/src/test/scala/org/typelevel/sbt/kernel/VSuite.scala b/kernel/src/test/scala/org/typelevel/sbt/kernel/VSuite.scala new file mode 100644 index 00000000..073a7379 --- /dev/null +++ b/kernel/src/test/scala/org/typelevel/sbt/kernel/VSuite.scala @@ -0,0 +1,181 @@ +/* + * 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.kernel + +import munit.FunSuite + +class VSuite extends FunSuite { + + test("V.apply constructs V") { + assertEquals(V("0.0"), Some(V(0, 0, None, None))) + assertEquals(V("0.0-M1"), Some(V(0, 0, None, Some("M1")))) + assertEquals(V("0.0.0-M1"), Some(V(0, 0, Some(0), Some("M1")))) + assertEquals(V("10.0"), Some(V(10, 0, None, None))) + assertEquals(V("10.100"), Some(V(10, 100, None, None))) + assertEquals(V("10.100.1000"), Some(V(10, 100, Some(1000), None))) + } + + test("x.y.z-M1 is a prerelease") { + assert(V(0, 0, None, Some("M1")).isPrerelease) + assert(V(0, 0, Some(1), Some("M1")).isPrerelease) + assert(V(0, 1, None, Some("M1")).isPrerelease) + assert(V(0, 1, Some(1), Some("M1")).isPrerelease) + assert(V(1, 0, None, Some("M1")).isPrerelease) + assert(V(1, 0, Some(1), Some("M1")).isPrerelease) + assert(V(1, 1, None, Some("M1")).isPrerelease) + assert(V(1, 1, Some(1), Some("M1")).isPrerelease) + } + + test("x.y.2 is the same series as x.y.1") { + assert(V(0, 0, Some(1), None).isSameSeries(V(0, 0, Some(2), None))) + assert(V(0, 1, Some(1), None).isSameSeries(V(0, 1, Some(2), None))) + assert(V(1, 1, Some(1), None).isSameSeries(V(1, 1, Some(2), None))) + } + + test("1.1 needs bincompat with 1.0") { + val currentV = V(1, 1, None, None) + val prevV = V(1, 0, None, None) + assertEquals(currentV.mustBeBinCompatWith(prevV), true) + } + + test("1.1 does not need bincompat with 1.2") { + val currentV = V(1, 1, None, None) + val nextV = V(1, 2, None, None) + assertEquals(currentV.mustBeBinCompatWith(nextV), false) + } + + test("2.0 does not need bincompat with 1.9") { + val currentV = V(2, 0, None, None) + val prevV = V(1, 9, None, None) + assertEquals(currentV.mustBeBinCompatWith(prevV), false) + } + + test("1.1.1 needs bincompat with 1.1.0") { + val currentV = V(1, 1, Some(1), None) + val prevV = V(1, 1, Some(0), None) + assertEquals(currentV.mustBeBinCompatWith(prevV), true) + } + + test("1.1 does not need bincompat with 1.0-M5") { + val currentV = V(1, 1, None, None) + val prevV = V(1, 0, None, Some("M5")) + assertEquals(currentV.mustBeBinCompatWith(prevV), false) + } + + test("0.5 does not need bincompat with 0.4") { + val currentV = V(0, 5, None, None) + val prevV = V(0, 4, None, None) + assertEquals(currentV.mustBeBinCompatWith(prevV), false) + } + + test("0.5.1 needs bincompat with 0.5.0") { + val currentV = V(0, 5, Some(1), None) + val prevV = V(0, 5, Some(0), None) + assertEquals(currentV.mustBeBinCompatWith(prevV), true) + } + + test("0.5.1 needs bincompat with 0.5.0") { + val currentV = V(0, 5, Some(1), None) + val prevV = V(0, 5, Some(0), None) + assertEquals(currentV.mustBeBinCompatWith(prevV), true) + } + + test("0.0.2 does not need bincompat with 0.0.1") { + val currentV = V(0, 0, Some(1), None) + val prevV = V(0, 0, Some(0), None) + assertEquals(currentV.mustBeBinCompatWith(prevV), false) + } + + test("all versions > 0.0 that are not prereleases need bincompat with self") { + val vs = List( + V(0, 5, None, None), + V(0, 5, Some(1), None), + V(1, 0, None, None), + V(1, 0, Some(1), None), + V(1, 5, None, None), + V(1, 5, Some(1), None) + ) + vs.foreach(v => assert(v.mustBeBinCompatWith(v), s"$v did not need bincompat with itself")) + } + + test("all versions < 0.0 need bincompat with self".fail) { + val vs = List( + V(0, 0, None, None), + V(0, 0, Some(1), None) + ) + vs.foreach(v => assert(v.mustBeBinCompatWith(v), s"$v did not need bincompat with itself")) + } + + // We current don't compare prereleases correctly + test("all versions that are prerelease need bincompat with self".fail) { + val vs = List( + V(0, 0, None, Some("M1")), + V(0, 0, Some(1), Some("M1")), + V(0, 5, None, Some("M1")), + V(0, 5, Some(1), Some("M1")), + V(1, 0, None, Some("M1")), + V(1, 0, Some(1), Some("M1")), + V(1, 5, None, Some("M1")), + V(1, 5, Some(1), Some("M1")) + ) + vs.foreach(v => assert(v.mustBeBinCompatWith(v), s"$v did not need bincompat with itself")) + } + + test("x.y needs bincompat with self") { + val v0 = V(0, 5, None, None) + assertEquals(v0.mustBeBinCompatWith(v0), true) + val v1 = V(1, 5, None, None) + assertEquals(v1.mustBeBinCompatWith(v1), true) + } + + test("x.y.1 does not need bincompat with x.y") { + val currentV = V(1, 5, Some(1), None) + val prevV = V(1, 5, None, None) + assertEquals(currentV.mustBeBinCompatWith(prevV), false) + val currentV0 = V(0, 5, Some(1), None) + val prevV0 = V(0, 5, None, None) + assertEquals(currentV0.mustBeBinCompatWith(prevV0), false) + } + + test("x.y.1 < x.y") { + val patch0 = V(0, 5, Some(1), None) + val nopatch0 = V(0, 5, None, None) + assertEquals(patch0 < nopatch0, true) + val patch1 = V(1, 5, Some(1), None) + val nopatch1 = V(1, 5, None, None) + assertEquals(patch1 < nopatch1, true) + } + + test("x.y.1 < x.y.2") { + val patch0 = V(0, 5, Some(1), None) + val p1patch0 = V(0, 5, Some(2), None) + assertEquals(patch0 < p1patch0, true) + val patch1 = V(1, 5, Some(1), None) + val p1patch1 = V(1, 5, Some(2), None) + assertEquals(patch1 < p1patch1, true) + } + + test("x.y.1-M1 < x.y.1") { + val pre0 = V(0, 5, Some(1), Some("M1")) + val nopre0 = V(0, 5, Some(2), None) + assertEquals(pre0 < nopre0, true) + val pre1 = V(1, 5, Some(1), Some("M1")) + val nopre1 = V(1, 5, Some(2), None) + assertEquals(pre1 < nopre1, true) + } + +} 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 22147550..5f36afde 100644 --- a/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyAction.scala +++ b/mergify/src/main/scala/org/typelevel/sbt/mergify/MergifyAction.scala @@ -23,6 +23,8 @@ import io.circe.syntax._ import org.typelevel.sbt.mergify.MergifyAction.RequestReviews._ import sbt.librarymanagement.Developer +import scala.annotation.nowarn + sealed abstract class MergifyAction { private[mergify] def name = getClass.getSimpleName.toLowerCase } @@ -159,6 +161,6 @@ object MergifyAction { Encoder[JsonObject].contramap(_ => JsonObject.empty) } + @nowarn("cat=unused") 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 index ea6f8c29..9049ef7b 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 import io.circe.Encoder import io.circe.syntax._ +import scala.annotation.nowarn + sealed abstract class MergifyCondition object MergifyCondition { @@ -44,5 +46,6 @@ object MergifyCondition { implicit def encoder: Encoder[Or] = Encoder.forProduct1("or")(_.conditions) } + @nowarn("cat=unused") private[this] final object Dummy extends MergifyCondition // break exhaustivity checking } diff --git a/project/build.properties b/project/build.properties index c8fcab54..563a014d 100644 --- a/project/build.properties +++ b/project/build.properties @@ -1 +1 @@ -sbt.version=1.6.2 +sbt.version=1.7.2 diff --git a/scalafix/build.sbt b/scalafix/build.sbt index 85b31649..d88312a0 100644 --- a/scalafix/build.sbt +++ b/scalafix/build.sbt @@ -1 +1 @@ -addSbtPlugin("ch.epfl.scala" % "sbt-scalafix" % "0.10.3") +addSbtPlugin("ch.epfl.scala" % "sbt-scalafix" % "0.10.4") diff --git a/settings/build.sbt b/settings/build.sbt index efd496c4..ec66dc97 100644 --- a/settings/build.sbt +++ b/settings/build.sbt @@ -1,2 +1,2 @@ -addSbtPlugin("com.typesafe.sbt" % "sbt-git" % "1.0.2") +addSbtPlugin("com.github.sbt" % "sbt-git" % "2.0.0") addSbtPlugin("org.portable-scala" % "sbt-crossproject" % "1.2.0") diff --git a/settings/src/main/scala/org/typelevel/sbt/TypelevelSettingsPlugin.scala b/settings/src/main/scala/org/typelevel/sbt/TypelevelSettingsPlugin.scala index 2ffc3b29..08098f53 100644 --- a/settings/src/main/scala/org/typelevel/sbt/TypelevelSettingsPlugin.scala +++ b/settings/src/main/scala/org/typelevel/sbt/TypelevelSettingsPlugin.scala @@ -16,8 +16,8 @@ package org.typelevel.sbt -import com.typesafe.sbt.GitPlugin -import com.typesafe.sbt.SbtGit.git +import com.github.sbt.git.GitPlugin +import com.github.sbt.git.SbtGit.git import org.typelevel.sbt.kernel.GitHelper import org.typelevel.sbt.kernel.V import sbt._ @@ -26,6 +26,7 @@ import sbtcrossproject.CrossType import java.io.File import java.lang.management.ManagementFactory +import scala.annotation.nowarn import scala.util.Try import Keys._ @@ -47,12 +48,11 @@ object TypelevelSettingsPlugin extends AutoPlugin { override def globalSettings = Seq( tlFatalWarnings := false, - tlJdkRelease := None, + tlJdkRelease := Some(8), Def.derive(scalaVersion := crossScalaVersions.value.last, default = true) ) override def projectSettings = Seq( - versionScheme := Some("early-semver"), pomIncludeRepository := { _ => false }, libraryDependencies ++= { if (tlIsScala3.value) @@ -72,36 +72,49 @@ object TypelevelSettingsPlugin extends AutoPlugin { "-feature", "-unchecked"), scalacOptions ++= { - scalaVersion.value match { - case V(V(2, minor, _, _)) if minor < 13 => - Seq("-Yno-adapted-args", "-Ywarn-unused-import") - case _ => - Seq.empty - } - }, - scalacOptions ++= { - val warningsNsc = Seq("-Xlint", "-Ywarn-dead-code") + val warningsNsc = Seq( + "-Xlint", + "-Yno-adapted-args", // similar to '-Xlint:adapted-args' but fails compilation instead of just emitting a warning + "-Ywarn-dead-code", + "-Ywarn-unused-import" + ) - val warnings211 = - Seq("-Ywarn-numeric-widen") // In 2.10 this produces a some strange spurious error + val warnings211 = Seq( + "-Ywarn-numeric-widen" // In 2.10 this produces a some strange spurious error + ) - val warnings212 = Seq("-Xlint:-unused,_") + val removed212 = Set( + "-Xlint", + "-Yno-adapted-args", // mostly superseded by '-Xlint:adapted-args' + "-Ywarn-unused-import" // superseded by '-Ywarn-unused:imports' + ) + val warnings212 = Seq( + // Tune '-Xlint': + // - remove 'unused' because it is configured by '-Ywarn-unused' + "-Xlint:_,-unused", + // Tune '-Ywarn-unused': + // - remove 'nowarn' because 2.13 can detect more unused cases than 2.12 + // - remove 'privates' because 2.12 can incorrectly detect some private objects as unused + "-Ywarn-unused:_,-nowarn,-privates" + ) - val removed213 = Set("-Xlint:-unused,_", "-Xlint") + val removed213 = Set( + "-Xlint:_,-unused", // reconfigured for 2.13 + "-Ywarn-unused:_,-nowarn,-privates", // mostly superseded by "-Wunused" + "-Ywarn-dead-code", // superseded by "-Wdead-code" + "-Ywarn-numeric-widen" // superseded by "-Wnumeric-widen" + ) val warnings213 = Seq( - "-Xlint:deprecation", - "-Wunused:nowarn", "-Wdead-code", "-Wextra-implicit", "-Wnumeric-widen", - "-Wunused:implicits", - "-Wunused:explicits", - "-Wunused:imports", - "-Wunused:locals", - "-Wunused:params", - "-Wunused:patvars", - "-Wunused:privates", - "-Wvalue-discard" + "-Wunused", // all choices are enabled by default + "-Wvalue-discard", + // Tune '-Xlint': + // - remove 'implicit-recursion' due to backward incompatibility with 2.12 + // - remove 'recurse-with-default' due to backward incompatibility with 2.12 + // - remove 'unused' because it is configured by '-Wunused' + "-Xlint:_,-implicit-recursion,-recurse-with-default,-unused" ) val warningsDotty = Seq.empty @@ -111,10 +124,11 @@ object TypelevelSettingsPlugin extends AutoPlugin { warningsDotty case V(V(2, minor, _, _)) if minor >= 13 => - (warnings211 ++ warnings212 ++ warnings213 ++ warningsNsc).filterNot(removed213) + (warnings211 ++ warnings212 ++ warnings213 ++ warningsNsc) + .filterNot(removed212 ++ removed213) case V(V(2, minor, _, _)) if minor >= 12 => - warnings211 ++ warnings212 ++ warningsNsc + (warnings211 ++ warnings212 ++ warningsNsc).filterNot(removed212) case V(V(2, minor, _, _)) if minor >= 11 => warnings211 ++ warningsNsc @@ -163,18 +177,15 @@ object TypelevelSettingsPlugin extends AutoPlugin { else Seq("-Yrangepos") }, - Compile / console / scalacOptions --= Seq( - "-Xlint", - "-Ywarn-unused-import", - "-Wextra-implicit", - "-Wunused:implicits", - "-Wunused:explicits", - "-Wunused:imports", - "-Wunused:locals", - "-Wunused:params", - "-Wunused:patvars", - "-Wunused:privates" - ), + Compile / console / scalacOptions := scalacOptions.value.filterNot { opt => + opt.startsWith("-Xlint") || + PartialFunction.cond(scalaVersion.value) { + case V(V(2, minor, _, _)) if minor >= 13 => + opt.startsWith("-Wunused") || opt == "-Wextra-implicit" + case V(V(2, minor, _, _)) if minor >= 12 => + opt.startsWith("-Ywarn-unused") + } + }, Test / console / scalacOptions := (Compile / console / scalacOptions).value, Compile / doc / scalacOptions ++= { Seq("-sourcepath", (LocalRootProject / baseDirectory).value.getAbsolutePath) @@ -319,4 +330,7 @@ object TypelevelSettingsPlugin extends AutoPlugin { oldSchool ++ newSchool ++ old } } + + @nowarn("cat=unused") + private[this] def unused(): Unit = () } diff --git a/site/build.sbt b/site/build.sbt index 4ba74bc6..12eefc9f 100644 --- a/site/build.sbt +++ b/site/build.sbt @@ -1,2 +1,2 @@ -addSbtPlugin("org.scalameta" % "sbt-mdoc" % "2.2.24") +addSbtPlugin("org.scalameta" % "sbt-mdoc" % "2.3.6") addSbtPlugin("org.planet42" % "laika-sbt" % "0.19.0") diff --git a/site/src/main/resources/org/typelevel/sbt/site/helium/default.template.html b/site/src/main/resources/org/typelevel/sbt/site/helium/default.template.html deleted file mode 100644 index 2335d4c9..00000000 --- a/site/src/main/resources/org/typelevel/sbt/site/helium/default.template.html +++ /dev/null @@ -1,106 +0,0 @@ - - - - - - - - ${cursor.currentDocument.title} - @:for(laika.site.metadata.authors) - - @:@ - @:for(laika.site.metadata.description) - - @:@ - @:for(helium.favIcons) - - @:@ - @:for(helium.webFonts) - - @:@ - @:linkCSS { paths = ${helium.site.includeCSS} } - @:linkJS { paths = ${helium.site.includeJS} } - @:heliumInitVersions - @:heliumInitPreview(container) - - - - - -
- - - - ${?helium.topBar.home} - - ${?helium.topBar.links} - -
- - - -
- - - -
- - ${cursor.currentDocument.content} - -
- - -
- -
- - - diff --git a/site/src/main/resources/org/typelevel/sbt/site/helium/site/styles.css b/site/src/main/resources/org/typelevel/sbt/site/helium/site/styles.css deleted file mode 100644 index 8d9ff940..00000000 --- a/site/src/main/resources/org/typelevel/sbt/site/helium/site/styles.css +++ /dev/null @@ -1,5 +0,0 @@ -header img { - height: 40px; - width: auto; - margin-top: 6px; -} diff --git a/site/src/main/scala/org/typelevel/sbt/TypelevelSitePlugin.scala b/site/src/main/scala/org/typelevel/sbt/TypelevelSitePlugin.scala index c2220ed0..ec5bd1d7 100644 --- a/site/src/main/scala/org/typelevel/sbt/TypelevelSitePlugin.scala +++ b/site/src/main/scala/org/typelevel/sbt/TypelevelSitePlugin.scala @@ -16,42 +16,45 @@ package org.typelevel.sbt -import laika.ast.LengthUnit._ -import laika.ast._ import laika.helium.Helium -import laika.helium.config.Favicon -import laika.helium.config.HeliumIcon -import laika.helium.config.IconLink -import laika.helium.config.ImageLink import laika.sbt.LaikaPlugin +import laika.sbt.LaikaPlugin.autoImport._ +import laika.sbt.Tasks import laika.theme.ThemeProvider import mdoc.MdocPlugin +import mdoc.MdocPlugin.autoImport._ +import org.typelevel.sbt.TypelevelKernelPlugin._ +import org.typelevel.sbt.gha.GenerativePlugin +import org.typelevel.sbt.gha.GenerativePlugin.autoImport._ import org.typelevel.sbt.site._ +import sbt.Keys._ import sbt._ import scala.annotation.nowarn -import Keys._ -import MdocPlugin.autoImport._ -import LaikaPlugin.autoImport._ -import gha.GenerativePlugin -import GenerativePlugin.autoImport._ -import TypelevelKernelPlugin._ -import TypelevelKernelPlugin.autoImport._ - object TypelevelSitePlugin extends AutoPlugin { object autoImport { + + @deprecated("Use tlSiteHelium", "0.5.0") lazy val tlSiteHeliumConfig = settingKey[Helium]("The Typelevel Helium configuration") + @deprecated("Use tlSiteHelium", "0.5.0") lazy val tlSiteHeliumExtensions = settingKey[ThemeProvider]("The Typelevel Helium extensions") + @deprecated("Use .site.mainNavigation(appendLinks = ...) in tlSiteHelium", "0.5.0") + lazy val tlSiteRelatedProjects = + settingKey[Seq[(String, URL)]]("A list of related projects (default: empty)") + + lazy val tlSiteHelium = settingKey[Helium]("The Helium theme configuration and extensions") + lazy val tlSiteIsTypelevelProject = + settingKey[Boolean]( + "Indicates whether the generated site should be pre-populated with UI elements specific to Typelevel projects (default: false)") + lazy val tlSiteApiUrl = settingKey[Option[URL]]("URL to the API docs") lazy val tlSiteApiModule = settingKey[Option[ModuleID]]("The module that publishes API docs") lazy val tlSiteApiPackage = settingKey[Option[String]]( "The top-level package for your API docs (e.g. org.typlevel.sbt)") - lazy val tlSiteRelatedProjects = - settingKey[Seq[(String, URL)]]("A list of related projects (default: cats)") lazy val tlSiteKeepFiles = settingKey[Boolean]("Whether to keep existing files when deploying site (default: true)") @@ -68,8 +71,6 @@ object TypelevelSitePlugin extends AutoPlugin { "Start a live-reload preview server (combines mdoc --watch with laikaPreview)") val TypelevelProject = site.TypelevelProject - implicit def tlLaikaThemeProviderOps(provider: ThemeProvider): LaikaThemeProviderOps = - new site.LaikaThemeProviderOps(provider) } import autoImport._ @@ -82,12 +83,13 @@ object TypelevelSitePlugin extends AutoPlugin { tlSiteApiModule := None ) + @nowarn("cat=deprecation") override def buildSettings = Seq( tlSitePublishBranch := Some("main"), tlSitePublishTags := tlSitePublishBranch.value.isEmpty, tlSiteApiUrl := None, tlSiteApiPackage := None, - tlSiteRelatedProjects := Seq(TypelevelProject.Cats), + tlSiteRelatedProjects := Nil, tlSiteKeepFiles := true, homepage := { gitHubUserRepo.value.map { @@ -97,6 +99,7 @@ object TypelevelSitePlugin extends AutoPlugin { } ) + @nowarn("cat=deprecation") override def projectSettings = Seq( tlSite := Def .sequential( @@ -106,7 +109,7 @@ object TypelevelSitePlugin extends AutoPlugin { .value: @nowarn("cat=other-pure-statement"), tlSitePreview := previewTask.value, Laika / sourceDirectories := Seq(mdocOut.value), - laikaTheme := tlSiteHeliumConfig.value.build.extend(tlSiteHeliumExtensions.value), + laikaTheme := tlSiteHelium.value.build, mdocVariables := { mdocVariables.value ++ Map( @@ -116,12 +119,13 @@ object TypelevelSitePlugin extends AutoPlugin { ) ++ tlSiteApiUrl.value.map("API_URL" -> _.toString).toMap }, - tlSiteHeliumExtensions := TypelevelHeliumExtensions( - licenses.value.headOption, - tlSiteRelatedProjects.value, - tlIsScala3.value, - tlSiteApiUrl.value - ), + tlSiteIsTypelevelProject := organization.value == "org.typelevel", + tlSiteHeliumConfig := TypelevelSiteSettings.defaults.value, + tlSiteHeliumExtensions := GenericSiteSettings.themeExtensions.value, + tlSiteHelium := { + if (tlSiteIsTypelevelProject.value) tlSiteHeliumConfig.value + else GenericSiteSettings.defaults.value + }, tlSiteApiUrl := { val javadocioUrl = for { moduleId <- (ThisProject / tlSiteApiModule).value @@ -145,54 +149,6 @@ object TypelevelSitePlugin extends AutoPlugin { tlSiteApiUrl.value.orElse(javadocioUrl).orElse(fallbackUrl) }, - tlSiteHeliumConfig := { - Helium - .defaults - .site - .metadata( - title = gitHubUserRepo.value.map(_._2), - authors = developers.value.map(_.name), - language = Some("en"), - version = Some(version.value.toString) - ) - .site - .layout( - contentWidth = px(860), - navigationWidth = px(275), - topBarHeight = px(50), - defaultBlockSpacing = px(10), - defaultLineHeight = 1.5, - anchorPlacement = laika.helium.config.AnchorPlacement.Right - ) - .site - .darkMode - .disabled - .site - .favIcons( - Favicon.external("https://typelevel.org/img/favicon.png", "32x32", "image/png") - ) - .site - .topNavigationBar( - homeLink = ImageLink.external( - "https://typelevel.org", - Image.external(s"https://typelevel.org/img/logo.svg") - ), - navLinks = tlSiteApiUrl.value.toList.map { url => - IconLink.external( - url.toString, - HeliumIcon.api, - options = Styles("svg-link") - ) - } ++ List( - IconLink.external( - scmInfo.value.fold("https://github.com/typelevel")(_.browseUrl.toString), - HeliumIcon.github, - options = Styles("svg-link")), - IconLink.external("https://discord.gg/XF3CXcMzqD", HeliumIcon.chat), - IconLink.external("https://twitter.com/typelevel", HeliumIcon.twitter) - ) - ) - }, tlSiteGenerate := List( WorkflowStep.Sbt( List(s"${thisProject.value.id}/${tlSite.key.toString}"), @@ -256,49 +212,18 @@ object TypelevelSitePlugin extends AutoPlugin { private def previewTask = Def .taskDyn { - // inlined from https://github.com/planet42/Laika/blob/9022f6f37c9017f7612fa59398f246c8e8c42c3e/sbt/src/main/scala/laika/sbt/Tasks.scala#L192 - import cats.effect.IO import cats.effect.unsafe.implicits._ - import laika.sbt.Settings - import laika.sbt.Tasks.generateAPI - import laika.preview.{ServerBuilder, ServerConfig} val logger = streams.value.log logger.info("Initializing server...") - def applyIf( - flag: Boolean, - f: ServerConfig => ServerConfig): ServerConfig => ServerConfig = - if (flag) f else identity - - val previewConfig = laikaPreviewConfig.value - val _ = generateAPI.value - - val applyFlags = applyIf(laikaIncludeEPUB.value, _.withEPUBDownloads) - .andThen(applyIf(laikaIncludePDF.value, _.withPDFDownloads)) - .andThen( - applyIf(laikaIncludeAPI.value, _.withAPIDirectory(Settings.apiTargetDirectory.value))) - .andThen(applyIf(previewConfig.isVerbose, _.verbose)) - - val config = ServerConfig - .defaults - .withArtifactBasename(name.value) - // .withHost(previewConfig.host) - .withPort(previewConfig.port) - .withPollInterval(previewConfig.pollInterval) - - val (_, cancel) = ServerBuilder[IO](Settings.parser.value, laikaInputs.value.delegate) - .withLogger(s => IO(logger.info(s))) - .withConfig(applyFlags(config)) - .build - .allocated - .unsafeRunSync() + val (_, cancel) = Tasks.buildPreviewServer.value.allocated.unsafeRunSync() - logger.info(s"Preview server started on port ${previewConfig.port}.") + logger.info(s"Preview server started on port ${laikaPreviewConfig.value.port}.") // watch but no-livereload b/c we don't need an mdoc server mdoc.toTask(" --watch --no-livereload").andFinally { - logger.info(s"Shutting down preview server.") + logger.info(s"Shutting down preview server...") cancel.unsafeRunSync() } } diff --git a/site/src/main/scala/org/typelevel/sbt/site/GenericSiteSettings.scala b/site/src/main/scala/org/typelevel/sbt/site/GenericSiteSettings.scala new file mode 100644 index 00000000..3b6e589f --- /dev/null +++ b/site/src/main/scala/org/typelevel/sbt/site/GenericSiteSettings.scala @@ -0,0 +1,86 @@ +/* + * 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.site + +import cats.data.NonEmptyList +import laika.helium.Helium +import laika.helium.config.HeliumIcon +import laika.helium.config.IconLink +import laika.helium.config.TextLink +import laika.helium.config.ThemeNavigationSection +import laika.theme.ThemeProvider +import org.typelevel.sbt.TypelevelGitHubPlugin.gitHubUserRepo +import org.typelevel.sbt.TypelevelKernelPlugin.autoImport.tlIsScala3 +import org.typelevel.sbt.TypelevelSitePlugin.autoImport.tlSiteApiUrl +import org.typelevel.sbt.TypelevelSitePlugin.autoImport.tlSiteHeliumExtensions +import org.typelevel.sbt.TypelevelSitePlugin.autoImport.tlSiteRelatedProjects +import sbt.Def._ +import sbt.Keys.developers +import sbt.Keys.scmInfo +import sbt.Keys.version + +import scala.annotation.nowarn + +object GenericSiteSettings { + + val apiLink: Initialize[Option[IconLink]] = setting { + tlSiteApiUrl.value.map { url => IconLink.external(url.toString, HeliumIcon.api) } + } + + val githubLink: Initialize[Option[IconLink]] = setting { + scmInfo.value.map { info => IconLink.external(info.browseUrl.toString, HeliumIcon.github) } + } + + @nowarn("cat=deprecation") + val themeExtensions: Initialize[ThemeProvider] = setting { + // TODO - inline when deprecated class gets removed + TypelevelHeliumExtensions( + tlIsScala3.value, + tlSiteApiUrl.value + ) + } + + @nowarn("cat=deprecation") + private val legacyRelatedProjects: Initialize[Option[ThemeNavigationSection]] = setting { + NonEmptyList.fromList(tlSiteRelatedProjects.value.toList).map { projects => + ThemeNavigationSection( + "Related Projects", + projects.map { case (name, url) => TextLink.external(url.toString, name) }) + } + } + + @nowarn("cat=deprecation") + val defaults: Initialize[Helium] = setting { + Helium + .defaults + .extendWith(tlSiteHeliumExtensions.value) + .site + .metadata( + title = gitHubUserRepo.value.map(_._2), + authors = developers.value.map(_.name), + language = Some("en"), + version = Some(version.value) + ) + .site + .mainNavigation(appendLinks = legacyRelatedProjects.value.toList) + .site + .topNavigationBar( + navLinks = apiLink.value.toList ++ githubLink.value.toList + ) + } + +} diff --git a/site/src/main/scala/org/typelevel/sbt/site/ThemeProviderOps.scala b/site/src/main/scala/org/typelevel/sbt/site/ThemeProviderOps.scala deleted file mode 100644 index 6d3750d6..00000000 --- a/site/src/main/scala/org/typelevel/sbt/site/ThemeProviderOps.scala +++ /dev/null @@ -1,53 +0,0 @@ -/* - * 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.site - -import cats.effect.Sync -import laika.bundle.ExtensionBundle -import laika.factory.Format -import laika.io.model.InputTree -import laika.theme.Theme -import laika.theme.ThemeProvider - -final class LaikaThemeProviderOps private[sbt] (provider: ThemeProvider) { - - def extend(extensions: ThemeProvider): ThemeProvider = new ThemeProvider { - def build[F[_]: Sync] = for { - base <- provider.build - ext <- extensions.build - } yield { - def overrideInputs(base: InputTree[F], overrides: InputTree[F]): InputTree[F] = { - val overridePaths = overrides.allPaths.toSet - val filteredBaseInputs = InputTree( - textInputs = base.textInputs.filterNot(in => overridePaths.contains(in.path)), - binaryInputs = base.binaryInputs.filterNot(in => overridePaths.contains(in.path)), - parsedResults = base.parsedResults.filterNot(in => overridePaths.contains(in.path)), - sourcePaths = base.sourcePaths - ) - overrides ++ filteredBaseInputs - } - - new Theme[F] { - override def inputs: InputTree[F] = overrideInputs(base.inputs, ext.inputs) - override def extensions: Seq[ExtensionBundle] = base.extensions ++ ext.extensions - override def treeProcessor: Format => Theme.TreeProcessor[F] = fmt => - base.treeProcessor(fmt).andThen(ext.treeProcessor(fmt)) - } - } - } - -} diff --git a/site/src/main/scala/org/typelevel/sbt/site/TypelevelHeliumExtensions.scala b/site/src/main/scala/org/typelevel/sbt/site/TypelevelHeliumExtensions.scala index 2e5d468f..87be973a 100644 --- a/site/src/main/scala/org/typelevel/sbt/site/TypelevelHeliumExtensions.scala +++ b/site/src/main/scala/org/typelevel/sbt/site/TypelevelHeliumExtensions.scala @@ -16,98 +16,66 @@ package org.typelevel.sbt.site -import cats.effect.Resource -import cats.effect.Sync +import cats.effect.Async +import cats.effect.kernel.Resource import laika.ast.Path -import laika.config.Config import laika.io.model.InputTree import laika.markdown.github.GitHubFlavor import laika.parse.code.SyntaxHighlighting import laika.parse.code.languages.DottySyntax -import laika.rewrite.DefaultTemplatePath import laika.theme.Theme import laika.theme.ThemeBuilder import laika.theme.ThemeProvider import java.net.URL +@deprecated("Use GenericSiteSettings.extensions", "0.5.0") object TypelevelHeliumExtensions { - @deprecated("Use overload with API url and scala3 parameter", "0.4.7") + @deprecated("Use overload with scala3 and apiURL parameter", "0.4.7") def apply(license: Option[(String, URL)], related: Seq[(String, URL)]): ThemeProvider = apply(license, related, false) - @deprecated("Use overload with API url and scala3 parameter", "0.4.13") + @deprecated("Use overload with scala3 and apiURL parameter", "0.4.13") def apply( license: Option[(String, URL)], related: Seq[(String, URL)], scala3: Boolean): ThemeProvider = apply(license, related, false, None) + @deprecated("Use overload with scala3 and apiURL parameter", "0.5.0") + def apply( + license: Option[(String, URL)], + related: Seq[(String, URL)], + scala3: Boolean, + apiUrl: Option[URL] + ): ThemeProvider = apply(scala3, apiUrl) + /** - * @param license - * name and [[java.net.URL]] of project license - * @param related - * name and [[java.net.URL]] of related projects * @param scala3 * whether to use Scala 3 syntax highlighting * @param apiUrl * url to API docs */ def apply( - license: Option[(String, URL)], - related: Seq[(String, URL)], scala3: Boolean, apiUrl: Option[URL] ): ThemeProvider = new ThemeProvider { - def build[F[_]](implicit F: Sync[F]): Resource[F, Theme[F]] = - ThemeBuilder[F]("Typelevel Helium Extensions") + def build[F[_]](implicit F: Async[F]): Resource[F, Theme[F]] = + ThemeBuilder[F]("sbt-typelevel-site Helium Extensions") .addInputs( - InputTree[F] - .addStream( - F.blocking(getClass.getResourceAsStream("helium/default.template.html")), - DefaultTemplatePath.forHTML - ) - .addStream( - F.blocking(getClass.getResourceAsStream("helium/site/styles.css")), - Path.Root / "site" / "styles.css" - ) - .merge( - apiUrl.fold(InputTree[F]) { url => - InputTree[F].addString(htmlForwarder(url), Path.Root / "api" / "index.html") - } - ) + apiUrl.fold(InputTree[F]) { url => + InputTree[F].addString(htmlForwarder(url), Path.Root / "api" / "index.html") + } ) .addExtensions( GitHubFlavor, if (scala3) SyntaxHighlighting.withSyntaxBinding("scala", DottySyntax) else SyntaxHighlighting ) - .addBaseConfig(licenseConfig(license).withFallback(relatedConfig(related))) .build } - private def licenseConfig(license: Option[(String, URL)]) = - license.fold(Config.empty) { - case (name, url) => - Config - .empty - .withValue("typelevel.site.license.name", name) - .withValue("typelevel.site.license.url", url.toString) - .build - } - - private def relatedConfig(related: Seq[(String, URL)]) = - Config - .empty - .withValue( - "typelevel.site.related", - related.map { - case (name, url) => - Map("name" -> name, "url" -> url.toString) - }) - .build - private def htmlForwarder(to: URL) = s"""| | diff --git a/site/src/main/scala/org/typelevel/sbt/site/TypelevelSiteSettings.scala b/site/src/main/scala/org/typelevel/sbt/site/TypelevelSiteSettings.scala new file mode 100644 index 00000000..e2a3e66f --- /dev/null +++ b/site/src/main/scala/org/typelevel/sbt/site/TypelevelSiteSettings.scala @@ -0,0 +1,80 @@ +/* + * 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.site + +import laika.ast.Image +import laika.ast.LengthUnit.px +import laika.ast.Span +import laika.ast.TemplateString +import laika.helium.Helium +import laika.helium.config._ +import org.typelevel.sbt.TypelevelGitHubPlugin.gitHubUserRepo +import sbt.Def._ +import sbt.Keys.licenses + +object TypelevelSiteSettings { + + val defaultHomeLink: ThemeLink = ImageLink.external( + "https://typelevel.org", + Image.external(s"https://typelevel.org/img/logo.svg") + ) + + val defaultFooter: Initialize[Seq[Span]] = setting { + val title = gitHubUserRepo.value.map(_._2) + title.fold(Seq[Span]()) { title => + val licensePhrase = licenses.value.headOption.fold("") { + case (name, url) => + s""" distributed under the $name license""" + } + Seq(TemplateString( + s"""$title is a Typelevel project$licensePhrase.""" + )) + } + } + + val chatLink: IconLink = IconLink.external("https://discord.gg/XF3CXcMzqD", HeliumIcon.chat) + + val twitterLink: IconLink = + IconLink.external("https://twitter.com/typelevel", HeliumIcon.twitter) + + val favIcons: Seq[Favicon] = Seq( + Favicon.external("https://typelevel.org/img/favicon.png", "32x32", "image/png") + ) + + val defaults: Initialize[Helium] = setting { + GenericSiteSettings + .defaults + .value + .site + .layout( + topBarHeight = px(50) + ) + .site + .darkMode + .disabled + .site + .favIcons(favIcons: _*) + .site + .footer(defaultFooter.value: _*) + .site + .topNavigationBar( + homeLink = defaultHomeLink, + navLinks = List(chatLink, twitterLink) + ) + } + +} diff --git a/sonatype-ci-release/src/main/scala/org/typelevel/sbt/TypelevelSonatypeCiReleasePlugin.scala b/sonatype-ci-release/src/main/scala/org/typelevel/sbt/TypelevelSonatypeCiReleasePlugin.scala index 36bd941d..427fca94 100644 --- a/sonatype-ci-release/src/main/scala/org/typelevel/sbt/TypelevelSonatypeCiReleasePlugin.scala +++ b/sonatype-ci-release/src/main/scala/org/typelevel/sbt/TypelevelSonatypeCiReleasePlugin.scala @@ -41,10 +41,6 @@ object TypelevelSonatypeCiReleasePlugin extends AutoPlugin { Seq(tlCiReleaseTags := true, tlCiReleaseBranches := Seq()) override def buildSettings = Seq( - githubWorkflowEnv ++= List( - "SONATYPE_USERNAME", - "SONATYPE_PASSWORD", - "SONATYPE_CREDENTIAL_HOST").map(k => k -> s"$${{ secrets.$k }}").toMap, githubWorkflowPublishTargetBranches := { val branches = tlCiReleaseBranches.value.map(b => RefPredicate.Equals(Ref.Branch(b))) @@ -59,7 +55,11 @@ object TypelevelSonatypeCiReleasePlugin extends AutoPlugin { }, githubWorkflowTargetTags += "v*", githubWorkflowPublish := Seq( - WorkflowStep.Sbt(List("tlRelease"), name = Some("Publish")) + WorkflowStep.Sbt(List("tlRelease"), name = Some("Publish"), env = env) ) ) + + private val env = List("SONATYPE_USERNAME", "SONATYPE_PASSWORD", "SONATYPE_CREDENTIAL_HOST") + .map(k => k -> s"$${{ secrets.$k }}") + .toMap } diff --git a/versioning/build.sbt b/versioning/build.sbt index 688233f6..34e16ce0 100644 --- a/versioning/build.sbt +++ b/versioning/build.sbt @@ -1 +1 @@ -addSbtPlugin("com.typesafe.sbt" % "sbt-git" % "1.0.2") +addSbtPlugin("com.github.sbt" % "sbt-git" % "2.0.0") diff --git a/versioning/src/main/scala/org/typelevel/sbt/TypelevelVersioningPlugin.scala b/versioning/src/main/scala/org/typelevel/sbt/TypelevelVersioningPlugin.scala index aeb13a55..f6688634 100644 --- a/versioning/src/main/scala/org/typelevel/sbt/TypelevelVersioningPlugin.scala +++ b/versioning/src/main/scala/org/typelevel/sbt/TypelevelVersioningPlugin.scala @@ -16,15 +16,17 @@ package org.typelevel.sbt -import com.typesafe.sbt.GitPlugin -import com.typesafe.sbt.SbtGit.git +import com.github.sbt.git.GitPlugin +import com.github.sbt.git.SbtGit.git import org.typelevel.sbt.TypelevelKernelPlugin._ import org.typelevel.sbt.kernel.GitHelper import org.typelevel.sbt.kernel.V import sbt._ +import sbt._ import scala.util.Try +import Keys._ import Keys._ object TypelevelVersioningPlugin extends AutoPlugin {