Refactor MimeType/MediaType specificity
This commit makes several changes to MimeType and MediaType related to the topic of specificity. This commit deprecates the MimeType and MediaType Comparators. Comparators require a transitive relationship, and the desired order for these types is not transitive (see #27488). Instead, this commit introduces two new MimeType methods: isMoreSpecific and isLessSpecific, both of which return booleans. MediaType overrides these methods to include the quality factor (q) in the comparison. All MediaType sorting methods have been deprecated in favor of MimeTypeUtils::sortBySpecificity. This sorting method now uses MimeType::isLessSpecific in combination a bubble sort algorithm (which does not require a transitive compare function). Closes gh-27580
This commit is contained in:
parent
a7789db067
commit
6d9136013e
|
@ -26,7 +26,6 @@ import java.util.Collections;
|
|||
import java.util.Comparator;
|
||||
import java.util.Iterator;
|
||||
import java.util.LinkedHashMap;
|
||||
import java.util.List;
|
||||
import java.util.Locale;
|
||||
import java.util.Map;
|
||||
import java.util.TreeSet;
|
||||
|
@ -529,7 +528,6 @@ public class MimeType implements Comparable<MimeType>, Serializable {
|
|||
/**
|
||||
* Compares this MIME Type to another alphabetically.
|
||||
* @param other the MIME Type to compare to
|
||||
* @see MimeTypeUtils#sortBySpecificity(List)
|
||||
*/
|
||||
@Override
|
||||
public int compareTo(MimeType other) {
|
||||
|
@ -592,6 +590,88 @@ public class MimeType implements Comparable<MimeType>, Serializable {
|
|||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Indicates whether this {@code MimeType} is more specific than the given
|
||||
* type.
|
||||
* <ol>
|
||||
* <li>if this mime type has a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does not, then this method returns {@code false}.</li>
|
||||
* <li>if this mime type does not have a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does, then this method returns {@code true}.</li>
|
||||
* <li>if this mime type has a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does not, then this method returns {@code false}.</li>
|
||||
* <li>if this mime type does not have a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does, then this method returns {@code true}.</li>
|
||||
* <li>if the two mime types have identical {@linkplain #getType() type} and
|
||||
* {@linkplain #getSubtype() subtype}, then the mime type with the most
|
||||
* parameters is more specific than the other.</li>
|
||||
* <li>Otherwise, this method returns {@code false}.</li>
|
||||
* </ol>
|
||||
* @param other the {@code MimeType} to be compared
|
||||
* @return the result of the comparison
|
||||
* @since 6.0
|
||||
* @see #isLessSpecific(MimeType)
|
||||
* @see <a href="https://tools.ietf.org/html/rfc7231#section-5.3.2">HTTP 1.1: Semantics
|
||||
* and Content, section 5.3.2</a>
|
||||
*/
|
||||
public boolean isMoreSpecific(MimeType other) {
|
||||
Assert.notNull(other, "Other must not be null");
|
||||
boolean thisWildcard = isWildcardType();
|
||||
boolean otherWildcard = other.isWildcardType();
|
||||
if (thisWildcard && !otherWildcard) { // */* > audio/*
|
||||
return false;
|
||||
}
|
||||
else if (!thisWildcard && otherWildcard) { // audio/* < */*
|
||||
return true;
|
||||
}
|
||||
else {
|
||||
boolean thisWildcardSubtype = isWildcardSubtype();
|
||||
boolean otherWildcardSubtype = other.isWildcardSubtype();
|
||||
if (thisWildcardSubtype && !otherWildcardSubtype) { // audio/* > audio/basic
|
||||
return false;
|
||||
}
|
||||
else if (!thisWildcardSubtype && otherWildcardSubtype) { // audio/basic < audio/*
|
||||
return true;
|
||||
}
|
||||
else if (getType().equals(other.getType()) && getSubtype().equals(other.getSubtype())) {
|
||||
int paramsSize1 = getParameters().size();
|
||||
int paramsSize2 = other.getParameters().size();
|
||||
return paramsSize1 > paramsSize2;
|
||||
}
|
||||
else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Indicates whether this {@code MimeType} is more less than the given type.
|
||||
* <ol>
|
||||
* <li>if this mime type has a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does not, then this method returns {@code true}.</li>
|
||||
* <li>if this mime type does not have a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does, then this method returns {@code false}.</li>
|
||||
* <li>if this mime type has a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does not, then this method returns {@code true}.</li>
|
||||
* <li>if this mime type does not have a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does, then this method returns {@code false}.</li>
|
||||
* <li>if the two mime types have identical {@linkplain #getType() type} and
|
||||
* {@linkplain #getSubtype() subtype}, then the mime type with the least
|
||||
* parameters is less specific than the other.</li>
|
||||
* <li>Otherwise, this method returns {@code false}.</li>
|
||||
* </ol>
|
||||
* @param other the {@code MimeType} to be compared
|
||||
* @return the result of the comparison
|
||||
* @since 6.0
|
||||
* @see #isMoreSpecific(MimeType)
|
||||
* @see <a href="https://tools.ietf.org/html/rfc7231#section-5.3.2">HTTP 1.1: Semantics
|
||||
* and Content, section 5.3.2</a>
|
||||
*/
|
||||
public boolean isLessSpecific(MimeType other) {
|
||||
Assert.notNull(other, "Other must not be null");
|
||||
return other.isMoreSpecific(this);
|
||||
}
|
||||
|
||||
private void readObject(ObjectInputStream ois) throws IOException, ClassNotFoundException {
|
||||
// Rely on default serialization, just initialize state after deserialization.
|
||||
ois.defaultReadObject();
|
||||
|
@ -625,7 +705,9 @@ public class MimeType implements Comparable<MimeType>, Serializable {
|
|||
* Comparator to sort {@link MimeType MimeTypes} in order of specificity.
|
||||
*
|
||||
* @param <T> the type of mime types that may be compared by this comparator
|
||||
* @deprecated As of 6.0, with no direct replacement
|
||||
*/
|
||||
@Deprecated
|
||||
public static class SpecificityComparator<T extends MimeType> implements Comparator<T> {
|
||||
|
||||
@Override
|
||||
|
|
|
@ -28,6 +28,7 @@ import java.util.LinkedHashMap;
|
|||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.Random;
|
||||
import java.util.function.BiPredicate;
|
||||
import java.util.stream.Collectors;
|
||||
|
||||
import org.springframework.lang.Nullable;
|
||||
|
@ -51,8 +52,10 @@ public abstract class MimeTypeUtils {
|
|||
'V', 'W', 'X', 'Y', 'Z'};
|
||||
|
||||
/**
|
||||
* Comparator used by {@link #sortBySpecificity(List)}.
|
||||
* Comparator formally used by {@link #sortBySpecificity(List)}.
|
||||
* @deprecated As of 6.0, with no direct replacement
|
||||
*/
|
||||
@Deprecated
|
||||
public static final Comparator<MimeType> SPECIFICITY_COMPARATOR = new MimeType.SpecificityComparator<>();
|
||||
|
||||
/**
|
||||
|
@ -334,37 +337,52 @@ public abstract class MimeTypeUtils {
|
|||
}
|
||||
|
||||
/**
|
||||
* Sorts the given list of {@code MimeType} objects by specificity.
|
||||
* <p>Given two mime types:
|
||||
* <ol>
|
||||
* <li>if either mime type has a {@linkplain MimeType#isWildcardType() wildcard type},
|
||||
* then the mime type without the wildcard is ordered before the other.</li>
|
||||
* <li>if the two mime types have different {@linkplain MimeType#getType() types},
|
||||
* then they are considered equal and remain their current order.</li>
|
||||
* <li>if either mime type has a {@linkplain MimeType#isWildcardSubtype() wildcard subtype}
|
||||
* , then the mime type without the wildcard is sorted before the other.</li>
|
||||
* <li>if the two mime types have different {@linkplain MimeType#getSubtype() subtypes},
|
||||
* then they are considered equal and remain their current order.</li>
|
||||
* <li>if the two mime types have a different amount of
|
||||
* {@linkplain MimeType#getParameter(String) parameters}, then the mime type with the most
|
||||
* parameters is ordered before the other.</li>
|
||||
* </ol>
|
||||
* <p>For example: <blockquote>audio/basic < audio/* < */*</blockquote>
|
||||
* <blockquote>audio/basic;level=1 < audio/basic</blockquote>
|
||||
* <blockquote>audio/basic == text/html</blockquote> <blockquote>audio/basic ==
|
||||
* audio/wave</blockquote>
|
||||
* Sorts the given list of {@code MimeType} objects by
|
||||
* {@linkplain MimeType#isMoreSpecific(MimeType) specificity}.
|
||||
*
|
||||
* <p>Because of the computational cost, this method throws an exception
|
||||
* when the given list contains too many elements.
|
||||
* @param mimeTypes the list of mime types to be sorted
|
||||
* @throws IllegalArgumentException if {@code mimeTypes} contains more
|
||||
* than 50 elements
|
||||
* @see <a href="https://tools.ietf.org/html/rfc7231#section-5.3.2">HTTP 1.1: Semantics
|
||||
* and Content, section 5.3.2</a>
|
||||
* @see MimeType#isMoreSpecific(MimeType)
|
||||
*/
|
||||
public static void sortBySpecificity(List<MimeType> mimeTypes) {
|
||||
public static <T extends MimeType> void sortBySpecificity(List<T> mimeTypes) {
|
||||
Assert.notNull(mimeTypes, "'mimeTypes' must not be null");
|
||||
if (mimeTypes.size() > 1) {
|
||||
mimeTypes.sort(SPECIFICITY_COMPARATOR);
|
||||
Assert.isTrue(mimeTypes.size() <= 50, "Too many elements");
|
||||
|
||||
bubbleSort(mimeTypes, MimeType::isLessSpecific);
|
||||
}
|
||||
|
||||
static <T> void bubbleSort(List<T> list, BiPredicate<? super T, ? super T> swap) {
|
||||
int len = list.size();
|
||||
for (int i = 0; i < len; i++) {
|
||||
for (int j = 1; j < len - i ; j++) {
|
||||
T prev = list.get(j - 1);
|
||||
T cur = list.get(j);
|
||||
if (swap.test(prev, cur)) {
|
||||
list.set(j, prev);
|
||||
list.set(j - 1, cur);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Generate a random MIME boundary as bytes, often used in multipart mime types.
|
||||
*/
|
||||
public static byte[] generateMultipartBoundary() {
|
||||
Random randomToUse = initRandom();
|
||||
byte[] boundary = new byte[randomToUse.nextInt(11) + 30];
|
||||
for (int i = 0; i < boundary.length; i++) {
|
||||
boundary[i] = BOUNDARY_CHARS[randomToUse.nextInt(BOUNDARY_CHARS.length)];
|
||||
}
|
||||
return boundary;
|
||||
}
|
||||
|
||||
/**
|
||||
* Lazily initialize the {@link SecureRandom} for {@link #generateMultipartBoundary()}.
|
||||
*/
|
||||
|
@ -382,17 +400,6 @@ public abstract class MimeTypeUtils {
|
|||
return randomToUse;
|
||||
}
|
||||
|
||||
/**
|
||||
* Generate a random MIME boundary as bytes, often used in multipart mime types.
|
||||
*/
|
||||
public static byte[] generateMultipartBoundary() {
|
||||
Random randomToUse = initRandom();
|
||||
byte[] boundary = new byte[randomToUse.nextInt(11) + 30];
|
||||
for (int i = 0; i < boundary.length; i++) {
|
||||
boundary[i] = BOUNDARY_CHARS[randomToUse.nextInt(BOUNDARY_CHARS.length)];
|
||||
}
|
||||
return boundary;
|
||||
}
|
||||
|
||||
/**
|
||||
* Generate a random MIME boundary as String, often used in multipart mime types.
|
||||
|
|
|
@ -402,6 +402,69 @@ class MimeTypeTests {
|
|||
assertThat(m2.compareTo(m1) != 0).as("Invalid comparison result").isTrue();
|
||||
}
|
||||
|
||||
@Test
|
||||
void isMoreSpecific() {
|
||||
MimeType audioBasic = new MimeType("audio", "basic");
|
||||
MimeType audio = new MimeType("audio");
|
||||
MimeType audioWave = new MimeType("audio", "wave");
|
||||
MimeType audioBasicLevel = new MimeType("audio", "basic", singletonMap("level", "1"));
|
||||
|
||||
assertThat(audioBasic.isMoreSpecific(audioBasicLevel)).isFalse();
|
||||
assertThat(audioBasicLevel.isMoreSpecific(audioBasic)).isTrue();
|
||||
|
||||
assertThat(audio.isMoreSpecific(MimeTypeUtils.ALL)).isTrue();
|
||||
assertThat(MimeTypeUtils.ALL.isMoreSpecific(audio)).isFalse();
|
||||
|
||||
assertThat(audioBasicLevel.isMoreSpecific(audioBasic)).isTrue();
|
||||
assertThat(audioBasic.isMoreSpecific(audioBasicLevel)).isFalse();
|
||||
|
||||
assertThat(audioBasic.isMoreSpecific(MimeTypeUtils.TEXT_HTML)).isFalse();
|
||||
assertThat(audioBasic.isMoreSpecific(audioWave)).isFalse();
|
||||
assertThat(audioBasicLevel.isMoreSpecific(MimeTypeUtils.TEXT_HTML)).isFalse();
|
||||
}
|
||||
|
||||
@Test
|
||||
void isLessSpecific() {
|
||||
MimeType audioBasic = new MimeType("audio", "basic");
|
||||
MimeType audio = new MimeType("audio");
|
||||
MimeType audioWave = new MimeType("audio", "wave");
|
||||
MimeType audioBasicLevel = new MimeType("audio", "basic", singletonMap("level", "1"));
|
||||
|
||||
assertThat(audioBasic.isLessSpecific(audioBasicLevel)).isTrue();
|
||||
assertThat(audioBasicLevel.isLessSpecific(audioBasic)).isFalse();
|
||||
|
||||
assertThat(audio.isLessSpecific(MimeTypeUtils.ALL)).isFalse();
|
||||
assertThat(MimeTypeUtils.ALL.isLessSpecific(audio)).isTrue();
|
||||
|
||||
assertThat(audioBasicLevel.isLessSpecific(audioBasic)).isFalse();
|
||||
assertThat(audioBasic.isLessSpecific(audioBasicLevel)).isTrue();
|
||||
|
||||
assertThat(audioBasic.isLessSpecific(MimeTypeUtils.TEXT_HTML)).isFalse();
|
||||
assertThat(audioBasic.isLessSpecific(audioWave)).isFalse();
|
||||
assertThat(audioBasicLevel.isLessSpecific(MimeTypeUtils.TEXT_HTML)).isFalse();
|
||||
}
|
||||
|
||||
@Test
|
||||
void sortBySpecificity() {
|
||||
MimeType audioBasic = new MimeType("audio", "basic");
|
||||
MimeType audio = new MimeType("audio");
|
||||
MimeType audioWave = new MimeType("audio", "wave");
|
||||
MimeType audioBasicLevel = new MimeType("audio", "basic", singletonMap("level", "1"));
|
||||
|
||||
List<MimeType> mimeTypes = new ArrayList<>(List.of(MimeTypeUtils.ALL, audio, audioWave, audioBasic,
|
||||
audioBasicLevel));
|
||||
MimeTypeUtils.sortBySpecificity(mimeTypes);
|
||||
|
||||
assertThat(mimeTypes).containsExactly(audioWave, audioBasicLevel, audioBasic, audio, MimeTypeUtils.ALL);
|
||||
}
|
||||
|
||||
@Test
|
||||
void bubbleSort() {
|
||||
List<Integer> list = new ArrayList<>(List.of(10, 9, 8, 7, 6, 5, 4, 3, 2, 1));
|
||||
MimeTypeUtils.bubbleSort(list, (i1, i2) -> i1 > i2);
|
||||
assertThat(list).containsExactly(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
|
||||
}
|
||||
|
||||
@Test // SPR-13157
|
||||
void equalsIsCaseInsensitiveForCharsets() {
|
||||
MimeType m1 = new MimeType("text", "plain", singletonMap("charset", "UTF-8"));
|
||||
|
|
|
@ -534,6 +534,82 @@ public class MediaType extends MimeType implements Serializable {
|
|||
return (qualityFactor != null ? Double.parseDouble(unquote(qualityFactor)) : 1D);
|
||||
}
|
||||
|
||||
/**
|
||||
* Indicates whether this {@code MediaType} more specific than the given type.
|
||||
* <ol>
|
||||
* <li>if this media type has a {@linkplain #getQualityValue() quality factor} higher than the other,
|
||||
* then this method returns {@code true}.</li>
|
||||
* <li>if this media type has a {@linkplain #getQualityValue() quality factor} lower than the other,
|
||||
* then this method returns {@code false}.</li>
|
||||
* <li>if this mime type has a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does not, then this method returns {@code false}.</li>
|
||||
* <li>if this mime type does not have a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does, then this method returns {@code true}.</li>
|
||||
* <li>if this mime type has a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does not, then this method returns {@code false}.</li>
|
||||
* <li>if this mime type does not have a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does, then this method returns {@code true}.</li>
|
||||
* <li>if the two mime types have identical {@linkplain #getType() type} and
|
||||
* {@linkplain #getSubtype() subtype}, then the mime type with the most
|
||||
* parameters is more specific than the other.</li>
|
||||
* <li>Otherwise, this method returns {@code false}.</li>
|
||||
* </ol>
|
||||
* @param other the {@code MimeType} to be compared
|
||||
* @return the result of the comparison
|
||||
* @since 6.0
|
||||
* @see #isLessSpecific(MimeType)
|
||||
* @see <a href="https://tools.ietf.org/html/rfc7231#section-5.3.2">HTTP 1.1: Semantics
|
||||
* and Content, section 5.3.2</a>
|
||||
*/
|
||||
@Override
|
||||
public boolean isMoreSpecific(MimeType other) {
|
||||
Assert.notNull(other, "Other must not be null");
|
||||
if (other instanceof MediaType otherMediaType) {
|
||||
double quality1 = getQualityValue();
|
||||
double quality2 = otherMediaType.getQualityValue();
|
||||
if (quality1 > quality2) {
|
||||
return true;
|
||||
}
|
||||
else if (quality1 < quality2) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return super.isMoreSpecific(other);
|
||||
}
|
||||
|
||||
/**
|
||||
* Indicates whether this {@code MediaType} more specific than the given type.
|
||||
* <ol>
|
||||
* <li>if this media type has a {@linkplain #getQualityValue() quality factor} higher than the other,
|
||||
* then this method returns {@code false}.</li>
|
||||
* <li>if this media type has a {@linkplain #getQualityValue() quality factor} lower than the other,
|
||||
* then this method returns {@code true}.</li>
|
||||
* <li>if this mime type has a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does not, then this method returns {@code true}.</li>
|
||||
* <li>if this mime type does not have a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does, then this method returns {@code false}.</li>
|
||||
* <li>if this mime type has a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does not, then this method returns {@code true}.</li>
|
||||
* <li>if this mime type does not have a {@linkplain #isWildcardType() wildcard type},
|
||||
* and the other does, then this method returns {@code false}.</li>
|
||||
* <li>if the two mime types have identical {@linkplain #getType() type} and
|
||||
* {@linkplain #getSubtype() subtype}, then the mime type with the least
|
||||
* parameters is less specific than the other.</li>
|
||||
* <li>Otherwise, this method returns {@code false}.</li>
|
||||
* </ol>
|
||||
* @param other the {@code MimeType} to be compared
|
||||
* @return the result of the comparison
|
||||
* @since 6.0
|
||||
* @see #isMoreSpecific(MimeType)
|
||||
* @see <a href="https://tools.ietf.org/html/rfc7231#section-5.3.2">HTTP 1.1: Semantics
|
||||
* and Content, section 5.3.2</a>
|
||||
*/
|
||||
@Override
|
||||
public boolean isLessSpecific(MimeType other) {
|
||||
Assert.notNull(other, "Other must not be null");
|
||||
return other.isMoreSpecific(this);
|
||||
}
|
||||
|
||||
/**
|
||||
* Indicate whether this {@code MediaType} includes the given media type.
|
||||
* <p>For instance, {@code text/*} includes {@code text/plain} and {@code text/html},
|
||||
|
@ -731,9 +807,9 @@ public class MediaType extends MimeType implements Serializable {
|
|||
* <blockquote>audio/basic == text/html</blockquote>
|
||||
* <blockquote>audio/basic == audio/wave</blockquote>
|
||||
* @param mediaTypes the list of media types to be sorted
|
||||
* @see <a href="https://tools.ietf.org/html/rfc7231#section-5.3.2">HTTP 1.1: Semantics
|
||||
* and Content, section 5.3.2</a>
|
||||
* @deprecated As of 6.0, in favor of {@link MimeTypeUtils#sortBySpecificity(List)}
|
||||
*/
|
||||
@Deprecated
|
||||
public static void sortBySpecificity(List<MediaType> mediaTypes) {
|
||||
Assert.notNull(mediaTypes, "'mediaTypes' must not be null");
|
||||
if (mediaTypes.size() > 1) {
|
||||
|
@ -760,7 +836,9 @@ public class MediaType extends MimeType implements Serializable {
|
|||
* </ol>
|
||||
* @param mediaTypes the list of media types to be sorted
|
||||
* @see #getQualityValue()
|
||||
* @deprecated As of 6.0, with no direct replacement
|
||||
*/
|
||||
@Deprecated
|
||||
public static void sortByQualityValue(List<MediaType> mediaTypes) {
|
||||
Assert.notNull(mediaTypes, "'mediaTypes' must not be null");
|
||||
if (mediaTypes.size() > 1) {
|
||||
|
@ -771,9 +849,9 @@ public class MediaType extends MimeType implements Serializable {
|
|||
/**
|
||||
* Sorts the given list of {@code MediaType} objects by specificity as the
|
||||
* primary criteria and quality value the secondary.
|
||||
* @see MediaType#sortBySpecificity(List)
|
||||
* @see MediaType#sortByQualityValue(List)
|
||||
* @deprecated As of 6.0, in favor of {@link MimeTypeUtils#sortBySpecificity(List)}
|
||||
*/
|
||||
@Deprecated
|
||||
public static void sortBySpecificityAndQuality(List<MediaType> mediaTypes) {
|
||||
Assert.notNull(mediaTypes, "'mediaTypes' must not be null");
|
||||
if (mediaTypes.size() > 1) {
|
||||
|
@ -784,7 +862,9 @@ public class MediaType extends MimeType implements Serializable {
|
|||
|
||||
/**
|
||||
* Comparator used by {@link #sortByQualityValue(List)}.
|
||||
* @deprecated As of 6.0, with no direct replacement
|
||||
*/
|
||||
@Deprecated
|
||||
public static final Comparator<MediaType> QUALITY_VALUE_COMPARATOR = (mediaType1, mediaType2) -> {
|
||||
double quality1 = mediaType1.getQualityValue();
|
||||
double quality2 = mediaType2.getQualityValue();
|
||||
|
@ -822,7 +902,9 @@ public class MediaType extends MimeType implements Serializable {
|
|||
|
||||
/**
|
||||
* Comparator used by {@link #sortBySpecificity(List)}.
|
||||
* @deprecated As of 6.0, with no direct replacement
|
||||
*/
|
||||
@Deprecated
|
||||
public static final Comparator<MediaType> SPECIFICITY_COMPARATOR = new SpecificityComparator<>() {
|
||||
|
||||
@Override
|
||||
|
|
|
@ -28,6 +28,7 @@ import org.junit.jupiter.api.Test;
|
|||
import org.springframework.core.convert.ConversionService;
|
||||
import org.springframework.core.convert.support.DefaultConversionService;
|
||||
import org.springframework.core.testfixture.io.SerializationTestUtils;
|
||||
import org.springframework.util.MimeTypeUtils;
|
||||
|
||||
import static org.assertj.core.api.Assertions.assertThat;
|
||||
import static org.assertj.core.api.Assertions.assertThatExceptionOfType;
|
||||
|
@ -240,6 +241,44 @@ public class MediaTypeTests {
|
|||
|
||||
}
|
||||
|
||||
@Test
|
||||
void isMoreSpecific() {
|
||||
MediaType audio = new MediaType("audio");
|
||||
MediaType audioBasic = new MediaType("audio", "basic");
|
||||
MediaType audioBasic07 = new MediaType("audio", "basic", 0.7);
|
||||
MediaType audioBasic03 = new MediaType("audio", "basic", 0.3);
|
||||
|
||||
assertThat(audioBasic.isMoreSpecific(audio)).isTrue();
|
||||
assertThat(audio.isMoreSpecific(audioBasic)).isFalse();
|
||||
|
||||
assertThat(audio.isMoreSpecific(audioBasic07)).isTrue();
|
||||
assertThat(audioBasic07.isMoreSpecific(audio)).isFalse();
|
||||
|
||||
assertThat(audioBasic07.isMoreSpecific(audioBasic03)).isTrue();
|
||||
assertThat(audioBasic03.isMoreSpecific(audioBasic07)).isFalse();
|
||||
|
||||
assertThat(audioBasic.isMoreSpecific(MediaType.TEXT_HTML)).isFalse();
|
||||
}
|
||||
|
||||
@Test
|
||||
void isLessSpecific() {
|
||||
MediaType audio = new MediaType("audio");
|
||||
MediaType audioBasic = new MediaType("audio", "basic");
|
||||
MediaType audioBasic07 = new MediaType("audio", "basic", 0.7);
|
||||
MediaType audioBasic03 = new MediaType("audio", "basic", 0.3);
|
||||
|
||||
assertThat(audioBasic.isLessSpecific(audio)).isFalse();
|
||||
assertThat(audio.isLessSpecific(audioBasic)).isTrue();
|
||||
|
||||
assertThat(audio.isLessSpecific(audioBasic07)).isFalse();
|
||||
assertThat(audioBasic07.isLessSpecific(audio)).isTrue();
|
||||
|
||||
assertThat(audioBasic07.isLessSpecific(audioBasic03)).isFalse();
|
||||
assertThat(audioBasic03.isLessSpecific(audioBasic07)).isTrue();
|
||||
|
||||
assertThat(audioBasic.isLessSpecific(MediaType.TEXT_HTML)).isFalse();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void specificityComparator() throws Exception {
|
||||
MediaType audioBasic = new MediaType("audio", "basic");
|
||||
|
@ -470,4 +509,33 @@ public class MediaTypeTests {
|
|||
assertThat(original).isEqualTo(deserialized);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void sortBySpecificity() {
|
||||
MediaType audioBasic = new MediaType("audio", "basic");
|
||||
MediaType audio = new MediaType("audio");
|
||||
MediaType audio03 = new MediaType("audio", "*", 0.3);
|
||||
MediaType audio07 = new MediaType("audio", "*", 0.7);
|
||||
MediaType audioBasicLevel = new MediaType("audio", "basic", Collections.singletonMap("level", "1"));
|
||||
MediaType all = MediaType.ALL;
|
||||
|
||||
List<MediaType> expected = new ArrayList<>();
|
||||
expected.add(audioBasicLevel);
|
||||
expected.add(audioBasic);
|
||||
expected.add(audio);
|
||||
expected.add(all);
|
||||
expected.add(audio07);
|
||||
expected.add(audio03);
|
||||
|
||||
List<MediaType> result = new ArrayList<>(expected);
|
||||
Random rnd = new Random();
|
||||
// shuffle & sort 10 times
|
||||
for (int i = 0; i < 10; i++) {
|
||||
Collections.shuffle(result, rnd);
|
||||
MimeTypeUtils.sortBySpecificity(result);
|
||||
|
||||
assertThat(result).containsExactlyElementsOf(expected);
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue