project/MultiNode.scala (73 lines of code) (raw):
/*
* 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
*
* This file is part of the Apache Pekko project, which was derived from Akka.
*/
/*
* Copyright (C) 2009-2020 Lightbend Inc. <https://www.lightbend.com>
*/
import com.typesafe.sbt.MultiJvmPlugin
import MultiJvmPlugin.autoImport._
import sbt._
import sbt.Keys._
object MultiNode extends AutoPlugin {
object CliOptions {
val multiNode = CliOption("pekko.test.multi-node", false)
val sbtLogNoFormat = CliOption("sbt.log.noformat", false)
def seqWithProperty(name: String) = Option(System.getProperty(name)).toSeq
val hostsFileName = seqWithProperty("pekko.test.multi-node.hostsFileName")
val javaName = seqWithProperty("pekko.test.multi-node.java")
val targetDirName = seqWithProperty("pekko.test.multi-node.targetDirName")
}
val multiExecuteTests =
CliOptions.multiNode.ifTrue(MultiJvm / multiNodeExecuteTests).getOrElse(MultiJvm / executeTests)
val multiTest = CliOptions.multiNode.ifTrue(MultiJvm / multiNodeTest).getOrElse(MultiJvm / test)
override def trigger = noTrigger
override def requires = plugins.JvmPlugin && MultiJvmPlugin
override lazy val projectSettings = multiJvmSettings
private val defaultMultiJvmOptions: Seq[String] = {
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
// -DMultiJvm.pekko.cluster.Stress.nrOfNodes=15
val MultinodeJvmArgs = "multinode\\.(D|X)(.*)".r
val knownPrefix = Set("multinode.", "pekko.", "MultiJvm.")
val pekkoProperties =
System.getProperties.propertyNames.asInstanceOf[java.util.Enumeration[String]].asScala.toList.collect {
case MultinodeJvmArgs(a, b) =>
val value = System.getProperty("multinode." + a + b)
"-" + a + b + (if (value == "") "" else "=" + value)
case key: String if knownPrefix.exists(pre => key.startsWith(pre)) => "-D" + key + "=" + System.getProperty(key)
}
"-Xmx256m" :: pekkoProperties ::: CliOptions.sbtLogNoFormat.ifTrue("-Dpekko.test.nocolor=true").toList
}
private val multiJvmSettings =
MultiJvmPlugin.multiJvmSettings ++
inConfig(MultiJvm)(org.scalafmt.sbt.ScalafmtPlugin.scalafmtConfigSettings) ++
inConfig(MultiJvm)(Seq(
MultiJvm / jvmOptions := defaultMultiJvmOptions,
MultiJvm / scalacOptions := (Test / scalacOptions).value,
MultiJvm / compile := (MultiJvm / compile).triggeredBy(Test / compile).value)) ++
CliOptions.hostsFileName.map(MultiJvm / multiNodeHostsFileName := _) ++
CliOptions.javaName.map(MultiJvm / multiNodeJavaName := _) ++
CliOptions.targetDirName.map(MultiJvm / multiNodeTargetDirName := _) ++
// make sure that MultiJvm tests are executed by the default test target,
// and combine the results from ordinary test and multi-jvm tests
(Test / executeTests := {
val testResults = (Test / executeTests).value
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)
})
implicit class TestResultOps(val self: TestResult) extends AnyVal {
def id: Int = self match {
case TestResult.Passed => 0
case TestResult.Failed => 1
case TestResult.Error => 2
}
}
}
/**
* Additional settings for scalatest.
*/
object MultiNodeScalaTest extends AutoPlugin {
override def requires = MultiNode
override lazy val projectSettings = Seq(
MultiJvm / extraOptions := {
val src = (MultiJvm / sourceDirectory).value
(name: String) => (src ** (name + ".conf")).get.headOption.map("-Dpekko.config=" + _.absolutePath).toSeq
},
MultiJvm / scalatestOptions := {
Seq("-C", "org.scalatest.extra.QuietReporter")
})
}