|
| 1 | +package io.micronaut.repro.http |
| 2 | + |
| 3 | +import io.micronaut.http.HttpHeaders |
| 4 | +import io.micronaut.http.MutableHttpRequest |
| 5 | +import io.micronaut.http.client.netty.NettyClientHttpRequestFactory |
| 6 | +import io.micronaut.http.cookie.Cookie |
| 7 | +import io.micronaut.http.simple.SimpleHttpRequestFactory |
| 8 | +import org.junit.jupiter.api.Assertions.assertEquals |
| 9 | +import org.junit.jupiter.api.Assertions.assertFalse |
| 10 | +import org.junit.jupiter.api.Assertions.assertTrue |
| 11 | +import org.junit.jupiter.api.Test |
| 12 | + |
| 13 | +class CookieHeaderSingleValueReproTest { |
| 14 | + |
| 15 | + private val cookies = linkedSetOf( |
| 16 | + Cookie.of("a", "1"), |
| 17 | + Cookie.of("b", "2"), |
| 18 | + Cookie.of("a", "3") // latest value for 'a' should win |
| 19 | + ) |
| 20 | + |
| 21 | + // Simple request factory |
| 22 | + @Test |
| 23 | + fun simpleRequest_addCookies_oneByOne_resultsInSingleCookieHeader() { |
| 24 | + val request: MutableHttpRequest<Any> = SimpleHttpRequestFactory().get<Any>("/") |
| 25 | + assertSingleCookieHeaderAfterAddingCookiesIndividually(request) |
| 26 | + } |
| 27 | + |
| 28 | + @Test |
| 29 | + fun simpleRequest_addCookies_bulk_resultsInSingleCookieHeader() { |
| 30 | + val request: MutableHttpRequest<Any> = SimpleHttpRequestFactory().get<Any>("/") |
| 31 | + assertSingleCookieHeaderAfterAddingCookiesBulk(request) |
| 32 | + } |
| 33 | + |
| 34 | + // Netty client request factory |
| 35 | + @Test |
| 36 | + fun nettyClientRequest_addCookies_oneByOne_resultsInSingleCookieHeader() { |
| 37 | + val request: MutableHttpRequest<Any> = NettyClientHttpRequestFactory().get<Any>("/") |
| 38 | + assertSingleCookieHeaderAfterAddingCookiesIndividually(request) |
| 39 | + } |
| 40 | + |
| 41 | + @Test |
| 42 | + fun nettyClientRequest_addCookies_bulk_resultsInSingleCookieHeader() { |
| 43 | + val request: MutableHttpRequest<Any> = NettyClientHttpRequestFactory().get<Any>("/") |
| 44 | + assertSingleCookieHeaderAfterAddingCookiesBulk(request) |
| 45 | + } |
| 46 | + |
| 47 | + private fun assertSingleCookieHeaderAfterAddingCookiesIndividually(request: MutableHttpRequest<*>) { |
| 48 | + cookies.forEach { request.cookie(it) } |
| 49 | + assertSingleCookieHeaderAndValues(request) |
| 50 | + } |
| 51 | + |
| 52 | + private fun assertSingleCookieHeaderAfterAddingCookiesBulk(request: MutableHttpRequest<*>) { |
| 53 | + request.cookies(cookies) |
| 54 | + assertSingleCookieHeaderAndValues(request) |
| 55 | + } |
| 56 | + |
| 57 | + private fun assertSingleCookieHeaderAndValues(request: MutableHttpRequest<*>) { |
| 58 | + val cookieHeaders = request.headers.getAll(HttpHeaders.COOKIE) |
| 59 | + assertEquals(1, cookieHeaders.size, "Expected a single Cookie header value") |
| 60 | + |
| 61 | + val value = cookieHeaders[0] |
| 62 | + // Validate semantics (latest value wins, all expected cookies present) |
| 63 | + val parts = value.split(";".toRegex()).map { it.trim() }.filter { it.isNotEmpty() } |
| 64 | + assertTrue(parts.contains("a=3"), "Expected latest value for cookie 'a' to be present") |
| 65 | + assertTrue(parts.contains("b=2"), "Expected cookie 'b' to be present") |
| 66 | + assertFalse(parts.contains("a=1"), "Expected old value for cookie 'a' to be replaced") |
| 67 | + } |
| 68 | +} |
0 commit comments