All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.eclipse.rdf4j.common.webapp.util.HttpServerUtil Maven / Gradle / Ivy

/*******************************************************************************
 * Copyright (c) 2015 Eclipse RDF4J contributors, Aduna, and others.
 *
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Distribution License v1.0
 * which accompanies this distribution, and is available at
 * http://www.eclipse.org/org/documents/edl-v10.php.
 *
 * SPDX-License-Identifier: BSD-3-Clause
 *******************************************************************************/
package org.eclipse.rdf4j.common.webapp.util;

import java.util.ArrayList;
import java.util.Enumeration;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import javax.servlet.http.HttpServletRequest;

public class HttpServerUtil {

	/**
	 * Extracts the MIME type from the specified content type string. This method parses the content type string and
	 * returns just the MIME type, ignoring any parameters that are included.
	 *
	 * @param contentType A content type string, e.g. application/xml; charset=utf-8 .
	 * @return The MIME type part of the specified content type string, or null if the specified content type
	 *         string was null.
	 */
	public static String getMIMEType(String contentType) {
		if (contentType == null) {
			return null;
		}

		return HeaderElement.parse(contentType).getValue();
	}

	/**
	 * Selects from a set of MIME types, the MIME type that has the highest quality score when matched with the Accept
	 * headers in the supplied request.
	 *
	 * @param mimeTypes The set of available MIME types.
	 * @param request   The request to match the MIME types against.
	 * @return The MIME type that best matches the types that the client finds acceptable, or null in case no
	 *         acceptable MIME type could be found.
	 */
	public static String selectPreferredMIMEType(Iterator mimeTypes, HttpServletRequest request) {
		List acceptElements = getHeaderElements(request, "Accept");

		if (acceptElements.isEmpty()) {
			// Client does not specify any requirements, return first MIME type
			// from the list
			if (mimeTypes.hasNext()) {
				return mimeTypes.next();
			} else {
				return null;
			}
		}

		String result = null;
		HeaderElement matchingAcceptType = null;

		double highestQuality = 0.0;

		while (mimeTypes.hasNext()) {
			String mimeType = mimeTypes.next();
			HeaderElement acceptType = matchAcceptHeader(mimeType, acceptElements);

			if (acceptType != null) {
				// quality defaults to 1.0
				double quality = 1.0;

				String qualityStr = acceptType.getParameterValue("q");
				if (qualityStr != null) {
					try {
						quality = Double.parseDouble(qualityStr);
					} catch (NumberFormatException e) {
						// Illegal quality value, assume it has a different meaning
						// and ignore it
					}
				}

				if (quality > highestQuality) {
					result = mimeType;
					matchingAcceptType = acceptType;
					highestQuality = quality;
				} else if (quality == highestQuality) {
					// found a match with equal quality preference. check if the
					// accept type is more specific
					// than the previous match.
					if (isMoreSpecificType(acceptType, matchingAcceptType)) {
						result = mimeType;
						matchingAcceptType = acceptType;
					}
				}
			}
		}

		return result;
	}

	/**
	 * Checks if the first supplied MIME type is more specific than the second supplied MIME type.
	 *
	 * @param leftMimeTypeElem
	 * @param rightMimeTypeElem
	 * @return true iff leftMimeTypeElem is a more specific MIME type spec than rightMimeTypeElem, false otherwise.
	 */
	private static boolean isMoreSpecificType(HeaderElement leftMimeTypeElem, HeaderElement rightMimeTypeElem) {

		String[] leftMimeType = splitMIMEType(leftMimeTypeElem.getValue());
		String[] rightMimeType = splitMIMEType(rightMimeTypeElem.getValue());

		if (rightMimeType != null) {
			if (rightMimeType[1].equals("*")) {
				if (!leftMimeType[1].equals("*")) {
					return true;
				}
			}
			if (rightMimeType[0].equals("*")) {
				if (!leftMimeType[0].equals("*")) {
					return true;
				}
			}

			return false;
		} else {
			return true;
		}
	}

	private static String[] splitMIMEType(String mimeTypeString) {
		int slashIdx = mimeTypeString.indexOf('/');
		if (slashIdx > 0) {
			String type = mimeTypeString.substring(0, slashIdx);
			String subType = mimeTypeString.substring(slashIdx + 1);
			return new String[] { type, subType };
		} else {
			// invalid mime type
			return null;
		}
	}

	/**
	 * Gets the elements of the request header with the specified name.
	 *
	 * @param request    The request to get the header from.
	 * @param headerName The name of the header to get the elements of.
	 * @return A List of {@link HeaderElement} objects.
	 */
	public static List getHeaderElements(HttpServletRequest request, String headerName) {
		List elemList = new ArrayList<>(8);

		@SuppressWarnings("unchecked")
		Enumeration headerValues = request.getHeaders(headerName);
		while (headerValues.hasMoreElements()) {
			String value = headerValues.nextElement();

			List subValues = splitHeaderString(value, ',');

			for (String subValue : subValues) {
				// Ignore any empty header elements
				subValue = subValue.trim();
				if (!subValue.isEmpty()) {
					elemList.add(HeaderElement.parse(subValue));
				}
			}
		}

		return elemList;
	}

