2023-01-08 17:13:31 +08:00
|
|
|
/*
|
|
|
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
|
|
|
* license agreements; and to You under the Apache License, version 2.0:
|
|
|
|
|
*
|
|
|
|
|
* https://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
*
|
2023-06-22 14:19:26 +01:00
|
|
|
* This file is part of the Apache Pekko project, which was derived from Akka.
|
2023-01-08 17:13:31 +08:00
|
|
|
*/
|
|
|
|
|
|
2019-01-02 18:55:26 +08:00
|
|
|
/*
|
2022-02-04 12:36:44 +01:00
|
|
|
* Copyright (C) 2009-2022 Lightbend Inc. <https://www.lightbend.com>
|
2015-05-21 09:48:49 +03:00
|
|
|
*/
|
2018-03-13 23:45:55 +09:00
|
|
|
|
2023-07-30 11:55:14 +02:00
|
|
|
import TestExtras.Filter.Keys._
|
2024-01-21 11:27:34 +11:00
|
|
|
import MultiJvmPlugin.autoImport.multiJvmCreateLogger
|
|
|
|
|
import MultiJvmPlugin.autoImport._
|
2014-05-07 14:49:35 +02:00
|
|
|
|
2018-04-24 16:03:55 +01:00
|
|
|
import sbt.{ Def, _ }
|
2015-05-21 09:48:49 +03:00
|
|
|
import sbt.Keys._
|
2018-04-24 16:03:55 +01:00
|
|
|
import de.heikoseeberger.sbtheader.HeaderPlugin.autoImport._
|
2019-02-09 14:21:16 +01:00
|
|
|
import org.scalafmt.sbt.ScalafmtPlugin.scalafmtConfigSettings
|
2019-09-06 14:59:57 +02:00
|
|
|
import sbtassembly.MergeStrategy
|
|
|
|
|
import sbtassembly.AssemblyKeys._
|
2015-05-21 09:48:49 +03:00
|
|
|
|
|
|
|
|
object MultiNode extends AutoPlugin {
|
2017-10-06 10:30:28 +02:00
|
|
|
|
2019-03-12 11:43:02 +00:00
|
|
|
object autoImport {
|
2024-01-22 07:15:16 +01:00
|
|
|
lazy val validateCompile = taskKey[Unit]("Validates compile for any project it is enabled")
|
2019-03-12 11:43:02 +00:00
|
|
|
}
|
|
|
|
|
import autoImport._
|
|
|
|
|
|
2016-11-29 08:33:36 +01:00
|
|
|
// MultiJvm tests can be excluded from normal test target an validatePullRequest
|
2022-12-02 04:53:48 -08:00
|
|
|
// with -Dpekko.test.multi-in-test=false
|
2024-01-22 07:15:16 +01:00
|
|
|
lazy val multiNodeTestInTest: Boolean = sys.props.getOrElse("pekko.test.multi-in-test", "true").toBoolean
|
2015-05-21 09:48:49 +03:00
|
|
|
|
|
|
|
|
object CliOptions {
|
2024-01-22 07:15:16 +01:00
|
|
|
lazy val multiNode = CliOption("pekko.test.multi-node", false)
|
|
|
|
|
lazy val sbtLogNoFormat = CliOption("sbt.log.noformat", false)
|
2015-05-21 09:48:49 +03:00
|
|
|
|
2024-01-22 07:15:16 +01:00
|
|
|
lazy val hostsFileName = sys.props.get("pekko.test.multi-node.hostsFileName").toSeq
|
|
|
|
|
lazy val javaName = sys.props.get("pekko.test.multi-node.java").toSeq
|
|
|
|
|
lazy val targetDirName = sys.props.get("pekko.test.multi-node.targetDirName").toSeq
|
2015-05-21 09:48:49 +03:00
|
|
|
}
|
2014-05-07 14:49:35 +02:00
|
|
|
|
2024-01-22 07:15:16 +01:00
|
|
|
lazy val multiExecuteTests =
|
2021-05-25 12:50:51 +02:00
|
|
|
CliOptions.multiNode.ifTrue(MultiJvm / multiNodeExecuteTests).getOrElse(MultiJvm / executeTests)
|
2024-01-22 07:15:16 +01:00
|
|
|
lazy val multiTest = CliOptions.multiNode.ifTrue(MultiJvm / multiNodeTest).getOrElse(MultiJvm / test)
|
2014-05-07 14:49:35 +02:00
|
|
|
|
2024-01-22 07:15:16 +01:00
|
|
|
override lazy val trigger = noTrigger
|
|
|
|
|
override lazy val requires = plugins.JvmPlugin && MultiJvmPlugin
|
2014-05-07 14:49:35 +02:00
|
|
|
|
2018-04-24 16:03:55 +01:00
|
|
|
override lazy val projectSettings: Seq[Def.Setting[_]] = multiJvmSettings
|
2015-05-21 09:48:49 +03:00
|
|
|
|
2024-01-22 07:15:16 +01:00
|
|
|
private lazy val defaultMultiJvmOptions: Seq[String] = {
|
2014-05-07 14:49:35 +02:00
|
|
|
import scala.collection.JavaConverters._
|
|
|
|
|
// multinode.D= and multinode.X= makes it possible to pass arbitrary
|
|
|
|
|
// -D or -X arguments to the forked jvm, e.g.
|
|
|
|
|
// -Dmultinode.Djava.net.preferIPv4Stack=true -Dmultinode.Xmx512m -Dmultinode.XX:MaxPermSize=256M
|
2023-02-22 12:48:15 +01:00
|
|
|
// -DMultiJvm.pekko.cluster.Stress.nrOfNodes=15
|
2014-05-07 14:49:35 +02:00
|
|
|
val MultinodeJvmArgs = "multinode\\.(D|X)(.*)".r
|
2023-02-22 12:48:15 +01:00
|
|
|
val knownPrefix = Set("pekko.", "akka.", "MultiJvm.", "aeron.")
|
2023-02-17 10:49:40 +01:00
|
|
|
val pekkoProperties = System.getProperties.stringPropertyNames.asScala.toList.collect {
|
2019-02-09 15:25:39 +01:00
|
|
|
case MultinodeJvmArgs(a, b) =>
|
2014-05-07 14:49:35 +02:00
|
|
|
val value = System.getProperty("multinode." + a + b)
|
|
|
|
|
"-" + a + b + (if (value == "") "" else "=" + value)
|
2019-02-09 15:25:39 +01:00
|
|
|
case key: String if knownPrefix.exists(pre => key.startsWith(pre)) => "-D" + key + "=" + System.getProperty(key)
|
2014-05-07 14:49:35 +02:00
|
|
|
}
|
|
|
|
|
|
2023-02-17 10:49:40 +01:00
|
|
|
"-Xmx256m" :: pekkoProperties ::: CliOptions.sbtLogNoFormat.ifTrue("-Dpekko.test.nocolor=true").toList
|
2022-01-27 16:14:29 +01:00
|
|
|
} ++ JdkOptions.versionSpecificJavaOptions
|
2014-05-07 14:49:35 +02:00
|
|
|
|
2024-01-22 07:15:16 +01:00
|
|
|
private lazy val anyConfigsInThisProject = ScopeFilter(configurations = inAnyConfiguration)
|
2019-03-12 11:43:02 +00:00
|
|
|
|
2024-01-22 07:15:16 +01:00
|
|
|
private lazy val multiJvmSettings =
|
2023-07-30 11:55:14 +02:00
|
|
|
MultiJvmPlugin.multiJvmSettings ++
|
2025-07-07 19:13:19 +08:00
|
|
|
scalafmtConfigSettings(MultiJvm) ++
|
2019-09-06 14:59:57 +02:00
|
|
|
Seq(
|
|
|
|
|
// Hack because 'provided' dependencies by default are not picked up by the multi-jvm plugin:
|
2021-05-25 12:50:51 +02:00
|
|
|
MultiJvm / managedClasspath ++= (Compile / managedClasspath).value.filter(_.data.name.contains("silencer-lib")),
|
|
|
|
|
MultiJvm / jvmOptions := defaultMultiJvmOptions,
|
|
|
|
|
MultiJvm / scalacOptions := (Test / scalacOptions).value,
|
|
|
|
|
multiJvmCreateLogger / logLevel := Level.Debug, // to see ssh establishment
|
2021-08-02 19:27:03 +02:00
|
|
|
MultiJvm / assembly / assemblyMergeStrategy := {
|
2021-02-01 15:38:29 +00:00
|
|
|
case n if n.endsWith("logback-test.xml") => MergeStrategy.first
|
2023-07-15 21:39:02 +08:00
|
|
|
case n if n.endsWith("io.netty.versions.properties") => MergeStrategy.first
|
2019-09-09 12:36:05 -07:00
|
|
|
case n if n.toLowerCase.matches("meta-inf.*\\.default") => MergeStrategy.first
|
2021-05-25 12:50:51 +02:00
|
|
|
case n => (MultiJvm / assembly / assemblyMergeStrategy).value.apply(n)
|
2019-09-06 14:59:57 +02:00
|
|
|
},
|
2021-05-25 12:50:51 +02:00
|
|
|
MultiJvm / multiJvmCreateLogger := { // to use normal sbt logging infra instead of custom sbt-multijvm-one
|
|
|
|
|
val previous = (MultiJvm / multiJvmCreateLogger).value
|
2019-09-06 14:59:57 +02:00
|
|
|
val logger = streams.value.log
|
|
|
|
|
(name: String) =>
|
|
|
|
|
new Logger {
|
|
|
|
|
def trace(t: => Throwable): Unit = { logger.trace(t) }
|
2021-08-19 20:03:07 -04:00
|
|
|
def success(message: => String): Unit = { logger.success(message) }
|
2019-09-06 14:59:57 +02:00
|
|
|
def log(level: Level.Value, message: => String): Unit =
|
|
|
|
|
logger.log(level, s"[${scala.Console.BLUE}$name${scala.Console.RESET}] $message")
|
2017-10-06 10:30:28 +02:00
|
|
|
}
|
2019-09-06 14:59:57 +02:00
|
|
|
}) ++
|
2021-05-25 12:50:51 +02:00
|
|
|
CliOptions.hostsFileName.map(MultiJvm / multiNodeHostsFileName := _) ++
|
|
|
|
|
CliOptions.javaName.map(MultiJvm / multiNodeJavaName := _) ++
|
|
|
|
|
CliOptions.targetDirName.map(MultiJvm / multiNodeTargetDirName := _) ++
|
2019-09-06 14:59:57 +02:00
|
|
|
(if (multiNodeTestInTest) {
|
|
|
|
|
// make sure that MultiJvm tests are executed by the default test target,
|
|
|
|
|
// and combine the results from ordinary test and multi-jvm tests
|
2021-05-25 12:50:51 +02:00
|
|
|
(Test / executeTests) := {
|
|
|
|
|
val testResults = (Test / executeTests).value
|
2019-09-06 14:59:57 +02:00
|
|
|
val multiNodeResults = multiExecuteTests.value
|
|
|
|
|
val overall =
|
|
|
|
|
if (testResults.overall.id < multiNodeResults.overall.id)
|
|
|
|
|
multiNodeResults.overall
|
|
|
|
|
else
|
|
|
|
|
testResults.overall
|
|
|
|
|
|
|
|
|
|
Tests.Output(
|
|
|
|
|
overall,
|
|
|
|
|
testResults.events ++ multiNodeResults.events,
|
|
|
|
|
testResults.summaries ++ multiNodeResults.summaries)
|
|
|
|
|
}
|
|
|
|
|
} else Nil) ++
|
2021-05-25 12:50:51 +02:00
|
|
|
Def.settings((MultiJvm / compile) := {
|
|
|
|
|
(MultiJvm / headerCreate).value
|
|
|
|
|
(MultiJvm / compile).value
|
2019-09-06 14:59:57 +02:00
|
|
|
}) ++ headerSettings(MultiJvm) ++ Seq(validateCompile := compile.?.all(anyConfigsInThisProject).value)
|
2017-10-30 03:13:14 +02:00
|
|
|
|
|
|
|
|
implicit class TestResultOps(val self: TestResult) extends AnyVal {
|
|
|
|
|
def id: Int = self match {
|
2019-02-09 15:25:39 +01:00
|
|
|
case TestResult.Passed => 0
|
|
|
|
|
case TestResult.Failed => 1
|
|
|
|
|
case TestResult.Error => 2
|
2017-10-30 03:13:14 +02:00
|
|
|
}
|
|
|
|
|
}
|
2015-05-21 09:48:49 +03:00
|
|
|
}
|
2014-05-12 15:36:49 +02:00
|
|
|
|
2015-05-21 09:48:49 +03:00
|
|
|
/**
|
|
|
|
|
* Additional settings for scalatest.
|
|
|
|
|
*/
|
|
|
|
|
object MultiNodeScalaTest extends AutoPlugin {
|
|
|
|
|
|
2024-01-22 07:15:16 +01:00
|
|
|
override lazy val requires = MultiNode
|
2015-05-21 09:48:49 +03:00
|
|
|
|
2019-09-06 14:59:57 +02:00
|
|
|
override lazy val projectSettings =
|
|
|
|
|
Seq(
|
2021-05-25 12:50:51 +02:00
|
|
|
MultiJvm / extraOptions := {
|
|
|
|
|
val src = (MultiJvm / sourceDirectory).value
|
2022-12-02 04:53:48 -08:00
|
|
|
(name: String) => (src ** (name + ".conf")).get.headOption.map("-Dpekko.config=" + _.absolutePath).toSeq
|
2019-09-06 14:59:57 +02:00
|
|
|
},
|
2021-05-25 12:50:51 +02:00
|
|
|
MultiJvm / scalatestOptions := {
|
2019-09-06 14:59:57 +02:00
|
|
|
Seq("-C", "org.scalatest.extra.QuietReporter") ++
|
|
|
|
|
(if (excludeTestTags.value.isEmpty) Seq.empty
|
|
|
|
|
else
|
|
|
|
|
Seq(
|
|
|
|
|
"-l",
|
|
|
|
|
if (MultiNode.CliOptions.multiNode.get) excludeTestTags.value.mkString("\"", " ", "\"")
|
|
|
|
|
else excludeTestTags.value.mkString(" "))) ++
|
|
|
|
|
(if (onlyTestTags.value.isEmpty) Seq.empty
|
|
|
|
|
else
|
|
|
|
|
Seq(
|
|
|
|
|
"-n",
|
|
|
|
|
if (MultiNode.CliOptions.multiNode.get) onlyTestTags.value.mkString("\"", " ", "\"")
|
|
|
|
|
else onlyTestTags.value.mkString(" ")))
|
|
|
|
|
})
|
2014-05-07 14:49:35 +02:00
|
|
|
}
|