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

org.appng.api.support.HttpHeaderUtils Maven / Gradle / Ivy

/*
 * Copyright 2011-2021 the original author or authors.
 *
 * 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 org.appng.api.support;

import java.io.IOException;
import java.io.OutputStream;
import java.text.ParseException;
import java.util.Date;
import java.util.Enumeration;
import java.util.HashSet;
import java.util.List;
import java.util.Locale;
import java.util.Set;
import java.util.TimeZone;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.commons.collections.EnumerationUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.time.FastDateFormat;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpStatus;
import org.springframework.http.MediaType;
import org.springframework.http.RequestEntity;
import org.springframework.http.RequestEntity.BodyBuilder;

import lombok.extern.slf4j.Slf4j;

/**
 * Utility-class that helps dealing with several HTTP-Headers
 * 
 * @author Matthias Müller
 */
@Slf4j
public class HttpHeaderUtils {

	private static final String HTTP_DATE_FORMAT = "EEE, dd MMM yyyy HH:mm:ss zzz";
	protected static final FastDateFormat HTTP_DATE = FastDateFormat.getInstance(HTTP_DATE_FORMAT,
			TimeZone.getTimeZone("GMT"), Locale.ENGLISH);

	/**
	 * Handles the {@value org.springframework.http.HttpHeaders#LAST_MODIFIED} and
	 * {@value org.springframework.http.HttpHeaders#IF_MODIFIED_SINCE} headers for the give request/response pair. Sets
	 * the {@value org.springframework.http.HttpHeaders#LAST_MODIFIED}-header for the response and reads the
	 * {@value org.springframework.http.HttpHeaders#IF_MODIFIED_SINCE}-header from the request.
	 * 
	 * @param servletRequest
	 *                        the current {@link HttpServletRequest}
	 * @param servletResponse
	 *                        the current {@link HttpServletResponse}
	 * @param resource
	 *                        the {@link HttpResource} to process
	 * @param output
	 *                        whether or not to write the response data to the {@link OutputStream} of the
	 *                        {@link HttpServletResponse}, also setting the content-type and content length
	 * 
	 * @return the response-data, which may have a length of 0 in case
	 *         {@value org.springframework.http.HttpHeaders#IF_MODIFIED_SINCE} was set for the request and the
	 *         {@link HttpResource} has'nt been updated since then
	 * 
	 * @throws IOException
	 *                     if an error occurred while reading/updating the {@link HttpResource}
	 */
	public static byte[] handleModifiedHeaders(HttpServletRequest servletRequest, HttpServletResponse servletResponse,
			HttpResource resource, boolean output) throws IOException {
		byte[] result = new byte[0];

		String sessionId = servletRequest.getSession().getId();

		long lastModified = resource.update();

		Date ifModifiedSince = null;
		String ifModifiedSinceHeader = servletRequest.getHeader(HttpHeaders.IF_MODIFIED_SINCE);
		boolean hasModifiedSince = StringUtils.isNotBlank(ifModifiedSinceHeader);
		if (hasModifiedSince) {
			try {
				LOGGER.debug("[{}] received {}={} for {}?{}", sessionId, HttpHeaders.IF_MODIFIED_SINCE,
						ifModifiedSinceHeader, servletRequest.getServletPath(), servletRequest.getQueryString());
				ifModifiedSince = HTTP_DATE.parse(ifModifiedSinceHeader);
			} catch (ParseException e) {
				hasModifiedSince = false;
				LOGGER.debug("[{}] error parsing header {}={} for {}?{} ({})", sessionId, HttpHeaders.IF_MODIFIED_SINCE,
						ifModifiedSinceHeader, servletRequest.getServletPath(), servletRequest.getQueryString(),
						e.getMessage());
			}
		}

		boolean isModifiedAfter = hasModifiedSince && lastModified > ifModifiedSince.getTime();
		if (resource.needsUpdate() || isModifiedAfter || !hasModifiedSince) {
			result = resource.getData();
		}
		if (hasModifiedSince && !isModifiedAfter) {
			servletResponse.setStatus(HttpServletResponse.SC_NOT_MODIFIED);
			LOGGER.debug("[{}] setting status {} for {}?{}", sessionId, HttpServletResponse.SC_NOT_MODIFIED,
					servletRequest.getServletPath(), servletRequest.getQueryString());
		} else {
			String lastModifiedHeader = HTTP_DATE.format(new Date(lastModified));
			servletResponse.setHeader(HttpHeaders.LAST_MODIFIED, lastModifiedHeader);
			LOGGER.debug("[{}] setting {}={} for {}?{}", sessionId, HttpHeaders.LAST_MODIFIED, lastModifiedHeader,
					servletRequest.getServletPath(), servletRequest.getQueryString());
			if (output) {
				LOGGER.debug("[{}] setting content type {} ({}B) for {}?{}", sessionId, resource.getContentType(),
						result.length, servletRequest.getServletPath(), servletRequest.getQueryString());
				servletResponse.setContentType(resource.getContentType());
				servletResponse.setContentLength(result.length);
				servletResponse.getOutputStream().write(result);
			}
		}

		return result;
	}

