com.caucho.server.http.WebSocketContextImpl Maven / Gradle / Ivy
/*
* Copyright (c) 1998-2018 Caucho Technology -- all rights reserved
*
* This file is part of Resin(R) Open Source
*
* Each copy or derived work must preserve the copyright notice and this
* notice unmodified.
*
* Resin Open Source is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* Resin Open Source is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, or any warranty
* of NON-INFRINGEMENT. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License
* along with Resin Open Source; if not, write to the
*
* Free Software Foundation, Inc.
* 59 Temple Place, Suite 330
* Boston, MA 02111-1307 USA
*
* @author Scott Ferguson
*/
package com.caucho.server.http;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.logging.Level;
import java.util.logging.Logger;
import com.caucho.network.listen.SocketLinkDuplexController;
import com.caucho.network.listen.SocketLinkDuplexListener;
import com.caucho.remote.websocket.WebSocketBlockingQueue;
import com.caucho.remote.websocket.WebSocketConstants;
import com.caucho.remote.websocket.WebSocketInputStream;
import com.caucho.remote.websocket.WebSocketOutputStream;
import com.caucho.remote.websocket.WebSocketPrintWriter;
import com.caucho.remote.websocket.WebSocketReader;
import com.caucho.remote.websocket.WebSocketWriter;
import com.caucho.remote.websocket.FrameInputStream;
import com.caucho.util.IoUtil;
import com.caucho.util.L10N;
import com.caucho.vfs.ReadStream;
import com.caucho.vfs.TempBuffer;
import com.caucho.vfs.WriteStream;
import com.caucho.websocket.WebSocketContext;
import com.caucho.websocket.WebSocketEncoder;
import com.caucho.websocket.WebSocketListener;
/**
* User facade for http requests.
*/
class WebSocketContextImpl
implements WebSocketContext, WebSocketConstants, SocketLinkDuplexListener
{
private static final L10N L = new L10N(WebSocketContextImpl.class);
private static final Logger log
= Logger.getLogger(WebSocketContextImpl.class.getName());
private final HttpServletRequestImpl _request;
private final WebSocketListener _listener;
private SocketLinkDuplexController _controller;
private FrameInputStream _is;
private WebSocketOutputStream _binaryOut;
private WebSocketInputStream _binaryIn;
private WebSocketWriter _textOut;
private PrintWriter _textWriter;
// private WebSocketReader _textIn;
private boolean _isReadClosed;
private AtomicBoolean _isWriteClosed = new AtomicBoolean();
WebSocketContextImpl(HttpServletRequestImpl request,
HttpServletResponseImpl response,
WebSocketListener listener,
FrameInputStream is)
{
_request = request;
_listener = listener;
_is = is;
}
public void setController(SocketLinkDuplexController controller)
{
_controller = controller;
_is.init(this, controller.getReadStream());
}
@Override
public void setTimeout(long timeout)
{
_controller.setIdleTimeMax(timeout);
}
@Override
public long getTimeout()
{
return _controller.getIdleTimeMax();
}
@Override
public BlockingQueue createOutputQueue(WebSocketEncoder encoder)
{
return new WebSocketBlockingQueue(this, encoder, 256);
}
@Override
public void setAutoFlush(boolean isAutoFlush)
{
}
@Override
public boolean isAutoFlush()
{
return false;
}
@Override
public OutputStream startBinaryMessage()
throws IOException
{
if (_isWriteClosed.get())
throw new IllegalStateException(L.l("{0} is closed for writing.",
this));
if (_binaryOut == null)
_binaryOut = new WebSocketOutputStream(_controller.getWriteStream(),
TempBuffer.allocate().getBuffer());
_binaryOut.init();
return _binaryOut;
}
@Override
public PrintWriter startTextMessage()
throws IOException
{
if (_textOut == null) {
_textOut = new WebSocketWriter(_controller.getWriteStream(),
TempBuffer.allocate().getBuffer());
_textWriter = new WebSocketPrintWriter(_textOut);
}
_textOut.init();
return _textWriter;
}
@Override
public void ping(byte []value)
throws IOException
{
WriteStream out = _controller.getWriteStream();
byte []bytes = value;
out.write(0x89);
out.write(bytes.length);
out.write(bytes);
out.flush();
}
@Override
public void pong(byte []value)
throws IOException
{
WriteStream out = _controller.getWriteStream();
byte []bytes = value;
out.write(0x8a);
out.write(bytes.length);
out.write(bytes);
out.flush();
}
public boolean isClosed()
{
return _isWriteClosed.get();
}
@Override
public void close()
{
close(1000, "ok");
}
@Override
public void close(int code, String message)
{
if (_isWriteClosed.getAndSet(true))
return;
WriteStream out = _controller.getWriteStream();
try {
if (code <= 0) {
out.write(0x88);
out.write(0x00);
}
else {
byte []bytes = message.getBytes("utf-8");
out.write(0x88);
out.write(0x02 + bytes.length);
out.write((code >> 8) & 0xff);
out.write(code & 0xff);
out.write(bytes);
}
} catch (IOException e) {
log.log(Level.WARNING, e.toString(), e);
} finally {
IoUtil.close(out);
disconnect();
}
}
@Override
public void disconnect()
{
_isWriteClosed.set(true);
try {
_controller.complete();
} finally {
IoUtil.close(_is);
}
}
//
// duplex callbacks
//
void onStart()
throws IOException
{
_listener.onStart(this);
}
@Override
public void flush()
throws IOException
{
WriteStream out = _controller.getWriteStream();
out.flush();
}
@Override
public void onStart(SocketLinkDuplexController context)
throws IOException
{
}
@Override
public void onRead(SocketLinkDuplexController duplex)
throws IOException
{
do {
if (! readFrame()) {
return;
}
} while (_request.getBufferAvailable() > 0);
}
private boolean readFrame()
throws IOException
{
if (! _is.readFrameHeader()) {
return false;
}
int opcode = _is.getOpcode();
switch (opcode) {
case OP_BINARY:
if (_binaryIn == null)
_binaryIn = createWebSocketInputStream(_is);
_binaryIn.init();
try {
_listener.onReadBinary(this, _binaryIn);
} finally {
_binaryIn.close();
}
break;
case OP_TEXT:
WebSocketReader textIn = _is.initReader(_is.getLength(), _is.isFinal());
try {
_listener.onReadText(this, textIn);
} finally {
textIn.close();
}
break;
default:
log.fine(this + " unexpected opcode " + opcode);
// XXX:
disconnect();
return false;
}
return true;
}
protected WebSocketInputStream createWebSocketInputStream(FrameInputStream is)
throws IOException
{
return new WebSocketInputStream(is);
}
@Override
public void onDisconnect(SocketLinkDuplexController duplex)
throws IOException
{
_listener.onDisconnect(this);
}
@Override
public void onTimeout(SocketLinkDuplexController duplex)
throws IOException
{
_listener.onTimeout(this);
}
@Override
public void onClose(int closeCode, String closeMessage)
{
try {
_listener.onClose(this);
} catch (Exception e) {
log.log(Level.FINER, e.toString(), e);
}
}
@Override
public String toString()
{
return getClass().getSimpleName() + "[" + _listener + "]";
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy