/*
 * Copyright 2002-2020 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.http.server;

import java.io.IOException;
import java.util.concurrent.atomic.AtomicBoolean;

import javax.servlet.AsyncContext;
import javax.servlet.AsyncEvent;
import javax.servlet.AsyncListener;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.springframework.lang.Nullable;
import org.springframework.util.Assert;

A ServerHttpAsyncRequestControl to use on Servlet containers (Servlet 3.0+).
Author:Rossen Stoyanchev
Since:4.0
/** * A {@link ServerHttpAsyncRequestControl} to use on Servlet containers (Servlet 3.0+). * * @author Rossen Stoyanchev * @since 4.0 */
public class ServletServerHttpAsyncRequestControl implements ServerHttpAsyncRequestControl, AsyncListener { private static final long NO_TIMEOUT_VALUE = Long.MIN_VALUE; private final ServletServerHttpRequest request; private final ServletServerHttpResponse response; @Nullable private AsyncContext asyncContext; private AtomicBoolean asyncCompleted = new AtomicBoolean();
Constructor accepting a request and response pair that are expected to be of type ServletServerHttpRequest and ServletServerHttpResponse respectively.
/** * Constructor accepting a request and response pair that are expected to be of type * {@link ServletServerHttpRequest} and {@link ServletServerHttpResponse} * respectively. */
public ServletServerHttpAsyncRequestControl(ServletServerHttpRequest request, ServletServerHttpResponse response) { Assert.notNull(request, "request is required"); Assert.notNull(response, "response is required"); Assert.isTrue(request.getServletRequest().isAsyncSupported(), "Async support must be enabled on a servlet and for all filters involved " + "in async request processing. This is done in Java code using the Servlet API " + "or by adding \"<async-supported>true</async-supported>\" to servlet and " + "filter declarations in web.xml. Also you must use a Servlet 3.0+ container"); this.request = request; this.response = response; } @Override public boolean isStarted() { return (this.asyncContext != null && this.request.getServletRequest().isAsyncStarted()); } @Override public boolean isCompleted() { return this.asyncCompleted.get(); } @Override public void start() { start(NO_TIMEOUT_VALUE); } @Override public void start(long timeout) { Assert.state(!isCompleted(), "Async processing has already completed"); if (isStarted()) { return; } HttpServletRequest servletRequest = this.request.getServletRequest(); HttpServletResponse servletResponse = this.response.getServletResponse(); this.asyncContext = servletRequest.startAsync(servletRequest, servletResponse); this.asyncContext.addListener(this); if (timeout != NO_TIMEOUT_VALUE) { this.asyncContext.setTimeout(timeout); } } @Override public void complete() { if (this.asyncContext != null && isStarted() && !isCompleted()) { this.asyncContext.complete(); } } // --------------------------------------------------------------------- // Implementation of AsyncListener methods // --------------------------------------------------------------------- @Override public void onComplete(AsyncEvent event) throws IOException { this.asyncContext = null; this.asyncCompleted.set(true); } @Override public void onStartAsync(AsyncEvent event) throws IOException { } @Override public void onError(AsyncEvent event) throws IOException { } @Override public void onTimeout(AsyncEvent event) throws IOException { } }