Метод обработчика в Spring Boot с Джерси не найден в тесте
Spring Boot / Jersey не может найти метод-обработчик, когда приложение запускается и обращается к нему из теста. Если я запускаю приложение отдельно и получаю доступ http://localhost:8080/demo
с браузером все нормально.
В журнале написано: "Не найден метод обработчика для [/demo]". Соответствующий вывод журнала:
2018-06-18 17:04:31.071 DEBUG 7628 --- [nio-8080-exec-1] o.s.web.reactive.DispatcherHandler : Processing GET request for [http://localhost:8080/demo]
2018-06-18 17:04:31.083 DEBUG 7628 --- [nio-8080-exec-1] s.w.r.r.m.a.RequestMappingHandlerMapping : Looking up handler method for path /demo
2018-06-18 17:04:31.085 DEBUG 7628 --- [nio-8080-exec-1] s.w.r.r.m.a.RequestMappingHandlerMapping : Did not find handler method for [/demo]
2018-06-18 17:04:31.087 DEBUG 7628 --- [nio-8080-exec-1] o.s.w.r.handler.SimpleUrlHandlerMapping : Matching pattern for request [[path='/demo']] is /**
Приложение состоит из следующих классов (написано на Kotlin):
Ресурс
import org.springframework.stereotype.Component
import javax.ws.rs.GET
import javax.ws.rs.Path
import javax.ws.rs.core.Response
@Component
@Path("/")
class Resource {
@GET
@Path("demo")
fun test() = Response.ok("Hi!").encoding("UTF-8").build()
}
JerseyConfig
import org.glassfish.jersey.server.ResourceConfig
import org.springframework.stereotype.Component
@Component
class JerseyConfig : ResourceConfig() {
init {
register(Resource::class.java)
}
}
Приложение:
import org.springframework.boot.autoconfigure.SpringBootApplication
import org.springframework.boot.runApplication
@SpringBootApplication
class App
fun main(args: Array<String>) {
runApplication<App>(*args)
}
Неудачный тест:
import org.junit.jupiter.api.Test
import org.junit.jupiter.api.TestInstance
import org.junit.jupiter.api.extension.ExtendWith
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.context.SpringBootTest
import org.springframework.test.context.junit.jupiter.SpringExtension
import org.springframework.test.web.reactive.server.WebTestClient
@ExtendWith(SpringExtension::class)
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.DEFINED_PORT)
class ResourceTest {
@Autowired
lateinit var client: WebTestClient
@Test
fun getTest() {
client.get().uri("demo").exchange().expectStatus().isOk
}
}
Я получаю ту же ошибку, если использую клиент Джерси для тестирования:
@Test
fun testWithJersey() {
val client = ClientBuilder.newClient()
val response = client.target("http://localhost:8080/demo").request().get()
assertThat(response.status).isEqualTo(200)
}
build.gradle:
buildscript {
ext {
kotlinVersion = '1.2.50'
springBootVersion = '2.0.3.RELEASE'
}
repositories {
mavenCentral()
}
dependencies {
classpath("org.springframework.boot:spring-boot-gradle-plugin:${springBootVersion}")
classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:${kotlinVersion}")
classpath("org.jetbrains.kotlin:kotlin-allopen:${kotlinVersion}")
}
}
apply plugin: 'kotlin'
apply plugin: 'kotlin-spring'
apply plugin: 'org.springframework.boot'
apply plugin: 'io.spring.dependency-management'
compileKotlin {
kotlinOptions {
freeCompilerArgs = ["-Xjsr305=strict"]
jvmTarget = "1.8"
}
}
compileTestKotlin {
kotlinOptions {
freeCompilerArgs = ["-Xjsr305=strict"]
jvmTarget = "1.8"
}
}
repositories {
mavenCentral()
}
dependencies {
compile("org.jetbrains.kotlin:kotlin-stdlib-jdk8")
compile("org.jetbrains.kotlin:kotlin-reflect")
compile('org.springframework.boot:spring-boot-starter-jersey')
testCompile("org.springframework.boot:spring-boot-starter-test") {
exclude group: "junit", module: "junit"
}
testCompile('org.springframework.boot:spring-boot-starter-webflux')
testCompile("org.junit.jupiter:junit-jupiter-api")
testRuntime("org.junit.jupiter:junit-jupiter-engine")
}
Сам тестовый код вроде бы в порядке, так как, когда я заменяю тело тестового метода на Thread.sleep(...)
и затем получить доступ к серверу из браузера, я получаю ту же ошибку (404 из-за "Не найден метод обработчика для [/demo]").
Почему метод-обработчик не найден в тестах? Что я должен изменить?
2 ответа
Уверены ли вы WebTestClient
является универсальным клиентом (который выполняет реальные сетевые запросы) и предназначен не только для использования с Spring MVC, как MockMvc
является? Ошибка звучит так, будто она ищет метод-обработчик Spring MVC. Если бы это был универсальный клиент, в сообщении об ошибке ничего не говорилось бы о методах-обработчиках, но вместо этого, вероятно, говорилось бы что-то о URL.
Я полагаю, что вам нужно сделать реальный сетевой запрос с реальным клиентом. Например, если вы использовали клиент Джерси, вы могли бы сделать что-то вроде
@LocalServerPort
private int port;
private Client client = ClientBuilder.newClient();
@Test
public void testCustomerLocationOnPost() {
URI resourceUri = UriBuilder.fromUri("http://localhost")
.port(port).path("demo").build();
Respons response = client.target(resourcrUri).request().get();
assertThat(response.getStatus()).isEqualTo(200);
}
С помощью замечательных парней из Pivotal я решил проблему: WebTestClient
потребности WebFlux
как зависимость. Но у WebFlux есть не только маленький помощник по тестированию, но и полноценный веб-фреймворк, который автоматически настраивается Spring Boot и конфликтует с Джерси.
Решение состоит в том, чтобы удалить WebFlux из зависимостей
dependencies {
compile("org.jetbrains.kotlin:kotlin-stdlib-jdk8")
compile("org.jetbrains.kotlin:kotlin-reflect")
compile('org.springframework.boot:spring-boot-starter-jersey')
testCompile("org.springframework.boot:spring-boot-starter-test") {
exclude group: "junit", module: "junit"
}
testCompile("org.junit.jupiter:junit-jupiter-api")
testRuntime("org.junit.jupiter:junit-jupiter-engine")
}
и используйте, например, клиент Джерси http:
@ExtendWith(SpringExtension::class)
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.DEFINED_PORT)
class ResourceTest {
@Test
fun testWithJerseyClient() {
val client = ClientBuilder.newClient()
val response = client.target("http://localhost:8080/demo").request().get()
assertThat(response.status).isEqualTo(200)
}
}
Но любой другой http-клиент подойдет, если WebFlux не является зависимостью.
Обновление: это ошибка в Spring Boot, которая будет исправлена в Spring Boot 2.0.4! Затем вы можете использовать WebFlux для тестирования вместе с Джерси на стороне сервера.