blade.kit.MimeParse Maven / Gradle / Ivy
/**
* Copyright (c) 2015, biezhi 王爵 ([email protected])
*
* 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
*
* http://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 blade.kit;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
/**
* MIME-Type解析类
*
* @author biezhi
* @since 1.0
*/
public class MimeParse {
/**
* Constant for no mime type
*/
public static final String NO_MIME_TYPE = "";
/**
* Parse results container
*/
private static class ParseResults {
String type;
String subType;
// !a dictionary of all the parameters for the media range
Map params;
@Override
public String toString() {
StringBuffer s = new StringBuffer("('" + type + "', '" + subType + "', {");
for (String k : params.keySet()) {
s.append("'" + k + "':'" + params.get(k) + "',");
}
return s.append("})").toString();
}
}
/**
* Carves up a mime-type and returns a ParseResults object
* For example, the media range 'application/xhtml;q=0.5' would get parsed
* into:
* ('application', 'xhtml', {'q', '0.5'})
*/
private static ParseResults parseMimeType(String mimeType) {
String[] parts = StringKit.split(mimeType, ";");
ParseResults results = new ParseResults();
results.params = new HashMap();
for (int i = 1; i < parts.length; ++i) {
String p = parts[i];
String[] subParts = StringKit.split(p, "=");
if (subParts.length == 2) {
results.params.put(subParts[0].trim(), subParts[1].trim());
}
}
String fullType = parts[0].trim();
// Java URLConnection class sends an Accept header that includes a
// single "*" - Turn it into a legal wildcard.
if (fullType.equals("*")) {
fullType = "*/*";
}
int slashIndex = fullType.indexOf('/');
if (slashIndex != -1) {
results.type = fullType.substring(0, slashIndex);
results.subType = fullType.substring(slashIndex + 1);
} else {
//If the type is invalid, attempt to turn into a wildcard
results.type = fullType;
results.subType = "*";
}
return results;
}
/**
* Carves up a media range and returns a ParseResults.
* For example, the media range 'application/*;q=0.5' would get parsed into:
* ('application', '*', {'q', '0.5'})
* In addition this function also guarantees that there is a value for 'q'
* in the params dictionary, filling it in with a proper default if
* necessary.
*
* @param range
*/
private static ParseResults parseMediaRange(String range) {
ParseResults results = parseMimeType(range);
String q = results.params.get("q");
float f = toFloat(q, 1);
if (isBlank(q) || f < 0 || f > 1) {
results.params.put("q", "1");
}
return results;
}
/**
* Structure for holding a fitness/quality combo
*/
private static class FitnessAndQuality implements Comparable {
int fitness;
float quality;
String mimeType; // optionally used
private FitnessAndQuality(int fitness, float quality) {
this.fitness = fitness;
this.quality = quality;
}
public int compareTo(FitnessAndQuality o) {
if (fitness == o.fitness) {
if (quality == o.quality) {
return 0;
} else {
return quality < o.quality ? -1 : 1;
}
} else {
return fitness < o.fitness ? -1 : 1;
}
}
}
/**
* Find the best match for a given mimeType against a list of media_ranges
* that have already been parsed by MimeParse.parseMediaRange(). Returns a
* tuple of the fitness value and the value of the 'q' quality parameter of
* the best match, or (-1, 0) if no match was found. Just as for
* quality_parsed(), 'parsed_ranges' must be a list of parsed media ranges.
*
* @param mimeType
* @param parsedRanges
*/
private static FitnessAndQuality fitnessAndQualityParsed(String mimeType, Collection parsedRanges) {
int bestFitness = -1;
float bestFitQ = 0;
ParseResults target = parseMediaRange(mimeType);
for (ParseResults range : parsedRanges) {
if ((target.type.equals(range.type) || range.type.equals("*") || target.type.equals("*"))
&& (target.subType.equals(range.subType) || range.subType.equals("*")
|| target.subType.equals("*"))) {
for (String k : target.params.keySet()) {
int paramMatches = 0;
if (!k.equals("q") && range.params.containsKey(k)
&& target.params.get(k).equals(range.params.get(k))) {
paramMatches++;
}
int fitness = (range.type.equals(target.type)) ? 100 : 0;
fitness += (range.subType.equals(target.subType)) ? 10 : 0;
fitness += paramMatches;
if (fitness > bestFitness) {
bestFitness = fitness;
bestFitQ = toFloat(range.params.get("q"), 0);
}
}
}
}
return new FitnessAndQuality(bestFitness, bestFitQ);
}
/**
* Finds best match
*
* @param supported the supported types
* @param header the header
* @return the best match
*/
public static String bestMatch(Collection supported, String header) {
List parseResults = new LinkedList();
List weightedMatches = new LinkedList();
String[] headers = StringKit.split(header, ",");
for (String r : headers) {
parseResults.add(parseMediaRange(r));
}
for (String s : supported) {
FitnessAndQuality fitnessAndQuality = fitnessAndQualityParsed(s, parseResults);
fitnessAndQuality.mimeType = s;
weightedMatches.add(fitnessAndQuality);
}
Collections.sort(weightedMatches);
FitnessAndQuality lastOne = weightedMatches.get(weightedMatches.size() - 1);
return Float.compare(lastOne.quality, 0) != 0 ? lastOne.mimeType : NO_MIME_TYPE;
}
private static boolean isBlank(String s) {
return s == null || "".equals(s.trim());
}
private static float toFloat(final String str, final float defaultValue) {
if (str == null) {
return defaultValue;
}
try {
return Float.parseFloat(str);
} catch (final NumberFormatException nfe) {
return defaultValue;
}
}
private MimeParse() {
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy