Skip to content

Commit 768802f

Browse files
committed
ResourceHttpMessageConverter reads Content-Disposition header to expose filename through Resource
Issue: SPR-15191
1 parent f51b896 commit 768802f

File tree

2 files changed

+27
-11
lines changed

2 files changed

+27
-11
lines changed

spring-web/src/main/java/org/springframework/http/converter/ResourceHttpMessageConverter.java

Lines changed: 13 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -82,12 +82,23 @@ protected boolean supports(Class<?> clazz) {
8282
protected Resource readInternal(Class<? extends Resource> clazz, HttpInputMessage inputMessage)
8383
throws IOException, HttpMessageNotReadableException {
8484

85+
final String filename = inputMessage.getHeaders().getContentDisposition().getFilename();
8586
if (this.supportsReadStreaming && InputStreamResource.class == clazz) {
86-
return new InputStreamResource(inputMessage.getBody());
87+
return new InputStreamResource(inputMessage.getBody()) {
88+
@Override
89+
public String getFilename() {
90+
return filename;
91+
}
92+
};
8793
}
8894
else if (clazz.isAssignableFrom(ByteArrayResource.class)) {
8995
byte[] body = StreamUtils.copyToByteArray(inputMessage.getBody());
90-
return new ByteArrayResource(body);
96+
return new ByteArrayResource(body) {
97+
@Override
98+
public String getFilename() {
99+
return filename;
100+
}
101+
};
91102
}
92103
else {
93104
throw new IllegalStateException("Unsupported resource class: " + clazz);

spring-web/src/test/java/org/springframework/http/converter/ResourceHttpMessageConverterTests.java

Lines changed: 14 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@
2929
import org.springframework.core.io.ClassPathResource;
3030
import org.springframework.core.io.InputStreamResource;
3131
import org.springframework.core.io.Resource;
32+
import org.springframework.http.ContentDisposition;
3233
import org.springframework.http.MediaType;
3334
import org.springframework.http.MockHttpInputMessage;
3435
import org.springframework.http.MockHttpOutputMessage;
@@ -69,18 +70,24 @@ public void shouldReadImageResource() throws IOException {
6970
byte[] body = FileCopyUtils.copyToByteArray(getClass().getResourceAsStream("logo.jpg"));
7071
MockHttpInputMessage inputMessage = new MockHttpInputMessage(body);
7172
inputMessage.getHeaders().setContentType(MediaType.IMAGE_JPEG);
73+
inputMessage.getHeaders().setContentDisposition(
74+
ContentDisposition.builder("attachment").filename("yourlogo.jpg").build());
7275
Resource actualResource = converter.read(Resource.class, inputMessage);
7376
assertThat(FileCopyUtils.copyToByteArray(actualResource.getInputStream()), is(body));
77+
assertEquals("yourlogo.jpg", actualResource.getFilename());
7478
}
7579

7680
@Test // SPR-13443
7781
public void shouldReadInputStreamResource() throws IOException {
7882
try (InputStream body = getClass().getResourceAsStream("logo.jpg") ) {
7983
MockHttpInputMessage inputMessage = new MockHttpInputMessage(body);
8084
inputMessage.getHeaders().setContentType(MediaType.IMAGE_JPEG);
85+
inputMessage.getHeaders().setContentDisposition(
86+
ContentDisposition.builder("attachment").filename("yourlogo.jpg").build());
8187
Resource actualResource = converter.read(InputStreamResource.class, inputMessage);
8288
assertThat(actualResource, instanceOf(InputStreamResource.class));
8389
assertThat(actualResource.getInputStream(), is(body));
90+
assertEquals("yourlogo.jpg", actualResource.getFilename());
8491
}
8592
}
8693

@@ -100,6 +107,7 @@ public void shouldWriteImageResource() throws IOException {
100107
MockHttpOutputMessage outputMessage = new MockHttpOutputMessage();
101108
Resource body = new ClassPathResource("logo.jpg", getClass());
102109
converter.write(body, null, outputMessage);
110+
103111
assertEquals("Invalid content-type", MediaType.IMAGE_JPEG,
104112
outputMessage.getHeaders().getContentType());
105113
assertEquals("Invalid content-length", body.getFile().length(), outputMessage.getHeaders().getContentLength());
@@ -111,45 +119,42 @@ public void writeByteArrayNullMediaType() throws IOException {
111119
byte[] byteArray = {1, 2, 3};
112120
Resource body = new ByteArrayResource(byteArray);
113121
converter.write(body, null, outputMessage);
122+
114123
assertTrue(Arrays.equals(byteArray, outputMessage.getBodyAsBytes()));
115124
}
116125

117-
// SPR-12999
118-
@Test @SuppressWarnings("unchecked")
126+
@Test // SPR-12999
127+
@SuppressWarnings("unchecked")
119128
public void writeContentNotGettingInputStream() throws Exception {
120129
MockHttpOutputMessage outputMessage = new MockHttpOutputMessage();
121130
Resource resource = mock(Resource.class);
122131
given(resource.getInputStream()).willThrow(FileNotFoundException.class);
123-
124132
converter.write(resource, MediaType.APPLICATION_OCTET_STREAM, outputMessage);
125133

126134
assertEquals(0, outputMessage.getHeaders().getContentLength());
127135
}
128136

129-
// SPR-12999
130-
@Test
137+
@Test // SPR-12999
131138
public void writeContentNotClosingInputStream() throws Exception {
132139
MockHttpOutputMessage outputMessage = new MockHttpOutputMessage();
133140
Resource resource = mock(Resource.class);
134141
InputStream inputStream = mock(InputStream.class);
135142
given(resource.getInputStream()).willReturn(inputStream);
136143
given(inputStream.read(any())).willReturn(-1);
137144
doThrow(new NullPointerException()).when(inputStream).close();
138-
139145
converter.write(resource, MediaType.APPLICATION_OCTET_STREAM, outputMessage);
140146

141147
assertEquals(0, outputMessage.getHeaders().getContentLength());
142148
}
143149

144-
// SPR-13620
145-
@Test @SuppressWarnings("unchecked")
150+
@Test // SPR-13620
151+
@SuppressWarnings("unchecked")
146152
public void writeContentInputStreamThrowingNullPointerException() throws Exception {
147153
MockHttpOutputMessage outputMessage = new MockHttpOutputMessage();
148154
Resource resource = mock(Resource.class);
149155
InputStream in = mock(InputStream.class);
150156
given(resource.getInputStream()).willReturn(in);
151157
given(in.read(any())).willThrow(NullPointerException.class);
152-
153158
converter.write(resource, MediaType.APPLICATION_OCTET_STREAM, outputMessage);
154159

155160
assertEquals(0, outputMessage.getHeaders().getContentLength());

0 commit comments

Comments
 (0)