Set 304 status on ServerResponse when ETag/LastModified match
This commit checks the Etag/LastModified headers on the incoming request, and sets a 304 Not Modified status with no body when they match, by delegating to ServerWebExchange.checkNotModified. Issue: SPR-16348
This commit is contained in:
		
							parent
							
								
									c211e3998b
								
							
						
					
					
						commit
						c53c8bfc5a
					
				| 
						 | 
				
			
			@ -224,10 +224,8 @@ class DefaultEntityResponseBuilder<T> implements EntityResponse.Builder<T> {
 | 
			
		|||
		}
 | 
			
		||||
 | 
			
		||||
		@Override
 | 
			
		||||
		public Mono<Void> writeTo(ServerWebExchange exchange, Context context) {
 | 
			
		||||
			ServerHttpResponse response = exchange.getResponse();
 | 
			
		||||
			writeStatusAndHeaders(response);
 | 
			
		||||
			return inserter().insert(response, new BodyInserter.Context() {
 | 
			
		||||
		protected Mono<Void> writeToInternal(ServerWebExchange exchange, Context context) {
 | 
			
		||||
			return inserter().insert(exchange.getResponse(), new BodyInserter.Context() {
 | 
			
		||||
				@Override
 | 
			
		||||
				public List<HttpMessageWriter<?>> messageWriters() {
 | 
			
		||||
					return context.messageWriters();
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -35,7 +35,6 @@ import org.springframework.http.HttpHeaders;
 | 
			
		|||
import org.springframework.http.HttpStatus;
 | 
			
		||||
import org.springframework.http.MediaType;
 | 
			
		||||
import org.springframework.http.ResponseCookie;
 | 
			
		||||
import org.springframework.http.server.reactive.ServerHttpResponse;
 | 
			
		||||
import org.springframework.lang.Nullable;
 | 
			
		||||
import org.springframework.util.Assert;
 | 
			
		||||
import org.springframework.util.LinkedMultiValueMap;
 | 
			
		||||
| 
						 | 
				
			
			@ -184,9 +183,7 @@ class DefaultRenderingResponseBuilder implements RenderingResponse.Builder {
 | 
			
		|||
		}
 | 
			
		||||
 | 
			
		||||
		@Override
 | 
			
		||||
		public Mono<Void> writeTo(ServerWebExchange exchange, Context context) {
 | 
			
		||||
			ServerHttpResponse response = exchange.getResponse();
 | 
			
		||||
			writeStatusAndHeaders(response);
 | 
			
		||||
		protected Mono<Void> writeToInternal(ServerWebExchange exchange, Context context) {
 | 
			
		||||
			MediaType responseContentType = exchange.getResponse().getHeaders().getContentType();
 | 
			
		||||
			Locale locale = LocaleContextHolder.getLocale(exchange.getLocaleContext());
 | 
			
		||||
			Stream<ViewResolver> viewResolverStream = context.viewResolvers().stream();
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -17,8 +17,10 @@
 | 
			
		|||
package org.springframework.web.reactive.function.server;
 | 
			
		||||
 | 
			
		||||
import java.net.URI;
 | 
			
		||||
import java.time.Instant;
 | 
			
		||||
import java.time.ZonedDateTime;
 | 
			
		||||
import java.util.Arrays;
 | 
			
		||||
import java.util.EnumSet;
 | 
			
		||||
import java.util.HashMap;
 | 
			
		||||
import java.util.LinkedHashSet;
 | 
			
		||||
import java.util.List;
 | 
			
		||||
| 
						 | 
				
			
			@ -278,6 +280,8 @@ class DefaultServerResponseBuilder implements ServerResponse.BodyBuilder {
 | 
			
		|||
 | 
			
		||||
	static abstract class AbstractServerResponse implements ServerResponse {
 | 
			
		||||
 | 
			
		||||
		private static final Set<HttpMethod> SAFE_METHODS = EnumSet.of(HttpMethod.GET, HttpMethod.HEAD);
 | 
			
		||||
 | 
			
		||||
		final int statusCode;
 | 
			
		||||
 | 
			
		||||
		private final HttpHeaders headers;
 | 
			
		||||
| 
						 | 
				
			
			@ -319,7 +323,21 @@ class DefaultServerResponseBuilder implements ServerResponse.BodyBuilder {
 | 
			
		|||
			return this.cookies;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		protected void writeStatusAndHeaders(ServerHttpResponse response) {
 | 
			
		||||
		@Override
 | 
			
		||||
		public final Mono<Void> writeTo(ServerWebExchange exchange, Context context) {
 | 
			
		||||
			writeStatusAndHeaders(exchange.getResponse());
 | 
			
		||||
 | 
			
		||||
			Instant lastModified = Instant.ofEpochMilli(headers().getLastModified());
 | 
			
		||||
			HttpMethod httpMethod = exchange.getRequest().getMethod();
 | 
			
		||||
			if (SAFE_METHODS.contains(httpMethod) && exchange.checkNotModified(headers().getETag(), lastModified)) {
 | 
			
		||||
				return exchange.getResponse().setComplete();
 | 
			
		||||
			}
 | 
			
		||||
			else {
 | 
			
		||||
				return writeToInternal(exchange, context);
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		private void writeStatusAndHeaders(ServerHttpResponse response) {
 | 
			
		||||
			if (response instanceof AbstractServerHttpResponse) {
 | 
			
		||||
				((AbstractServerHttpResponse) response).setStatusCodeValue(this.statusCode);
 | 
			
		||||
			}
 | 
			
		||||
| 
						 | 
				
			
			@ -335,6 +353,8 @@ class DefaultServerResponseBuilder implements ServerResponse.BodyBuilder {
 | 
			
		|||
			copy(this.cookies, response.getCookies());
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		protected abstract Mono<Void> writeToInternal(ServerWebExchange exchange, Context context);
 | 
			
		||||
 | 
			
		||||
		private static <K,V> void copy(MultiValueMap<K,V> src, MultiValueMap<K,V> dst) {
 | 
			
		||||
			if (!src.isEmpty()) {
 | 
			
		||||
				src.entrySet().stream()
 | 
			
		||||
| 
						 | 
				
			
			@ -358,8 +378,7 @@ class DefaultServerResponseBuilder implements ServerResponse.BodyBuilder {
 | 
			
		|||
		}
 | 
			
		||||
 | 
			
		||||
		@Override
 | 
			
		||||
		public Mono<Void> writeTo(ServerWebExchange exchange, Context context) {
 | 
			
		||||
			writeStatusAndHeaders(exchange.getResponse());
 | 
			
		||||
		protected Mono<Void> writeToInternal(ServerWebExchange exchange, Context context) {
 | 
			
		||||
			return this.writeFunction.apply(exchange, context);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
| 
						 | 
				
			
			@ -381,10 +400,8 @@ class DefaultServerResponseBuilder implements ServerResponse.BodyBuilder {
 | 
			
		|||
		}
 | 
			
		||||
 | 
			
		||||
		@Override
 | 
			
		||||
		public Mono<Void> writeTo(ServerWebExchange exchange, Context context) {
 | 
			
		||||
			ServerHttpResponse response = exchange.getResponse();
 | 
			
		||||
			writeStatusAndHeaders(response);
 | 
			
		||||
			return this.inserter.insert(response, new BodyInserter.Context() {
 | 
			
		||||
		protected Mono<Void> writeToInternal(ServerWebExchange exchange, Context context) {
 | 
			
		||||
			return this.inserter.insert(exchange.getResponse(), new BodyInserter.Context() {
 | 
			
		||||
				@Override
 | 
			
		||||
				public List<HttpMessageWriter<?>> messageWriters() {
 | 
			
		||||
					return context.messageWriters();
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2017 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 *
 | 
			
		||||
 * Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
 * you may not use this file except in compliance with the License.
 | 
			
		||||
| 
						 | 
				
			
			@ -18,6 +18,8 @@ package org.springframework.web.reactive.function.server;
 | 
			
		|||
 | 
			
		||||
import java.nio.ByteBuffer;
 | 
			
		||||
import java.time.ZonedDateTime;
 | 
			
		||||
import java.time.format.DateTimeFormatter;
 | 
			
		||||
import java.time.temporal.ChronoUnit;
 | 
			
		||||
import java.util.Collections;
 | 
			
		||||
import java.util.EnumSet;
 | 
			
		||||
import java.util.List;
 | 
			
		||||
| 
						 | 
				
			
			@ -44,6 +46,7 @@ import org.springframework.http.codec.EncoderHttpMessageWriter;
 | 
			
		|||
import org.springframework.http.codec.HttpMessageWriter;
 | 
			
		||||
import org.springframework.http.server.reactive.ServerHttpResponse;
 | 
			
		||||
import org.springframework.mock.http.server.reactive.test.MockServerHttpRequest;
 | 
			
		||||
import org.springframework.mock.http.server.reactive.test.MockServerHttpResponse;
 | 
			
		||||
import org.springframework.mock.web.test.server.MockServerWebExchange;
 | 
			
		||||
import org.springframework.util.LinkedMultiValueMap;
 | 
			
		||||
import org.springframework.util.MultiValueMap;
 | 
			
		||||
| 
						 | 
				
			
			@ -232,4 +235,52 @@ public class DefaultEntityResponseBuilderTests {
 | 
			
		|||
		assertNotNull(exchange.getResponse().getBody());
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Test
 | 
			
		||||
	public void notModifiedEtag() {
 | 
			
		||||
		String etag = "\"foo\"";
 | 
			
		||||
		EntityResponse<String> responseMono = EntityResponse.fromObject("bar")
 | 
			
		||||
				.eTag(etag)
 | 
			
		||||
				.build()
 | 
			
		||||
				.block();
 | 
			
		||||
 | 
			
		||||
		MockServerHttpRequest request = MockServerHttpRequest.get("http://example.com")
 | 
			
		||||
				.header(HttpHeaders.IF_NONE_MATCH, etag)
 | 
			
		||||
				.build();
 | 
			
		||||
		MockServerWebExchange exchange = MockServerWebExchange.from(request);
 | 
			
		||||
 | 
			
		||||
		responseMono.writeTo(exchange, DefaultServerResponseBuilderTests.EMPTY_CONTEXT);
 | 
			
		||||
 | 
			
		||||
		MockServerHttpResponse response = exchange.getResponse();
 | 
			
		||||
		assertEquals(HttpStatus.NOT_MODIFIED, response.getStatusCode());
 | 
			
		||||
		StepVerifier.create(response.getBody())
 | 
			
		||||
				.expectError(IllegalStateException.class)
 | 
			
		||||
				.verify();
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	@Test
 | 
			
		||||
	public void notModifiedLastModified() {
 | 
			
		||||
		ZonedDateTime now = ZonedDateTime.now();
 | 
			
		||||
		ZonedDateTime oneMinuteBeforeNow = now.minus(1, ChronoUnit.MINUTES);
 | 
			
		||||
 | 
			
		||||
		EntityResponse<String> responseMono = EntityResponse.fromObject("bar")
 | 
			
		||||
				.lastModified(oneMinuteBeforeNow)
 | 
			
		||||
				.build()
 | 
			
		||||
				.block();
 | 
			
		||||
 | 
			
		||||
		MockServerHttpRequest request = MockServerHttpRequest.get("http://example.com")
 | 
			
		||||
				.header(HttpHeaders.IF_MODIFIED_SINCE,
 | 
			
		||||
						DateTimeFormatter.RFC_1123_DATE_TIME.format(now))
 | 
			
		||||
				.build();
 | 
			
		||||
		MockServerWebExchange exchange = MockServerWebExchange.from(request);
 | 
			
		||||
 | 
			
		||||
		responseMono.writeTo(exchange, DefaultServerResponseBuilderTests.EMPTY_CONTEXT);
 | 
			
		||||
 | 
			
		||||
		MockServerHttpResponse response = exchange.getResponse();
 | 
			
		||||
		assertEquals(HttpStatus.NOT_MODIFIED, response.getStatusCode());
 | 
			
		||||
		StepVerifier.create(response.getBody())
 | 
			
		||||
				.expectError(IllegalStateException.class)
 | 
			
		||||
				.verify();
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2017 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 *
 | 
			
		||||
 * Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
 * you may not use this file except in compliance with the License.
 | 
			
		||||
| 
						 | 
				
			
			@ -16,6 +16,9 @@
 | 
			
		|||
 | 
			
		||||
package org.springframework.web.reactive.function.server;
 | 
			
		||||
 | 
			
		||||
import java.time.ZonedDateTime;
 | 
			
		||||
import java.time.format.DateTimeFormatter;
 | 
			
		||||
import java.time.temporal.ChronoUnit;
 | 
			
		||||
import java.util.ArrayList;
 | 
			
		||||
import java.util.Collections;
 | 
			
		||||
import java.util.List;
 | 
			
		||||
| 
						 | 
				
			
			@ -28,9 +31,11 @@ import reactor.core.publisher.Mono;
 | 
			
		|||
import reactor.test.StepVerifier;
 | 
			
		||||
 | 
			
		||||
import org.springframework.http.HttpHeaders;
 | 
			
		||||
import org.springframework.http.HttpStatus;
 | 
			
		||||
import org.springframework.http.MediaType;
 | 
			
		||||
import org.springframework.http.ResponseCookie;
 | 
			
		||||
import org.springframework.mock.http.server.reactive.test.MockServerHttpRequest;
 | 
			
		||||
import org.springframework.mock.http.server.reactive.test.MockServerHttpResponse;
 | 
			
		||||
import org.springframework.mock.web.test.server.MockServerWebExchange;
 | 
			
		||||
import org.springframework.util.LinkedMultiValueMap;
 | 
			
		||||
import org.springframework.util.MultiValueMap;
 | 
			
		||||
| 
						 | 
				
			
			@ -40,7 +45,7 @@ import org.springframework.web.reactive.result.view.ViewResolver;
 | 
			
		|||
import org.springframework.web.reactive.result.view.ViewResolverSupport;
 | 
			
		||||
import org.springframework.web.server.ServerWebExchange;
 | 
			
		||||
 | 
			
		||||
import static org.junit.Assert.assertEquals;
 | 
			
		||||
import static org.junit.Assert.*;
 | 
			
		||||
import static org.mockito.Mockito.*;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
| 
						 | 
				
			
			@ -181,4 +186,52 @@ public class DefaultRenderingResponseTests {
 | 
			
		|||
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Test
 | 
			
		||||
	public void notModifiedEtag() {
 | 
			
		||||
		String etag = "\"foo\"";
 | 
			
		||||
		RenderingResponse responseMono = RenderingResponse.create("bar")
 | 
			
		||||
				.header(HttpHeaders.ETAG, etag)
 | 
			
		||||
				.build()
 | 
			
		||||
				.block();
 | 
			
		||||
 | 
			
		||||
		MockServerHttpRequest request = MockServerHttpRequest.get("http://example.com")
 | 
			
		||||
				.header(HttpHeaders.IF_NONE_MATCH, etag)
 | 
			
		||||
				.build();
 | 
			
		||||
		MockServerWebExchange exchange = MockServerWebExchange.from(request);
 | 
			
		||||
 | 
			
		||||
		responseMono.writeTo(exchange, DefaultServerResponseBuilderTests.EMPTY_CONTEXT);
 | 
			
		||||
 | 
			
		||||
		MockServerHttpResponse response = exchange.getResponse();
 | 
			
		||||
		assertEquals(HttpStatus.NOT_MODIFIED, response.getStatusCode());
 | 
			
		||||
		StepVerifier.create(response.getBody())
 | 
			
		||||
				.expectError(IllegalStateException.class)
 | 
			
		||||
				.verify();
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Test
 | 
			
		||||
	public void notModifiedLastModified() {
 | 
			
		||||
		ZonedDateTime now = ZonedDateTime.now();
 | 
			
		||||
		ZonedDateTime oneMinuteBeforeNow = now.minus(1, ChronoUnit.MINUTES);
 | 
			
		||||
 | 
			
		||||
		RenderingResponse responseMono = RenderingResponse.create("bar")
 | 
			
		||||
				.header(HttpHeaders.LAST_MODIFIED, DateTimeFormatter.RFC_1123_DATE_TIME.format(oneMinuteBeforeNow))
 | 
			
		||||
				.build()
 | 
			
		||||
				.block();
 | 
			
		||||
 | 
			
		||||
		MockServerHttpRequest request = MockServerHttpRequest.get("http://example.com")
 | 
			
		||||
				.header(HttpHeaders.IF_MODIFIED_SINCE,
 | 
			
		||||
						DateTimeFormatter.RFC_1123_DATE_TIME.format(now))
 | 
			
		||||
				.build();
 | 
			
		||||
		MockServerWebExchange exchange = MockServerWebExchange.from(request);
 | 
			
		||||
 | 
			
		||||
		responseMono.writeTo(exchange, DefaultServerResponseBuilderTests.EMPTY_CONTEXT);
 | 
			
		||||
 | 
			
		||||
		MockServerHttpResponse response = exchange.getResponse();
 | 
			
		||||
		assertEquals(HttpStatus.NOT_MODIFIED, response.getStatusCode());
 | 
			
		||||
		StepVerifier.create(response.getBody())
 | 
			
		||||
				.expectError(IllegalStateException.class)
 | 
			
		||||
				.verify();
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1,5 +1,5 @@
 | 
			
		|||
/*
 | 
			
		||||
 * Copyright 2002-2017 the original author or authors.
 | 
			
		||||
 * Copyright 2002-2018 the original author or authors.
 | 
			
		||||
 *
 | 
			
		||||
 * Licensed under the Apache License, Version 2.0 (the "License");
 | 
			
		||||
 * you may not use this file except in compliance with the License.
 | 
			
		||||
| 
						 | 
				
			
			@ -18,6 +18,8 @@ package org.springframework.web.reactive.function.server;
 | 
			
		|||
 | 
			
		||||
import java.net.URI;
 | 
			
		||||
import java.time.ZonedDateTime;
 | 
			
		||||
import java.time.format.DateTimeFormatter;
 | 
			
		||||
import java.time.temporal.ChronoUnit;
 | 
			
		||||
import java.util.Collections;
 | 
			
		||||
import java.util.EnumSet;
 | 
			
		||||
import java.util.List;
 | 
			
		||||
| 
						 | 
				
			
			@ -33,19 +35,33 @@ import org.springframework.http.HttpMethod;
 | 
			
		|||
import org.springframework.http.HttpStatus;
 | 
			
		||||
import org.springframework.http.MediaType;
 | 
			
		||||
import org.springframework.http.ResponseCookie;
 | 
			
		||||
import org.springframework.http.codec.HttpMessageWriter;
 | 
			
		||||
import org.springframework.mock.http.server.reactive.test.MockServerHttpRequest;
 | 
			
		||||
import org.springframework.mock.http.server.reactive.test.MockServerHttpResponse;
 | 
			
		||||
import org.springframework.mock.web.test.server.MockServerWebExchange;
 | 
			
		||||
import org.springframework.util.LinkedMultiValueMap;
 | 
			
		||||
import org.springframework.util.MultiValueMap;
 | 
			
		||||
import org.springframework.web.server.ServerWebExchange;
 | 
			
		||||
import org.springframework.web.reactive.result.view.ViewResolver;
 | 
			
		||||
 | 
			
		||||
import static org.junit.Assert.*;
 | 
			
		||||
import static org.mockito.Mockito.*;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * @author Arjen Poutsma
 | 
			
		||||
 */
 | 
			
		||||
public class DefaultServerResponseBuilderTests {
 | 
			
		||||
 | 
			
		||||
	static final ServerResponse.Context EMPTY_CONTEXT = new ServerResponse.Context() {
 | 
			
		||||
		@Override
 | 
			
		||||
		public List<HttpMessageWriter<?>> messageWriters() {
 | 
			
		||||
			return Collections.emptyList();
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		@Override
 | 
			
		||||
		public List<ViewResolver> viewResolvers() {
 | 
			
		||||
			return Collections.emptyList();
 | 
			
		||||
		}
 | 
			
		||||
	};
 | 
			
		||||
 | 
			
		||||
	@Test
 | 
			
		||||
	public void from() throws Exception {
 | 
			
		||||
		ServerResponse other = ServerResponse.ok().header("foo", "bar").build().block();
 | 
			
		||||
| 
						 | 
				
			
			@ -287,13 +303,12 @@ public class DefaultServerResponseBuilderTests {
 | 
			
		|||
				.header("MyKey", "MyValue")
 | 
			
		||||
				.cookie(cookie).build();
 | 
			
		||||
 | 
			
		||||
		ServerWebExchange exchange = mock(ServerWebExchange.class);
 | 
			
		||||
		MockServerHttpResponse response = new MockServerHttpResponse();
 | 
			
		||||
		when(exchange.getResponse()).thenReturn(response);
 | 
			
		||||
		ServerResponse.Context context = mock(ServerResponse.Context.class);
 | 
			
		||||
		MockServerHttpRequest request = MockServerHttpRequest.get("http://example.com").build();
 | 
			
		||||
		MockServerWebExchange exchange = MockServerWebExchange.from(request);
 | 
			
		||||
 | 
			
		||||
		result.flatMap(res -> res.writeTo(exchange, context)).block();
 | 
			
		||||
		result.flatMap(res -> res.writeTo(exchange, EMPTY_CONTEXT)).block();
 | 
			
		||||
 | 
			
		||||
		MockServerHttpResponse response = exchange.getResponse();
 | 
			
		||||
		assertEquals(HttpStatus.CREATED, response.getStatusCode());
 | 
			
		||||
		assertEquals("MyValue", response.getHeaders().getFirst("MyKey"));
 | 
			
		||||
		assertEquals("value", response.getCookies().getFirst("name").getValue());
 | 
			
		||||
| 
						 | 
				
			
			@ -305,13 +320,12 @@ public class DefaultServerResponseBuilderTests {
 | 
			
		|||
		Mono<Void> mono = Mono.empty();
 | 
			
		||||
		Mono<ServerResponse> result = ServerResponse.ok().build(mono);
 | 
			
		||||
 | 
			
		||||
		ServerWebExchange exchange = mock(ServerWebExchange.class);
 | 
			
		||||
		MockServerHttpResponse response = new MockServerHttpResponse();
 | 
			
		||||
		when(exchange.getResponse()).thenReturn(response);
 | 
			
		||||
		ServerResponse.Context context = mock(ServerResponse.Context.class);
 | 
			
		||||
		MockServerHttpRequest request = MockServerHttpRequest.get("http://example.com").build();
 | 
			
		||||
		MockServerWebExchange exchange = MockServerWebExchange.from(request);
 | 
			
		||||
 | 
			
		||||
		result.flatMap(res -> res.writeTo(exchange, context)).block();
 | 
			
		||||
		result.flatMap(res -> res.writeTo(exchange, EMPTY_CONTEXT)).block();
 | 
			
		||||
 | 
			
		||||
		MockServerHttpResponse response = exchange.getResponse();
 | 
			
		||||
		StepVerifier.create(response.getBody()).expectComplete().verify();
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -322,5 +336,52 @@ public class DefaultServerResponseBuilderTests {
 | 
			
		|||
		ServerResponse.ok().syncBody(mono);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Test
 | 
			
		||||
	public void notModifiedEtag() {
 | 
			
		||||
		String etag = "\"foo\"";
 | 
			
		||||
		ServerResponse responseMono = ServerResponse.ok()
 | 
			
		||||
				.eTag(etag)
 | 
			
		||||
				.syncBody("bar")
 | 
			
		||||
				.block();
 | 
			
		||||
 | 
			
		||||
		MockServerHttpRequest request = MockServerHttpRequest.get("http://example.com")
 | 
			
		||||
				.header(HttpHeaders.IF_NONE_MATCH, etag)
 | 
			
		||||
				.build();
 | 
			
		||||
		MockServerWebExchange exchange = MockServerWebExchange.from(request);
 | 
			
		||||
 | 
			
		||||
		responseMono.writeTo(exchange, EMPTY_CONTEXT);
 | 
			
		||||
 | 
			
		||||
		MockServerHttpResponse response = exchange.getResponse();
 | 
			
		||||
		assertEquals(HttpStatus.NOT_MODIFIED, response.getStatusCode());
 | 
			
		||||
		StepVerifier.create(response.getBody())
 | 
			
		||||
				.expectError(IllegalStateException.class)
 | 
			
		||||
				.verify();
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	@Test
 | 
			
		||||
	public void notModifiedLastModified() {
 | 
			
		||||
		ZonedDateTime now = ZonedDateTime.now();
 | 
			
		||||
		ZonedDateTime oneMinuteBeforeNow = now.minus(1, ChronoUnit.MINUTES);
 | 
			
		||||
 | 
			
		||||
		ServerResponse responseMono = ServerResponse.ok()
 | 
			
		||||
				.lastModified(oneMinuteBeforeNow)
 | 
			
		||||
				.syncBody("bar")
 | 
			
		||||
				.block();
 | 
			
		||||
 | 
			
		||||
		MockServerHttpRequest request = MockServerHttpRequest.get("http://example.com")
 | 
			
		||||
				.header(HttpHeaders.IF_MODIFIED_SINCE,
 | 
			
		||||
						DateTimeFormatter.RFC_1123_DATE_TIME.format(now))
 | 
			
		||||
				.build();
 | 
			
		||||
		MockServerWebExchange exchange = MockServerWebExchange.from(request);
 | 
			
		||||
 | 
			
		||||
		responseMono.writeTo(exchange, EMPTY_CONTEXT);
 | 
			
		||||
 | 
			
		||||
		MockServerHttpResponse response = exchange.getResponse();
 | 
			
		||||
		assertEquals(HttpStatus.NOT_MODIFIED, response.getStatusCode());
 | 
			
		||||
		StepVerifier.create(response.getBody())
 | 
			
		||||
				.expectError(IllegalStateException.class)
 | 
			
		||||
				.verify();
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
		Loading…
	
		Reference in New Issue