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") }) }