0

我正在尝试检查从 webtestclient 返回的数据是否与我的预期相同。但是,当我将 Jackson 应用于 webtestclient 编解码器时,来自 User 数据类的 ZonedDateTime 不仅显示为日期,还显示为时间戳。示例:2021-12-09T16:39:43.225207700+01:00转换为1639064383.225207700while I expect nothing to change。有人可以解释我做错了什么。(在测试之外调用此端点时使用此杰克逊配置会给出日期不是时间戳)

WebTestClientUtil:

object WebTestClientUtil {
    fun webTestClient(routerFunction: RouterFunction<ServerResponse>): WebTestClient {
        return WebTestClient
            .bindToRouterFunction(routerFunction)
            .configureClient()
            .codecs { configurer: ClientCodecConfigurer ->
                configurer.defaultCodecs().jackson2JsonEncoder(Jackson2JsonEncoder(objectMapper, MediaType.APPLICATION_JSON))
                configurer.defaultCodecs().jackson2JsonDecoder(Jackson2JsonDecoder(objectMapper, MediaType.APPLICATION_JSON))
            }
            .build()
    }
}

测试用例:

@Test
fun `get user when given correct data`() {
    val user = GlobalMocks.mockedUser
    coEvery { userRepository.getUserWithData(any()) } returns user

    val result = webTestClient.get()
        .uri("/api/v1/user/${user.userId}")
        .exchange()
        .expectStatus().is2xxSuccessful
        .expectBody<Result>().returnResult().responseBody?.payload


    assertEquals(user, result)
}

data class Result(
    val payload: User
)

杰克逊配置:

class JacksonConfig {
    companion object {
        val serializationDateFormat: DateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ssXX")
        val deserializationDateFormat: DateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm[:ss][XXX][X]")
        val objectMapper = jacksonObjectMapper().applyDefaultSettings()

        private fun ObjectMapper.applyDefaultSettings() =
            apply {
                disable(DeserializationFeature.ADJUST_DATES_TO_CONTEXT_TIME_ZONE)
                disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS)
                disable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES)

                enable(DeserializationFeature.READ_UNKNOWN_ENUM_VALUES_USING_DEFAULT_VALUE)
                setSerializationInclusion(JsonInclude.Include.NON_NULL)

                registerModule(Jdk8Module())
                registerModule(ParameterNamesModule())
                registerModule(JsonComponentModule())
                registerModule(
                    JavaTimeModule().apply {
                        addSerializer(ZonedDateTime::class.java, ZonedDateTimeSerializer(serializationDateFormat))
                        addDeserializer(ZonedDateTime::class.java, ZonedDateTimeDeserializer())
                    }
                )
            }
    }

    class ZonedDateTimeDeserializer : JsonDeserializer<ZonedDateTime>() {
        override fun deserialize(jsonParser: JsonParser, deserializationContext: DeserializationContext): ZonedDateTime {
            val epochTime = jsonParser.text.toLongOrNull()
            return if (epochTime != null) {
                ZonedDateTime.ofInstant(
                    Instant.ofEpochSecond(epochTime),
                    currentZone
                )
            } else {
                ZonedDateTime.parse(jsonParser.text, deserializationDateFormat)
            }
        }
    }
}

编辑:还发现了这个问题,这让我认为它可能与bindToRouterFunction.

4

1 回答 1

1

您需要定义一个ObjectMapperbean,以便不使用自动配置的bean:

@Configuration(proxyBeanMethods = false)
class JacksonConfiguration {

    companion object {
        val serializationDateFormat: DateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ssXX")
        val deserializationDateFormat: DateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm[:ss][XXX][X]")
    }

    @Bean
    fun objectMapper() = jacksonObjectMapper().applyDefaultSettings ()

    private fun ObjectMapper.applyDefaultSettings() =
        apply {
            disable(DeserializationFeature.ADJUST_DATES_TO_CONTEXT_TIME_ZONE)
            disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS)
            disable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES)

            enable(DeserializationFeature.READ_UNKNOWN_ENUM_VALUES_USING_DEFAULT_VALUE)
            setSerializationInclusion(JsonInclude.Include.NON_NULL)

            registerModule(Jdk8Module())
            registerModule(ParameterNamesModule())
            registerModule(JsonComponentModule())
            registerModule(
                JavaTimeModule().apply {
                    addSerializer(ZonedDateTime::class.java, ZonedDateTimeSerializer(serializationDateFormat))
                    addDeserializer(ZonedDateTime::class.java, ZonedDateTimeDeserializer())
                }
            )
        }

    class ZonedDateTimeDeserializer : JsonDeserializer<ZonedDateTime>() {
        override fun deserialize(jsonParser: JsonParser, deserializationContext: DeserializationContext): ZonedDateTime {
            val epochTime = jsonParser.text.toLongOrNull()
            return if (epochTime != null) {
                ZonedDateTime.ofInstant(
                    Instant.ofEpochSecond(epochTime),
                    currentZone
                )
            } else {
                ZonedDateTime.parse(jsonParser.text, deserializationDateFormat)
            }
        }
    }
}
于 2021-12-09T16:58:51.400 回答