2011-09-26 14:51:15 +02:00
|
|
|
/*
|
2012-06-04 22:31:44 +02:00
|
|
|
* Copyright 2012 The Netty Project
|
2011-09-26 14:51:15 +02:00
|
|
|
*
|
2011-12-09 06:18:34 +01:00
|
|
|
* The Netty Project 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:
|
2011-09-26 14:51:15 +02:00
|
|
|
*
|
2012-06-04 22:31:44 +02:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2011-09-26 14:51:15 +02:00
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
2011-12-09 06:18:34 +01:00
|
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
2011-09-26 14:51:15 +02:00
|
|
|
* License for the specific language governing permissions and limitations
|
|
|
|
* under the License.
|
|
|
|
*/
|
2011-12-09 04:38:59 +01:00
|
|
|
package io.netty.handler.codec.http.websocketx;
|
2011-09-26 14:51:15 +02:00
|
|
|
|
2012-06-10 04:08:43 +02:00
|
|
|
import io.netty.buffer.ByteBuf;
|
2013-07-19 08:28:07 +02:00
|
|
|
import io.netty.buffer.Unpooled;
|
2011-12-09 04:38:59 +01:00
|
|
|
import io.netty.channel.ChannelHandler.Sharable;
|
2012-06-07 07:52:33 +02:00
|
|
|
import io.netty.channel.ChannelHandlerContext;
|
2013-07-19 08:28:07 +02:00
|
|
|
import io.netty.handler.codec.MessageToMessageEncoder;
|
|
|
|
|
|
|
|
import java.util.List;
|
2011-09-26 14:51:15 +02:00
|
|
|
|
|
|
|
/**
|
2012-06-10 04:08:43 +02:00
|
|
|
* Encodes a {@link WebSocketFrame} into a {@link ByteBuf}.
|
2011-09-26 14:51:15 +02:00
|
|
|
* <p>
|
2011-12-15 06:09:09 +01:00
|
|
|
* For the detailed instruction on adding add Web Socket support to your HTTP server, take a look into the
|
|
|
|
* <tt>WebSocketServer</tt> example located in the {@code io.netty.example.http.websocket} package.
|
2011-09-26 14:51:15 +02:00
|
|
|
*/
|
|
|
|
@Sharable
|
2013-07-19 08:28:07 +02:00
|
|
|
public class WebSocket00FrameEncoder extends MessageToMessageEncoder<WebSocketFrame> implements WebSocketFrameEncoder {
|
|
|
|
private static final ByteBuf _0X00 = Unpooled.unreleasableBuffer(
|
|
|
|
Unpooled.directBuffer(1, 1).writeByte((byte) 0x00));
|
|
|
|
private static final ByteBuf _0XFF = Unpooled.unreleasableBuffer(
|
|
|
|
Unpooled.directBuffer(1, 1).writeByte((byte) 0xFF));
|
|
|
|
private static final ByteBuf _0XFF_0X00 = Unpooled.unreleasableBuffer(
|
|
|
|
Unpooled.directBuffer(2, 2).writeByte((byte) 0xFF).writeByte((byte) 0x00));
|
2011-09-26 14:51:15 +02:00
|
|
|
|
2011-12-15 12:25:40 +01:00
|
|
|
@Override
|
2013-07-19 08:28:07 +02:00
|
|
|
protected void encode(ChannelHandlerContext ctx, WebSocketFrame msg, List<Object> out) throws Exception {
|
2012-05-23 20:42:10 +02:00
|
|
|
if (msg instanceof TextWebSocketFrame) {
|
|
|
|
// Text frame
|
2013-05-01 10:04:43 +02:00
|
|
|
ByteBuf data = msg.content();
|
2013-07-19 08:28:07 +02:00
|
|
|
|
|
|
|
out.add(_0X00.duplicate());
|
|
|
|
out.add(data.retain());
|
|
|
|
out.add(_0XFF.duplicate());
|
2012-05-23 20:42:10 +02:00
|
|
|
} else if (msg instanceof CloseWebSocketFrame) {
|
2014-08-14 09:50:40 +02:00
|
|
|
// Close frame, needs to call duplicate to allow multiple writes.
|
|
|
|
// See https://github.com/netty/netty/issues/2768
|
|
|
|
out.add(_0XFF_0X00.duplicate());
|
2012-05-23 20:42:10 +02:00
|
|
|
} else {
|
|
|
|
// Binary frame
|
2013-05-01 10:04:43 +02:00
|
|
|
ByteBuf data = msg.content();
|
2012-05-23 20:42:10 +02:00
|
|
|
int dataLen = data.readableBytes();
|
2011-09-26 14:51:15 +02:00
|
|
|
|
2013-07-19 08:28:07 +02:00
|
|
|
ByteBuf buf = ctx.alloc().buffer(5);
|
|
|
|
boolean release = true;
|
|
|
|
try {
|
|
|
|
// Encode type.
|
|
|
|
buf.writeByte((byte) 0x80);
|
2011-09-26 14:51:15 +02:00
|
|
|
|
2013-07-19 08:28:07 +02:00
|
|
|
// Encode length.
|
|
|
|
int b1 = dataLen >>> 28 & 0x7F;
|
|
|
|
int b2 = dataLen >>> 14 & 0x7F;
|
|
|
|
int b3 = dataLen >>> 7 & 0x7F;
|
|
|
|
int b4 = dataLen & 0x7F;
|
|
|
|
if (b1 == 0) {
|
|
|
|
if (b2 == 0) {
|
|
|
|
if (b3 == 0) {
|
|
|
|
buf.writeByte(b4);
|
|
|
|
} else {
|
|
|
|
buf.writeByte(b3 | 0x80);
|
|
|
|
buf.writeByte(b4);
|
|
|
|
}
|
2011-12-15 12:25:40 +01:00
|
|
|
} else {
|
2013-07-19 08:28:07 +02:00
|
|
|
buf.writeByte(b2 | 0x80);
|
|
|
|
buf.writeByte(b3 | 0x80);
|
|
|
|
buf.writeByte(b4);
|
2011-12-15 12:25:40 +01:00
|
|
|
}
|
|
|
|
} else {
|
2013-07-19 08:28:07 +02:00
|
|
|
buf.writeByte(b1 | 0x80);
|
|
|
|
buf.writeByte(b2 | 0x80);
|
|
|
|
buf.writeByte(b3 | 0x80);
|
|
|
|
buf.writeByte(b4);
|
2011-12-15 12:25:40 +01:00
|
|
|
}
|
2012-05-23 20:42:10 +02:00
|
|
|
|
2013-07-19 08:28:07 +02:00
|
|
|
// Encode binary data.
|
|
|
|
out.add(buf);
|
|
|
|
out.add(data.retain());
|
|
|
|
release = false;
|
|
|
|
} finally {
|
|
|
|
if (release) {
|
|
|
|
buf.release();
|
|
|
|
}
|
|
|
|
}
|
2011-12-15 12:25:40 +01:00
|
|
|
}
|
|
|
|
}
|
2011-09-26 14:51:15 +02:00
|
|
|
}
|