Browse Source

Merge pull request #30252 from luozhenyu:content-disposition

* gh-30252:
  Polish contribution
  Quote question marks in content-disposition
pull/30420/head
Arjen Poutsma 3 years ago
parent
commit
685548b58b
  1. 27
      spring-web/src/main/java/org/springframework/http/ContentDisposition.java
  2. 22
      spring-web/src/test/java/org/springframework/http/ContentDispositionTests.java

27
spring-web/src/main/java/org/springframework/http/ContentDisposition.java

@ -23,6 +23,7 @@ import java.time.ZonedDateTime; @@ -23,6 +23,7 @@ import java.time.ZonedDateTime;
import java.time.format.DateTimeParseException;
import java.util.ArrayList;
import java.util.Base64;
import java.util.BitSet;
import java.util.List;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
@ -58,6 +59,19 @@ public final class ContentDisposition { @@ -58,6 +59,19 @@ public final class ContentDisposition {
private static final String INVALID_HEADER_FIELD_PARAMETER_FORMAT =
"Invalid header field parameter format (as defined in RFC 5987)";
private static final BitSet PRINTABLE = new BitSet(256);
static {
// RFC 2045, Section 6.7, and RFC 2047, Section 4.2
for (int i=33; i<= 126; i++) {
PRINTABLE.set(i);
}
PRINTABLE.set(61, false); // =
PRINTABLE.set(63, false); // ?
PRINTABLE.set(95, false); // _
}
@Nullable
private final String type;
@ -545,7 +559,7 @@ public final class ContentDisposition { @@ -545,7 +559,7 @@ public final class ContentDisposition {
int index = 0;
while (index < value.length) {
byte b = value[index];
if (b == '_') {
if (b == '_') { // RFC 2047, section 4.2, rule (2)
baos.write(' ');
index++;
}
@ -583,7 +597,10 @@ public final class ContentDisposition { @@ -583,7 +597,10 @@ public final class ContentDisposition {
sb.append(charset.name());
sb.append("?Q?");
for (byte b : source) {
if (isPrintable(b)) {
if (b == 32) { // RFC 2047, section 4.2, rule (2)
sb.append('_');
}
else if (isPrintable(b)) {
sb.append((char) b);
}
else {
@ -599,7 +616,11 @@ public final class ContentDisposition { @@ -599,7 +616,11 @@ public final class ContentDisposition {
}
private static boolean isPrintable(byte c) {
return (c >= '!' && c <= '<') || (c >= '>' && c <= '~');
int b = c;
if (b < 0) {
b = 256 + b;
}
return PRINTABLE.get(b);
}
private static String encodeQuotedPairs(String filename) {

22
spring-web/src/test/java/org/springframework/http/ContentDispositionTests.java

@ -325,4 +325,26 @@ class ContentDispositionTests { @@ -325,4 +325,26 @@ class ContentDispositionTests {
assertThat(parsed.toString()).isEqualTo(cd.toString());
}
@Test // gh-30252
void parseFormattedWithQuestionMark() {
String filename = "filename with ?问号.txt";
ContentDisposition cd = ContentDisposition.attachment()
.filename(filename, StandardCharsets.UTF_8)
.build();
String result = cd.toString();
assertThat(result).isEqualTo("attachment; " +
"filename=\"=?UTF-8?Q?filename_with_=3F=E9=97=AE=E5=8F=B7.txt?=\"; " +
"filename*=UTF-8''filename%20with%20%3F%E9%97%AE%E5%8F%B7.txt");
String[] parts = result.split("; ");
String quotedPrintableFilename = parts[0] + "; " + parts[1];
assertThat(ContentDisposition.parse(quotedPrintableFilename).getFilename())
.isEqualTo(filename);
String rfc5987Filename = parts[0] + "; " + parts[2];
assertThat(ContentDisposition.parse(rfc5987Filename).getFilename())
.isEqualTo(filename);
}
}

Loading…
Cancel
Save