From 57353cd5b94d62ee37fb189ed2ba1779eaf1d0c3 Mon Sep 17 00:00:00 2001 From: Seth Tisue Date: Tue, 15 Sep 2020 15:19:15 -0700 Subject: [PATCH 1/6] bump Scala, sbt, Scala.js versions --- .travis.yml | 8 ++++---- .../test/scala/scala/xml/CompilerErrors.scala | 16 +++++++++++----- project/build.properties | 2 +- project/plugins.sbt | 2 +- 4 files changed, 17 insertions(+), 11 deletions(-) diff --git a/.travis.yml b/.travis.yml index 9c90c4644..bef473398 100644 --- a/.travis.yml +++ b/.travis.yml @@ -6,19 +6,19 @@ language: scala scala: - 2.11.12 - - 2.12.10 - - 2.13.1 + - 2.12.12 + - 2.13.3 env: - SCALAJS_VERSION= ADOPTOPENJDK=8 - - SCALAJS_VERSION=0.6.32 ADOPTOPENJDK=8 + - SCALAJS_VERSION=0.6.33 ADOPTOPENJDK=8 - SCALAJS_VERSION=1.0.1 ADOPTOPENJDK=8 - SCALAJS_VERSION= ADOPTOPENJDK=11 matrix: exclude: - scala: 2.11.12 - env: SCALAJS_VERSION=0.6.32 ADOPTOPENJDK=8 + env: SCALAJS_VERSION=0.6.33 ADOPTOPENJDK=8 - scala: 2.11.12 env: SCALAJS_VERSION=1.0.1 ADOPTOPENJDK=8 diff --git a/jvm/src/test/scala/scala/xml/CompilerErrors.scala b/jvm/src/test/scala/scala/xml/CompilerErrors.scala index 6096e5a60..995fd64ed 100644 --- a/jvm/src/test/scala/scala/xml/CompilerErrors.scala +++ b/jvm/src/test/scala/scala/xml/CompilerErrors.scala @@ -4,14 +4,20 @@ import org.junit.Test class CompilerErrors extends CompilerTesting { @Test - def t7185() = - expectXmlError("""|overloaded method value apply with alternatives: - | (f: scala.xml.Node => Boolean)scala.xml.NodeSeq - | (i: Int)scala.xml.Node - | cannot be applied to ()""".stripMargin, + def t7185() = { + // the error message here differs a bit by Scala version + import util.Properties.versionNumberString + val thing = + if (versionNumberString.startsWith("2.11") || versionNumberString.startsWith("2.12")) "method value" + else "method" + expectXmlError(s"""|overloaded $thing apply with alternatives: + | (f: scala.xml.Node => Boolean)scala.xml.NodeSeq + | (i: Int)scala.xml.Node + | cannot be applied to ()""".stripMargin, """|object Test { | () |}""") + } @Test def t1878_typer() = diff --git a/project/build.properties b/project/build.properties index a919a9b5f..0837f7a13 100644 --- a/project/build.properties +++ b/project/build.properties @@ -1 +1 @@ -sbt.version=1.3.8 +sbt.version=1.3.13 diff --git a/project/plugins.sbt b/project/plugins.sbt index c7107fa52..2f2ad8e24 100644 --- a/project/plugins.sbt +++ b/project/plugins.sbt @@ -1,5 +1,5 @@ val scalaJSVersion = - Option(System.getenv("SCALAJS_VERSION")).filter(_.nonEmpty).getOrElse("0.6.32") + Option(System.getenv("SCALAJS_VERSION")).filter(_.nonEmpty).getOrElse("0.6.33") addSbtPlugin("org.scala-lang.modules" % "sbt-scala-module" % "2.1.3") addSbtPlugin("org.portable-scala" % "sbt-scalajs-crossproject" % "1.0.0") From 975b95be3b1b690eddcba305cd8381a45ebce874 Mon Sep 17 00:00:00 2001 From: Lukas Rytz Date: Tue, 11 Jul 2023 12:31:33 +0200 Subject: [PATCH 2/6] Build definitions from main branch Unchanged, from revision 82d76cf2ad. --- .github/workflows/ci.yml | 25 ++++++ .github/workflows/release.yml | 21 +++++ build.sbt | 146 +++++++++++++++++++++++++++------- project/build.properties | 2 +- project/plugins.sbt | 12 +-- 5 files changed, 170 insertions(+), 36 deletions(-) create mode 100644 .github/workflows/ci.yml create mode 100644 .github/workflows/release.yml diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 000000000..7ea58b3d9 --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,25 @@ +name: test +on: + push: + branches: + - main + pull_request: +jobs: + test: + strategy: + fail-fast: false + matrix: + java: [8, 11, 17] + scala: [2.12.x, 2.13.x, 3.x] + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v2 + with: + fetch-depth: 0 + - uses: actions/setup-java@v3 + with: + distribution: temurin + java-version: ${{matrix.java}} + cache: sbt + - name: Test + run: sbt ++${{matrix.scala}} test headerCheck versionPolicyCheck package diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml new file mode 100644 index 000000000..d69ab7208 --- /dev/null +++ b/.github/workflows/release.yml @@ -0,0 +1,21 @@ +name: Release +on: + push: + tags: ["*"] +jobs: + publish: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v2 + with: + fetch-depth: 0 + - uses: actions/setup-java@v2 + with: + distribution: temurin + java-version: 8 + - run: sbt versionCheck ci-release + env: + PGP_PASSPHRASE: ${{secrets.PGP_PASSPHRASE}} + PGP_SECRET: ${{secrets.PGP_SECRET}} + SONATYPE_PASSWORD: ${{secrets.SONATYPE_PASSWORD}} + SONATYPE_USERNAME: ${{secrets.SONATYPE_USERNAME}} diff --git a/build.sbt b/build.sbt index a35583347..a7b82237a 100644 --- a/build.sbt +++ b/build.sbt @@ -1,37 +1,91 @@ import sbtcrossproject.CrossPlugin.autoImport.{crossProject, CrossType} -lazy val xml = crossProject(JSPlatform, JVMPlatform) +publish / skip := true // root project + +ThisBuild / startYear := Some(2002) +ThisBuild / licenses += (("Apache-2.0", url("https://www.apache.org/licenses/LICENSE-2.0"))) + +// because it doesn't declare it itself +ThisBuild / libraryDependencySchemes += "org.scala-js" %% "scalajs-library" % "semver-spec" +ThisBuild / apiURL := Some(url("https://javadoc.io/doc/org.scala-lang.modules/scala-xml_2.13/")) + +lazy val configSettings: Seq[Setting[?]] = Seq( + unmanagedSourceDirectories ++= { + unmanagedSourceDirectories.value.flatMap { dir => + def forVersion(version: String): File = file(dir.getPath ++ "-" ++ version) + CrossVersion.partialVersion(scalaVersion.value) match { + case Some((3, _)) => Seq(forVersion("3"), forVersion("2.13+")) + case Some((2, minor)) => + Seq(forVersion("2")) ++ (minor match { + case 13 => Seq(forVersion("2.13"), forVersion("2.13+")) + case 12 => Seq(forVersion("2.12")) + case _ => Seq() + }) + case _ => Seq() + } + } + } +) + +lazy val xml = crossProject(JSPlatform, JVMPlatform, NativePlatform) .withoutSuffixFor(JVMPlatform) .crossType(CrossType.Full) .in(file(".")) .settings(ScalaModulePlugin.scalaModuleSettings) - .jvmSettings(ScalaModulePlugin.scalaModuleSettingsJVM) + .jvmSettings(ScalaModulePlugin.scalaModuleOsgiSettings) .settings( name := "scala-xml", + scalaModuleAutomaticModuleName := Some("scala.xml"), + crossScalaVersions := Seq("2.13.11", "2.12.18", "3.3.0"), + scalaVersion := "2.12.18", - // Compiler team advised avoiding the -Xfuture option for releases. - // The output with -Xfuture should be periodically checked, though. - scalacOptions ++= "-deprecation:false -feature -Xlint:-stars-align,-nullary-unit,_".split("\\s+").to[Seq], - scalacOptions in Test += "-Xxml:coalescing", + scalacOptions ++= (CrossVersion.partialVersion(scalaVersion.value) match { + case Some((3, _)) => + Seq("-language:Scala2") + case _ => + // Compiler team advised avoiding the -Xsource:3 option for releases. + // The output with -Xsource:3 should be periodically checked, though. + Seq("-deprecation:false", "-feature", "-Xlint:-stars-align,-nullary-unit,_") + }), - scalaModuleMimaPreviousVersion := { - if (System.getenv("SCALAJS_VERSION") == "1.0.1") None - else Some("1.2.0") - }, + Test / scalacOptions += "-Xxml:coalescing", - unmanagedSourceDirectories in Compile ++= { - (unmanagedSourceDirectories in Compile).value.map { dir => - val sv = scalaVersion.value - CrossVersion.partialVersion(sv) match { - case Some((2, 13)) => file(dir.getPath ++ "-2.13") - case _ => file(dir.getPath ++ "-2.11-2.12") - } - } - }, + headerLicense := Some(HeaderLicense.Custom( + s"""|Scala (https://www.scala-lang.org) + | + |Copyright EPFL and Lightbend, Inc. + | + |Licensed under Apache License 2.0 + |(http://www.apache.org/licenses/LICENSE-2.0). + | + |See the NOTICE file distributed with this work for + |additional information regarding copyright ownership. + |""".stripMargin)), - apiURL := Some( - url(s"""https://scala.github.io/scala-xml/api/${"-.*".r.replaceAllIn(version.value, "")}/""") - ), + versionPolicyIntention := Compatibility.BinaryCompatible, + // Note: See discussion on non-JVM Mima in https://github.com/scala/scala-xml/pull/517 + mimaBinaryIssueFilters ++= { + //import com.typesafe.tools.mima.core.{} + //import com.typesafe.tools.mima.core.ProblemFilters + Seq( // exclusions for all Scala versions + ) ++ (CrossVersion.partialVersion(scalaVersion.value) match { + case Some((3, _)) => Seq( // Scala 3-specific exclusions + ) + case Some((2, minor)) => Seq( // Scala 2-specific exclusions + ) ++ (minor match { + case 13 => Seq( // Scala 2.13-specific exclusions + ) + case 12 => Seq( // Scala 2.12-specific exclusions + ) + }) + case _ => Seq() + }) + }, + // Mima signature checking stopped working after 3.0.2 upgrade, see #557 + mimaReportSignatureProblems := (CrossVersion.partialVersion(scalaVersion.value) match { + case Some((3, _)) => false + case _ => true + }), apiMappings ++= scalaInstance.value.libraryJars.filter { file => file.getName.startsWith("scala-library") && file.getName.endsWith(".jar") @@ -47,7 +101,7 @@ lazy val xml = crossProject(JSPlatform, JVMPlatform) file(jarPath) -> url("http://docs.oracle.com/javase/8/docs/api") ) - } getOrElse { + }.getOrElse { // If everything fails, jam in Java 11 modules. Map( file("/modules/java.base") @@ -58,16 +112,50 @@ lazy val xml = crossProject(JSPlatform, JVMPlatform) } } ) + .settings( + inConfig(Compile)(configSettings) ++ inConfig(Test)(configSettings) + ) .jvmSettings( OsgiKeys.exportPackage := Seq(s"scala.xml.*;version=${version.value}"), - libraryDependencies += "junit" % "junit" % "4.13" % "test", - libraryDependencies += "com.novocode" % "junit-interface" % "0.11" % "test", - libraryDependencies += "org.apache.commons" % "commons-lang3" % "3.9" % "test", - libraryDependencies += ("org.scala-lang" % "scala-compiler" % scalaVersion.value % "test").exclude("org.scala-lang.modules", s"scala-xml_${scalaBinaryVersion.value}") + libraryDependencies += "junit" % "junit" % "4.13.2" % Test, + libraryDependencies += "com.github.sbt" % "junit-interface" % "0.13.3" % Test, + libraryDependencies += "org.apache.commons" % "commons-lang3" % "3.12.0" % Test, + libraryDependencies += "xerces" % "xercesImpl" % "2.12.2" % Test, + libraryDependencies ++= (CrossVersion.partialVersion(scalaVersion.value) match { + case Some((3, _)) => + Seq() + case _ => + Seq("org.scala-lang" % "scala-compiler" % scalaVersion.value % Test) + }), ) .jsSettings( // Scala.js cannot run forked tests - fork in Test := false + Test / fork := false + ) + .jsEnablePlugins(ScalaJSJUnitPlugin) + .nativeSettings( + versionPolicyCheck / skip := true, + versionCheck / skip := true, + // Scala Native cannot run forked tests + Test / fork := false, + libraryDependencies += "org.scala-native" %%% "junit-runtime" % nativeVersion % Test, + Test / scalacOptions += { + val log = streams.value.log + val retrieveDir = baseDirectory.value / "scala-native-junit-plugin-jars" + val lm = dependencyResolution.value + val cp = lm + .retrieve( + "org.scala-native" % s"junit-plugin_${scalaVersion.value}" % nativeVersion, + scalaModuleInfo = None, + retrieveDir, + log + ) + .fold(w => throw w.resolveException, identity) + val jarPath = cp + .find(_.toString.contains("junit-plugin")) + .getOrElse(throw new Exception("Can't find Scala Native junit-plugin jar")) + s"-Xplugin:$jarPath" + }, + Test / testOptions += Tests.Argument(TestFrameworks.JUnit, "-a", "-s", "-v"), ) - .jsConfigure(_.enablePlugins(ScalaJSJUnitPlugin)) diff --git a/project/build.properties b/project/build.properties index 0837f7a13..875b706a8 100644 --- a/project/build.properties +++ b/project/build.properties @@ -1 +1 @@ -sbt.version=1.3.13 +sbt.version=1.9.2 diff --git a/project/plugins.sbt b/project/plugins.sbt index 2f2ad8e24..6f6beee39 100644 --- a/project/plugins.sbt +++ b/project/plugins.sbt @@ -1,6 +1,6 @@ -val scalaJSVersion = - Option(System.getenv("SCALAJS_VERSION")).filter(_.nonEmpty).getOrElse("0.6.33") - -addSbtPlugin("org.scala-lang.modules" % "sbt-scala-module" % "2.1.3") -addSbtPlugin("org.portable-scala" % "sbt-scalajs-crossproject" % "1.0.0") -addSbtPlugin("org.scala-js" % "sbt-scalajs" % scalaJSVersion) +addSbtPlugin("org.scala-lang.modules" % "sbt-scala-module" % "3.1.0") +addSbtPlugin("org.portable-scala" % "sbt-scalajs-crossproject" % "1.3.2") +addSbtPlugin("org.portable-scala" % "sbt-scala-native-crossproject" % "1.3.2") +addSbtPlugin("org.scala-js" % "sbt-scalajs" % "1.13.2") +addSbtPlugin("org.scala-native" % "sbt-scala-native" % "0.4.14") +addSbtPlugin("de.heikoseeberger" % "sbt-header" % "5.10.0") From ff85eced6e6e15906c0b0a38fa4a6956bb314f10 Mon Sep 17 00:00:00 2001 From: Lukas Rytz Date: Wed, 12 Jul 2023 14:49:01 +0200 Subject: [PATCH 3/6] adjust build to 1.x branch --- .github/workflows/ci.yml | 4 +- .travis.yml | 28 -------------- build.sbt | 80 ++++------------------------------------ build.sh | 54 --------------------------- 4 files changed, 9 insertions(+), 157 deletions(-) delete mode 100644 .travis.yml delete mode 100755 build.sh diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 7ea58b3d9..2d6c9651e 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -10,7 +10,7 @@ jobs: fail-fast: false matrix: java: [8, 11, 17] - scala: [2.12.x, 2.13.x, 3.x] + scala: [2.11.x, 2.12.x, 2.13.x] runs-on: ubuntu-latest steps: - uses: actions/checkout@v2 @@ -22,4 +22,4 @@ jobs: java-version: ${{matrix.java}} cache: sbt - name: Test - run: sbt ++${{matrix.scala}} test headerCheck versionPolicyCheck package + run: sbt ++${{matrix.scala}} test versionPolicyCheck package diff --git a/.travis.yml b/.travis.yml deleted file mode 100644 index bef473398..000000000 --- a/.travis.yml +++ /dev/null @@ -1,28 +0,0 @@ -version: ~> 1.0 # needed for imports - -import: scala/scala-dev:travis/default.yml - -language: scala - -scala: - - 2.11.12 - - 2.12.12 - - 2.13.3 - -env: - - SCALAJS_VERSION= ADOPTOPENJDK=8 - - SCALAJS_VERSION=0.6.33 ADOPTOPENJDK=8 - - SCALAJS_VERSION=1.0.1 ADOPTOPENJDK=8 - - SCALAJS_VERSION= ADOPTOPENJDK=11 - -matrix: - exclude: - - scala: 2.11.12 - env: SCALAJS_VERSION=0.6.33 ADOPTOPENJDK=8 - - scala: 2.11.12 - env: SCALAJS_VERSION=1.0.1 ADOPTOPENJDK=8 - -install: - - git fetch --tags # get all tags for sbt-dynver - -script: ./build.sh diff --git a/build.sbt b/build.sbt index a7b82237a..72f829f69 100644 --- a/build.sbt +++ b/build.sbt @@ -14,20 +14,14 @@ lazy val configSettings: Seq[Setting[?]] = Seq( unmanagedSourceDirectories.value.flatMap { dir => def forVersion(version: String): File = file(dir.getPath ++ "-" ++ version) CrossVersion.partialVersion(scalaVersion.value) match { - case Some((3, _)) => Seq(forVersion("3"), forVersion("2.13+")) - case Some((2, minor)) => - Seq(forVersion("2")) ++ (minor match { - case 13 => Seq(forVersion("2.13"), forVersion("2.13+")) - case 12 => Seq(forVersion("2.12")) - case _ => Seq() - }) - case _ => Seq() + case Some((2, 13)) => Seq(forVersion("2.13")) + case _ => Seq(forVersion("2.11-2.12")) } } } ) -lazy val xml = crossProject(JSPlatform, JVMPlatform, NativePlatform) +lazy val xml = crossProject(JSPlatform, JVMPlatform) .withoutSuffixFor(JVMPlatform) .crossType(CrossType.Full) .in(file(".")) @@ -36,17 +30,10 @@ lazy val xml = crossProject(JSPlatform, JVMPlatform, NativePlatform) .settings( name := "scala-xml", scalaModuleAutomaticModuleName := Some("scala.xml"), - crossScalaVersions := Seq("2.13.11", "2.12.18", "3.3.0"), + crossScalaVersions := Seq("2.13.11", "2.12.18", "2.11.12"), scalaVersion := "2.12.18", - scalacOptions ++= (CrossVersion.partialVersion(scalaVersion.value) match { - case Some((3, _)) => - Seq("-language:Scala2") - case _ => - // Compiler team advised avoiding the -Xsource:3 option for releases. - // The output with -Xsource:3 should be periodically checked, though. - Seq("-deprecation:false", "-feature", "-Xlint:-stars-align,-nullary-unit,_") - }), + scalacOptions ++= Seq("-deprecation:false", "-feature", "-Xlint:-stars-align,-nullary-unit,_"), Test / scalacOptions += "-Xxml:coalescing", @@ -63,29 +50,6 @@ lazy val xml = crossProject(JSPlatform, JVMPlatform, NativePlatform) |""".stripMargin)), versionPolicyIntention := Compatibility.BinaryCompatible, - // Note: See discussion on non-JVM Mima in https://github.com/scala/scala-xml/pull/517 - mimaBinaryIssueFilters ++= { - //import com.typesafe.tools.mima.core.{} - //import com.typesafe.tools.mima.core.ProblemFilters - Seq( // exclusions for all Scala versions - ) ++ (CrossVersion.partialVersion(scalaVersion.value) match { - case Some((3, _)) => Seq( // Scala 3-specific exclusions - ) - case Some((2, minor)) => Seq( // Scala 2-specific exclusions - ) ++ (minor match { - case 13 => Seq( // Scala 2.13-specific exclusions - ) - case 12 => Seq( // Scala 2.12-specific exclusions - ) - }) - case _ => Seq() - }) - }, - // Mima signature checking stopped working after 3.0.2 upgrade, see #557 - mimaReportSignatureProblems := (CrossVersion.partialVersion(scalaVersion.value) match { - case Some((3, _)) => false - case _ => true - }), apiMappings ++= scalaInstance.value.libraryJars.filter { file => file.getName.startsWith("scala-library") && file.getName.endsWith(".jar") @@ -121,41 +85,11 @@ lazy val xml = crossProject(JSPlatform, JVMPlatform, NativePlatform) libraryDependencies += "junit" % "junit" % "4.13.2" % Test, libraryDependencies += "com.github.sbt" % "junit-interface" % "0.13.3" % Test, libraryDependencies += "org.apache.commons" % "commons-lang3" % "3.12.0" % Test, - libraryDependencies += "xerces" % "xercesImpl" % "2.12.2" % Test, - libraryDependencies ++= (CrossVersion.partialVersion(scalaVersion.value) match { - case Some((3, _)) => - Seq() - case _ => - Seq("org.scala-lang" % "scala-compiler" % scalaVersion.value % Test) - }), + libraryDependencies += ("org.scala-lang" % "scala-compiler" % scalaVersion.value % Test).exclude("org.scala-lang.modules", s"scala-xml_${scalaBinaryVersion.value}") ) .jsSettings( + crossScalaVersions := crossScalaVersions.value.filterNot(_.startsWith("2.11")), // Scala.js cannot run forked tests Test / fork := false ) .jsEnablePlugins(ScalaJSJUnitPlugin) - .nativeSettings( - versionPolicyCheck / skip := true, - versionCheck / skip := true, - // Scala Native cannot run forked tests - Test / fork := false, - libraryDependencies += "org.scala-native" %%% "junit-runtime" % nativeVersion % Test, - Test / scalacOptions += { - val log = streams.value.log - val retrieveDir = baseDirectory.value / "scala-native-junit-plugin-jars" - val lm = dependencyResolution.value - val cp = lm - .retrieve( - "org.scala-native" % s"junit-plugin_${scalaVersion.value}" % nativeVersion, - scalaModuleInfo = None, - retrieveDir, - log - ) - .fold(w => throw w.resolveException, identity) - val jarPath = cp - .find(_.toString.contains("junit-plugin")) - .getOrElse(throw new Exception("Can't find Scala Native junit-plugin jar")) - s"-Xplugin:$jarPath" - }, - Test / testOptions += Tests.Argument(TestFrameworks.JUnit, "-a", "-s", "-v"), - ) diff --git a/build.sh b/build.sh deleted file mode 100755 index 11ca2bb0a..000000000 --- a/build.sh +++ /dev/null @@ -1,54 +0,0 @@ -#!/bin/bash - -set -e - -# Builds of tagged revisions are published to sonatype staging. - -# Travis runs a build on new revisions and on new tags, so a tagged revision is built twice. -# Builds for a tag have TRAVIS_TAG defined, which we use for identifying tagged builds. - -# sbt-dynver sets the version number from the tag -# sbt-travisci sets the Scala version from the travis job matrix - -# To back-publish an existing release for a new Scala / Scala.js / Scala Native version: -# - check out the tag for the version that needs to be published -# - change `.travis.yml` to adjust the version numbers and trim down the build matrix as necessary -# - commit the changes and tag this new revision with an arbitrary suffix after a hash, e.g., -# `v1.2.3#dotty-0.27` (the suffix is ignored, the version will be `1.2.3`) - -# We release on JDK 8 (for Scala 2.x and Dotty 0.x) -isReleaseJob() { - if [[ "$ADOPTOPENJDK" == "8" ]]; then - true - else - false - fi -} - -if [[ "$SCALAJS_VERSION" == "" ]]; then - projectPrefix="xml" -else - projectPrefix="xmlJS" -fi - -verPat="[0-9]+\.[0-9]+\.[0-9]+(-[A-Za-z0-9-]+)?" -tagPat="^v$verPat(#.*)?$" - -if [[ "$TRAVIS_TAG" =~ $tagPat ]]; then - releaseTask="ci-release" - if ! isReleaseJob; then - echo "Not releasing on Java $ADOPTOPENJDK with Scala $TRAVIS_SCALA_VERSION" - exit 0 - fi -fi - -# default is +publishSigned; we cross-build with travis jobs, not sbt's crossScalaVersions -export CI_RELEASE="$projectPrefix/publishSigned" -export CI_SNAPSHOT_RELEASE="$projectPrefix/publish" - -# default is sonatypeBundleRelease, which closes and releases the staging repo -# see https://github.com/xerial/sbt-sonatype#commands -# for now, until we're confident in the new release scripts, just close the staging repo. -export CI_SONATYPE_RELEASE="; sonatypePrepare; sonatypeBundleUpload; sonatypeClose" - -sbt clean $projectPrefix/test $projectPrefix/publishLocal $releaseTask From 63e24f51e7579c08824afccd507be6b675d2e0d0 Mon Sep 17 00:00:00 2001 From: Lukas Rytz Date: Wed, 21 Jun 2023 14:50:40 +0200 Subject: [PATCH 4/6] make Node.toString stack safe --- shared/src/main/scala/scala/xml/Utility.scala | 110 +++++++++++------- .../test/scala/scala/xml/UtilityTest.scala | 6 + 2 files changed, 72 insertions(+), 44 deletions(-) diff --git a/shared/src/main/scala/scala/xml/Utility.scala b/shared/src/main/scala/scala/xml/Utility.scala index 9374ff990..2d27bb0e8 100755 --- a/shared/src/main/scala/scala/xml/Utility.scala +++ b/shared/src/main/scala/scala/xml/Utility.scala @@ -9,6 +9,7 @@ package scala package xml +import scala.annotation.tailrec import scala.collection.mutable import scala.language.implicitConversions import scala.collection.Seq @@ -187,9 +188,7 @@ object Utility extends AnyRef with parsing.TokenTests { decodeEntities: Boolean = true, preserveWhitespace: Boolean = false, minimizeTags: Boolean = false): StringBuilder = - { serialize(x, pscope, sb, stripComments, decodeEntities, preserveWhitespace, if (minimizeTags) MinimizeMode.Always else MinimizeMode.Never) - } /** * Serialize an XML Node to a StringBuilder. @@ -206,35 +205,67 @@ object Utility extends AnyRef with parsing.TokenTests { stripComments: Boolean = false, decodeEntities: Boolean = true, preserveWhitespace: Boolean = false, - minimizeTags: MinimizeMode.Value = MinimizeMode.Default): StringBuilder = - { - x match { - case c: Comment => if (!stripComments) c buildString sb; sb - case s: SpecialNode => s buildString sb - case g: Group => - for (c <- g.nodes) serialize(c, g.scope, sb, stripComments, decodeEntities, preserveWhitespace, minimizeTags); sb - case el: Elem => - // print tag with namespace declarations - sb.append('<') - el.nameToString(sb) - if (el.attributes ne null) el.attributes.buildString(sb) - el.scope.buildString(sb, pscope) - if (el.child.isEmpty && - (minimizeTags == MinimizeMode.Always || - (minimizeTags == MinimizeMode.Default && el.minimizeEmpty))) { - // no children, so use short form: - sb.append("/>") - } else { - // children, so use long form: ... - sb.append('>') - sequenceToXML(el.child, el.scope, sb, stripComments, decodeEntities, preserveWhitespace, minimizeTags) - sb.append("') - } - case _ => throw new IllegalArgumentException("Don't know how to serialize a " + x.getClass.getName) - } + minimizeTags: MinimizeMode.Value = MinimizeMode.Default + ): StringBuilder = { + serializeImpl(List(x), pscope, false, stripComments, minimizeTags, sb) + sb + } + + private def serializeImpl( + ns: Seq[Node], + pscope: NamespaceBinding, + spaced: Boolean, + stripComments: Boolean, + minimizeTags: MinimizeMode.Value, + sb: StringBuilder + ): Unit = { + @tailrec def ser(nss: List[Seq[Node]], pscopes: List[NamespaceBinding], spaced: List[Boolean], toClose: List[Node]): Unit = nss match { + case List(ns) if ns.isEmpty => + case ns :: rests if ns.isEmpty => + if (toClose.head != null) { + sb.append("') + } + ser(rests, pscopes.tail, spaced.tail, toClose.tail) + case ns1 :: r => + val (n, ns) = (ns1.head, ns1.tail) + def sp(): Unit = if (ns.nonEmpty && spaced.head) sb.append(' ') + n match { + case c: Comment => + if (!stripComments) { + c.buildString(sb) + sp() + } + ser(ns :: r, pscopes, spaced, toClose) + case s: SpecialNode => + s.buildString(sb) + sp() + ser(ns :: r, pscopes, spaced, toClose) + case g: Group => + ser(g.nodes :: ns :: r, g.scope :: pscopes, false :: spaced, null :: toClose) + case e: Elem => + sb.append('<') + e.nameToString(sb) + if (e.attributes.ne(null)) e.attributes.buildString(sb) + e.scope.buildString(sb, pscopes.head) + if (e.child.isEmpty && + (minimizeTags == MinimizeMode.Always || + (minimizeTags == MinimizeMode.Default && e.minimizeEmpty))) { + // no children, so use short form: + sb.append("/>") + sp() + ser(ns :: r, pscopes, spaced, toClose) + } else { + sb.append('>') + val csp = e.child.forall(isAtomAndNotText) + ser(e.child :: ns :: r, e.scope :: pscopes, csp :: spaced, e :: toClose) + } + case n => throw new IllegalArgumentException("Don't know how to serialize a " + n.getClass.getName) + } } + ser(List(ns), List(pscope), List(spaced), Nil) + } def sequenceToXML( children: Seq[Node], @@ -243,20 +274,11 @@ object Utility extends AnyRef with parsing.TokenTests { stripComments: Boolean = false, decodeEntities: Boolean = true, preserveWhitespace: Boolean = false, - minimizeTags: MinimizeMode.Value = MinimizeMode.Default): Unit = - { - if (children.isEmpty) return - else if (children forall isAtomAndNotText) { // add space - val it = children.iterator - val f = it.next() - serialize(f, pscope, sb, stripComments, decodeEntities, preserveWhitespace, minimizeTags) - while (it.hasNext) { - val x = it.next() - sb.append(' ') - serialize(x, pscope, sb, stripComments, decodeEntities, preserveWhitespace, minimizeTags) - } - } else children foreach { serialize(_, pscope, sb, stripComments, decodeEntities, preserveWhitespace, minimizeTags) } - } + minimizeTags: MinimizeMode.Value = MinimizeMode.Default + ): Unit = if (children.nonEmpty) { + val spaced = children.forall(isAtomAndNotText) + serializeImpl(children, pscope, spaced, stripComments, minimizeTags, sb) + } /** * Returns prefix of qualified name if any. diff --git a/shared/src/test/scala/scala/xml/UtilityTest.scala b/shared/src/test/scala/scala/xml/UtilityTest.scala index a27fb0b1a..b7eb95a80 100644 --- a/shared/src/test/scala/scala/xml/UtilityTest.scala +++ b/shared/src/test/scala/scala/xml/UtilityTest.scala @@ -217,4 +217,10 @@ class UtilityTest { val x =
{Text(" My name ")}{Text(" is ")}{Text(" Harry ")}
assertEquals(
My name is Harry
, Utility.trim(x)) } + + @Test + def toStringStackSafe(): Unit = { + val xml = (1 to 5000).foldRight() { case (_, n) => {n} } + xml.toString + } } From c9c8f2186d8238918247f17ba5dc08f2c8a194d5 Mon Sep 17 00:00:00 2001 From: Lukas Rytz Date: Thu, 22 Jun 2023 10:58:58 +0200 Subject: [PATCH 5/6] Use List[Node] in toString --- shared/src/main/scala/scala/xml/Utility.scala | 15 +++++++-------- 1 file changed, 7 insertions(+), 8 deletions(-) diff --git a/shared/src/main/scala/scala/xml/Utility.scala b/shared/src/main/scala/scala/xml/Utility.scala index 2d27bb0e8..4385e5807 100755 --- a/shared/src/main/scala/scala/xml/Utility.scala +++ b/shared/src/main/scala/scala/xml/Utility.scala @@ -219,17 +219,16 @@ object Utility extends AnyRef with parsing.TokenTests { minimizeTags: MinimizeMode.Value, sb: StringBuilder ): Unit = { - @tailrec def ser(nss: List[Seq[Node]], pscopes: List[NamespaceBinding], spaced: List[Boolean], toClose: List[Node]): Unit = nss match { - case List(ns) if ns.isEmpty => - case ns :: rests if ns.isEmpty => + @tailrec def ser(nss: List[List[Node]], pscopes: List[NamespaceBinding], spaced: List[Boolean], toClose: List[Node]): Unit = nss match { + case List(Nil) => + case Nil :: rests => if (toClose.head != null) { sb.append("') } ser(rests, pscopes.tail, spaced.tail, toClose.tail) - case ns1 :: r => - val (n, ns) = (ns1.head, ns1.tail) + case (n :: ns) :: r => def sp(): Unit = if (ns.nonEmpty && spaced.head) sb.append(' ') n match { case c: Comment => @@ -243,7 +242,7 @@ object Utility extends AnyRef with parsing.TokenTests { sp() ser(ns :: r, pscopes, spaced, toClose) case g: Group => - ser(g.nodes :: ns :: r, g.scope :: pscopes, false :: spaced, null :: toClose) + ser(g.nodes.toList :: ns :: r, g.scope :: pscopes, false :: spaced, null :: toClose) case e: Elem => sb.append('<') e.nameToString(sb) @@ -259,12 +258,12 @@ object Utility extends AnyRef with parsing.TokenTests { } else { sb.append('>') val csp = e.child.forall(isAtomAndNotText) - ser(e.child :: ns :: r, e.scope :: pscopes, csp :: spaced, e :: toClose) + ser(e.child.toList :: ns :: r, e.scope :: pscopes, csp :: spaced, e :: toClose) } case n => throw new IllegalArgumentException("Don't know how to serialize a " + n.getClass.getName) } } - ser(List(ns), List(pscope), List(spaced), Nil) + ser(List(ns.toList), List(pscope), List(spaced), Nil) } def sequenceToXML( From 46971ed510256d289af0f2cf94dee552b3827753 Mon Sep 17 00:00:00 2001 From: Lukas Rytz Date: Mon, 17 Jul 2023 11:46:41 +0200 Subject: [PATCH 6/6] update version policy for minor release --- build.sbt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.sbt b/build.sbt index 72f829f69..cbd2c98ef 100644 --- a/build.sbt +++ b/build.sbt @@ -49,7 +49,7 @@ lazy val xml = crossProject(JSPlatform, JVMPlatform) |additional information regarding copyright ownership. |""".stripMargin)), - versionPolicyIntention := Compatibility.BinaryCompatible, + versionPolicyIntention := Compatibility.BinaryAndSourceCompatible, apiMappings ++= scalaInstance.value.libraryJars.filter { file => file.getName.startsWith("scala-library") && file.getName.endsWith(".jar")