fixing ServerSpec flaky tests

This commit is contained in:
andrea 2018-10-09 10:20:01 +01:00
parent c9a0698a18
commit a23479dfb1
1 changed files with 109 additions and 74 deletions

View File

@ -11,6 +11,7 @@ import org.scalatest._
import scala.concurrent._
import scala.annotation.tailrec
import sbt.protocol.ClientSocket
import scala.util.Try
import TestServer.withTestServer
import java.io.File
import sbt.io.syntax._
@ -19,101 +20,129 @@ import sbt.RunFromSourceMain
import scala.concurrent.ExecutionContext
import java.util.concurrent.ForkJoinPool
class ServerSpec extends AsyncFreeSpec with Matchers {
class ServerSpec extends fixture.AsyncFreeSpec with fixture.AsyncTestDataFixture with Matchers {
"server" - {
"should start" in withTestServer("handshake") { p =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id": "3", "method": "sbt/setting", "params": { "setting": "root/name" } }"""
)
assert(p.waitForString(10) { s =>
s contains """"id":"3""""
})
"should start" in { implicit td =>
withTestServer("handshake") { p =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id": "3", "method": "sbt/setting", "params": { "setting": "root/name" } }"""
)
assert(p.waitForString(10) { s =>
s contains """"id":"3""""
})
}
}
"return number id when number id is sent" in withTestServer("handshake") { p =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id": 3, "method": "sbt/setting", "params": { "setting": "root/name" } }"""
)
assert(p.waitForString(10) { s =>
s contains """"id":3"""
})
"return number id when number id is sent" in { implicit td =>
withTestServer("handshake") { p =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id": 3, "method": "sbt/setting", "params": { "setting": "root/name" } }"""
)
assert(p.waitForString(10) { s =>
s contains """"id":3"""
})
}
}
"report task failures in case of exceptions" in withTestServer("events") { p =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id": 11, "method": "sbt/exec", "params": { "commandLine": "hello" } }"""
)
assert(p.waitForString(10) { s =>
(s contains """"id":11""") && (s contains """"error":""")
})
"report task failures in case of exceptions" in { implicit td =>
withTestServer("events") { p =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id": 11, "method": "sbt/exec", "params": { "commandLine": "hello" } }"""
)
assert(p.waitForString(10) { s =>
(s contains """"id":11""") && (s contains """"error":""")
})
}
}
"return error if cancelling non-matched task id" in withTestServer("events") { p =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id":12, "method": "sbt/exec", "params": { "commandLine": "run" } }"""
)
p.writeLine(
"""{ "jsonrpc": "2.0", "id":13, "method": "sbt/cancelRequest", "params": { "id": "55" } }"""
)
"return error if cancelling non-matched task id" in { implicit td =>
withTestServer("events") { p =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id":12, "method": "sbt/exec", "params": { "commandLine": "run" } }"""
)
p.writeLine(
"""{ "jsonrpc": "2.0", "id":13, "method": "sbt/cancelRequest", "params": { "id": "55" } }"""
)
assert(p.waitForString(20) { s =>
(s contains """"error":{"code":-32800""")
})
assert(p.waitForString(20) { s =>
(s contains """"error":{"code":-32800""")
})
}
}
"cancel on-going task with numeric id" in withTestServer("events") { p =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id":12, "method": "sbt/exec", "params": { "commandLine": "run" } }"""
)
"cancel on-going task with numeric id" in { implicit td =>
withTestServer("events") { p =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id":12, "method": "sbt/exec", "params": { "commandLine": "run" } }"""
)
Thread.sleep(1000)
p.writeLine(
"""{ "jsonrpc": "2.0", "id":13, "method": "sbt/cancelRequest", "params": { "id": "12" } }"""
)
assert(p.waitForString(30) { s =>
s contains """"result":{"status":"Task cancelled""""
})
assert(p.waitForString(60) { s =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id":13, "method": "sbt/cancelRequest", "params": { "id": "12" } }"""
)
s contains """"result":{"status":"Task cancelled""""
})
}
}
"cancel on-going task with string id" in withTestServer("events") { p =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id": "foo", "method": "sbt/exec", "params": { "commandLine": "run" } }"""
)
"cancel on-going task with string id" in { implicit td =>
withTestServer("events") { p =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id": "foo", "method": "sbt/exec", "params": { "commandLine": "run" } }"""
)
Thread.sleep(1000)
p.writeLine(
"""{ "jsonrpc": "2.0", "id": "bar", "method": "sbt/cancelRequest", "params": { "id": "foo" } }"""
)
assert(p.waitForString(30) { s =>
s contains """"result":{"status":"Task cancelled""""
})
assert(p.waitForString(60) { s =>
p.writeLine(
"""{ "jsonrpc": "2.0", "id": "bar", "method": "sbt/cancelRequest", "params": { "id": "foo" } }"""
)
s contains """"result":{"status":"Task cancelled""""
})
}
}
}
}
object TestServer {
// The test server instance will be executed in a Thread pool separated from the tests
implicit val ec = ExecutionContext.fromExecutor(new ForkJoinPool())
private val serverTestBase: File = new File(".").getAbsoluteFile / "sbt" / "src" / "server-test"
def withTestServer(testBuild: String)(f: TestServer => Future[Assertion]): Future[Assertion] = {
def withTestServer(
testBuild: String
)(f: TestServer => Future[Assertion])(implicit td: TestData): Future[Assertion] = {
println(s"Starting test: ${td.name}")
IO.withTemporaryDirectory { temp =>
IO.copyDirectory(serverTestBase / testBuild, temp / testBuild)
withTestServer(temp / testBuild)(f)
withTestServer(testBuild, temp / testBuild)(f)
}
}
def withTestServer(baseDirectory: File)(f: TestServer => Future[Assertion]): Future[Assertion] = {
val testServer = TestServer(baseDirectory)
try {
f(testServer)
} finally {
try { testServer.bye() } finally {}
def withTestServer(testBuild: String, baseDirectory: File)(
f: TestServer => Future[Assertion]
)(implicit td: TestData): Future[Assertion] = {
// Each test server instance will be executed in a Thread pool separated from the tests
val testServer = TestServer(baseDirectory)(
ExecutionContext.fromExecutor(new ForkJoinPool())
)
// checking last log message after initialization
// if something goes wrong here the communication streams are corrupted, restarting
val init =
Try {
testServer.waitForString(30) { s =>
s contains """"message":"Done""""
}
}.toOption
init match {
case Some(_) =>
try {
f(testServer)
} finally {
try { testServer.bye() } finally {}
}
case _ =>
try { testServer.bye() } finally {}
hostLog("Server started but not connected properly... restarting...")
withTestServer(testBuild)(f)
}
}
@ -216,13 +245,19 @@ case class TestServer(baseDirectory: File)(implicit ec: ExecutionContext) {
@tailrec
final def waitForString(num: Int)(f: String => Boolean): Boolean = {
if (num < 0) false
else
readFrame match {
case Some(x) if f(x) => true
case _ =>
waitForString(num - 1)(f)
if (num < 0) { throw new Exception("Retries are over.") } else {
// readFrame should be called in another Thread in orrder to be able to time limit it's execution
val res = Future { readFrame }(ec)
import scala.concurrent.duration._
Try {
Await.result(res, 1.second)
}.toOption.flatten match {
// function f should be called in this Thread in order to be executed exactly once before eventually returning
case Some(str) if f(str) => true
case _ => waitForString(num - 1)(f)
}
}
}
def readLine: Option[String] = {