HttpServletResponseWrapper.java

/*
 * Copyright 2013-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
 *
 *      https://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.springframework.cloud.sleuth.instrument.web.servlet;

import java.util.Collection;

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

import org.springframework.cloud.sleuth.http.HttpServerResponse;
import org.springframework.lang.Nullable;

/**
 * This delegates to {@link HttpServletResponse} methods, taking care to portably handle
 * {@link #statusCode()}.
 *
 * @since 5.10
 */
public class HttpServletResponseWrapper implements HttpServerResponse {

	// not final for inner
	// subtype
	/**
	 * Returns the trace representation of a response.
	 * @param caught an exception caught serving the request.
	 * @return wrapped response
	 */
	public static HttpServerResponse create(@Nullable HttpServletRequest request, HttpServletResponse response,
			@Nullable Throwable caught) {
		return new HttpServletResponseWrapper(request, response, caught);
	}

	@Nullable
	final HttpServletRequestWrapper request;

	final HttpServletResponse response;

	@Nullable
	final Throwable caught;

	HttpServletResponseWrapper(@Nullable HttpServletRequest request, HttpServletResponse response,
			@Nullable Throwable caught) {
		if (response == null) {
			throw new NullPointerException("response == null");
		}
		this.request = request != null ? new HttpServletRequestWrapper(request) : null;
		this.response = response;
		this.caught = caught;
	}

	@Override
	public final Object unwrap() {
		return response;
	}

	@Override
	public Collection<String> headerNames() {
		return this.response.getHeaderNames();
	}

	@Override
	@Nullable
	public HttpServletRequestWrapper request() {
		return request;
	}

	@Override
	public Throwable error() {
		if (caught != null) {
			return caught;
		}
		if (request == null) {
			return null;
		}
		return request.maybeError();
	}

	@Override
	public int statusCode() {
		int result = ServletRuntime.get().status(response);
		if (caught != null && result == 200) { // We may have a potentially bad status due
												// to defaults
			// Servlet only seems to define one exception that has a built-in code. Logic
			// in Jetty
			// defaults the status to 500 otherwise.
			if (caught instanceof UnavailableException) {
				return ((UnavailableException) caught).isPermanent() ? 404 : 503;
			}
			return 500;
		}
		return result;
	}

}