/*
* Copyright 2017 Async-IO.org
*
* 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.atmosphere.nettosphere;
import io.netty.buffer.Unpooled;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFutureListener;
import io.netty.handler.codec.http.websocketx.BinaryWebSocketFrame;
import io.netty.handler.codec.http.websocketx.CloseWebSocketFrame;
import io.netty.handler.codec.http.websocketx.PingWebSocketFrame;
import io.netty.handler.codec.http.websocketx.PongWebSocketFrame;
import io.netty.handler.codec.http.websocketx.TextWebSocketFrame;
import org.atmosphere.cpr.ApplicationConfig;
import org.atmosphere.cpr.AtmosphereConfig;
import org.atmosphere.cpr.AtmosphereResource;
import org.atmosphere.nettosphere.util.Utils;
import org.atmosphere.util.IOUtils;
import org.atmosphere.websocket.WebSocket;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.IOException;
import java.net.InetSocketAddress;
import java.util.concurrent.Future;
import java.util.concurrent.atomic.AtomicBoolean;
import static org.atmosphere.nettosphere.util.Utils.REMOTELY_CLOSED;
public class NettyWebSocket extends WebSocket {
private static final Logger logger = LoggerFactory.getLogger(NettyWebSocket.class);
private final Channel channel;
private final AtomicBoolean firstWrite = new AtomicBoolean(false);
private boolean binaryWrite = false;
private final boolean noInternalAlloc;
private Future<?> closeFuture;
private final AtomicBoolean isClosed = new AtomicBoolean();
private io.netty.channel.ChannelId id;
public NettyWebSocket(Channel channel, AtmosphereConfig config, boolean noInternalAlloc, boolean binaryWrite) {
super(config);
this.channel = channel;
String s = config.getInitParameter(ApplicationConfig.WEBSOCKET_MAXBINARYSIZE);
if (s != null) {
Integer.valueOf(s);
}
s = config.getInitParameter(ApplicationConfig.WEBSOCKET_MAXTEXTSIZE);
if (s != null) {
Integer.valueOf(s);
}
this.noInternalAlloc = noInternalAlloc;
this.binaryWrite = binaryWrite;
this.lastWrite = System.currentTimeMillis();
}
public WebSocket resource(AtmosphereResource r) {
super.resource(r);
// TODO: Netty 4.1 id()
if (noInternalAlloc) {
this.uuid = BridgeRuntime.NETTY_41_PLUS ? channel.id().asLongText() : Utils.id(channel);
if (BridgeRuntime.NETTY_41_PLUS) {
id = channel.id();
}
}
if (!binaryWrite && r != null && r.getRequest() != null) {
try {
binaryWrite = IOUtils.isBodyBinary(r.getRequest());
} catch (Exception ex) {
logger.trace("", ex);
// Don't fail for any reason.
}
}
return this;
}
/**
* {@inheritDoc}
*/
@Override
public WebSocket write(String data) throws IOException {
firstWrite.set(true);
if (!channel.isOpen()) throw REMOTELY_CLOSED;
logger.trace("WebSocket.write() as binary {}", binaryWrite);
if (binaryWrite) {
channel.writeAndFlush(new BinaryWebSocketFrame(Unpooled.wrappedBuffer(data.getBytes("UTF-8"))));
} else {
channel.writeAndFlush(new TextWebSocketFrame(data));
}
lastWrite = System.currentTimeMillis();
return this;
}
public WebSocket write(byte[] data) throws IOException {
_write(data, 0, data.length);
return this;
}
/**
* {@inheritDoc}
*/
@Override
public WebSocket write(byte[] data, int offset, int length) throws IOException {
_write(data, offset, length);
return this;
}
void _write(byte[] data, int offset, int length) throws IOException {
firstWrite.set(true);
if (!channel.isOpen()) throw REMOTELY_CLOSED;
logger.trace("WebSocket.write() as binary {}", binaryWrite);
if (binaryWrite) {
channel.writeAndFlush(new BinaryWebSocketFrame(Unpooled.wrappedBuffer(data, offset, length)));
} else {
channel.writeAndFlush(new TextWebSocketFrame(Unpooled.wrappedBuffer(data, offset, length)));
}
lastWrite = System.currentTimeMillis();
}
@Override
public boolean isOpen() {
return channel.isOpen();
}
/**
* {@inheritDoc}
*/
@Override
public void close() {
if (isClosed.getAndSet(true)) return;
channel.writeAndFlush(new CloseWebSocketFrame()).addListener(ChannelFutureListener.CLOSE);
if (closeFuture != null) {
closeFuture.cancel(true);
}
}
/**
* Send a WebSocket Ping
*
* @param payload the bytes to send
* @return this
*/
public WebSocket sendPing(byte[] payload) {
channel.writeAndFlush(new PingWebSocketFrame(Unpooled.wrappedBuffer(payload)));
return this;
}
/**
* Send a WebSocket Pong
*
* @param payload the bytes to send
* @return this
*/
public WebSocket sendPong(byte[] payload) {
channel.writeAndFlush(new PongWebSocketFrame(Unpooled.wrappedBuffer(payload)));
return this;
}
/**
* The remote ip address.
*
* @return The remote ip address.
*/
public String address() {
return ((InetSocketAddress) channel.remoteAddress()).getAddress().getHostAddress();
}
protected WebSocket closeFuture(Future<?> closeFuture) {
this.closeFuture = closeFuture;
return this;
}
protected Future<?> closeFuture() {
return closeFuture;
}
public io.netty.channel.ChannelId id() {
return id;
}
}