Merge branch '5.1.x'

This commit is contained in:
Rossen Stoyanchev 2019-08-21 10:21:32 +03:00
commit 364c7c459d
10 changed files with 115 additions and 15 deletions

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2018 the original author or authors. * Copyright 2002-2019 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -24,6 +24,7 @@ import reactor.core.publisher.Mono;
import org.springframework.core.io.buffer.DataBuffer; import org.springframework.core.io.buffer.DataBuffer;
import org.springframework.core.io.buffer.DataBufferUtils; import org.springframework.core.io.buffer.DataBufferUtils;
import org.springframework.http.HttpHeaders;
/** /**
* {@link ServerHttpResponse} decorator for HTTP HEAD requests. * {@link ServerHttpResponse} decorator for HTTP HEAD requests.
@ -52,7 +53,11 @@ public class HttpHeadResponseDecorator extends ServerHttpResponseDecorator {
DataBufferUtils.release(buffer); DataBufferUtils.release(buffer);
return next; return next;
}) })
.doOnNext(count -> getHeaders().setContentLength(count)) .doOnNext(length -> {
if (length > 0 || getHeaders().getFirst(HttpHeaders.CONTENT_LENGTH) == null) {
getHeaders().setContentLength(length);
}
})
.then(); .then();
} }

View File

@ -0,0 +1,73 @@
/*
* Copyright 2002-2019 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.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.springframework.http.server.reactive;
import java.nio.charset.StandardCharsets;
import io.netty.buffer.PooledByteBufAllocator;
import org.junit.After;
import org.junit.Test;
import reactor.core.publisher.Flux;
import org.springframework.core.io.buffer.DataBuffer;
import org.springframework.core.io.buffer.LeakAwareDataBufferFactory;
import org.springframework.core.io.buffer.NettyDataBufferFactory;
import org.springframework.mock.http.server.reactive.test.MockServerHttpResponse;
import static org.junit.Assert.assertEquals;
/**
* Unit tests for {@link HttpHeadResponseDecorator}.
* @author Rossen Stoyanchev
*/
public class HttpHeadResponseDecoratorTests {
private final LeakAwareDataBufferFactory bufferFactory =
new LeakAwareDataBufferFactory(new NettyDataBufferFactory(PooledByteBufAllocator.DEFAULT));
private final ServerHttpResponse response =
new HttpHeadResponseDecorator(new MockServerHttpResponse(this.bufferFactory));
@After
public void tearDown() {
this.bufferFactory.checkForLeaks();
}
@Test
public void write() {
Flux<DataBuffer> body = Flux.just(toDataBuffer("data1"), toDataBuffer("data2"));
response.writeWith(body).block();
assertEquals(10, response.getHeaders().getContentLength());
}
@Test // gh-23484
public void writeWithGivenContentLength() {
int length = 15;
this.response.getHeaders().setContentLength(length);
this.response.writeWith(Flux.empty()).block();
assertEquals(length, this.response.getHeaders().getContentLength());
}
private DataBuffer toDataBuffer(String s) {
DataBuffer buffer = this.bufferFactory.allocateBuffer();
buffer.write(s.getBytes(StandardCharsets.UTF_8));
return buffer;
}
}

View File

@ -203,6 +203,9 @@ public class PathResourceResolver extends AbstractResourceResolver {
return true; return true;
} }
} }
catch (IllegalArgumentException ex) {
// May not be possible to decode...
}
catch (UnsupportedEncodingException ex) { catch (UnsupportedEncodingException ex) {
// Should never happen... // Should never happen...
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2018 the original author or authors. * Copyright 2002-2019 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -468,7 +468,10 @@ public class ResourceWebHandler implements WebHandler, InitializingBean {
return true; return true;
} }
} }
catch (IllegalArgumentException | UnsupportedEncodingException ex) { catch (IllegalArgumentException ex) {
// May not be possible to decode...
}
catch (UnsupportedEncodingException ex) {
// Should never happen... // Should never happen...
} }
} }

View File

@ -106,6 +106,13 @@ public class PathResourceResolverTests {
assertThat(actual).isNull(); assertThat(actual).isNull();
} }
@Test // gh-23463
public void ignoreInvalidEscapeSequence() throws IOException {
UrlResource location = new UrlResource(getClass().getResource("./test/"));
Resource resource = location.createRelative("test%file.txt");
assertTrue(this.resolver.checkResource(resource, location));
}
@Test @Test
public void checkResourceWithAllowedLocations() { public void checkResourceWithAllowedLocations() {
this.resolver.setAllowedLocations( this.resolver.setAllowedLocations(

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2018 the original author or authors. * Copyright 2002-2019 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -288,6 +288,9 @@ public class PathResourceResolver extends AbstractResourceResolver {
return true; return true;
} }
} }
catch (IllegalArgumentException ex) {
// May not be possible to decode...
}
catch (UnsupportedEncodingException ex) { catch (UnsupportedEncodingException ex) {
// Should never happen... // Should never happen...
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright 2002-2018 the original author or authors. * Copyright 2002-2019 the original author or authors.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -605,7 +605,10 @@ public class ResourceHttpRequestHandler extends WebContentGenerator
return true; return true;
} }
} }
catch (IllegalArgumentException | UnsupportedEncodingException ex) { catch (IllegalArgumentException ex) {
// May not be possible to decode...
}
catch (UnsupportedEncodingException ex) {
// Should never happen... // Should never happen...
} }
} }

View File

@ -91,6 +91,13 @@ public class PathResourceResolverTests {
assertThat(actual).isNull(); assertThat(actual).isNull();
} }
@Test // gh-23463
public void ignoreInvalidEscapeSequence() throws IOException {
UrlResource location = new UrlResource(getClass().getResource("./test/"));
Resource resource = location.createRelative("test%file.txt");
assertTrue(this.resolver.checkResource(resource, location));
}
@Test @Test
public void checkResourceWithAllowedLocations() { public void checkResourceWithAllowedLocations() {
this.resolver.setAllowedLocations( this.resolver.setAllowedLocations(
@ -104,8 +111,7 @@ public class PathResourceResolverTests {
assertThat(actual).isEqualTo("../testalternatepath/bar.css"); assertThat(actual).isEqualTo("../testalternatepath/bar.css");
} }
// SPR-12432 @Test // SPR-12432
@Test
public void checkServletContextResource() throws Exception { public void checkServletContextResource() throws Exception {
Resource classpathLocation = new ClassPathResource("test/", PathResourceResolver.class); Resource classpathLocation = new ClassPathResource("test/", PathResourceResolver.class);
MockServletContext context = new MockServletContext(); MockServletContext context = new MockServletContext();
@ -117,8 +123,7 @@ public class PathResourceResolverTests {
assertThat(this.resolver.checkResource(resource, servletContextLocation)).isTrue(); assertThat(this.resolver.checkResource(resource, servletContextLocation)).isTrue();
} }
// SPR-12624 @Test // SPR-12624
@Test
public void checkRelativeLocation() throws Exception { public void checkRelativeLocation() throws Exception {
String locationUrl= new UrlResource(getClass().getResource("./test/")).getURL().toExternalForm(); String locationUrl= new UrlResource(getClass().getResource("./test/")).getURL().toExternalForm();
Resource location = new UrlResource(locationUrl.replace("/springframework","/../org/springframework")); Resource location = new UrlResource(locationUrl.replace("/springframework","/../org/springframework"));
@ -126,15 +131,13 @@ public class PathResourceResolverTests {
assertThat(this.resolver.resolveResource(null, "main.css", Collections.singletonList(location), null)).isNotNull(); assertThat(this.resolver.resolveResource(null, "main.css", Collections.singletonList(location), null)).isNotNull();
} }
// SPR-12747 @Test // SPR-12747
@Test
public void checkFileLocation() throws Exception { public void checkFileLocation() throws Exception {
Resource resource = getResource("main.css"); Resource resource = getResource("main.css");
assertThat(this.resolver.checkResource(resource, resource)).isTrue(); assertThat(this.resolver.checkResource(resource, resource)).isTrue();
} }
// SPR-13241 @Test // SPR-13241
@Test
public void resolvePathRootResource() { public void resolvePathRootResource() {
Resource webjarsLocation = new ClassPathResource("/META-INF/resources/webjars/", PathResourceResolver.class); Resource webjarsLocation = new ClassPathResource("/META-INF/resources/webjars/", PathResourceResolver.class);
String path = this.resolver.resolveUrlPathInternal("", Collections.singletonList(webjarsLocation), null); String path = this.resolver.resolveUrlPathInternal("", Collections.singletonList(webjarsLocation), null);