Fix due to the added SessionManager.cs

This commit is contained in:
sta 2012-10-09 12:14:55 +09:00
parent 94385ea2bc
commit c560d4fba9
66 changed files with 345 additions and 245 deletions

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

View File

@ -32,8 +32,8 @@ using System.Collections;
using System.Collections.Generic; using System.Collections.Generic;
using System.Text; using System.Text;
namespace WebSocketSharp.Frame namespace WebSocketSharp.Frame {
{
public class WsFrame : IEnumerable<byte> public class WsFrame : IEnumerable<byte>
{ {
#region Field #region Field
@ -42,37 +42,6 @@ namespace WebSocketSharp.Frame
#endregion #endregion
#region Properties
public Fin Fin { get; private set; }
public Rsv Rsv1 { get; private set; }
public Rsv Rsv2 { get; private set; }
public Rsv Rsv3 { get; private set; }
public Opcode Opcode { get; private set; }
public Mask Masked { get; private set; }
public byte PayloadLen { get; private set; }
public byte[] ExtPayloadLen { get; private set; }
public byte[] MaskingKey { get; private set; }
public PayloadData PayloadData { get; private set; }
public ulong Length
{
get
{
return 2 + (ulong)(ExtPayloadLen.Length + MaskingKey.Length) + PayloadLength;
}
}
public ulong PayloadLength
{
get
{
return PayloadData.Length;
}
}
#endregion
#region Static Constructor #region Static Constructor
static WsFrame() static WsFrame()
@ -120,6 +89,37 @@ namespace WebSocketSharp.Frame
#endregion #endregion
#region Properties
public Fin Fin { get; private set; }
public Rsv Rsv1 { get; private set; }
public Rsv Rsv2 { get; private set; }
public Rsv Rsv3 { get; private set; }
public Opcode Opcode { get; private set; }
public Mask Masked { get; private set; }
public byte PayloadLen { get; private set; }
public byte[] ExtPayloadLen { get; private set; }
public byte[] MaskingKey { get; private set; }
public PayloadData PayloadData { get; private set; }
public ulong Length
{
get
{
return 2 + (ulong)(ExtPayloadLen.Length + MaskingKey.Length) + PayloadLength;
}
}
public ulong PayloadLength
{
get
{
return PayloadData.Length;
}
}
#endregion
#region Private Methods #region Private Methods
IEnumerator IEnumerable.GetEnumerator() IEnumerator IEnumerable.GetEnumerator()

View File

@ -1,6 +1,6 @@
// //
// EndPointManager.cs // EndPointManager.cs
// Copied from System.Net.EndPointManager // Copied from System.Net.EndPointManager.cs
// //
// Author: // Author:
// Gonzalo Paniagua Javier (gonzalo@ximian.com) // Gonzalo Paniagua Javier (gonzalo@ximian.com)

View File

@ -39,12 +39,12 @@ namespace WebSocketSharp.Server {
#region Fields #region Fields
private Thread _acceptRequestThread; private Thread _acceptRequestThread;
private bool _isWindows; private bool _isWindows;
private HttpListener _listener; private HttpListener _listener;
private int _port; private int _port;
private string _rootPath; private string _rootPath;
private Dictionary<string, IWebSocketServer> _wsServers; private Dictionary<string, IServiceHost> _services;
#endregion #endregion
@ -119,7 +119,7 @@ namespace WebSocketSharp.Server {
{ {
_isWindows = false; _isWindows = false;
_listener = new HttpListener(); _listener = new HttpListener();
_wsServers = new Dictionary<string, IWebSocketServer>(); _services = new Dictionary<string, IServiceHost>();
var os = Environment.OSVersion; var os = Environment.OSVersion;
if (os.Platform != PlatformID.Unix && os.Platform != PlatformID.MacOSX) if (os.Platform != PlatformID.Unix && os.Platform != PlatformID.MacOSX)
@ -248,7 +248,7 @@ namespace WebSocketSharp.Server {
var res = context.Response; var res = context.Response;
var path = req.RawUrl; var path = req.RawUrl;
if (!_wsServers.ContainsKey(path)) if (!_services.ContainsKey(path))
{ {
res.StatusCode = (int)HttpStatusCode.NotImplemented; res.StatusCode = (int)HttpStatusCode.NotImplemented;
return false; return false;
@ -256,8 +256,8 @@ namespace WebSocketSharp.Server {
var wsContext = context.AcceptWebSocket(path); var wsContext = context.AcceptWebSocket(path);
var socket = wsContext.WebSocket; var socket = wsContext.WebSocket;
var wsServer = _wsServers[path]; var service = _services[path];
wsServer.BindWebSocket(socket); service.BindWebSocket(socket);
return true; return true;
} }
@ -269,8 +269,8 @@ namespace WebSocketSharp.Server {
public void AddService<T>(string path) public void AddService<T>(string path)
where T : WebSocketService, new() where T : WebSocketService, new()
{ {
var server = new WebSocketServer<T>(); var service = new WebSocketServer<T>();
_wsServers.Add(path, server); _services.Add(path, service);
} }
public byte[] GetFile(string path) public byte[] GetFile(string path)
@ -295,9 +295,9 @@ namespace WebSocketSharp.Server {
{ {
_listener.Close(); _listener.Close();
_acceptRequestThread.Join(5 * 1000); _acceptRequestThread.Join(5 * 1000);
foreach (var server in _wsServers.Values) foreach (var service in _services.Values)
server.Stop(); service.Stop();
_wsServers.Clear(); _services.Clear();
} }
#endregion #endregion

View File

@ -1,6 +1,6 @@
#region MIT License #region MIT License
/** /**
* IWebSocketServer.cs * IServiceHost.cs
* *
* The MIT License * The MIT License
* *
@ -30,11 +30,10 @@ using System;
namespace WebSocketSharp.Server { namespace WebSocketSharp.Server {
public interface IWebSocketServer { public interface IServiceHost {
void BindWebSocket(WebSocket socket); void BindWebSocket(WebSocket socket);
void Start(); void Start();
void Stop(); void Stop();
void StopServices();
} }
} }

View File

@ -0,0 +1,184 @@
#region MIT License
/**
* SessionManager.cs
*
* The MIT License
*
* Copyright (c) 2012 sta.blockhead
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
#endregion
using System;
using System.Collections.Generic;
using WebSocketSharp.Frame;
namespace WebSocketSharp.Server {
public class SessionManager {
#region Private Fields
private bool _isStopped;
private Dictionary<string, WebSocketService> _sessions;
private object _syncRoot;
#endregion
#region Public Constructor
public SessionManager()
{
_isStopped = false;
_sessions = new Dictionary<string, WebSocketService>();
_syncRoot = new object();
}
#endregion
#region Properties
public int Count {
get {
lock (_syncRoot)
{
return _sessions.Count;
}
}
}
public object SyncRoot {
get {
return _syncRoot;
}
}
#endregion
#region Private Method
private Dictionary<string, WebSocketService> copySessions()
{
lock (_syncRoot)
{
return new Dictionary<string, WebSocketService>(_sessions);
}
}
private string getNewID()
{
return Guid.NewGuid().ToString("N");
}
#endregion
#region Public Methods
public string Add(WebSocketService service)
{
lock (_syncRoot)
{
if (_isStopped)
return null;
var id = getNewID();
_sessions.Add(id, service);
return id;
}
}
public void Broadcast(byte[] data)
{
lock (_syncRoot)
{
foreach (var service in _sessions.Values)
service.SendAsync(data);
}
}
public void Broadcast(string data)
{
lock (_syncRoot)
{
foreach (var service in _sessions.Values)
service.SendAsync(data);
}
}
public Dictionary<string, bool> Broadping(string data)
{
var result = new Dictionary<string, bool>();
foreach (var session in copySessions())
result.Add(session.Key, session.Value.Ping(data));
return result;
}
public IEnumerable<string> GetIDs()
{
lock (_syncRoot)
{
return _sessions.Keys;
}
}
public bool Remove(string id)
{
lock (_syncRoot)
{
if (_isStopped)
return false;
return _sessions.Remove(id);
}
}
public bool TryGetByID(string id, out WebSocketService service)
{
lock (_syncRoot)
{
return _sessions.TryGetValue(id, out service);
}
}
public void Stop()
{
Stop(CloseStatusCode.NORMAL, String.Empty);
}
public void Stop(CloseStatusCode code, string reason)
{
lock (_syncRoot)
{
if (_isStopped)
return;
_isStopped = true;
foreach (var service in _sessions.Values)
service.Stop(code, reason);
_sessions.Clear();
}
}
#endregion
}
}

View File

@ -29,10 +29,8 @@
#endregion #endregion
using System; using System;
using System.Collections;
using System.Collections.Generic; using System.Collections.Generic;
using System.Net.Sockets; using System.Net.Sockets;
using WebSocketSharp.Frame;
using WebSocketSharp.Net; using WebSocketSharp.Net;
using WebSocketSharp.Net.Sockets; using WebSocketSharp.Net.Sockets;
@ -42,7 +40,7 @@ namespace WebSocketSharp.Server {
{ {
#region Field #region Field
private Dictionary<string, IWebSocketServer> _servers; private Dictionary<string, IServiceHost> _services;
#endregion #endregion
@ -56,7 +54,7 @@ namespace WebSocketSharp.Server {
public WebSocketServer(int port) public WebSocketServer(int port)
: base(System.Net.IPAddress.Any, port) : base(System.Net.IPAddress.Any, port)
{ {
_servers = new Dictionary<string, IWebSocketServer>(); _services = new Dictionary<string, IServiceHost>();
} }
#endregion #endregion
@ -68,14 +66,14 @@ namespace WebSocketSharp.Server {
var context = client.AcceptWebSocket(); var context = client.AcceptWebSocket();
var socket = context.WebSocket; var socket = context.WebSocket;
var path = context.RequestUri.ToString(); var path = context.RequestUri.ToString();
if (!_servers.ContainsKey(path)) if (!_services.ContainsKey(path))
{ {
socket.Close(HttpStatusCode.NotImplemented); socket.Close(HttpStatusCode.NotImplemented);
return; return;
} }
var server = _servers[path]; var service = _services[path];
server.BindWebSocket(socket); service.BindWebSocket(socket);
} }
#endregion #endregion
@ -85,28 +83,28 @@ namespace WebSocketSharp.Server {
public void AddService<T>(string path) public void AddService<T>(string path)
where T : WebSocketService, new() where T : WebSocketService, new()
{ {
var server = new WebSocketServer<T>(); var service = new WebSocketServer<T>();
_servers.Add(path, server); _services.Add(path, service);
} }
public override void Stop() public override void Stop()
{ {
base.Stop(); base.Stop();
foreach (var server in _servers.Values) foreach (var service in _services.Values)
server.Stop(); service.Stop();
_servers.Clear(); _services.Clear();
} }
#endregion #endregion
} }
public class WebSocketServer<T> : WebSocketServerBase, IWebSocketServer public class WebSocketServer<T> : WebSocketServerBase, IServiceHost
where T : WebSocketService, new() where T : WebSocketService, new()
{ {
#region Fields #region Fields
private Dictionary<string, WebSocketService> _services; private SessionManager _sessions;
private Uri _uri; private Uri _uri;
#endregion #endregion
@ -162,7 +160,7 @@ namespace WebSocketSharp.Server {
private void init() private void init()
{ {
_services = new Dictionary<string, WebSocketService>(); _sessions = new SessionManager();
} }
#endregion #endregion
@ -182,29 +180,14 @@ namespace WebSocketSharp.Server {
public void BindWebSocket(WebSocket socket) public void BindWebSocket(WebSocket socket)
{ {
T service = new T(); T service = new T();
service.Bind(socket, _services); service.Bind(socket, _sessions);
service.Start(); service.Start();
} }
public override void Stop() public override void Stop()
{ {
base.Stop(); base.Stop();
StopServices(); _sessions.Stop();
}
public void StopServices()
{
StopServices(CloseStatusCode.NORMAL, String.Empty);
}
public void StopServices(CloseStatusCode code, string reason)
{
lock (((ICollection)_services).SyncRoot)
{
foreach (WebSocketService service in _services.Values)
service.Stop(code, reason);
_services.Clear();
}
} }
#endregion #endregion

View File

@ -27,7 +27,6 @@
#endregion #endregion
using System; using System;
using System.Collections;
using System.Collections.Generic; using System.Collections.Generic;
using System.Threading; using System.Threading;
using WebSocketSharp.Frame; using WebSocketSharp.Frame;
@ -38,16 +37,8 @@ namespace WebSocketSharp.Server
{ {
#region Private Fields #region Private Fields
private Dictionary<string, WebSocketService> _services; private SessionManager _sessions;
private WebSocket _socket; private WebSocket _socket;
#endregion
#region Properties
public string ID { get; private set; }
public bool IsBound { get; private set; }
public bool IsStop { get; private set; }
#endregion #endregion
@ -55,53 +46,37 @@ namespace WebSocketSharp.Server
public WebSocketService() public WebSocketService()
{ {
ID = String.Empty; ID = String.Empty;
IsBound = false; IsBound = false;
IsStop = false; IsStopped = false;
} }
#endregion #endregion
#region Properties
public string ID { get; private set; }
public bool IsBound { get; private set; }
public bool IsStopped { get; private set; }
#endregion
#region Private Method #region Private Method
private void addService(string id, WebSocketService service)
{
lock (((ICollection)_services).SyncRoot)
{
_services.Add(id, service);
}
}
private string getNewID()
{
return Guid.NewGuid().ToString("N");
}
private void defaultBind() private void defaultBind()
{ {
_socket.OnOpen += (sender, e) => _socket.OnOpen += (sender, e) =>
{ {
ID = getNewID(); ID = _sessions.Add(this);
addService(ID, this);
}; };
_socket.OnClose += (sender, e) => _socket.OnClose += (sender, e) =>
{ {
if (!IsStop) if (!IsStopped)
{ _sessions.Remove(ID);
removeService(ID);
}
}; };
} }
private void removeService(string id)
{
lock (((ICollection)_services).SyncRoot)
{
_services.Remove(id);
}
}
#endregion #endregion
#region Protected Methods #region Protected Methods
@ -126,10 +101,10 @@ namespace WebSocketSharp.Server
#region Public Methods #region Public Methods
public void Bind(WebSocket socket, Dictionary<string, WebSocketService> services) public void Bind(WebSocket socket, SessionManager sessions)
{ {
_socket = socket; _socket = socket;
_services = services; _sessions = sessions;
defaultBind(); defaultBind();
_socket.OnOpen += onOpen; _socket.OnOpen += onOpen;
@ -159,12 +134,9 @@ namespace WebSocketSharp.Server
public Dictionary<string, bool> PingAround(string data) public Dictionary<string, bool> PingAround(string data)
{ {
if (!IsBound) return null; return IsBound
? _sessions.Broadping(data)
lock (((ICollection)_services).SyncRoot) : null;
{
return PingTo(_services.Keys, data);
}
} }
public bool PingTo(string id) public bool PingTo(string id)
@ -172,120 +144,83 @@ namespace WebSocketSharp.Server
return PingTo(id, String.Empty); return PingTo(id, String.Empty);
} }
public Dictionary<string, bool> PingTo(IEnumerable<string> group)
{
return PingTo(group, String.Empty);
}
public bool PingTo(string id, string data) public bool PingTo(string id, string data)
{ {
if (!IsBound) return false; if (!IsBound)
return false;
lock (((ICollection)_services).SyncRoot) WebSocketService service;
{ return _sessions.TryGetByID(id, out service)
WebSocketService service; ? service.Ping(data)
: false;
return _services.TryGetValue(id, out service)
? service.Ping(data)
: false;
}
} }
public Dictionary<string, bool> PingTo(IEnumerable<string> group, string data) public void Publish(byte[] data)
{ {
if (!IsBound) return null; if (IsBound)
_sessions.Broadcast(data);
var result = new Dictionary<string, bool>();
lock (((ICollection)_services).SyncRoot)
{
foreach (string id in group)
{
result.Add(id, PingTo(id, data));
}
}
return result;
} }
public void Publish<TData>(TData data) public void Publish(string data)
{ {
if (!IsBound) return; if (IsBound)
_sessions.Broadcast(data);
WaitCallback broadcast = (state) =>
{
lock (((ICollection)_services).SyncRoot)
{
SendTo(_services.Keys, data);
}
};
ThreadPool.QueueUserWorkItem(broadcast);
} }
public void Send(byte[] data) public void Send(byte[] data)
{ {
if (IsBound) _socket.Send(data); if (IsBound)
_socket.Send(data);
} }
public void Send(string data) public void Send(string data)
{ {
if (IsBound) _socket.Send(data); if (IsBound)
_socket.Send(data);
} }
public void SendTo<TData>(string id, TData data) public void SendAsync(byte[] data)
{ {
if (!IsBound) return; WaitCallback sendCb = (state) =>
if (typeof(TData) != typeof(string) &&
typeof(TData) != typeof(byte[]))
{ {
var msg = "Type of data must be string or byte[]."; Send(data);
throw new ArgumentException(msg); };
} ThreadPool.QueueUserWorkItem(sendCb);
lock (((ICollection)_services).SyncRoot)
{
WebSocketService service;
if (_services.TryGetValue(id, out service))
{
if (typeof(TData) == typeof(string))
{
string data_ = (string)(object)data;
service.Send(data_);
}
else if (typeof(TData) == typeof(byte[]))
{
byte[] data_ = (byte[])(object)data;
service.Send(data_);
}
}
}
} }
public void SendTo<TData>(IEnumerable<string> group, TData data) public void SendAsync(string data)
{ {
if (!IsBound) return; WaitCallback sendCb = (state) =>
if (typeof(TData) != typeof(string) &&
typeof(TData) != typeof(byte[]))
{ {
var msg = "Type of data must be string or byte[]."; Send(data);
throw new ArgumentException(msg); };
} ThreadPool.QueueUserWorkItem(sendCb);
}
lock (((ICollection)_services).SyncRoot) public void SendTo(string id, byte[] data)
{ {
foreach (string id in group) if (!IsBound)
{ return;
SendTo(id, data);
} WebSocketService service;
} if (_sessions.TryGetByID(id, out service))
service.Send(data);
}
public void SendTo(string id, string data)
{
if (!IsBound)
return;
WebSocketService service;
if (_sessions.TryGetByID(id, out service))
service.Send(data);
} }
public void Start() public void Start()
{ {
if (IsBound) _socket.Connect(); if (IsBound)
_socket.Connect();
} }
public void Stop() public void Stop()
@ -295,9 +230,10 @@ namespace WebSocketSharp.Server
public void Stop(CloseStatusCode code, string reason) public void Stop(CloseStatusCode code, string reason)
{ {
if (!IsBound || IsStop) return; if (!IsBound || IsStopped)
return;
IsStop = true; IsStopped = true;
_socket.Close(code, reason); _socket.Close(code, reason);
} }

View File

@ -32,11 +32,9 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Collections.Specialized;
using System.Diagnostics; using System.Diagnostics;
using System.IO; using System.IO;
using System.Linq; using System.Linq;
using System.Net.Security;
using System.Net.Sockets; using System.Net.Sockets;
using System.Text; using System.Text;
using System.Threading; using System.Threading;
@ -588,12 +586,9 @@ namespace WebSocketSharp {
{ {
try try
{ {
MessageEventArgs eventArgs = receive(); var eventArgs = receive();
if (eventArgs != null) if (eventArgs != null)
{
OnMessage.Emit(this, eventArgs); OnMessage.Emit(this, eventArgs);
}
} }
catch (WsReceivedTooBigMessageException ex) catch (WsReceivedTooBigMessageException ex)
{ {
@ -609,7 +604,6 @@ namespace WebSocketSharp {
{ {
Action messageInvoker = (Action)ar.AsyncState; Action messageInvoker = (Action)ar.AsyncState;
messageInvoker.EndInvoke(ar); messageInvoker.EndInvoke(ar);
if (_readyState == WsState.OPEN) if (_readyState == WsState.OPEN)
{ {
messageInvoker.BeginInvoke(messageLoopCallback, messageInvoker); messageInvoker.BeginInvoke(messageLoopCallback, messageInvoker);
@ -619,6 +613,22 @@ namespace WebSocketSharp {
_exitMessageLoop.Set(); _exitMessageLoop.Set();
} }
private bool ping(string data, int millisecondsTimeout)
{
var buffer = Encoding.UTF8.GetBytes(data);
if (buffer.Length > 125)
{
var msg = "Ping frame must have a payload length of 125 bytes or less.";
error(msg);
return false;
}
if (!send(Fin.FINAL, Opcode.PING, buffer))
return false;
return _receivePong.WaitOne(millisecondsTimeout);
}
private void pong(PayloadData data) private void pong(PayloadData data)
{ {
var frame = createFrame(Fin.FINAL, Opcode.PONG, data); var frame = createFrame(Fin.FINAL, Opcode.PONG, data);
@ -643,12 +653,11 @@ namespace WebSocketSharp {
return frame; return frame;
} }
private WsFrame readFrameWithTimeout() private WsFrame readFrameWithTimeout(int millisecondsTimeout)
{ {
if (!_wsStream.DataAvailable) if (!_wsStream.DataAvailable)
{ {
var timeout = 1 * 100; Thread.Sleep(millisecondsTimeout);
Thread.Sleep(timeout);
if (!_wsStream.DataAvailable) if (!_wsStream.DataAvailable)
return null; return null;
} }
@ -663,7 +672,7 @@ namespace WebSocketSharp {
private MessageEventArgs receive() private MessageEventArgs receive()
{ {
var frame = _isClient ? readFrame() : readFrameWithTimeout(); var frame = _isClient ? readFrame() : readFrameWithTimeout(1 * 100);
if (frame == null) if (frame == null)
return null; return null;
@ -939,7 +948,6 @@ namespace WebSocketSharp {
private void startMessageThread() private void startMessageThread()
{ {
_exitMessageLoop = new AutoResetEvent(false); _exitMessageLoop = new AutoResetEvent(false);
Action messageInvoker = () => Action messageInvoker = () =>
{ {
if (_readyState == WsState.OPEN) if (_readyState == WsState.OPEN)
@ -1022,18 +1030,7 @@ namespace WebSocketSharp {
public bool Ping(string data) public bool Ping(string data)
{ {
var buffer = Encoding.UTF8.GetBytes(data); return ping(data, 5 * 1000);
if (buffer.Length > 125)
{
var msg = "Ping frame must have a payload length of 125 bytes or less.";
error(msg);
return false;
}
if (!send(Fin.FINAL, Opcode.PING, buffer))
return false;
return _receivePong.WaitOne(5 * 1000);
} }
public void Send(string data) public void Send(string data)

View File

@ -107,10 +107,11 @@
<Compile Include="Server\ResponseEventArgs.cs" /> <Compile Include="Server\ResponseEventArgs.cs" />
<Compile Include="Net\HttpVersion.cs" /> <Compile Include="Net\HttpVersion.cs" />
<Compile Include="Net\HttpStatusCode.cs" /> <Compile Include="Net\HttpStatusCode.cs" />
<Compile Include="Server\IWebSocketServer.cs" />
<Compile Include="Net\Sockets\TcpListenerWebSocketContext.cs" /> <Compile Include="Net\Sockets\TcpListenerWebSocketContext.cs" />
<Compile Include="Server\WebSocketServerBase.cs" /> <Compile Include="Server\WebSocketServerBase.cs" />
<Compile Include="Net\Security\SslStream.cs" /> <Compile Include="Net\Security\SslStream.cs" />
<Compile Include="Server\IServiceHost.cs" />
<Compile Include="Server\SessionManager.cs" />
</ItemGroup> </ItemGroup>
<Import Project="$(MSBuildBinPath)\Microsoft.CSharp.targets" /> <Import Project="$(MSBuildBinPath)\Microsoft.CSharp.targets" />
<ItemGroup> <ItemGroup>

Binary file not shown.