| /* |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You 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.apache.coyote.http11.upgrade; |
| |
| import java.io.IOException; |
| import java.nio.ByteBuffer; |
| import java.util.concurrent.Executor; |
| |
| import javax.servlet.ServletInputStream; |
| import javax.servlet.ServletOutputStream; |
| import javax.servlet.http.HttpUpgradeHandler; |
| import javax.servlet.http.WebConnection; |
| |
| import org.apache.coyote.Processor; |
| import org.apache.coyote.Request; |
| import org.apache.juli.logging.Log; |
| import org.apache.juli.logging.LogFactory; |
| import org.apache.tomcat.util.net.AbstractEndpoint.Handler.SocketState; |
| import org.apache.tomcat.util.net.SSLSupport; |
| import org.apache.tomcat.util.net.SocketStatus; |
| import org.apache.tomcat.util.net.SocketWrapperBase; |
| import org.apache.tomcat.util.res.StringManager; |
| |
| public class UpgradeProcessor<S> implements Processor<S>, WebConnection { |
| |
| private static final int INFINITE_TIMEOUT = -1; |
| |
| private static final Log log = LogFactory.getLog(UpgradeProcessor.class); |
| private static final StringManager sm = StringManager.getManager(UpgradeProcessor.class); |
| |
| private final HttpUpgradeHandler httpUpgradeHandler; |
| private final UpgradeServletInputStream upgradeServletInputStream; |
| private final UpgradeServletOutputStream upgradeServletOutputStream; |
| |
| |
| public UpgradeProcessor(SocketWrapperBase<?> wrapper, ByteBuffer leftOverInput, |
| HttpUpgradeHandler httpUpgradeHandler, int asyncWriteBufferSize) { |
| this.httpUpgradeHandler = httpUpgradeHandler; |
| this.upgradeServletInputStream = new UpgradeServletInputStream(wrapper); |
| this.upgradeServletOutputStream = |
| new UpgradeServletOutputStream(wrapper, asyncWriteBufferSize); |
| |
| wrapper.unRead(leftOverInput); |
| wrapper.setTimeout(INFINITE_TIMEOUT); |
| } |
| |
| |
| // --------------------------------------------------- AutoCloseable methods |
| |
| @Override |
| public void close() throws Exception { |
| upgradeServletInputStream.close(); |
| upgradeServletOutputStream.close(); |
| } |
| |
| |
| // --------------------------------------------------- WebConnection methods |
| |
| @Override |
| public ServletInputStream getInputStream() throws IOException { |
| return upgradeServletInputStream; |
| } |
| |
| @Override |
| public ServletOutputStream getOutputStream() throws IOException { |
| return upgradeServletOutputStream; |
| } |
| |
| |
| // ------------------------------------------- Implemented Processor methods |
| |
| @Override |
| public final boolean isUpgrade() { |
| return true; |
| } |
| |
| |
| @Override |
| public HttpUpgradeHandler getHttpUpgradeHandler() { |
| return httpUpgradeHandler; |
| } |
| |
| |
| @Override |
| public final SocketState upgradeDispatch(SocketStatus status) throws IOException { |
| if (status == SocketStatus.OPEN_READ) { |
| upgradeServletInputStream.onDataAvailable(); |
| } else if (status == SocketStatus.OPEN_WRITE) { |
| upgradeServletOutputStream.onWritePossible(); |
| } else if (status == SocketStatus.STOP) { |
| try { |
| upgradeServletInputStream.close(); |
| } catch (IOException ioe) { |
| log.debug(sm.getString("upgradeProcessor.isCloseFail", ioe)); |
| } |
| try { |
| upgradeServletOutputStream.close(); |
| } catch (IOException ioe) { |
| log.debug(sm.getString("upgradeProcessor.osCloseFail", ioe)); |
| } |
| return SocketState.CLOSED; |
| } else { |
| // Unexpected state |
| return SocketState.CLOSED; |
| } |
| if (upgradeServletInputStream.isCloseRequired() || |
| upgradeServletOutputStream.isCloseRequired()) { |
| return SocketState.CLOSED; |
| } |
| return SocketState.UPGRADED; |
| } |
| |
| |
| @Override |
| public final void recycle(boolean socketClosing) { |
| // Currently a NO-OP as upgrade processors are not recycled. |
| } |
| |
| |
| // ---------------------------- Processor methods that are NO-OP for upgrade |
| |
| @Override |
| public final Executor getExecutor() { |
| return null; |
| } |
| |
| |
| @Override |
| public final SocketState process(SocketWrapperBase<S> socketWrapper) throws IOException { |
| return null; |
| } |
| |
| |
| @Override |
| public final SocketState asyncDispatch(SocketStatus status) { |
| return null; |
| } |
| |
| |
| @Override |
| public void errorDispatch() { |
| // NO-OP |
| } |
| |
| |
| @Override |
| public final SocketState asyncPostProcess() { |
| return null; |
| } |
| |
| |
| @Override |
| public final boolean isAsync() { |
| return false; |
| } |
| |
| |
| @Override |
| public final Request getRequest() { |
| return null; |
| } |
| |
| |
| @Override |
| public final void setSslSupport(SSLSupport sslSupport) { |
| // NOOP |
| } |
| |
| |
| @Override |
| public ByteBuffer getLeftoverInput() { |
| return null; |
| } |
| } |