using System;
using System.Collections.Concurrent;
using System.Collections.Generic;
using System.Linq;
using System.Security.Cryptography;
using System.Text;
using System.Text.RegularExpressions;
using System.Threading;
using SuperSocket.Common;
using SuperSocket.SocketBase;
using SuperSocket.SocketBase.Protocol;
namespace SuperWebSocket.Protocol
{
///
/// http://tools.ietf.org/html/draft-ietf-hybi-thewebsocketprotocol-10
///
class DraftHybi10Processor : ProtocolProcessorBase
{
private const string m_Magic = "258EAFA5-E914-47DA-95CA-C5AB0DC85B11";
protected DraftHybi10Processor(int version, ICloseStatusCode closeStatusCode)
: base(version, closeStatusCode)
{
}
public DraftHybi10Processor()
: base(8, new CloseStatusCodeHybi10())
{
}
private const string m_OriginKey = "Sec-WebSocket-Origin";
protected virtual string OriginKey
{
get { return m_OriginKey; }
}
public override bool Handshake(IWebSocketSession session, WebSocketReceiveFilterBase previousFilter, out IReceiveFilter dataFrameReader)
{
if (!VersionTag.Equals(session.SecWebSocketVersion) && NextProcessor != null)
{
return NextProcessor.Handshake(session, previousFilter, out dataFrameReader);
}
dataFrameReader = null;
session.ProtocolProcessor = this;
if (!session.AppServer.ValidateHandshake(session, session.Items.GetValue(OriginKey, string.Empty)))
return false;
var secWebSocketKey = session.Items.GetValue(WebSocketConstant.SecWebSocketKey, string.Empty);
if (string.IsNullOrEmpty(secWebSocketKey))
{
return false;
}
var responseBuilder = new StringBuilder();
string secKeyAccept = string.Empty;
try
{
secKeyAccept = Convert.ToBase64String(SHA1.Create().ComputeHash(Encoding.ASCII.GetBytes(secWebSocketKey + m_Magic)));
}
catch (Exception)
{
return false;
}
responseBuilder.AppendWithCrCf(WebSocketConstant.ResponseHeadLine10);
responseBuilder.AppendWithCrCf(WebSocketConstant.ResponseUpgradeLine);
responseBuilder.AppendWithCrCf(WebSocketConstant.ResponseConnectionLine);
responseBuilder.AppendFormatWithCrCf(WebSocketConstant.ResponseAcceptLine, secKeyAccept);
var subProtocol = session.GetAvailableSubProtocol(session.Items.GetValue(WebSocketConstant.SecWebSocketProtocol, string.Empty));
if (!string.IsNullOrEmpty(subProtocol))
responseBuilder.AppendFormatWithCrCf(WebSocketConstant.ResponseProtocolLine, subProtocol);
responseBuilder.AppendWithCrCf();
byte[] data = Encoding.UTF8.GetBytes(responseBuilder.ToString());
session.SendRawData(data, 0, data.Length);
dataFrameReader = new WebSocketDataFrameReceiveFilter();
return true;
}
public override bool CanSendBinaryData
{
get { return true; }
}
public override void SendData(IWebSocketSession session, byte[] data, int offset, int length)
{
SendPackage(session, OpCode.Binary, data, offset, length);
}
public override void SendMessage(IWebSocketSession session, string message)
{
SendMessage(session, OpCode.Text, message);
}
public override void SendCloseHandshake(IWebSocketSession session, int statusCode, string closeReason)
{
byte[] playloadData = new byte[(string.IsNullOrEmpty(closeReason) ? 0 : Encoding.UTF8.GetMaxByteCount(closeReason.Length)) + 2];
int highByte = statusCode / 256;
int lowByte = statusCode % 256;
playloadData[0] = (byte)highByte;
playloadData[1] = (byte)lowByte;
var playloadLength = playloadData.Length;
if (!string.IsNullOrEmpty(closeReason))
{
int bytesCount = Encoding.UTF8.GetBytes(closeReason, 0, closeReason.Length, playloadData, 2);
playloadLength = bytesCount + 2;
}
SendPackage(session, OpCode.Close, playloadData, 0, playloadLength);
}
public override void SendPong(IWebSocketSession session, byte[] pong)
{
SendPackage(session, OpCode.Pong, pong, 0, pong.Length);
}
public override void SendPing(IWebSocketSession session, byte[] ping)
{
SendPackage(session, OpCode.Ping, ping, 0, ping.Length);
}
public override IList> GetEncodedPackage(int opCode, byte[] data, int offset, int length)
{
byte[] head;
if (length < 126)
{
head = new byte[2];
head[1] = (byte)length;
}
else if (length < 65536)
{
head = new byte[4];
head[1] = (byte)126;
head[2] = (byte)(length / 256);
head[3] = (byte)(length % 256);
}
else
{
head = new byte[10];
head[1] = (byte)127;
int left = length;
int unit = 256;
for (int i = 9; i > 1; i--)
{
head[i] = (byte)(left % unit);
left = left / unit;
if (left == 0)
break;
}
}
head[0] = (byte)(opCode | 0x80); //No mask by default
return new ArraySegment[] { new ArraySegment(head), new ArraySegment(data, offset, length) };
}
public override IList> GetEncodedPackage(int opCode, string message)
{
byte[] playloadData = Encoding.UTF8.GetBytes(message);
return GetEncodedPackage(opCode, playloadData, 0, playloadData.Length);
}
private byte[] GetPackageData(int opCode, byte[] data, int offset, int length)
{
byte[] fragment;
if (length < 126)
{
fragment = new byte[2 + length];
fragment[1] = (byte)length;
}
else if (length < 65536)
{
fragment = new byte[4 + length];
fragment[1] = (byte)126;
fragment[2] = (byte)(length / 256);
fragment[3] = (byte)(length % 256);
}
else
{
fragment = new byte[10 + length];
fragment[1] = (byte)127;
int left = length;
int unit = 256;
for (int i = 9; i > 1; i--)
{
fragment[i] = (byte)(left % unit);
left = left / unit;
if (left == 0)
break;
}
}
fragment[0] = (byte)(opCode | 0x80);
if (length > 0)
{
Buffer.BlockCopy(data, offset, fragment, fragment.Length - length, length);
}
return fragment;
}
private void SendPackage(IWebSocketSession session, int opCode, byte[] data, int offset, int length)
{
var fragment = GetPackageData(opCode, data, offset, length);
session.SendRawData(fragment, 0, fragment.Length);
}
private bool TrySendPackage(IWebSocketSession session, int opCode, byte[] data, int offset, int length)
{
var fragment = GetPackageData(opCode, data, offset, length);
return session.TrySendRawData(fragment, 0, fragment.Length);
}
private void SendMessage(IWebSocketSession session, int opCode, string message)
{
byte[] playloadData = Encoding.UTF8.GetBytes(message);
SendPackage(session, opCode, playloadData, 0, playloadData.Length);
}
private bool TrySendMessage(IWebSocketSession session, int opCode, string message)
{
byte[] playloadData = Encoding.UTF8.GetBytes(message);
return TrySendPackage(session, opCode, playloadData, 0, playloadData.Length);
}
public override bool IsValidCloseCode(int code)
{
var closeCode = this.CloseStatusClode;
if (code >= 0 && code <= 999)
return false;
if (code >= 1000 && code <= 1999)
{
if (code == closeCode.NormalClosure
|| code == closeCode.GoingAway
|| code == closeCode.ProtocolError
|| code == closeCode.NotAcceptableData
|| code == closeCode.TooLargeFrame
|| code == closeCode.InvalidUTF8)
{
return true;
}
return false;
}
if (code >= 2000 && code <= 4999)
return true;
return false;
}
public override bool TrySendMessage(IWebSocketSession session, string message)
{
return TrySendMessage(session, OpCode.Text, message);
}
public override bool TrySendData(IWebSocketSession session, byte[] data, int offset, int length)
{
return TrySendPackage(session, OpCode.Binary, data, offset, length);
}
}
}