	/**
	 * Handles the {@value org.springframework.http.HttpHeaders#LAST_MODIFIED} and
	 * {@value org.springframework.http.HttpHeaders#IF_MODIFIED_SINCE} headers for the give request/response pair. Sets
	 * the {@value org.springframework.http.HttpHeaders#LAST_MODIFIED}-header for the response and reads the
	 * {@value org.springframework.http.HttpHeaders#IF_MODIFIED_SINCE}-header from the request.
	 * 
	 * @param requestHeaders
	 *                        the headers the current {@link HttpServletRequest}
	 * @param responseHeaders
	 *                        the headers of the current {@link HttpServletResponse}
	 * @param resource
	 *                        the {@link HttpResource} to process
	 * 
	 * @return the response-data, which may have a length of 0 in case
	 *         {@value org.springframework.http.HttpHeaders#IF_MODIFIED_SINCE} was set for the request and the
	 *         {@link HttpResource} has'nt been updated since then
	 * 
	 * @throws IOException
	 */
	public static byte[] handleModifiedHeaders(HttpHeaders requestHeaders, HttpHeaders responseHeaders,
			HttpResource resource) throws IOException {
		byte[] result = new byte[0];
		long lastModified = resource.update();
		long ifModifiedSince = requestHeaders.getIfModifiedSince();
		if (ifModifiedSince > 0) {
			LOGGER.debug("received {}={}", HttpHeaders.IF_MODIFIED_SINCE, new Date(ifModifiedSince));
			if (ifModifiedSince > lastModified) {
				LOGGER.debug("setting status {}", HttpServletResponse.SC_NOT_MODIFIED);
				resource.setStatus(HttpStatus.NOT_MODIFIED);
			}
		} else {
			result = resource.getData();
			responseHeaders.setLastModified(lastModified);
			LOGGER.debug("setting {}={}", HttpHeaders.LAST_MODIFIED, new Date(lastModified));
			LOGGER.debug("setting content type {} ({}B)", resource.getContentType(), result.length);
			responseHeaders.setContentType(MediaType.parseMediaType(resource.getContentType()));
			responseHeaders.setContentLength(result.length);
		}

		return result;
	}

	/**
	 * A resource that has been requested by an {@link HttpServletRequest} and eventually needs to be updated.
	 * 
	 * @see HttpHeaderUtils#handleModifiedHeaders(HttpServletRequest, HttpServletResponse, HttpResource, boolean)
	 */
	public interface HttpResource {

		/**
		 * Updates this resource if necessary and returns the latest version.
		 * 
		 * @return the latest version of the resource
		 * 
		 * @throws IOException
		 *                     if an error occurs while updating
		 */
		long update() throws IOException;

		/**
		 * Returns the content-type for this resource
		 * 
		 * @return the content-type
		 */
		String getContentType();

		/**
		 * Whether or not this resource has been updated during {@link #update()}.
		 * 
		 * @return {@code true} if this resource has been updated
		 * 
		 * @see #update()
		 */
		boolean needsUpdate();

		/**
		 * Retrieves the data of this resource.
		 * 
		 * @return the data
		 * 
		 * @throws IOException
		 *                     if an error occurs while retrieving the data
		 */
		byte[] getData() throws IOException;

		/**
		 * Set the status for this response.
		 * 
		 * @param status
		 *               the status
		 */
		default void setStatus(HttpStatus status) {
		}

		/**
		 * Returns the {@link HttpStatus} for this resource. Default: {@code HttpStatus#OK}.
		 * 
		 * @return the status
		 */
		default HttpStatus getStatus() {
			return HttpStatus.OK;
		}

		/**
		 * Returns the {@link HttpHeaders} for this resource. Default: An empty {@link HttpHeaders} object
		 * 
		 * @return the HttpHeaders
		 */
		default HttpHeaders getHeaders() {
			return new HttpHeaders();
		}
	}

	/**
	 * Parses the string-based HTTP headers of the given {@link HttpServletRequest} to an {@link HttpHeaders} object.
	 * 
	 * @param httpServletRequest
	 *                           a {@link HttpServletRequest}
	 * 
	 * @return The immutable {@link HttpHeaders}
	 */
	public static HttpHeaders parse(HttpServletRequest httpServletRequest) {
		BodyBuilder builder = RequestEntity.method(null, null);
		if (null != httpServletRequest) {
			Enumeration headerNames = httpServletRequest.getHeaderNames();
			while (headerNames.hasMoreElements()) {
				String header = headerNames.nextElement();
				@SuppressWarnings("unchecked")
				List headerValues = EnumerationUtils.toList(httpServletRequest.getHeaders(header));
				builder.header(header, headerValues.toArray(new String[headerValues.size()]));
			}
		}
		return HttpHeaders.readOnlyHttpHeaders(builder.build().getHeaders());
	}

	/**
	 * Applies the given {@link HttpHeaders} to the {@link HttpServletResponse}.
	 * 
	 * @param httpServletResponse
	 *                            the response to set the headers for
	 * @param headers
	 *                            the headers to apply
	 */
	public static void applyHeaders(HttpServletResponse httpServletResponse, HttpHeaders headers) {
		if (null != headers) {
			Set headerNames = new HashSet<>(headers.keySet());
			if (headerNames.remove(HttpHeaders.CACHE_CONTROL)) {
				httpServletResponse.setHeader(HttpHeaders.CACHE_CONTROL, headers.getCacheControl());
			}
			if (headerNames.remove(HttpHeaders.ACCESS_CONTROL_ALLOW_ORIGIN)) {
				httpServletResponse.setHeader(HttpHeaders.ACCESS_CONTROL_ALLOW_ORIGIN,
						headers.getAccessControlAllowOrigin());
			}
			headerNames.forEach(h -> httpServletResponse.setHeader(h, headers.getFirst(h)));
		}
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy