2020-02-17 00:01:45 +01:00
|
|
|
/*
|
|
|
|
|
* sbt
|
|
|
|
|
* Copyright 2011 - 2018, Lightbend, Inc.
|
|
|
|
|
* Copyright 2008 - 2010, Mark Harrah
|
|
|
|
|
* Licensed under Apache License 2.0 (see LICENSE)
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
package testpkg
|
|
|
|
|
|
2021-08-27 09:02:47 +02:00
|
|
|
import sbt.internal.bsp._
|
2021-06-28 15:35:09 +02:00
|
|
|
import sbt.internal.langserver.ErrorCodes
|
|
|
|
|
import sbt.IO
|
2021-10-29 14:31:01 +02:00
|
|
|
import sbt.internal.protocol.JsonRpcRequestMessage
|
|
|
|
|
import sbt.internal.protocol.codec.JsonRPCProtocol._
|
|
|
|
|
import sjsonnew.JsonWriter
|
|
|
|
|
import sjsonnew.support.scalajson.unsafe.{ CompactPrinter, Converter }
|
2021-06-30 03:19:24 +02:00
|
|
|
|
2021-06-28 15:35:09 +02:00
|
|
|
import java.io.File
|
2021-10-29 14:31:01 +02:00
|
|
|
import java.net.URI
|
2021-08-27 15:58:46 +02:00
|
|
|
import java.nio.file.Paths
|
2023-01-27 10:22:23 +01:00
|
|
|
import java.util.concurrent.atomic.AtomicInteger
|
2020-02-17 00:01:45 +01:00
|
|
|
import scala.concurrent.duration._
|
|
|
|
|
|
|
|
|
|
// starts svr using server-test/buildserver and perform custom server tests
|
|
|
|
|
object BuildServerTest extends AbstractServerTest {
|
2021-06-30 03:19:24 +02:00
|
|
|
|
|
|
|
|
import sbt.internal.bsp.codec.JsonProtocol._
|
|
|
|
|
|
2020-02-17 00:01:45 +01:00
|
|
|
override val testDirectory: String = "buildserver"
|
|
|
|
|
|
2023-01-27 10:22:23 +01:00
|
|
|
private val idGen: AtomicInteger = new AtomicInteger(0)
|
|
|
|
|
private def nextId(): Int = idGen.getAndIncrement()
|
|
|
|
|
|
2020-02-17 00:01:45 +01:00
|
|
|
test("build/initialize") { _ =>
|
2023-01-27 10:22:23 +01:00
|
|
|
val id = nextId()
|
|
|
|
|
initializeRequest(id)
|
2020-02-17 00:01:45 +01:00
|
|
|
assert(svr.waitForString(10.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
(s contains s""""id":"${id}"""") &&
|
2022-07-28 04:40:18 +02:00
|
|
|
(s contains """"resourcesProvider":true""") &&
|
|
|
|
|
(s contains """"outputPathsProvider":true""")
|
2020-02-17 00:01:45 +01:00
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
test("workspace/buildTargets") { _ =>
|
|
|
|
|
svr.sendJsonRpc(
|
2023-01-27 10:22:23 +01:00
|
|
|
s"""{ "jsonrpc": "2.0", "id": "${nextId()}", "method": "workspace/buildTargets", "params": {} }"""
|
2020-02-17 00:01:45 +01:00
|
|
|
)
|
2021-03-09 10:18:12 +01:00
|
|
|
assert(processing("workspace/buildTargets"))
|
2021-06-30 03:19:24 +02:00
|
|
|
val result = svr.waitFor[WorkspaceBuildTargetsResult](10.seconds)
|
|
|
|
|
val utilTarget = result.targets.find(_.displayName.contains("util")).get
|
|
|
|
|
assert(utilTarget.id.uri.toString.endsWith("#util/Compile"))
|
|
|
|
|
val buildServerBuildTarget =
|
|
|
|
|
result.targets.find(_.displayName.contains("buildserver-build")).get
|
|
|
|
|
assert(buildServerBuildTarget.id.uri.toString.endsWith("#buildserver-build"))
|
2021-07-29 17:39:31 +02:00
|
|
|
assert(!result.targets.exists(_.displayName.contains("badBuildTarget")))
|
2020-02-17 00:01:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
test("buildTarget/sources") { _ =>
|
2021-06-28 15:35:09 +02:00
|
|
|
val buildTarget = buildTargetUri("util", "Compile")
|
2021-07-29 17:39:31 +02:00
|
|
|
val badBuildTarget = buildTargetUri("badBuildTarget", "Compile")
|
2023-01-27 10:22:23 +01:00
|
|
|
svr.sendJsonRpc(buildTargetSources(Seq(buildTarget, badBuildTarget)))
|
2021-03-09 10:18:12 +01:00
|
|
|
assert(processing("buildTarget/sources"))
|
2021-06-30 03:19:24 +02:00
|
|
|
val s = svr.waitFor[SourcesResult](10.seconds)
|
|
|
|
|
val sources = s.items.head.sources.map(_.uri)
|
|
|
|
|
assert(sources.contains(new File(svr.baseDirectory, "util/src/main/scala").toURI))
|
|
|
|
|
}
|
2021-10-29 14:31:01 +02:00
|
|
|
test("buildTarget/sources: base sources") { _ =>
|
|
|
|
|
val buildTarget = buildTargetUri("buildserver", "Compile")
|
2023-01-27 10:22:23 +01:00
|
|
|
svr.sendJsonRpc(buildTargetSources(Seq(buildTarget)))
|
2021-10-29 14:31:01 +02:00
|
|
|
assert(processing("buildTarget/sources"))
|
|
|
|
|
val s = svr.waitFor[SourcesResult](10.seconds)
|
|
|
|
|
val sources = s.items.head.sources
|
|
|
|
|
val expectedSource = SourceItem(
|
|
|
|
|
new File(svr.baseDirectory, "BaseSource.scala").toURI,
|
|
|
|
|
SourceItemKind.File,
|
|
|
|
|
generated = false
|
2021-06-30 03:19:24 +02:00
|
|
|
)
|
2021-10-29 14:31:01 +02:00
|
|
|
assert(sources.contains(expectedSource))
|
|
|
|
|
}
|
2022-06-22 00:00:31 +02:00
|
|
|
|
2021-10-29 14:31:01 +02:00
|
|
|
test("buildTarget/sources: sbt") { _ =>
|
|
|
|
|
val x = new URI(s"${svr.baseDirectory.getAbsoluteFile.toURI}#buildserver-build")
|
2023-01-27 10:22:23 +01:00
|
|
|
svr.sendJsonRpc(buildTargetSources(Seq(x)))
|
2021-06-30 03:19:24 +02:00
|
|
|
assert(processing("buildTarget/sources"))
|
|
|
|
|
val s = svr.waitFor[SourcesResult](10.seconds)
|
|
|
|
|
val sources = s.items.head.sources.map(_.uri).sorted
|
|
|
|
|
val expectedSources = Vector(
|
|
|
|
|
"build.sbt",
|
|
|
|
|
"project/A.scala",
|
|
|
|
|
"project/src/main/java",
|
|
|
|
|
"project/src/main/scala-2",
|
|
|
|
|
"project/src/main/scala-2.12",
|
|
|
|
|
"project/src/main/scala-sbt-1.0",
|
|
|
|
|
"project/src/main/scala/",
|
|
|
|
|
"project/target/scala-2.12/sbt-1.0/src_managed/main"
|
|
|
|
|
).map(rel => new File(svr.baseDirectory.getAbsoluteFile, rel).toURI).sorted
|
|
|
|
|
assert(sources == expectedSources)
|
2020-02-24 07:02:16 +01:00
|
|
|
}
|
2022-06-22 00:00:31 +02:00
|
|
|
|
2020-04-28 18:03:42 +02:00
|
|
|
test("buildTarget/compile") { _ =>
|
2021-06-28 15:35:09 +02:00
|
|
|
val buildTarget = buildTargetUri("util", "Compile")
|
2022-06-22 00:00:31 +02:00
|
|
|
|
2023-01-27 10:22:23 +01:00
|
|
|
compile(buildTarget, id = nextId())
|
2022-06-22 00:00:31 +02:00
|
|
|
|
2021-03-09 10:18:12 +01:00
|
|
|
assert(processing("buildTarget/compile"))
|
2021-07-29 17:39:31 +02:00
|
|
|
val res = svr.waitFor[BspCompileResult](10.seconds)
|
|
|
|
|
assert(res.statusCode == StatusCode.Success)
|
2020-04-28 18:03:42 +02:00
|
|
|
}
|
|
|
|
|
|
2021-09-10 00:02:26 +02:00
|
|
|
test("buildTarget/compile - reports compilation progress") { _ =>
|
|
|
|
|
val buildTarget = buildTargetUri("runAndTest", "Compile")
|
2022-06-22 00:00:31 +02:00
|
|
|
|
2023-01-27 10:22:23 +01:00
|
|
|
compile(buildTarget)
|
2021-09-10 00:02:26 +02:00
|
|
|
|
2021-09-19 00:21:55 +02:00
|
|
|
// This doesn't always come back in 10s on CI.
|
|
|
|
|
assert(svr.waitForString(60.seconds) { s =>
|
2021-09-10 00:02:26 +02:00
|
|
|
s.contains("build/taskStart") &&
|
|
|
|
|
s.contains(""""message":"Compiling runAndTest"""")
|
|
|
|
|
})
|
|
|
|
|
|
2021-09-19 00:21:55 +02:00
|
|
|
assert(svr.waitForString(60.seconds) { s =>
|
2021-09-10 00:02:26 +02:00
|
|
|
s.contains("build/taskProgress") &&
|
|
|
|
|
s.contains(""""message":"Compiling runAndTest (15%)"""")
|
|
|
|
|
})
|
|
|
|
|
|
2021-09-19 00:21:55 +02:00
|
|
|
assert(svr.waitForString(60.seconds) { s =>
|
2021-09-10 00:02:26 +02:00
|
|
|
s.contains("build/taskProgress") &&
|
|
|
|
|
s.contains(""""message":"Compiling runAndTest (100%)"""")
|
|
|
|
|
})
|
|
|
|
|
|
2022-03-21 13:35:11 +01:00
|
|
|
assert(svr.waitForString(60.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
s.contains("build/publishDiagnostics") &&
|
2022-03-21 13:35:11 +01:00
|
|
|
s.contains(""""diagnostics":[]""")
|
|
|
|
|
})
|
|
|
|
|
|
2021-09-19 00:21:55 +02:00
|
|
|
assert(svr.waitForString(60.seconds) { s =>
|
2021-09-10 00:02:26 +02:00
|
|
|
s.contains("build/taskFinish") &&
|
|
|
|
|
s.contains(""""message":"Compiled runAndTest"""")
|
|
|
|
|
})
|
2022-06-22 00:00:31 +02:00
|
|
|
}
|
2022-03-21 13:35:11 +01:00
|
|
|
|
2022-06-22 00:00:31 +02:00
|
|
|
test(
|
|
|
|
|
"buildTarget/compile [diagnostics] don't publish unnecessary for successful compilation case"
|
|
|
|
|
) { _ =>
|
|
|
|
|
val buildTarget = buildTargetUri("diagnostics", "Compile")
|
|
|
|
|
val mainFile = new File(svr.baseDirectory, "diagnostics/src/main/scala/Diagnostics.scala")
|
|
|
|
|
|
2023-01-27 10:22:23 +01:00
|
|
|
compile(buildTarget)
|
2022-06-22 00:00:31 +02:00
|
|
|
|
|
|
|
|
assert(svr.waitForString(30.seconds) { s =>
|
|
|
|
|
s.contains("build/taskFinish") &&
|
|
|
|
|
s.contains(""""message":"Compiled diagnostics"""")
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
// introduce compile error
|
|
|
|
|
IO.write(
|
|
|
|
|
mainFile,
|
|
|
|
|
"""|object Diagnostics {
|
|
|
|
|
| private val a: Int = ""
|
|
|
|
|
|}""".stripMargin
|
|
|
|
|
)
|
|
|
|
|
|
2023-01-27 10:22:23 +01:00
|
|
|
reloadWorkspace()
|
|
|
|
|
compile(buildTarget)
|
2022-06-22 00:00:31 +02:00
|
|
|
|
|
|
|
|
assert(
|
|
|
|
|
svr.waitForString(30.seconds) { s =>
|
|
|
|
|
s.contains("build/publishDiagnostics") &&
|
|
|
|
|
s.contains("Diagnostics.scala") &&
|
|
|
|
|
s.contains("\"message\":\"type mismatch")
|
|
|
|
|
},
|
|
|
|
|
"should send publishDiagnostics with type error for Main.scala"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// fix compilation error
|
|
|
|
|
IO.write(
|
|
|
|
|
mainFile,
|
|
|
|
|
"""|object Diagnostics {
|
|
|
|
|
| private val a: Int = 5
|
|
|
|
|
|}""".stripMargin
|
|
|
|
|
)
|
|
|
|
|
|
2023-01-27 10:22:23 +01:00
|
|
|
reloadWorkspace()
|
|
|
|
|
compile(buildTarget)
|
2022-06-22 00:00:31 +02:00
|
|
|
|
|
|
|
|
assert(
|
|
|
|
|
svr.waitForString(30.seconds) { s =>
|
|
|
|
|
s.contains("build/publishDiagnostics") &&
|
|
|
|
|
s.contains("Diagnostics.scala") &&
|
|
|
|
|
s.contains("\"diagnostics\":[]")
|
|
|
|
|
},
|
|
|
|
|
"should send publishDiagnostics with empty diagnostics"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// trigger no-op compilation
|
2023-01-27 10:22:23 +01:00
|
|
|
compile(buildTarget)
|
2022-06-22 00:00:31 +02:00
|
|
|
|
|
|
|
|
assert(
|
|
|
|
|
!svr.waitForString(20.seconds) { s =>
|
|
|
|
|
s.contains("build/publishDiagnostics") &&
|
|
|
|
|
s.contains("Diagnostics.scala")
|
|
|
|
|
},
|
|
|
|
|
"shouldn't send publishDiagnostics if there's no change in diagnostics (were empty, are empty)"
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
test("buildTarget/compile [diagnostics] clear stale warnings") { _ =>
|
|
|
|
|
val buildTarget = buildTargetUri("diagnostics", "Compile")
|
|
|
|
|
val testFile = new File(svr.baseDirectory, s"diagnostics/src/main/scala/PatternMatch.scala")
|
|
|
|
|
|
2023-01-27 10:22:23 +01:00
|
|
|
compile(buildTarget)
|
2022-06-22 00:00:31 +02:00
|
|
|
|
|
|
|
|
assert(
|
|
|
|
|
svr.waitForString(30.seconds) { s =>
|
|
|
|
|
s.contains("build/publishDiagnostics") &&
|
|
|
|
|
s.contains("PatternMatch.scala") &&
|
|
|
|
|
s.contains(""""message":"match may not be exhaustive""")
|
|
|
|
|
},
|
|
|
|
|
"should send publishDiagnostics with type error for PatternMatch.scala"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
IO.write(
|
|
|
|
|
testFile,
|
|
|
|
|
"""|class PatternMatch {
|
|
|
|
|
| val opt: Option[Int] = None
|
|
|
|
|
| opt match {
|
|
|
|
|
| case Some(value) => ()
|
|
|
|
|
| case None => ()
|
|
|
|
|
| }
|
|
|
|
|
|}
|
|
|
|
|
|""".stripMargin
|
|
|
|
|
)
|
|
|
|
|
|
2023-01-27 10:22:23 +01:00
|
|
|
reloadWorkspace()
|
|
|
|
|
compile(buildTarget)
|
2022-06-22 00:00:31 +02:00
|
|
|
|
|
|
|
|
assert(
|
|
|
|
|
svr.waitForString(30.seconds) { s =>
|
|
|
|
|
s.contains("build/publishDiagnostics") &&
|
|
|
|
|
s.contains("PatternMatch.scala") &&
|
|
|
|
|
s.contains("\"diagnostics\":[]")
|
|
|
|
|
},
|
|
|
|
|
"should send publishDiagnostics with empty diagnostics"
|
2022-03-21 13:35:11 +01:00
|
|
|
)
|
|
|
|
|
|
2021-09-10 00:02:26 +02:00
|
|
|
}
|
|
|
|
|
|
2023-01-27 11:59:04 +01:00
|
|
|
test("buildTarget/compile: Java diagnostics") { _ =>
|
|
|
|
|
val buildTarget = buildTargetUri("javaProj", "Compile")
|
|
|
|
|
|
|
|
|
|
compile(buildTarget)
|
|
|
|
|
|
|
|
|
|
assert(
|
|
|
|
|
svr.waitForString(10.seconds) { s =>
|
|
|
|
|
s.contains("build/publishDiagnostics") &&
|
|
|
|
|
s.contains("Hello.java") &&
|
|
|
|
|
s.contains(""""severity":2""") &&
|
|
|
|
|
s.contains("""missing type arguments for generic class java.util.List""")
|
|
|
|
|
},
|
|
|
|
|
"should send publishDiagnostics with serverity 2 for Hello.java"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
assert(
|
|
|
|
|
svr.waitForString(1.seconds) { s =>
|
|
|
|
|
s.contains("build/publishDiagnostics") &&
|
|
|
|
|
s.contains("Hello.java") &&
|
|
|
|
|
s.contains(""""severity":1""") &&
|
|
|
|
|
s.contains("""incompatible types: int cannot be converted to java.lang.String""")
|
|
|
|
|
},
|
|
|
|
|
"should send publishDiagnostics with serverity 1 for Hello.java"
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-24 07:02:16 +01:00
|
|
|
test("buildTarget/scalacOptions") { _ =>
|
2021-06-28 15:35:09 +02:00
|
|
|
val buildTarget = buildTargetUri("util", "Compile")
|
2021-07-29 17:39:31 +02:00
|
|
|
val badBuildTarget = buildTargetUri("badBuildTarget", "Compile")
|
2023-01-27 10:22:23 +01:00
|
|
|
val id = nextId()
|
2020-02-24 07:02:16 +01:00
|
|
|
svr.sendJsonRpc(
|
2023-01-27 10:22:23 +01:00
|
|
|
s"""{ "jsonrpc": "2.0", "id": "$id", "method": "buildTarget/scalacOptions", "params": {
|
2021-07-29 17:39:31 +02:00
|
|
|
| "targets": [{ "uri": "$buildTarget" }, { "uri": "$badBuildTarget" }]
|
2020-02-24 07:02:16 +01:00
|
|
|
|} }""".stripMargin
|
|
|
|
|
)
|
2021-03-09 10:18:12 +01:00
|
|
|
assert(processing("buildTarget/scalacOptions"))
|
2020-02-24 07:02:16 +01:00
|
|
|
assert(svr.waitForString(10.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
(s contains s""""id":"$id"""") &&
|
2023-06-03 22:27:45 +02:00
|
|
|
(s contains "scala-library-2.13.11.jar")
|
2020-02-17 00:01:45 +01:00
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-27 09:02:47 +02:00
|
|
|
test("buildTarget/cleanCache") { _ =>
|
2021-08-27 15:58:46 +02:00
|
|
|
def targetDir =
|
|
|
|
|
Paths
|
|
|
|
|
.get(
|
|
|
|
|
svr.baseDirectory.getAbsoluteFile.toString,
|
|
|
|
|
"run-and-test/target/scala-2.13/classes/main"
|
|
|
|
|
)
|
|
|
|
|
.toFile
|
|
|
|
|
|
|
|
|
|
val buildTarget = buildTargetUri("runAndTest", "Compile")
|
2023-01-27 10:22:23 +01:00
|
|
|
compile(buildTarget)
|
2021-08-27 15:58:46 +02:00
|
|
|
svr.waitFor[BspCompileResult](10.seconds)
|
|
|
|
|
assert(targetDir.list().contains("Main.class"))
|
|
|
|
|
|
2021-08-27 09:02:47 +02:00
|
|
|
svr.sendJsonRpc(
|
2023-01-27 10:22:23 +01:00
|
|
|
s"""{ "jsonrpc": "2.0", "id": "${nextId()}", "method": "buildTarget/cleanCache", "params": {
|
2021-08-27 09:02:47 +02:00
|
|
|
| "targets": [{ "uri": "$buildTarget" }]
|
|
|
|
|
|} }""".stripMargin
|
|
|
|
|
)
|
|
|
|
|
assert(processing("buildTarget/cleanCache"))
|
|
|
|
|
val res = svr.waitFor[CleanCacheResult](10.seconds)
|
|
|
|
|
assert(res.cleaned)
|
2021-08-27 15:58:46 +02:00
|
|
|
assert(targetDir.list().isEmpty)
|
2021-08-27 09:02:47 +02:00
|
|
|
}
|
|
|
|
|
|
2021-08-28 12:50:12 +02:00
|
|
|
test("buildTarget/cleanCache: rebuild project") { _ =>
|
|
|
|
|
svr.sendJsonRpc(
|
2023-01-27 10:22:23 +01:00
|
|
|
s"""{ "jsonrpc": "2.0", "id": "${nextId()}", "method": "workspace/buildTargets", "params": {} }"""
|
2021-08-28 12:50:12 +02:00
|
|
|
)
|
|
|
|
|
assert(processing("workspace/buildTargets"))
|
|
|
|
|
val result = svr.waitFor[WorkspaceBuildTargetsResult](10.seconds)
|
|
|
|
|
val allTargets = result.targets.map(_.id.uri)
|
|
|
|
|
|
|
|
|
|
svr.sendJsonRpc(
|
2023-01-27 10:22:23 +01:00
|
|
|
s"""{ "jsonrpc": "2.0", "id": "${nextId()}", "method": "buildTarget/cleanCache", "params": {
|
2021-08-28 12:50:12 +02:00
|
|
|
| "targets": [
|
|
|
|
|
| ${allTargets.map(uri => s"""{ "uri": "$uri" }""").mkString(",\n")}
|
|
|
|
|
| ]
|
|
|
|
|
|} }""".stripMargin
|
|
|
|
|
)
|
|
|
|
|
assert(processing("buildTarget/cleanCache"))
|
|
|
|
|
val res = svr.waitFor[CleanCacheResult](10.seconds)
|
|
|
|
|
assert(res.cleaned)
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-11 11:33:13 +02:00
|
|
|
test("workspace/reload") { _ =>
|
2023-01-27 10:22:23 +01:00
|
|
|
val id = nextId()
|
2020-09-11 11:33:13 +02:00
|
|
|
svr.sendJsonRpc(
|
2023-01-27 10:22:23 +01:00
|
|
|
s"""{ "jsonrpc": "2.0", "id": "$id", "method": "workspace/reload"}"""
|
2020-09-11 11:33:13 +02:00
|
|
|
)
|
2021-03-09 10:18:12 +01:00
|
|
|
assert(processing("workspace/reload"))
|
2020-09-11 11:33:13 +02:00
|
|
|
assert(svr.waitForString(10.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
(s contains s""""id":"$id"""") &&
|
2020-09-11 11:33:13 +02:00
|
|
|
(s contains """"result":null""")
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-28 15:35:09 +02:00
|
|
|
test("workspace/reload: send diagnostic and respond with error") { _ =>
|
|
|
|
|
// write an other-build.sbt file that does not compile
|
|
|
|
|
val otherBuildFile = new File(svr.baseDirectory, "other-build.sbt")
|
|
|
|
|
IO.write(
|
|
|
|
|
otherBuildFile,
|
|
|
|
|
"""
|
|
|
|
|
|val someSettings = Seq(
|
|
|
|
|
| scalacOptions ++= "-deprecation"
|
|
|
|
|
|)
|
|
|
|
|
|""".stripMargin
|
|
|
|
|
)
|
2023-01-27 10:22:23 +01:00
|
|
|
val id = nextId()
|
2021-06-28 15:35:09 +02:00
|
|
|
// reload
|
2023-01-27 10:22:23 +01:00
|
|
|
reloadWorkspace(id)
|
2021-06-28 15:35:09 +02:00
|
|
|
assert(
|
|
|
|
|
svr.waitForString(10.seconds) { s =>
|
|
|
|
|
s.contains(s""""buildTarget":{"uri":"$metaBuildTarget"}""") &&
|
|
|
|
|
s.contains(s""""textDocument":{"uri":"${otherBuildFile.toPath.toUri}"}""") &&
|
|
|
|
|
s.contains(""""severity":1""") &&
|
|
|
|
|
s.contains(""""reset":true""")
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
assert(
|
|
|
|
|
svr.waitForString(10.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
s.contains(s""""id":"$id"""") &&
|
2021-06-28 15:35:09 +02:00
|
|
|
s.contains(""""error"""") &&
|
|
|
|
|
s.contains(s""""code":${ErrorCodes.InternalError}""") &&
|
|
|
|
|
s.contains("Type error in expression")
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
// fix the other-build.sbt file and reload again
|
|
|
|
|
IO.write(
|
|
|
|
|
otherBuildFile,
|
|
|
|
|
"""
|
|
|
|
|
|val someSettings = Seq(
|
|
|
|
|
| scalacOptions += "-deprecation"
|
|
|
|
|
|)
|
|
|
|
|
|""".stripMargin
|
|
|
|
|
)
|
2023-01-27 10:22:23 +01:00
|
|
|
reloadWorkspace()
|
2021-06-28 15:35:09 +02:00
|
|
|
// assert received an empty diagnostic
|
|
|
|
|
assert(
|
|
|
|
|
svr.waitForString(10.seconds) { s =>
|
|
|
|
|
s.contains(s""""buildTarget":{"uri":"$metaBuildTarget"}""") &&
|
|
|
|
|
s.contains(s""""textDocument":{"uri":"${otherBuildFile.toPath.toUri}"}""") &&
|
|
|
|
|
s.contains(""""diagnostics":[]""") &&
|
|
|
|
|
s.contains(""""reset":true""")
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
IO.delete(otherBuildFile)
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-14 13:58:51 +02:00
|
|
|
test("buildTarget/scalaMainClasses") { _ =>
|
2021-06-28 15:35:09 +02:00
|
|
|
val buildTarget = buildTargetUri("runAndTest", "Compile")
|
2021-07-29 17:39:31 +02:00
|
|
|
val badBuildTarget = buildTargetUri("badBuildTarget", "Compile")
|
2023-01-27 10:22:23 +01:00
|
|
|
val id = nextId()
|
2020-09-14 13:58:51 +02:00
|
|
|
svr.sendJsonRpc(
|
2023-01-27 10:22:23 +01:00
|
|
|
s"""{ "jsonrpc": "2.0", "id": "$id", "method": "buildTarget/scalaMainClasses", "params": {
|
2021-07-29 17:39:31 +02:00
|
|
|
| "targets": [{ "uri": "$buildTarget" }, { "uri": "$badBuildTarget" }]
|
2020-09-14 13:58:51 +02:00
|
|
|
|} }""".stripMargin
|
|
|
|
|
)
|
2021-03-09 10:18:12 +01:00
|
|
|
assert(processing("buildTarget/scalaMainClasses"))
|
2020-09-21 09:18:07 +02:00
|
|
|
assert(svr.waitForString(30.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
(s contains s""""id":"$id"""") &&
|
2020-10-22 11:09:00 +02:00
|
|
|
(s contains """"class":"main.Main"""")
|
2020-09-14 13:58:51 +02:00
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-17 14:23:07 +02:00
|
|
|
test("buildTarget/run") { _ =>
|
2021-06-28 15:35:09 +02:00
|
|
|
val buildTarget = buildTargetUri("runAndTest", "Compile")
|
2023-01-27 10:22:23 +01:00
|
|
|
val id = nextId()
|
2020-09-17 14:23:07 +02:00
|
|
|
svr.sendJsonRpc(
|
2023-01-27 10:22:23 +01:00
|
|
|
s"""{ "jsonrpc": "2.0", "id": "$id", "method": "buildTarget/run", "params": {
|
2021-06-28 15:35:09 +02:00
|
|
|
| "target": { "uri": "$buildTarget" },
|
2020-09-17 14:23:07 +02:00
|
|
|
| "dataKind": "scala-main-class",
|
2020-10-22 11:09:00 +02:00
|
|
|
| "data": { "class": "main.Main" }
|
2020-09-17 14:23:07 +02:00
|
|
|
|} }""".stripMargin
|
|
|
|
|
)
|
2021-03-09 10:18:12 +01:00
|
|
|
assert(processing("buildTarget/run"))
|
2020-09-17 14:23:07 +02:00
|
|
|
assert(svr.waitForString(10.seconds) { s =>
|
|
|
|
|
(s contains "build/logMessage") &&
|
|
|
|
|
(s contains """"message":"Hello World!"""")
|
|
|
|
|
})
|
|
|
|
|
assert(svr.waitForString(10.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
(s contains s""""id":"$id"""") &&
|
2020-09-17 14:23:07 +02:00
|
|
|
(s contains """"statusCode":1""")
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-05 19:29:00 +02:00
|
|
|
test("buildTarget/jvmRunEnvironment") { _ =>
|
|
|
|
|
val buildTarget = buildTargetUri("runAndTest", "Compile")
|
2023-01-27 10:22:23 +01:00
|
|
|
val id = nextId()
|
2022-04-05 19:29:00 +02:00
|
|
|
svr.sendJsonRpc(
|
|
|
|
|
s"""|{ "jsonrpc": "2.0",
|
2023-01-27 10:22:23 +01:00
|
|
|
| "id": "$id",
|
2022-04-05 19:29:00 +02:00
|
|
|
| "method": "buildTarget/jvmRunEnvironment",
|
|
|
|
|
| "params": { "targets": [{ "uri": "$buildTarget" }] }
|
|
|
|
|
|}""".stripMargin
|
|
|
|
|
)
|
|
|
|
|
assert(processing("buildTarget/jvmRunEnvironment"))
|
|
|
|
|
assert {
|
|
|
|
|
svr.waitForString(10.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
(s contains s""""id":"$id"""") &&
|
2022-04-05 19:29:00 +02:00
|
|
|
(s contains "jsoniter-scala-core_2.13-2.13.11.jar") && // compile dependency
|
|
|
|
|
(s contains "\"jvmOptions\":[\"Xmx256M\"]") &&
|
|
|
|
|
(s contains "\"environmentVariables\":{\"KEY\":\"VALUE\"}") &&
|
|
|
|
|
(s contains "/buildserver/run-and-test/") // working directory
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
test("buildTarget/jvmTestEnvironment") { _ =>
|
|
|
|
|
val buildTarget = buildTargetUri("runAndTest", "Test")
|
2023-01-27 10:22:23 +01:00
|
|
|
val id = nextId()
|
2022-04-05 19:29:00 +02:00
|
|
|
svr.sendJsonRpc(
|
2022-06-22 00:00:31 +02:00
|
|
|
s"""|{ "jsonrpc": "2.0",
|
2023-01-27 10:22:23 +01:00
|
|
|
| "id": "$id",
|
2022-06-22 00:00:31 +02:00
|
|
|
| "method": "buildTarget/jvmTestEnvironment",
|
|
|
|
|
| "params": { "targets": [{ "uri": "$buildTarget" }] }
|
2022-04-05 19:29:00 +02:00
|
|
|
|}""".stripMargin
|
|
|
|
|
)
|
|
|
|
|
assert(processing("buildTarget/jvmTestEnvironment"))
|
|
|
|
|
assert {
|
|
|
|
|
svr.waitForString(10.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
(s contains s""""id":"$id"""") &&
|
2022-04-05 19:29:00 +02:00
|
|
|
// test depends on compile so it has dependencies from both
|
|
|
|
|
(s contains "jsoniter-scala-core_2.13-2.13.11.jar") && // compile dependency
|
|
|
|
|
(s contains "scalatest_2.13-3.0.8.jar") && // test dependency
|
|
|
|
|
(s contains "\"jvmOptions\":[\"Xmx512M\"]") &&
|
|
|
|
|
(s contains "\"environmentVariables\":{\"KEY_TEST\":\"VALUE_TEST\"}")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-17 15:58:14 +02:00
|
|
|
test("buildTarget/scalaTestClasses") { _ =>
|
2021-06-28 15:35:09 +02:00
|
|
|
val buildTarget = buildTargetUri("runAndTest", "Test")
|
2021-07-29 17:39:31 +02:00
|
|
|
val badBuildTarget = buildTargetUri("badBuildTarget", "Test")
|
2023-01-27 10:22:23 +01:00
|
|
|
val id = nextId()
|
2020-09-17 15:58:14 +02:00
|
|
|
svr.sendJsonRpc(
|
2023-01-27 10:22:23 +01:00
|
|
|
s"""{ "jsonrpc": "2.0", "id": "$id", "method": "buildTarget/scalaTestClasses", "params": {
|
2021-07-29 17:39:31 +02:00
|
|
|
| "targets": [{ "uri": "$buildTarget" }, { "uri": "$badBuildTarget" }]
|
2020-09-17 15:58:14 +02:00
|
|
|
|} }""".stripMargin
|
|
|
|
|
)
|
2021-03-09 10:18:12 +01:00
|
|
|
assert(processing("buildTarget/scalaTestClasses"))
|
2020-09-17 15:58:14 +02:00
|
|
|
assert(svr.waitForString(10.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
(s contains s""""id":"$id"""") &&
|
2020-10-22 11:09:00 +02:00
|
|
|
(s contains """"tests.FailingTest"""") &&
|
2022-03-10 19:56:49 +01:00
|
|
|
(s contains """"tests.PassingTest"""") &&
|
|
|
|
|
(s contains """"framework":"ScalaTest"""")
|
2020-09-21 12:16:31 +02:00
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
test("buildTarget/test: run all tests") { _ =>
|
2021-06-28 15:35:09 +02:00
|
|
|
val buildTarget = buildTargetUri("runAndTest", "Test")
|
2023-01-27 10:22:23 +01:00
|
|
|
val id = nextId()
|
2020-09-21 12:16:31 +02:00
|
|
|
svr.sendJsonRpc(
|
2023-01-27 10:22:23 +01:00
|
|
|
s"""{ "jsonrpc": "2.0", "id": "$id", "method": "buildTarget/test", "params": {
|
2021-06-28 15:35:09 +02:00
|
|
|
| "targets": [{ "uri": "$buildTarget" }]
|
2020-09-21 12:16:31 +02:00
|
|
|
|} }""".stripMargin
|
|
|
|
|
)
|
2021-03-09 10:18:12 +01:00
|
|
|
assert(processing("buildTarget/test"))
|
2020-09-21 12:16:31 +02:00
|
|
|
assert(svr.waitForString(10.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
(s contains s""""id":"$id"""") &&
|
2020-09-21 12:16:31 +02:00
|
|
|
(s contains """"statusCode":2""")
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
test("buildTarget/test: run one test class") { _ =>
|
2021-06-28 15:35:09 +02:00
|
|
|
val buildTarget = buildTargetUri("runAndTest", "Test")
|
2023-01-27 10:22:23 +01:00
|
|
|
val id = nextId()
|
2020-09-21 12:16:31 +02:00
|
|
|
svr.sendJsonRpc(
|
2023-01-27 10:22:23 +01:00
|
|
|
s"""{ "jsonrpc": "2.0", "id": "$id", "method": "buildTarget/test", "params": {
|
2021-06-28 15:35:09 +02:00
|
|
|
| "targets": [{ "uri": "$buildTarget" }],
|
2020-09-21 12:16:31 +02:00
|
|
|
| "dataKind": "scala-test",
|
|
|
|
|
| "data": {
|
|
|
|
|
| "testClasses": [
|
|
|
|
|
| {
|
2021-06-28 15:35:09 +02:00
|
|
|
| "target": { "uri": "$buildTarget" },
|
2020-10-22 11:09:00 +02:00
|
|
|
| "classes": ["tests.PassingTest"]
|
2020-09-21 12:16:31 +02:00
|
|
|
| }
|
|
|
|
|
| ]
|
|
|
|
|
| }
|
|
|
|
|
|} }""".stripMargin
|
|
|
|
|
)
|
2021-03-09 10:18:12 +01:00
|
|
|
assert(processing("buildTarget/test"))
|
2020-09-21 12:16:31 +02:00
|
|
|
assert(svr.waitForString(10.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
(s contains s""""id":"$id"""") &&
|
2020-09-21 12:16:31 +02:00
|
|
|
(s contains """"statusCode":1""")
|
2020-09-17 15:58:14 +02:00
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-22 11:09:00 +02:00
|
|
|
test("buildTarget/compile: report error") { _ =>
|
2021-06-28 15:35:09 +02:00
|
|
|
val buildTarget = buildTargetUri("reportError", "Compile")
|
2023-01-27 10:22:23 +01:00
|
|
|
compile(buildTarget)
|
2020-10-22 11:09:00 +02:00
|
|
|
assert(svr.waitForString(10.seconds) { s =>
|
2021-06-28 15:35:09 +02:00
|
|
|
(s contains s""""buildTarget":{"uri":"$buildTarget"}""") &&
|
2020-10-22 11:09:00 +02:00
|
|
|
(s contains """"severity":1""") &&
|
|
|
|
|
(s contains """"reset":true""")
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
test("buildTarget/compile: report warning") { _ =>
|
2021-06-28 15:35:09 +02:00
|
|
|
val buildTarget = buildTargetUri("reportWarning", "Compile")
|
2023-01-27 10:22:23 +01:00
|
|
|
compile(buildTarget)
|
2020-10-22 11:09:00 +02:00
|
|
|
assert(svr.waitForString(10.seconds) { s =>
|
2021-06-28 15:35:09 +02:00
|
|
|
(s contains s""""buildTarget":{"uri":"$buildTarget"}""") &&
|
2020-10-22 11:09:00 +02:00
|
|
|
(s contains """"severity":2""") &&
|
|
|
|
|
(s contains """"reset":true""")
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-24 18:14:12 +02:00
|
|
|
test("buildTarget/compile: respond error") { _ =>
|
2021-06-28 15:35:09 +02:00
|
|
|
val buildTarget = buildTargetUri("respondError", "Compile")
|
2023-01-27 10:22:23 +01:00
|
|
|
val id = nextId()
|
|
|
|
|
compile(buildTarget, id)
|
2021-06-24 18:14:12 +02:00
|
|
|
assert(svr.waitForString(10.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
s.contains(s""""id":"$id"""") &&
|
2021-06-24 18:14:12 +02:00
|
|
|
s.contains(""""error"""") &&
|
2021-06-28 15:35:09 +02:00
|
|
|
s.contains(s""""code":${ErrorCodes.InternalError}""") &&
|
2021-06-24 18:14:12 +02:00
|
|
|
s.contains("custom message")
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2021-06-17 11:09:08 +02:00
|
|
|
test("buildTarget/resources") { _ =>
|
2021-06-28 15:35:09 +02:00
|
|
|
val buildTarget = buildTargetUri("util", "Compile")
|
2021-07-29 17:39:31 +02:00
|
|
|
val badBuildTarget = buildTargetUri("badBuildTarget", "Compile")
|
2023-01-27 10:22:23 +01:00
|
|
|
val id = nextId()
|
2021-06-17 11:09:08 +02:00
|
|
|
svr.sendJsonRpc(
|
2023-01-27 10:22:23 +01:00
|
|
|
s"""{ "jsonrpc": "2.0", "id": "$id", "method": "buildTarget/resources", "params": {
|
2021-07-29 17:39:31 +02:00
|
|
|
| "targets": [{ "uri": "$buildTarget" }, { "uri": "$badBuildTarget" }]
|
2021-06-17 11:09:08 +02:00
|
|
|
|} }""".stripMargin
|
|
|
|
|
)
|
|
|
|
|
assert(processing("buildTarget/resources"))
|
|
|
|
|
assert(svr.waitForString(10.seconds) { s =>
|
2023-01-27 10:22:23 +01:00
|
|
|
(s contains s""""id":"$id"""") && (s contains "util/src/main/resources/")
|
2021-06-17 11:09:08 +02:00
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-28 04:40:18 +02:00
|
|
|
test("buildTarget/outputPaths") { _ =>
|
|
|
|
|
val buildTarget = buildTargetUri("util", "Compile")
|
|
|
|
|
val badBuildTarget = buildTargetUri("badBuildTarget", "Compile")
|
|
|
|
|
svr.sendJsonRpc(
|
2023-01-27 10:22:23 +01:00
|
|
|
s"""{ "jsonrpc": "2.0", "id": "${nextId()}", "method": "buildTarget/outputPaths", "params": {
|
2022-07-28 04:40:18 +02:00
|
|
|
| "targets": [{ "uri": "$buildTarget" }, { "uri": "$badBuildTarget" }]
|
|
|
|
|
|} }""".stripMargin
|
|
|
|
|
)
|
|
|
|
|
assert(processing("buildTarget/outputPaths"))
|
|
|
|
|
val actualResult = svr.waitFor[OutputPathsResult](10.seconds)
|
|
|
|
|
val expectedResult = OutputPathsResult(
|
|
|
|
|
items = Vector(
|
|
|
|
|
OutputPathsItem(
|
|
|
|
|
target = BuildTargetIdentifier(buildTarget),
|
|
|
|
|
outputPaths = Vector(
|
|
|
|
|
OutputPathItem(
|
|
|
|
|
uri = new File(svr.baseDirectory, "util/custom-target").toURI,
|
|
|
|
|
kind = OutputPathItemKind.Directory
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
assert(actualResult == expectedResult)
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-27 10:22:23 +01:00
|
|
|
private def initializeRequest(id: Int): Unit = {
|
|
|
|
|
val params = InitializeBuildParams(
|
|
|
|
|
"test client",
|
|
|
|
|
"1.0.0",
|
|
|
|
|
"2.1.0-M1",
|
|
|
|
|
new URI("file://root/"),
|
|
|
|
|
BuildClientCapabilities(Vector("scala")),
|
|
|
|
|
None
|
2020-02-17 00:01:45 +01:00
|
|
|
)
|
2023-01-27 10:22:23 +01:00
|
|
|
svr.sendJsonRpc(request(id, "build/initialize", params))
|
2020-02-17 00:01:45 +01:00
|
|
|
}
|
2021-03-09 10:18:12 +01:00
|
|
|
|
|
|
|
|
private def processing(method: String, debug: Boolean = false): Boolean = {
|
|
|
|
|
svr.waitForString(10.seconds) { msg =>
|
|
|
|
|
if (debug) println(msg)
|
|
|
|
|
msg.contains("build/logMessage") &&
|
|
|
|
|
msg.contains(s""""message":"Processing $method"""")
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-06-28 15:35:09 +02:00
|
|
|
|
2023-01-27 10:22:23 +01:00
|
|
|
private def reloadWorkspace(id: Int = nextId()): Unit =
|
|
|
|
|
svr.sendJsonRpc(s"""{ "jsonrpc": "2.0", "id": "$id", "method": "workspace/reload"}""")
|
2022-06-22 00:00:31 +02:00
|
|
|
|
2023-01-27 10:22:23 +01:00
|
|
|
private def compile(buildTarget: URI, id: Int = nextId()): Unit = {
|
|
|
|
|
val params =
|
|
|
|
|
CompileParams(targets = Vector(BuildTargetIdentifier(buildTarget)), None, Vector.empty)
|
|
|
|
|
svr.sendJsonRpc(request(id, "buildTarget/compile", params))
|
|
|
|
|
}
|
2022-06-22 00:00:31 +02:00
|
|
|
|
2023-01-27 10:22:23 +01:00
|
|
|
private def buildTargetSources(buildTargets: Seq[URI], id: Int = nextId()): String = {
|
2021-10-29 14:31:01 +02:00
|
|
|
val targets = buildTargets.map(BuildTargetIdentifier.apply).toVector
|
|
|
|
|
request(id, "buildTarget/sources", SourcesParams(targets))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private def request[T: JsonWriter](id: Int, method: String, params: T): String = {
|
|
|
|
|
val request = JsonRpcRequestMessage("2.0", id.toString, method, Converter.toJson(params).get)
|
|
|
|
|
val json = Converter.toJson(request).get
|
|
|
|
|
CompactPrinter(json)
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-27 10:22:23 +01:00
|
|
|
private def buildTargetUri(project: String, config: String): URI =
|
2021-10-29 14:31:01 +02:00
|
|
|
new URI(s"${svr.baseDirectory.getAbsoluteFile.toURI}#$project/$config")
|
2021-06-28 15:35:09 +02:00
|
|
|
|
|
|
|
|
private def metaBuildTarget: String =
|
|
|
|
|
s"${svr.baseDirectory.getAbsoluteFile.toURI}project/#buildserver-build/Compile"
|
2020-02-17 00:01:45 +01:00
|
|
|
}
|