	/**
	 * Splits the supplied string into sub parts using the specified splitChar as a separator, ignoring occurrences of
	 * this character inside quoted strings.
	 *
	 * @param s         The header string to split into sub parts.
	 * @param splitChar The character to use as separator.
	 * @return A List of Strings.
	 */
	public static List splitHeaderString(String s, char splitChar) {
		List result = new ArrayList<>(8);

		boolean parsingQuotedString = false;
		int i, startIdx = 0;

		for (i = 0; i < s.length(); i++) {
			char c = s.charAt(i);

			if (c == splitChar && !parsingQuotedString) {
				result.add(s.substring(startIdx, i));
				startIdx = i + 1;
			} else if (c == '"') {
				parsingQuotedString = !parsingQuotedString;
			}
		}

		if (startIdx < s.length()) {
			result.add(s.substring(startIdx));
		}

		return result;
	}

	/**
	 * Tries to match the specified MIME type spec against the list of Accept header elements, returning the applicable
	 * header element if available.
	 *
	 * @param mimeTypeSpec   The MIME type to determine the quality for, e.g. "text/plain" or "application/xml;
	 *                       charset=utf-8".
	 * @param acceptElements A List of {@link HeaderElement} objects.
	 * @return The Accept header element that matches the MIME type spec most closely, or null if no such
	 *         header element could be found.
	 */
	public static HeaderElement matchAcceptHeader(String mimeTypeSpec, List acceptElements) {
		HeaderElement mimeTypeElem = HeaderElement.parse(mimeTypeSpec);

		while (mimeTypeElem != null) {
			for (HeaderElement acceptElem : acceptElements) {
				if (matchesAcceptHeader(mimeTypeElem, acceptElem)) {
					return acceptElem;
				}
			}

			// No match found, generalize the MIME type spec and try again
			mimeTypeElem = generalizeMIMEType(mimeTypeElem);
		}

		return null;
	}

	private static boolean matchesAcceptHeader(HeaderElement mimeTypeElem, HeaderElement acceptElem) {
		if (!mimeTypeElem.getValue().equals(acceptElem.getValue())) {
			return false;
		}

		// Values match, check parameters
		if (mimeTypeElem.getParameterCount() > acceptElem.getParameterCount()) {
			return false;
		}

		for (int i = 0; i < mimeTypeElem.getParameterCount(); i++) {
			if (!mimeTypeElem.getParameter(i).equals(acceptElem.getParameter(i))) {
				return false;
			}
		}

		return true;
	}

	/**
	 * Generalizes a MIME type element. The following steps are taken for generalization:
	 * 
    *
  • If the MIME type element has one or more parameters, the last parameter is removed. *
  • Otherwise, if the MIME type element's subtype is not equal to '*' then it is set to this value. *
  • Otherwise, if the MIME type element's type is not equal to '*' then it is set to this value. *
  • Otherwise, the MIME type is equal to "*&slash;*" and cannot be generalized any further; null is * returned. *
*

* Example generalizations: *

* * * * * * * * * * * * * * * * * * * * * *
inputresult
application/xml; charset=utf-8application/xml
application/xmlapplication/*
application/*&slash;*
&slash;*null
* * @param mimeTypeElem The MIME type element that should be generalized. * @return The generalized MIME type element, or null if it could not be generalized any further. */ private static HeaderElement generalizeMIMEType(HeaderElement mimeTypeElem) { int parameterCount = mimeTypeElem.getParameterCount(); if (parameterCount > 0) { // remove last parameter mimeTypeElem.removeParameter(parameterCount - 1); } else { String mimeType = mimeTypeElem.getValue(); int slashIdx = mimeType.indexOf('/'); if (slashIdx > 0) { String type = mimeType.substring(0, slashIdx); String subType = mimeType.substring(slashIdx + 1); if (!subType.equals("*")) { // generalize subtype mimeTypeElem.setValue(type + "/*"); } else if (!type.equals("*")) { // generalize type mimeTypeElem.setValue("*/*"); } else { // Cannot generalize any further mimeTypeElem = null; } } else { // invalid MIME type mimeTypeElem = null; } } return mimeTypeElem; } /** * Gets the trimmed value of a request parameter as a String. * * @return The trimmed value, or null if the parameter does not exist. */ public static String getPostDataParameter(Map formData, String name) { String result = null; try { Object param = formData.get(name); if (param instanceof String[]) { String[] paramArray = (String[]) param; if (paramArray.length > 0) { result = paramArray[0]; } } else if (param instanceof String) { result = (String) param; } if (result != null) { result = result.trim(); } } catch (ClassCastException cce) { // ignore, return null } return result; } /** * @return true if the string is either null or equal to "" */ public static boolean isEmpty(String string) { boolean result = false; if (string == null || string.trim().isEmpty()) { result = true; } return result; } /** * @return true if the string is !isEmpty and equal to "true" */ public static boolean isTrue(String string) { boolean result = false; if (!isEmpty(string) && (string.equalsIgnoreCase("true") || string.equalsIgnoreCase("on"))) { result = true; } return result; } /** * @return true if the string is !isEmpty and equal to "false" */ public static boolean isFalse(String string) { boolean result = false; if (!isEmpty(string) && (string.equalsIgnoreCase("false") || string.equalsIgnoreCase("off"))) { result = true; } return result; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy