Fix due to the added QueryString property in WebSocketService.cs
This commit is contained in:
parent
a7eef35c96
commit
c55b5d6479
Binary file not shown.
@ -77,7 +77,11 @@ namespace Example
|
|||||||
//using (WebSocket ws = new WebSocket("wss://echo.websocket.org", "echo"))
|
//using (WebSocket ws = new WebSocket("wss://echo.websocket.org", "echo"))
|
||||||
//using (WebSocket ws = new WebSocket("ws://localhost:4649"))
|
//using (WebSocket ws = new WebSocket("ws://localhost:4649"))
|
||||||
//using (WebSocket ws = new WebSocket("ws://localhost:4649/Echo"))
|
//using (WebSocket ws = new WebSocket("ws://localhost:4649/Echo"))
|
||||||
|
//using (WebSocket ws = new WebSocket("ws://localhost:4649/Echo?name=nobita"))
|
||||||
|
//using (WebSocket ws = new WebSocket("ws://localhost:4649/エコー?name=のび太"))
|
||||||
//using (WebSocket ws = new WebSocket("ws://localhost:4649/Chat"))
|
//using (WebSocket ws = new WebSocket("ws://localhost:4649/Chat"))
|
||||||
|
//using (WebSocket ws = new WebSocket("ws://localhost:4649/Chat?name=nobita"))
|
||||||
|
//using (WebSocket ws = new WebSocket("ws://localhost:4649/チャット?name=のび太"))
|
||||||
{
|
{
|
||||||
ws.OnOpen += (sender, e) =>
|
ws.OnOpen += (sender, e) =>
|
||||||
{
|
{
|
||||||
|
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.
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.
@ -6,9 +6,37 @@ namespace Example2
|
|||||||
{
|
{
|
||||||
public class Chat : WebSocketService
|
public class Chat : WebSocketService
|
||||||
{
|
{
|
||||||
protected override void onMessage(object sender, MessageEventArgs e)
|
private static object _forId = new object();
|
||||||
|
private static uint _id = 0;
|
||||||
|
|
||||||
|
private string _name;
|
||||||
|
|
||||||
|
private string getName()
|
||||||
{
|
{
|
||||||
Publish(e.Data);
|
lock (_forId)
|
||||||
|
{
|
||||||
|
return QueryString.Exists("name")
|
||||||
|
? QueryString["name"]
|
||||||
|
: "anon#" + (++_id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void OnOpen(object sender, EventArgs e)
|
||||||
|
{
|
||||||
|
_name = getName();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void OnMessage(object sender, MessageEventArgs e)
|
||||||
|
{
|
||||||
|
|
||||||
|
var msg = String.Format("{0}: {1}", _name, e.Data);
|
||||||
|
Publish(msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void OnClose(object sender, CloseEventArgs e)
|
||||||
|
{
|
||||||
|
var msg = String.Format("{0} got logged off...", _name);
|
||||||
|
Publish(msg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,18 +2,16 @@ using System;
|
|||||||
using WebSocketSharp;
|
using WebSocketSharp;
|
||||||
using WebSocketSharp.Server;
|
using WebSocketSharp.Server;
|
||||||
|
|
||||||
namespace Example2
|
namespace Example2 {
|
||||||
{
|
|
||||||
public class Echo : WebSocketService
|
public class Echo : WebSocketService
|
||||||
{
|
{
|
||||||
protected override void onMessage(object sender, MessageEventArgs e)
|
protected override void OnMessage(object sender, MessageEventArgs e)
|
||||||
{
|
{
|
||||||
Send(e.Data);
|
var msg = QueryString.Exists("name")
|
||||||
}
|
? String.Format("'{0}' returns to {1}", e.Data, QueryString["name"])
|
||||||
|
: e.Data;
|
||||||
protected override void onClose(object sender, CloseEventArgs e)
|
Send(msg);
|
||||||
{
|
|
||||||
Console.WriteLine("[Echo] Close({0})", e.Code);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Binary file not shown.
@ -8,10 +8,18 @@ namespace Example2
|
|||||||
public static void Main(string[] args)
|
public static void Main(string[] args)
|
||||||
{
|
{
|
||||||
/* Single service server
|
/* Single service server
|
||||||
var wssv = new WebSocketServer<Echo>("ws://localhost:4649");
|
//var wssv = new WebSocketServer<Echo>("ws://localhost:4649");
|
||||||
|
var wssv = new WebSocketServer<Echo>("ws://localhost:4649/Echo");
|
||||||
|
//var wssv = new WebSocketServer<Echo>("ws://localhost:4649/エコー");
|
||||||
//var wssv = new WebSocketServer<Echo>(4649);
|
//var wssv = new WebSocketServer<Echo>(4649);
|
||||||
|
//var wssv = new WebSocketServer<Echo>(4649, "/Echo");
|
||||||
|
//var wssv = new WebSocketServer<Echo>(4649, "/エコー");
|
||||||
//var wssv = new WebSocketServer<Chat>("ws://localhost:4649");
|
//var wssv = new WebSocketServer<Chat>("ws://localhost:4649");
|
||||||
|
//var wssv = new WebSocketServer<Chat>("ws://localhost:4649/Chat");
|
||||||
|
//var wssv = new WebSocketServer<Chat>("ws://localhost:4649/チャット");
|
||||||
//var wssv = new WebSocketServer<Chat>(4649);
|
//var wssv = new WebSocketServer<Chat>(4649);
|
||||||
|
//var wssv = new WebSocketServer<Chat>(4649, "/Chat");
|
||||||
|
//var wssv = new WebSocketServer<Chat>(4649, "/チャット");
|
||||||
|
|
||||||
wssv.Start();
|
wssv.Start();
|
||||||
Console.WriteLine(
|
Console.WriteLine(
|
||||||
@ -22,7 +30,9 @@ namespace Example2
|
|||||||
// Multi services server
|
// Multi services server
|
||||||
var wssv = new WebSocketServer(4649);
|
var wssv = new WebSocketServer(4649);
|
||||||
wssv.AddService<Echo>("/Echo");
|
wssv.AddService<Echo>("/Echo");
|
||||||
|
wssv.AddService<Echo>("/エコー");
|
||||||
wssv.AddService<Chat>("/Chat");
|
wssv.AddService<Chat>("/Chat");
|
||||||
|
wssv.AddService<Chat>("/チャット");
|
||||||
|
|
||||||
wssv.Start();
|
wssv.Start();
|
||||||
Console.WriteLine(
|
Console.WriteLine(
|
||||||
|
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.
@ -6,9 +6,37 @@ namespace Example3
|
|||||||
{
|
{
|
||||||
public class Chat : WebSocketService
|
public class Chat : WebSocketService
|
||||||
{
|
{
|
||||||
protected override void onMessage(object sender, MessageEventArgs e)
|
private static object _forId = new object();
|
||||||
|
private static uint _id = 0;
|
||||||
|
|
||||||
|
private string _name;
|
||||||
|
|
||||||
|
private string getName()
|
||||||
{
|
{
|
||||||
Publish(e.Data);
|
lock (_forId)
|
||||||
|
{
|
||||||
|
return QueryString.Exists("name")
|
||||||
|
? QueryString["name"]
|
||||||
|
: "anon#" + (++_id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void OnOpen(object sender, EventArgs e)
|
||||||
|
{
|
||||||
|
_name = getName();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void OnMessage(object sender, MessageEventArgs e)
|
||||||
|
{
|
||||||
|
|
||||||
|
var msg = String.Format("{0}: {1}", _name, e.Data);
|
||||||
|
Publish(msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void OnClose(object sender, CloseEventArgs e)
|
||||||
|
{
|
||||||
|
var msg = String.Format("{0} got logged off...", _name);
|
||||||
|
Publish(msg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,14 +6,12 @@ namespace Example3
|
|||||||
{
|
{
|
||||||
public class Echo : WebSocketService
|
public class Echo : WebSocketService
|
||||||
{
|
{
|
||||||
protected override void onMessage(object sender, MessageEventArgs e)
|
protected override void OnMessage(object sender, MessageEventArgs e)
|
||||||
{
|
{
|
||||||
Send(e.Data);
|
var msg = QueryString.Exists("name")
|
||||||
}
|
? String.Format("'{0}' returns to {1}", e.Data, QueryString["name"])
|
||||||
|
: e.Data;
|
||||||
protected override void onClose(object sender, CloseEventArgs e)
|
Send(msg);
|
||||||
{
|
|
||||||
Console.WriteLine("[Echo] Close({0})", e.Code);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Binary file not shown.
@ -12,7 +12,8 @@ namespace Example3
|
|||||||
public static void Main(string[] args)
|
public static void Main(string[] args)
|
||||||
{
|
{
|
||||||
_httpsv = new HttpServer(4649);
|
_httpsv = new HttpServer(4649);
|
||||||
_httpsv.AddService<Echo>("/");
|
_httpsv.AddService<Echo>("/Echo");
|
||||||
|
_httpsv.AddService<Chat>("/Chat");
|
||||||
|
|
||||||
_httpsv.OnGet += (sender, e) =>
|
_httpsv.OnGet += (sender, e) =>
|
||||||
{
|
{
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var wsUri = "ws://localhost:4649/";
|
var wsUri = "ws://localhost:4649/Echo";
|
||||||
var output;
|
var output;
|
||||||
|
|
||||||
function init(){
|
function init(){
|
||||||
|
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.
@ -164,7 +164,7 @@ using WebSocketSharp.Server;
|
|||||||
|
|
||||||
public class Echo : WebSocketService
|
public class Echo : WebSocketService
|
||||||
{
|
{
|
||||||
protected override void onMessage(object sender, MessageEventArgs e)
|
protected override void OnMessage(object sender, MessageEventArgs e)
|
||||||
{
|
{
|
||||||
Send(e.Data);
|
Send(e.Data);
|
||||||
}
|
}
|
||||||
@ -180,16 +180,16 @@ using WebSocketSharp.Server;
|
|||||||
|
|
||||||
public class Chat : WebSocketService
|
public class Chat : WebSocketService
|
||||||
{
|
{
|
||||||
protected override void onMessage(object sender, MessageEventArgs e)
|
protected override void OnMessage(object sender, MessageEventArgs e)
|
||||||
{
|
{
|
||||||
Publish(e.Data);
|
Publish(e.Data);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
If you override the `onMessage` method, it is bound to the server side `WebSocket.OnMessage` event.
|
If you override the `OnMessage` method, it is bound to the server side `WebSocket.OnMessage` event.
|
||||||
|
|
||||||
In addition, if you override the `onOpen`, `onError` and `onClose` methods, each of them is bound to the `WebSocket.OnOpen`, `WebSocket.OnError` and `WebSocket.OnClose` events.
|
In addition, if you override the `OnOpen`, `OnError` and `OnClose` methods, each of them is bound to the `WebSocket.OnOpen`, `WebSocket.OnError` and `WebSocket.OnClose` events.
|
||||||
|
|
||||||
#### Step 3 ####
|
#### Step 3 ####
|
||||||
|
|
||||||
|
@ -52,23 +52,46 @@ namespace WebSocketSharp {
|
|||||||
return new TcpListenerWebSocketContext(client);
|
return new TcpListenerWebSocketContext(client);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Emit the specified <see cref="EventHandler"/> delegate if is not <see langword="null"/>.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="eventHandler">
|
||||||
|
/// An <see cref="EventHandler"/> to be emitted.
|
||||||
|
/// </param>
|
||||||
|
/// <param name="sender">
|
||||||
|
/// An <see cref="object"/> that emits this <paramref name="eventHandler"/>.
|
||||||
|
/// </param>
|
||||||
|
/// <param name="e">
|
||||||
|
/// An <see cref="EventArgs"/> that contains no event data.
|
||||||
|
/// </param>
|
||||||
public static void Emit(
|
public static void Emit(
|
||||||
this EventHandler eventHandler, object sender, EventArgs e)
|
this EventHandler eventHandler, object sender, EventArgs e)
|
||||||
{
|
{
|
||||||
if (eventHandler != null)
|
if (!eventHandler.IsNull())
|
||||||
{
|
|
||||||
eventHandler(sender, e);
|
eventHandler(sender, e);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Emit the specified <see cref="EventHandler<TEventArgs>"/> delegate if is not <see langword="null"/>.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="eventHandler">
|
||||||
|
/// An <see cref="EventHandler<TEventArgs>"/> to be emitted.
|
||||||
|
/// </param>
|
||||||
|
/// <param name="sender">
|
||||||
|
/// An <see cref="object"/> that emits this <paramref name="eventHandler"/>.
|
||||||
|
/// </param>
|
||||||
|
/// <param name="e">
|
||||||
|
/// An <see cref="EventArgs"/> that contains the event data.
|
||||||
|
/// </param>
|
||||||
|
/// <typeparam name="TEventArgs">
|
||||||
|
/// The type of the event data generated by the event.
|
||||||
|
/// </typeparam>
|
||||||
public static void Emit<TEventArgs>(
|
public static void Emit<TEventArgs>(
|
||||||
this EventHandler<TEventArgs> eventHandler, object sender, TEventArgs e)
|
this EventHandler<TEventArgs> eventHandler, object sender, TEventArgs e)
|
||||||
where TEventArgs : EventArgs
|
where TEventArgs : EventArgs
|
||||||
{
|
{
|
||||||
if (eventHandler != null)
|
if (!eventHandler.IsNull())
|
||||||
{
|
|
||||||
eventHandler(sender, e);
|
eventHandler(sender, e);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static bool EqualsAndSaveTo(this int value, char c, List<byte> dest)
|
public static bool EqualsAndSaveTo(this int value, char c, List<byte> dest)
|
||||||
@ -81,17 +104,17 @@ namespace WebSocketSharp {
|
|||||||
return b == Convert.ToByte(c);
|
return b == Convert.ToByte(c);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static bool Exists(this NameValueCollection headers, string name)
|
public static bool Exists(this NameValueCollection collections, string name)
|
||||||
{
|
{
|
||||||
return headers[name] != null
|
return collections[name] != null
|
||||||
? true
|
? true
|
||||||
: false;
|
: false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static bool Exists(this NameValueCollection headers, string name, string value)
|
public static bool Exists(this NameValueCollection collections, string name, string value)
|
||||||
{
|
{
|
||||||
var values = headers[name];
|
var values = collections[name];
|
||||||
if (values == null)
|
if (values.IsNull())
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
foreach (string v in values.Split(','))
|
foreach (string v in values.Split(','))
|
||||||
@ -101,6 +124,27 @@ namespace WebSocketSharp {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static string GetAbsolutePath(this Uri uri)
|
||||||
|
{
|
||||||
|
if (uri.IsAbsoluteUri)
|
||||||
|
return uri.AbsolutePath;
|
||||||
|
|
||||||
|
var uriString = uri.OriginalString;
|
||||||
|
|
||||||
|
var i = uriString.IndexOf('/');
|
||||||
|
if (i != 0)
|
||||||
|
{
|
||||||
|
var msg = "Not absolute path: " + uriString;
|
||||||
|
throw new ArgumentException(msg, "uri");
|
||||||
|
}
|
||||||
|
|
||||||
|
var j = uriString.IndexOfAny(new []{'?', '#'});
|
||||||
|
if (j > 0)
|
||||||
|
return uriString.Substring(0, j);
|
||||||
|
|
||||||
|
return uriString;
|
||||||
|
}
|
||||||
|
|
||||||
public static string GetDescription(this HttpStatusCode code)
|
public static string GetDescription(this HttpStatusCode code)
|
||||||
{
|
{
|
||||||
return ((int)code).GetStatusDescription();
|
return ((int)code).GetStatusDescription();
|
||||||
@ -162,11 +206,32 @@ namespace WebSocketSharp {
|
|||||||
return String.Empty;
|
return String.Empty;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static string GetName(this string nameAndValue, string separator)
|
||||||
|
{
|
||||||
|
var i = nameAndValue.IndexOf(separator);
|
||||||
|
if (i <= 0)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
return nameAndValue.Substring(0, i).Trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static KeyValuePair<string, string> GetNameAndValue(this string nameAndValue, string separator)
|
||||||
|
{
|
||||||
|
var i = nameAndValue.IndexOf(separator);
|
||||||
|
if (i <= 0 || i == nameAndValue.Length - 1)
|
||||||
|
return new KeyValuePair<string, string>(null, null);
|
||||||
|
|
||||||
|
var name = nameAndValue.Substring(0, i).Trim();
|
||||||
|
var value = nameAndValue.Substring(i + 1).Trim();
|
||||||
|
|
||||||
|
return new KeyValuePair<string, string>(name, value);
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the value from a <see cref="string"/> that contains a pair of name and value are separated by a separator string.
|
/// Gets the value from a <see cref="string"/> that contains a pair of name and value are separated by a separator string.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <returns>
|
/// <returns>
|
||||||
/// A <see cref="string"/> that contains the value if can get; otherwise, <c>null</c>.
|
/// A <see cref="string"/> that contains the value if any; otherwise, <c>null</c>.
|
||||||
/// </returns>
|
/// </returns>
|
||||||
/// <param name="nameAndValue">
|
/// <param name="nameAndValue">
|
||||||
/// A <see cref="string"/> that contains a pair of name and value are separated by a separator string.
|
/// A <see cref="string"/> that contains a pair of name and value are separated by a separator string.
|
||||||
@ -177,7 +242,7 @@ namespace WebSocketSharp {
|
|||||||
public static string GetValue(this string nameAndValue, string separator)
|
public static string GetValue(this string nameAndValue, string separator)
|
||||||
{
|
{
|
||||||
var i = nameAndValue.IndexOf(separator);
|
var i = nameAndValue.IndexOf(separator);
|
||||||
if (i <= 0)
|
if (i == -1 || i == nameAndValue.Length - 1)
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
return nameAndValue.Substring(i + 1).Trim();
|
return nameAndValue.Substring(i + 1).Trim();
|
||||||
@ -195,10 +260,42 @@ namespace WebSocketSharp {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static bool IsNullDo<T>(this T value, Action act)
|
/// <summary>
|
||||||
|
/// Determines whether the specified <see cref="string"/> is <see cref="String.Empty"/>.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>
|
||||||
|
/// <c>true</c> if the <paramref name="value"/> is <see cref="String.Empty"/>; otherwise, <c>false</c>.
|
||||||
|
/// </returns>
|
||||||
|
/// <param name="value">
|
||||||
|
/// A <see cref="string"/> to test.
|
||||||
|
/// </param>
|
||||||
|
public static bool IsEmpty(this string value)
|
||||||
|
{
|
||||||
|
return value == String.Empty ? true : false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Determines whether the specified object is <see langword="null"/>.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>
|
||||||
|
/// <c>true</c> if the specified object is <see langword="null"/>; otherwise, <c>false</c>.
|
||||||
|
/// </returns>
|
||||||
|
/// <param name="obj">
|
||||||
|
/// An <see cref="class"/> to test.
|
||||||
|
/// </param>
|
||||||
|
/// <typeparam name="T">
|
||||||
|
/// The type of this <paramref name="obj"/>.
|
||||||
|
/// </typeparam>
|
||||||
|
public static bool IsNull<T>(this T obj)
|
||||||
where T : class
|
where T : class
|
||||||
{
|
{
|
||||||
if (value == null)
|
return obj == null ? true : false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static bool IsNullDo<T>(this T obj, Action act)
|
||||||
|
where T : class
|
||||||
|
{
|
||||||
|
if (obj.IsNull())
|
||||||
{
|
{
|
||||||
act();
|
act();
|
||||||
return true;
|
return true;
|
||||||
@ -207,10 +304,50 @@ namespace WebSocketSharp {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Determines whether the specified <see cref="string"/> is <see langword="null"/> or <see cref="String.Empty"/>.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>
|
||||||
|
/// <c>true</c> if the specified <see cref="string"/> is <see langword="null"/> or <see cref="String.Empty"/>; otherwise, <c>false</c>.
|
||||||
|
/// </returns>
|
||||||
|
/// <param name="value">
|
||||||
|
/// A <see cref="string"/> to test.
|
||||||
|
/// </param>
|
||||||
|
public static bool IsNullOrEmpty(this string value)
|
||||||
|
{
|
||||||
|
return String.IsNullOrEmpty(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static bool IsValidAbsolutePath(this string absPath, out string message)
|
||||||
|
{
|
||||||
|
if (absPath.IsEmpty())
|
||||||
|
{
|
||||||
|
message = "Must not be empty.";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
var i = absPath.IndexOf('/');
|
||||||
|
if (i != 0)
|
||||||
|
{
|
||||||
|
message = "Not absolute path: " + absPath;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
var j = absPath.IndexOfAny(new []{'?', '#'});
|
||||||
|
if (j != -1)
|
||||||
|
{
|
||||||
|
message = "Must not contain either or both query and fragment components: " + absPath;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
message = String.Empty;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
// Derived from System.Uri.IsPredefinedScheme method
|
// Derived from System.Uri.IsPredefinedScheme method
|
||||||
public static bool IsPredefinedScheme(this string scheme)
|
public static bool IsPredefinedScheme(this string scheme)
|
||||||
{
|
{
|
||||||
if (scheme == null && scheme.Length < 2)
|
if (scheme.IsNull() && scheme.Length < 2)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
char c = scheme[0];
|
char c = scheme[0];
|
||||||
@ -243,56 +380,6 @@ namespace WebSocketSharp {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Determines whether <paramref name="uri"/> is valid WebSocket URI.
|
|
||||||
/// </summary>
|
|
||||||
/// <returns>
|
|
||||||
/// <c>true</c> if <paramref name="uri"/> is valid WebSocket URI; otherwise, <c>false</c>.
|
|
||||||
/// </returns>
|
|
||||||
/// <param name="uri">
|
|
||||||
/// A <see cref="Uri"/> that contains a WebSocket URI.
|
|
||||||
/// </param>
|
|
||||||
/// <param name="message">
|
|
||||||
/// A <see cref="string"/> that contains a error message if <paramref name="uri"/> is invalid WebSocket URI; otherwise, <c>String.Empty</c>.
|
|
||||||
/// </param>
|
|
||||||
public static bool IsValidWebSocketUri(this Uri uri, out string message)
|
|
||||||
{
|
|
||||||
if (!uri.IsAbsoluteUri)
|
|
||||||
{
|
|
||||||
message = "Not absolute URI: " + uri.ToString();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
var scheme = uri.Scheme;
|
|
||||||
if (scheme != "ws" && scheme != "wss")
|
|
||||||
{
|
|
||||||
message = "Unsupported WebSocket URI scheme: " + scheme;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
var original = uri.OriginalString;
|
|
||||||
if (original.Contains('#'))
|
|
||||||
{
|
|
||||||
message = "WebSocket URI must not contain a fragment component: " + original;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
var port = uri.Port;
|
|
||||||
if (port > 0)
|
|
||||||
{
|
|
||||||
if ((scheme == "ws" && port == 443) ||
|
|
||||||
(scheme == "wss" && port == 80))
|
|
||||||
{
|
|
||||||
message = String.Format(
|
|
||||||
"Invalid pair of WebSocket URI scheme and port: {0}, {1}", scheme, port);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
message = String.Empty;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Derived from System.Uri.MaybeUri method
|
// Derived from System.Uri.MaybeUri method
|
||||||
public static bool MaybeUri(this string uriString)
|
public static bool MaybeUri(this string uriString)
|
||||||
{
|
{
|
||||||
@ -372,12 +459,11 @@ namespace WebSocketSharp {
|
|||||||
public static T[] SubArray<T>(this T[] array, int startIndex, int length)
|
public static T[] SubArray<T>(this T[] array, int startIndex, int length)
|
||||||
{
|
{
|
||||||
if (startIndex == 0 && array.Length == length)
|
if (startIndex == 0 && array.Length == length)
|
||||||
{
|
|
||||||
return array;
|
return array;
|
||||||
}
|
|
||||||
|
|
||||||
T[] subArray = new T[length];
|
T[] subArray = new T[length];
|
||||||
Array.Copy(array, startIndex, subArray, 0, length);
|
Array.Copy(array, startIndex, subArray, 0, length);
|
||||||
|
|
||||||
return subArray;
|
return subArray;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -564,6 +650,85 @@ namespace WebSocketSharp {
|
|||||||
return new Uri(uriString);
|
return new Uri(uriString);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Tries to create a new WebSocket <see cref="Uri"/> using <paramref name="uriString"/>.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>
|
||||||
|
/// <c>true</c> if the WebSocket <see cref="Uri"/> was successfully created; otherwise, <c>false</c>.
|
||||||
|
/// </returns>
|
||||||
|
/// <param name="uriString">
|
||||||
|
/// A <see cref="string"/> that contains a WebSocket URI.
|
||||||
|
/// </param>
|
||||||
|
/// <param name="result">
|
||||||
|
/// When this method returns, contains a created WebSocket <see cref="Uri"/> if <paramref name="uriString"/> is valid WebSocket URI; otherwise, <see langword="null"/>.
|
||||||
|
/// </param>
|
||||||
|
/// <param name="message">
|
||||||
|
/// When this method returns, contains a error message <see cref="string"/> if <paramref name="uriString"/> is invalid WebSocket URI; otherwise, <c>String.Empty</c>.
|
||||||
|
/// </param>
|
||||||
|
/// <exception cref="ArgumentNullException">
|
||||||
|
/// Is thrown when <paramref name="uriString"/> passed to a method is invalid because it is <see langword="null"/>.
|
||||||
|
/// </exception>
|
||||||
|
public static bool TryCreateWebSocketUri(this string uriString, out Uri result, out string message)
|
||||||
|
{
|
||||||
|
if (uriString == null)
|
||||||
|
throw new ArgumentNullException("uriString");
|
||||||
|
|
||||||
|
result = null;
|
||||||
|
if (uriString == String.Empty)
|
||||||
|
{
|
||||||
|
message = "Must not be empty.";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
var uri = uriString.ToUri();
|
||||||
|
if (!uri.IsAbsoluteUri)
|
||||||
|
{
|
||||||
|
message = "Not absolute URI: " + uriString;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
var scheme = uri.Scheme;
|
||||||
|
if (scheme != "ws" && scheme != "wss")
|
||||||
|
{
|
||||||
|
message = "Unsupported scheme: " + scheme;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
var fragment = uri.Fragment;
|
||||||
|
if (!String.IsNullOrEmpty(fragment))
|
||||||
|
{
|
||||||
|
message = "Must not contain the fragment component: " + uriString;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
var port = uri.Port;
|
||||||
|
if (port > 0)
|
||||||
|
{
|
||||||
|
if ((scheme == "ws" && port == 443) ||
|
||||||
|
(scheme == "wss" && port == 80))
|
||||||
|
{
|
||||||
|
message = String.Format(
|
||||||
|
"Invalid pair of scheme and port: {0}, {1}", scheme, port);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
result = uri;
|
||||||
|
message = String.Empty;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static string UrlDecode(this string s)
|
||||||
|
{
|
||||||
|
return HttpUtility.UrlDecode(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static string UrlEncode(this string s)
|
||||||
|
{
|
||||||
|
return HttpUtility.UrlEncode(s);
|
||||||
|
}
|
||||||
|
|
||||||
public static void WriteContent(this HttpListenerResponse response, byte[] content)
|
public static void WriteContent(this HttpListenerResponse response, byte[] content)
|
||||||
{
|
{
|
||||||
var output = response.OutputStream;
|
var output = response.OutputStream;
|
||||||
|
@ -169,9 +169,9 @@ namespace WebSocketSharp.Net {
|
|||||||
|
|
||||||
#region Public Method
|
#region Public Method
|
||||||
|
|
||||||
public HttpListenerWebSocketContext AcceptWebSocket (string path)
|
public HttpListenerWebSocketContext AcceptWebSocket ()
|
||||||
{
|
{
|
||||||
return new HttpListenerWebSocketContext (path, this);
|
return new HttpListenerWebSocketContext (this);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
@ -39,67 +39,113 @@ namespace WebSocketSharp.Net {
|
|||||||
private WebSocket _socket;
|
private WebSocket _socket;
|
||||||
private WsStream _stream;
|
private WsStream _stream;
|
||||||
|
|
||||||
internal HttpListenerWebSocketContext(string path, HttpListenerContext context)
|
internal HttpListenerWebSocketContext(HttpListenerContext context)
|
||||||
{
|
{
|
||||||
_context = context;
|
_context = context;
|
||||||
_stream = WsStream.CreateServerStream(context);
|
_stream = WsStream.CreateServerStream(context);
|
||||||
_socket = new WebSocket(path.ToUri(), this);
|
_socket = new WebSocket(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
internal HttpListenerContext BaseContext {
|
internal HttpListenerContext BaseContext {
|
||||||
get { return _context; }
|
get {
|
||||||
|
return _context;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal WsStream Stream {
|
internal WsStream Stream {
|
||||||
get { return _stream; }
|
get {
|
||||||
|
return _stream;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override CookieCollection CookieCollection {
|
public override CookieCollection CookieCollection {
|
||||||
get { return _context.Request.Cookies; }
|
get {
|
||||||
|
return _context.Request.Cookies;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override NameValueCollection Headers {
|
public override NameValueCollection Headers {
|
||||||
get { return _context.Request.Headers; }
|
get {
|
||||||
|
return _context.Request.Headers;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override bool IsAuthenticated {
|
public override bool IsAuthenticated {
|
||||||
get { return _context.Request.IsAuthenticated; }
|
get {
|
||||||
|
return _context.Request.IsAuthenticated;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override bool IsSecureConnection {
|
public override bool IsSecureConnection {
|
||||||
get { return _context.Request.IsSecureConnection; }
|
get {
|
||||||
|
return _context.Request.IsSecureConnection;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override bool IsLocal {
|
public override bool IsLocal {
|
||||||
get { return _context.Request.IsLocal; }
|
get {
|
||||||
|
return _context.Request.IsLocal;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override string Origin {
|
public override string Origin {
|
||||||
get { return Headers["Origin"]; }
|
get {
|
||||||
|
return Headers["Origin"];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual string Path {
|
||||||
|
get {
|
||||||
|
return RequestUri.GetAbsolutePath();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override Uri RequestUri {
|
public override Uri RequestUri {
|
||||||
get { return _context.Request.RawUrl.ToUri(); }
|
get {
|
||||||
|
return _context.Request.RawUrl.ToUri();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override string SecWebSocketKey {
|
public override string SecWebSocketKey {
|
||||||
get { return Headers["Sec-WebSocket-Key"]; }
|
get {
|
||||||
|
return Headers["Sec-WebSocket-Key"];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override IEnumerable<string> SecWebSocketProtocols {
|
public override IEnumerable<string> SecWebSocketProtocols {
|
||||||
get { return Headers.GetValues("Sec-WebSocket-Protocol"); }
|
get {
|
||||||
|
return Headers.GetValues("Sec-WebSocket-Protocol");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override string SecWebSocketVersion {
|
public override string SecWebSocketVersion {
|
||||||
get { return Headers["Sec-WebSocket-Version"]; }
|
get {
|
||||||
|
return Headers["Sec-WebSocket-Version"];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual System.Net.IPEndPoint ServerEndPoint {
|
||||||
|
get {
|
||||||
|
return _context.Connection.LocalEndPoint;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override IPrincipal User {
|
public override IPrincipal User {
|
||||||
get { return _context.User; }
|
get {
|
||||||
|
return _context.User;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual System.Net.IPEndPoint UserEndPoint {
|
||||||
|
get {
|
||||||
|
return _context.Connection.RemoteEndPoint;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override WebSocket WebSocket {
|
public override WebSocket WebSocket {
|
||||||
get { return _socket; }
|
get {
|
||||||
|
return _socket;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -29,6 +29,7 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Collections.Specialized;
|
using System.Collections.Specialized;
|
||||||
|
using System.Net;
|
||||||
using System.Net.Sockets;
|
using System.Net.Sockets;
|
||||||
using System.Security.Principal;
|
using System.Security.Principal;
|
||||||
|
|
||||||
@ -44,72 +45,113 @@ namespace WebSocketSharp.Net.Sockets {
|
|||||||
|
|
||||||
internal TcpListenerWebSocketContext(TcpClient client)
|
internal TcpListenerWebSocketContext(TcpClient client)
|
||||||
{
|
{
|
||||||
_client = client;
|
_client = client;
|
||||||
init();
|
_stream = WsStream.CreateServerStream(client);
|
||||||
}
|
|
||||||
|
|
||||||
internal TcpClient Client {
|
|
||||||
get { return _client; }
|
|
||||||
}
|
|
||||||
|
|
||||||
internal WsStream Stream {
|
|
||||||
get { return _stream; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public override CookieCollection CookieCollection {
|
|
||||||
get { throw new NotImplementedException(); }
|
|
||||||
}
|
|
||||||
|
|
||||||
public override NameValueCollection Headers {
|
|
||||||
get { return _request.Headers; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public override bool IsAuthenticated {
|
|
||||||
get { throw new NotImplementedException(); }
|
|
||||||
}
|
|
||||||
|
|
||||||
public override bool IsSecureConnection {
|
|
||||||
get { return _isSecure; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public override bool IsLocal {
|
|
||||||
get { throw new NotImplementedException(); }
|
|
||||||
}
|
|
||||||
|
|
||||||
public override string Origin {
|
|
||||||
get { return Headers["Origin"]; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public override Uri RequestUri {
|
|
||||||
get { return _request.RequestUri; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public override string SecWebSocketKey {
|
|
||||||
get { return Headers["Sec-WebSocket-Key"]; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public override IEnumerable<string> SecWebSocketProtocols {
|
|
||||||
get { return Headers.GetValues("Sec-WebSocket-Protocol"); }
|
|
||||||
}
|
|
||||||
|
|
||||||
public override string SecWebSocketVersion {
|
|
||||||
get { return Headers["Sec-WebSocket-Version"]; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public override IPrincipal User {
|
|
||||||
get { throw new NotImplementedException(); }
|
|
||||||
}
|
|
||||||
|
|
||||||
public override WebSocket WebSocket {
|
|
||||||
get { return _socket; }
|
|
||||||
}
|
|
||||||
|
|
||||||
private void init()
|
|
||||||
{
|
|
||||||
_stream = WsStream.CreateServerStream(_client);
|
|
||||||
_isSecure = _stream.IsSecure;
|
_isSecure = _stream.IsSecure;
|
||||||
_request = RequestHandshake.Parse(_stream.ReadHandshake());
|
_request = RequestHandshake.Parse(_stream.ReadHandshake());
|
||||||
_socket = new WebSocket(this);
|
_socket = new WebSocket(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal TcpClient Client {
|
||||||
|
get {
|
||||||
|
return _client;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
internal WsStream Stream {
|
||||||
|
get {
|
||||||
|
return _stream;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override CookieCollection CookieCollection {
|
||||||
|
get {
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override NameValueCollection Headers {
|
||||||
|
get {
|
||||||
|
return _request.Headers;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override bool IsAuthenticated {
|
||||||
|
get {
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override bool IsSecureConnection {
|
||||||
|
get {
|
||||||
|
return _isSecure;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override bool IsLocal {
|
||||||
|
get {
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override string Origin {
|
||||||
|
get {
|
||||||
|
return Headers["Origin"];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual string Path {
|
||||||
|
get {
|
||||||
|
return _request.RequestUri.GetAbsolutePath();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override Uri RequestUri {
|
||||||
|
get {
|
||||||
|
return _request.RequestUri;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override string SecWebSocketKey {
|
||||||
|
get {
|
||||||
|
return Headers["Sec-WebSocket-Key"];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override IEnumerable<string> SecWebSocketProtocols {
|
||||||
|
get {
|
||||||
|
return Headers.GetValues("Sec-WebSocket-Protocol");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override string SecWebSocketVersion {
|
||||||
|
get {
|
||||||
|
return Headers["Sec-WebSocket-Version"];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual IPEndPoint ServerEndPoint {
|
||||||
|
get {
|
||||||
|
return (IPEndPoint)_client.Client.LocalEndPoint;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override IPrincipal User {
|
||||||
|
get {
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual IPEndPoint UserEndPoint {
|
||||||
|
get {
|
||||||
|
return (IPEndPoint)_client.Client.RemoteEndPoint;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override WebSocket WebSocket {
|
||||||
|
get {
|
||||||
|
return _socket;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -35,6 +35,12 @@ namespace WebSocketSharp {
|
|||||||
|
|
||||||
public class RequestHandshake : Handshake
|
public class RequestHandshake : Handshake
|
||||||
{
|
{
|
||||||
|
#region Private Field
|
||||||
|
|
||||||
|
private NameValueCollection _queryString;
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
#region Private Constructor
|
#region Private Constructor
|
||||||
|
|
||||||
private RequestHandshake()
|
private RequestHandshake()
|
||||||
@ -43,7 +49,6 @@ namespace WebSocketSharp {
|
|||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
|
||||||
#region Public Constructor
|
#region Public Constructor
|
||||||
|
|
||||||
public RequestHandshake(string uriString)
|
public RequestHandshake(string uriString)
|
||||||
@ -59,7 +64,7 @@ namespace WebSocketSharp {
|
|||||||
|
|
||||||
#region Properties
|
#region Properties
|
||||||
|
|
||||||
public string HttpMethod { get; internal set; }
|
public string HttpMethod { get; private set; }
|
||||||
|
|
||||||
public bool IsWebSocketRequest {
|
public bool IsWebSocketRequest {
|
||||||
|
|
||||||
@ -89,7 +94,44 @@ namespace WebSocketSharp {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public Uri RequestUri { get; internal set; }
|
public NameValueCollection QueryString {
|
||||||
|
get {
|
||||||
|
if (_queryString == null)
|
||||||
|
{
|
||||||
|
_queryString = new NameValueCollection();
|
||||||
|
|
||||||
|
var i = RawUrl.IndexOf('?');
|
||||||
|
if (i > 0)
|
||||||
|
{
|
||||||
|
var query = RawUrl.Substring(i + 1);
|
||||||
|
var components = query.Split('&');
|
||||||
|
foreach (var c in components)
|
||||||
|
{
|
||||||
|
var nv = c.GetNameAndValue("=");
|
||||||
|
if (nv.Key != null)
|
||||||
|
{
|
||||||
|
var name = nv.Key.UrlDecode();
|
||||||
|
var val = nv.Value.UrlDecode();
|
||||||
|
_queryString.Add(name, val);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return _queryString;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public string RawUrl {
|
||||||
|
get {
|
||||||
|
if (RequestUri.IsAbsoluteUri)
|
||||||
|
return RequestUri.PathAndQuery;
|
||||||
|
|
||||||
|
return RequestUri.OriginalString;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public Uri RequestUri { get; private set; }
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
@ -109,7 +151,10 @@ namespace WebSocketSharp {
|
|||||||
{
|
{
|
||||||
var requestLine = request[0].Split(' ');
|
var requestLine = request[0].Split(' ');
|
||||||
if (requestLine.Length != 3)
|
if (requestLine.Length != 3)
|
||||||
throw new ArgumentException("Invalid request line.");
|
{
|
||||||
|
var msg = "Invalid HTTP Request-Line: " + request[0];
|
||||||
|
throw new ArgumentException(msg, "request");
|
||||||
|
}
|
||||||
|
|
||||||
var headers = new WebHeaderCollection();
|
var headers = new WebHeaderCollection();
|
||||||
for (int i = 1; i < request.Length; i++)
|
for (int i = 1; i < request.Length; i++)
|
||||||
@ -130,14 +175,11 @@ namespace WebSocketSharp {
|
|||||||
public override string ToString()
|
public override string ToString()
|
||||||
{
|
{
|
||||||
var buffer = new StringBuilder();
|
var buffer = new StringBuilder();
|
||||||
|
buffer.AppendFormat("{0} {1} HTTP/{2}{3}", HttpMethod, RawUrl, ProtocolVersion, _crlf);
|
||||||
buffer.AppendFormat("{0} {1} HTTP/{2}{3}", HttpMethod, RequestUri, ProtocolVersion, _crlf);
|
|
||||||
|
|
||||||
foreach (string key in Headers.AllKeys)
|
foreach (string key in Headers.AllKeys)
|
||||||
buffer.AppendFormat("{0}: {1}{2}", key, Headers[key], _crlf);
|
buffer.AppendFormat("{0}: {1}{2}", key, Headers[key], _crlf);
|
||||||
|
|
||||||
buffer.Append(_crlf);
|
buffer.Append(_crlf);
|
||||||
|
|
||||||
return buffer.ToString();
|
return buffer.ToString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -29,6 +29,7 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Configuration;
|
using System.Configuration;
|
||||||
|
using System.Diagnostics;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using WebSocketSharp.Net;
|
using WebSocketSharp.Net;
|
||||||
@ -95,7 +96,7 @@ namespace WebSocketSharp.Server {
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
var context = _listener.GetContext();
|
var context = _listener.GetContext();
|
||||||
respond(context);
|
respondAsync(context);
|
||||||
}
|
}
|
||||||
catch (HttpListenerException)
|
catch (HttpListenerException)
|
||||||
{
|
{
|
||||||
@ -104,7 +105,7 @@ namespace WebSocketSharp.Server {
|
|||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
OnError.Emit(this, new ErrorEventArgs(ex.Message));
|
onError(ex.Message);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -143,36 +144,17 @@ namespace WebSocketSharp.Server {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void respond(HttpListenerContext context)
|
private void onError(string message)
|
||||||
{
|
{
|
||||||
WaitCallback respondCb = (state) =>
|
#if DEBUG
|
||||||
{
|
var callerFrame = new StackFrame(1);
|
||||||
var req = context.Request;
|
var caller = callerFrame.GetMethod();
|
||||||
var res = context.Response;
|
Console.WriteLine("HTTPSV: Error@{0}: {1}", caller.Name, message);
|
||||||
|
#endif
|
||||||
try
|
OnError.Emit(this, new ErrorEventArgs(message));
|
||||||
{
|
|
||||||
if (isUpgrade(req, "websocket"))
|
|
||||||
{
|
|
||||||
if (upgradeToWebSocket(context))
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
respondToClient(context);
|
|
||||||
}
|
|
||||||
|
|
||||||
res.Close();
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
OnError.Emit(this, new ErrorEventArgs(ex.Message));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
ThreadPool.QueueUserWorkItem(respondCb);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void respondToClient(HttpListenerContext context)
|
private void respond(HttpListenerContext context)
|
||||||
{
|
{
|
||||||
var req = context.Request;
|
var req = context.Request;
|
||||||
var res = context.Response;
|
var res = context.Response;
|
||||||
@ -235,6 +217,36 @@ namespace WebSocketSharp.Server {
|
|||||||
res.StatusCode = (int)HttpStatusCode.NotImplemented;
|
res.StatusCode = (int)HttpStatusCode.NotImplemented;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void respondAsync(HttpListenerContext context)
|
||||||
|
{
|
||||||
|
WaitCallback respondCb = (state) =>
|
||||||
|
{
|
||||||
|
var req = context.Request;
|
||||||
|
var res = context.Response;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
if (isUpgrade(req, "websocket"))
|
||||||
|
{
|
||||||
|
if (upgradeToWebSocket(context))
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
respond(context);
|
||||||
|
}
|
||||||
|
|
||||||
|
res.Close();
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
onError(ex.Message);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
ThreadPool.QueueUserWorkItem(respondCb);
|
||||||
|
}
|
||||||
|
|
||||||
private void startAcceptRequestThread()
|
private void startAcceptRequestThread()
|
||||||
{
|
{
|
||||||
_acceptRequestThread = new Thread(new ThreadStart(acceptRequest));
|
_acceptRequestThread = new Thread(new ThreadStart(acceptRequest));
|
||||||
@ -244,19 +256,17 @@ namespace WebSocketSharp.Server {
|
|||||||
|
|
||||||
private bool upgradeToWebSocket(HttpListenerContext context)
|
private bool upgradeToWebSocket(HttpListenerContext context)
|
||||||
{
|
{
|
||||||
var req = context.Request;
|
var res = context.Response;
|
||||||
var res = context.Response;
|
var wsContext = context.AcceptWebSocket();
|
||||||
|
var path = wsContext.Path.UrlDecode();
|
||||||
var path = req.RawUrl;
|
|
||||||
if (!_services.ContainsKey(path))
|
if (!_services.ContainsKey(path))
|
||||||
{
|
{
|
||||||
res.StatusCode = (int)HttpStatusCode.NotImplemented;
|
res.StatusCode = (int)HttpStatusCode.NotImplemented;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
var wsContext = context.AcceptWebSocket(path);
|
var socket = wsContext.WebSocket;
|
||||||
var socket = wsContext.WebSocket;
|
var service = _services[path];
|
||||||
var service = _services[path];
|
|
||||||
service.BindWebSocket(socket);
|
service.BindWebSocket(socket);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
@ -266,11 +276,18 @@ namespace WebSocketSharp.Server {
|
|||||||
|
|
||||||
#region Public Methods
|
#region Public Methods
|
||||||
|
|
||||||
public void AddService<T>(string path)
|
public void AddService<T>(string absPath)
|
||||||
where T : WebSocketService, new()
|
where T : WebSocketService, new()
|
||||||
{
|
{
|
||||||
|
string msg;
|
||||||
|
if (!absPath.IsValidAbsolutePath(out msg))
|
||||||
|
{
|
||||||
|
onError(msg);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
var service = new WebSocketServer<T>();
|
var service = new WebSocketServer<T>();
|
||||||
_services.Add(path, service);
|
_services.Add(absPath, service);
|
||||||
}
|
}
|
||||||
|
|
||||||
public byte[] GetFile(string path)
|
public byte[] GetFile(string path)
|
||||||
|
@ -52,7 +52,33 @@ namespace WebSocketSharp.Server {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public WebSocketServer(int port)
|
public WebSocketServer(int port)
|
||||||
: base(System.Net.IPAddress.Any, port)
|
: this(System.Net.IPAddress.Any, port)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public WebSocketServer(string url)
|
||||||
|
: base(url)
|
||||||
|
{
|
||||||
|
if (BaseUri.AbsolutePath != "/")
|
||||||
|
{
|
||||||
|
var msg = "Must not contain the path component: " + url;
|
||||||
|
throw new ArgumentException(msg, "url");
|
||||||
|
}
|
||||||
|
|
||||||
|
init();
|
||||||
|
}
|
||||||
|
|
||||||
|
public WebSocketServer(System.Net.IPAddress address, int port)
|
||||||
|
: base(address, port)
|
||||||
|
{
|
||||||
|
init();
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region Private Method
|
||||||
|
|
||||||
|
private void init()
|
||||||
{
|
{
|
||||||
_services = new Dictionary<string, IServiceHost>();
|
_services = new Dictionary<string, IServiceHost>();
|
||||||
}
|
}
|
||||||
@ -61,17 +87,20 @@ namespace WebSocketSharp.Server {
|
|||||||
|
|
||||||
#region Protected Method
|
#region Protected Method
|
||||||
|
|
||||||
protected override void bindSocket(TcpClient client)
|
protected override void AcceptWebSocket(TcpClient client)
|
||||||
{
|
{
|
||||||
var context = client.AcceptWebSocket();
|
var context = client.AcceptWebSocket();
|
||||||
var socket = context.WebSocket;
|
var socket = context.WebSocket;
|
||||||
var path = context.RequestUri.ToString();
|
var path = context.Path.UrlDecode();
|
||||||
if (!_services.ContainsKey(path))
|
if (!_services.ContainsKey(path))
|
||||||
{
|
{
|
||||||
socket.Close(HttpStatusCode.NotImplemented);
|
socket.Close(HttpStatusCode.NotImplemented);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (BaseUri.IsAbsoluteUri)
|
||||||
|
socket.Url = new Uri(BaseUri, path);
|
||||||
|
|
||||||
var service = _services[path];
|
var service = _services[path];
|
||||||
service.BindWebSocket(socket);
|
service.BindWebSocket(socket);
|
||||||
}
|
}
|
||||||
@ -80,11 +109,18 @@ namespace WebSocketSharp.Server {
|
|||||||
|
|
||||||
#region Public Methods
|
#region Public Methods
|
||||||
|
|
||||||
public void AddService<T>(string path)
|
public void AddService<T>(string absPath)
|
||||||
where T : WebSocketService, new()
|
where T : WebSocketService, new()
|
||||||
{
|
{
|
||||||
|
string msg;
|
||||||
|
if (!absPath.IsValidAbsolutePath(out msg))
|
||||||
|
{
|
||||||
|
Error(msg);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
var service = new WebSocketServer<T>();
|
var service = new WebSocketServer<T>();
|
||||||
_services.Add(path, service);
|
_services.Add(absPath, service);
|
||||||
}
|
}
|
||||||
|
|
||||||
public override void Stop()
|
public override void Stop()
|
||||||
@ -104,7 +140,6 @@ namespace WebSocketSharp.Server {
|
|||||||
#region Fields
|
#region Fields
|
||||||
|
|
||||||
private SessionManager _sessions;
|
private SessionManager _sessions;
|
||||||
private Uri _uri;
|
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
@ -119,29 +154,25 @@ namespace WebSocketSharp.Server {
|
|||||||
|
|
||||||
#region Public Constructors
|
#region Public Constructors
|
||||||
|
|
||||||
public WebSocketServer(string url)
|
|
||||||
: base(url)
|
|
||||||
{
|
|
||||||
_uri = url.ToUri();
|
|
||||||
init();
|
|
||||||
}
|
|
||||||
|
|
||||||
public WebSocketServer(int port)
|
public WebSocketServer(int port)
|
||||||
: this(port, "/")
|
: this(port, "/")
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
public WebSocketServer(int port, string path)
|
public WebSocketServer(string url)
|
||||||
: base(System.Net.IPAddress.Any, port)
|
: base(url)
|
||||||
{
|
{
|
||||||
var uri = path.ToUri();
|
init();
|
||||||
if (uri.IsAbsoluteUri)
|
}
|
||||||
{
|
|
||||||
var msg = "Not absolute path: " + path;
|
|
||||||
throw new ArgumentException(msg, "path");
|
|
||||||
}
|
|
||||||
|
|
||||||
_uri = uri;
|
public WebSocketServer(int port, string absPath)
|
||||||
|
: this(System.Net.IPAddress.Any, port, absPath)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public WebSocketServer(System.Net.IPAddress address, int port, string absPath)
|
||||||
|
: base(address, port, absPath)
|
||||||
|
{
|
||||||
init();
|
init();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -149,9 +180,10 @@ namespace WebSocketSharp.Server {
|
|||||||
|
|
||||||
#region Property
|
#region Property
|
||||||
|
|
||||||
public Uri Uri
|
public Uri Uri {
|
||||||
{
|
get {
|
||||||
get { return _uri; }
|
return BaseUri;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
@ -167,9 +199,20 @@ namespace WebSocketSharp.Server {
|
|||||||
|
|
||||||
#region Protected Method
|
#region Protected Method
|
||||||
|
|
||||||
protected override void bindSocket(TcpClient client)
|
protected override void AcceptWebSocket(TcpClient client)
|
||||||
{
|
{
|
||||||
var socket = new WebSocket(_uri, client);
|
var context = client.AcceptWebSocket();
|
||||||
|
var socket = context.WebSocket;
|
||||||
|
var path = context.Path.UrlDecode();
|
||||||
|
if (path != Uri.GetAbsolutePath().UrlDecode())
|
||||||
|
{
|
||||||
|
socket.Close(HttpStatusCode.NotImplemented);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Uri.IsAbsoluteUri)
|
||||||
|
socket.Url = new Uri(Uri, path);
|
||||||
|
|
||||||
BindWebSocket(socket);
|
BindWebSocket(socket);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -34,8 +34,8 @@ using System.Threading;
|
|||||||
|
|
||||||
namespace WebSocketSharp.Server {
|
namespace WebSocketSharp.Server {
|
||||||
|
|
||||||
public abstract class WebSocketServerBase
|
public abstract class WebSocketServerBase {
|
||||||
{
|
|
||||||
#region Fields
|
#region Fields
|
||||||
|
|
||||||
private Thread _acceptClientThread;
|
private Thread _acceptClientThread;
|
||||||
@ -43,6 +43,7 @@ namespace WebSocketSharp.Server {
|
|||||||
private bool _isSelfHost;
|
private bool _isSelfHost;
|
||||||
private int _port;
|
private int _port;
|
||||||
private TcpListener _tcpListener;
|
private TcpListener _tcpListener;
|
||||||
|
private Uri _uri;
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
@ -55,31 +56,72 @@ namespace WebSocketSharp.Server {
|
|||||||
|
|
||||||
protected WebSocketServerBase(string url)
|
protected WebSocketServerBase(string url)
|
||||||
{
|
{
|
||||||
init(url);
|
if (url.IsNull())
|
||||||
|
throw new ArgumentNullException("url");
|
||||||
|
|
||||||
|
Uri uri;
|
||||||
|
string msg;
|
||||||
|
if (!tryCreateUri(url, out uri, out msg))
|
||||||
|
throw new ArgumentException(msg, "url");
|
||||||
|
|
||||||
|
init(uri);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected WebSocketServerBase(IPAddress address, int port)
|
protected WebSocketServerBase(IPAddress address, int port)
|
||||||
|
: this(address, port, "/")
|
||||||
{
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
protected WebSocketServerBase(IPAddress address, int port, string absPath)
|
||||||
|
{
|
||||||
|
if (address.IsNull())
|
||||||
|
throw new ArgumentNullException("address");
|
||||||
|
|
||||||
|
if (absPath.IsNull())
|
||||||
|
throw new ArgumentNullException("absPath");
|
||||||
|
|
||||||
|
string msg;
|
||||||
|
if (!absPath.IsValidAbsolutePath(out msg))
|
||||||
|
throw new ArgumentException(msg, "absPath");
|
||||||
|
|
||||||
_address = address;
|
_address = address;
|
||||||
_port = port <= 0 ? 80 : port;
|
_port = port <= 0 ? 80 : port;
|
||||||
|
_uri = absPath.ToUri();
|
||||||
|
|
||||||
init();
|
init();
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Property
|
#region Protected Property
|
||||||
|
|
||||||
|
protected Uri BaseUri
|
||||||
|
{
|
||||||
|
get {
|
||||||
|
return _uri;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region Public Properties
|
||||||
|
|
||||||
public IPAddress Address {
|
public IPAddress Address {
|
||||||
get { return _address; }
|
get {
|
||||||
|
return _address;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public bool IsSelfHost {
|
public bool IsSelfHost {
|
||||||
get { return _isSelfHost; }
|
get {
|
||||||
|
return _isSelfHost;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public int Port {
|
public int Port {
|
||||||
get { return _port; }
|
get {
|
||||||
|
return _port;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
@ -99,7 +141,7 @@ namespace WebSocketSharp.Server {
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
var client = _tcpListener.AcceptTcpClient();
|
var client = _tcpListener.AcceptTcpClient();
|
||||||
acceptSocket(client);
|
acceptSocketAsync(client);
|
||||||
}
|
}
|
||||||
catch (SocketException)
|
catch (SocketException)
|
||||||
{
|
{
|
||||||
@ -108,36 +150,27 @@ namespace WebSocketSharp.Server {
|
|||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
error(ex.Message);
|
onError(ex.Message);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void acceptSocket(TcpClient client)
|
private void acceptSocketAsync(TcpClient client)
|
||||||
{
|
{
|
||||||
WaitCallback acceptSocketCb = (state) =>
|
WaitCallback acceptSocketCb = (state) =>
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
bindSocket(client);
|
AcceptWebSocket(client);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
error(ex.Message);
|
onError(ex.Message);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
ThreadPool.QueueUserWorkItem(acceptSocketCb);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void error(string message)
|
ThreadPool.QueueUserWorkItem(acceptSocketCb);
|
||||||
{
|
|
||||||
#if DEBUG
|
|
||||||
var callerFrame = new StackFrame(1);
|
|
||||||
var caller = callerFrame.GetMethod();
|
|
||||||
Console.WriteLine("WSSV: Error@{0}: {1}", caller.Name, message);
|
|
||||||
#endif
|
|
||||||
OnError.Emit(this, new ErrorEventArgs(message));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void init()
|
private void init()
|
||||||
@ -146,14 +179,9 @@ namespace WebSocketSharp.Server {
|
|||||||
_isSelfHost = true;
|
_isSelfHost = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void init(string url)
|
private void init(Uri uri)
|
||||||
{
|
{
|
||||||
var uri = url.ToUri();
|
_uri = uri;
|
||||||
|
|
||||||
string msg;
|
|
||||||
if (!uri.IsValidWebSocketUri(out msg))
|
|
||||||
throw new ArgumentException(msg, "url");
|
|
||||||
|
|
||||||
var scheme = uri.Scheme;
|
var scheme = uri.Scheme;
|
||||||
var port = uri.Port;
|
var port = uri.Port;
|
||||||
var host = uri.DnsSafeHost;
|
var host = uri.DnsSafeHost;
|
||||||
@ -168,6 +196,16 @@ namespace WebSocketSharp.Server {
|
|||||||
init();
|
init();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void onError(string message)
|
||||||
|
{
|
||||||
|
#if DEBUG
|
||||||
|
var callerFrame = new StackFrame(1);
|
||||||
|
var caller = callerFrame.GetMethod();
|
||||||
|
Console.WriteLine("WSSV: Error@{0}: {1}", caller.Name, message);
|
||||||
|
#endif
|
||||||
|
OnError.Emit(this, new ErrorEventArgs(message));
|
||||||
|
}
|
||||||
|
|
||||||
private void startAcceptClientThread()
|
private void startAcceptClientThread()
|
||||||
{
|
{
|
||||||
_acceptClientThread = new Thread(new ThreadStart(acceptClient));
|
_acceptClientThread = new Thread(new ThreadStart(acceptClient));
|
||||||
@ -175,11 +213,31 @@ namespace WebSocketSharp.Server {
|
|||||||
_acceptClientThread.Start();
|
_acceptClientThread.Start();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private bool tryCreateUri(string uriString, out Uri result, out string message)
|
||||||
|
{
|
||||||
|
if (!uriString.TryCreateWebSocketUri(out result, out message))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (!result.Query.IsNullOrEmpty())
|
||||||
|
{
|
||||||
|
result = null;
|
||||||
|
message = "Must not contain the query component: " + uriString;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Protected Method
|
#region Protected Method
|
||||||
|
|
||||||
protected abstract void bindSocket(TcpClient client);
|
protected abstract void AcceptWebSocket(TcpClient client);
|
||||||
|
|
||||||
|
protected virtual void Error(string message)
|
||||||
|
{
|
||||||
|
onError(message);
|
||||||
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
@ -28,6 +28,7 @@
|
|||||||
|
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using System.Collections.Specialized;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using WebSocketSharp.Frame;
|
using WebSocketSharp.Frame;
|
||||||
|
|
||||||
@ -53,9 +54,15 @@ namespace WebSocketSharp.Server
|
|||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Protected Property
|
#region Protected Properties
|
||||||
|
|
||||||
protected SessionManager sessions {
|
protected NameValueCollection QueryString {
|
||||||
|
get {
|
||||||
|
return _socket.QueryString;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected SessionManager Sessions {
|
||||||
get {
|
get {
|
||||||
return _sessions;
|
return _sessions;
|
||||||
}
|
}
|
||||||
@ -91,19 +98,19 @@ namespace WebSocketSharp.Server
|
|||||||
|
|
||||||
#region Protected Methods
|
#region Protected Methods
|
||||||
|
|
||||||
protected virtual void onOpen(object sender, EventArgs e)
|
protected virtual void OnClose(object sender, CloseEventArgs e)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
protected virtual void onMessage(object sender, MessageEventArgs e)
|
protected virtual void OnError(object sender, ErrorEventArgs e)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
protected virtual void onError(object sender, ErrorEventArgs e)
|
protected virtual void OnMessage(object sender, MessageEventArgs e)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
protected virtual void onClose(object sender, CloseEventArgs e)
|
protected virtual void OnOpen(object sender, EventArgs e)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -117,10 +124,10 @@ namespace WebSocketSharp.Server
|
|||||||
_sessions = sessions;
|
_sessions = sessions;
|
||||||
|
|
||||||
defaultBind();
|
defaultBind();
|
||||||
_socket.OnOpen += onOpen;
|
_socket.OnOpen += OnOpen;
|
||||||
_socket.OnMessage += onMessage;
|
_socket.OnMessage += OnMessage;
|
||||||
_socket.OnError += onError;
|
_socket.OnError += OnError;
|
||||||
_socket.OnClose += onClose;
|
_socket.OnClose += OnClose;
|
||||||
|
|
||||||
IsBound = true;
|
IsBound = true;
|
||||||
}
|
}
|
||||||
|
@ -32,6 +32,7 @@
|
|||||||
|
|
||||||
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;
|
||||||
@ -75,6 +76,7 @@ namespace WebSocketSharp {
|
|||||||
private bool _isSecure;
|
private bool _isSecure;
|
||||||
private string _protocol;
|
private string _protocol;
|
||||||
private string _protocols;
|
private string _protocols;
|
||||||
|
private NameValueCollection _queryString;
|
||||||
private volatile WsState _readyState;
|
private volatile WsState _readyState;
|
||||||
private AutoResetEvent _receivePong;
|
private AutoResetEvent _receivePong;
|
||||||
private TcpClient _tcpClient;
|
private TcpClient _tcpClient;
|
||||||
@ -100,39 +102,28 @@ namespace WebSocketSharp {
|
|||||||
|
|
||||||
#region Internal Constructor
|
#region Internal Constructor
|
||||||
|
|
||||||
internal WebSocket(TcpListenerWebSocketContext context)
|
internal WebSocket(HttpListenerWebSocketContext context)
|
||||||
: this()
|
: this()
|
||||||
{
|
{
|
||||||
_uri = context.RequestUri;
|
_uri = context.Path.ToUri();
|
||||||
_context = context;
|
|
||||||
_tcpClient = context.Client;
|
|
||||||
_wsStream = context.Stream;
|
|
||||||
_endPoint = (System.Net.IPEndPoint)_tcpClient.Client.LocalEndPoint;
|
|
||||||
_isClient = false;
|
|
||||||
_isSecure = context.IsSecureConnection;
|
|
||||||
}
|
|
||||||
|
|
||||||
internal WebSocket(Uri uri, HttpListenerWebSocketContext context)
|
|
||||||
: this()
|
|
||||||
{
|
|
||||||
_uri = uri;
|
|
||||||
_context = context;
|
_context = context;
|
||||||
_baseContext = context.BaseContext;
|
_baseContext = context.BaseContext;
|
||||||
_wsStream = context.Stream;
|
_wsStream = context.Stream;
|
||||||
_endPoint = _baseContext.Connection.LocalEndPoint;
|
_endPoint = context.ServerEndPoint;
|
||||||
_isClient = false;
|
_isClient = false;
|
||||||
_isSecure = context.IsSecureConnection;
|
_isSecure = context.IsSecureConnection;
|
||||||
}
|
}
|
||||||
|
|
||||||
internal WebSocket(Uri uri, TcpClient tcpClient)
|
internal WebSocket(TcpListenerWebSocketContext context)
|
||||||
: this()
|
: this()
|
||||||
{
|
{
|
||||||
_uri = uri;
|
_uri = context.Path.ToUri();
|
||||||
_tcpClient = tcpClient;
|
_context = context;
|
||||||
_wsStream = WsStream.CreateServerStream(tcpClient);
|
_tcpClient = context.Client;
|
||||||
_endPoint = (System.Net.IPEndPoint)tcpClient.Client.LocalEndPoint;
|
_wsStream = context.Stream;
|
||||||
|
_endPoint = context.ServerEndPoint;
|
||||||
_isClient = false;
|
_isClient = false;
|
||||||
_isSecure = _wsStream.IsSecure;
|
_isSecure = context.IsSecureConnection;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
@ -157,16 +148,18 @@ namespace WebSocketSharp {
|
|||||||
public WebSocket(string url, params string[] protocols)
|
public WebSocket(string url, params string[] protocols)
|
||||||
: this()
|
: this()
|
||||||
{
|
{
|
||||||
if (url == null)
|
if (url.IsNull())
|
||||||
throw new ArgumentNullException("url");
|
throw new ArgumentNullException("url");
|
||||||
|
|
||||||
|
Uri uri;
|
||||||
string msg;
|
string msg;
|
||||||
if (!isValidUrl(url, out msg))
|
if (!tryCreateUri(url, out uri, out msg))
|
||||||
throw new ArgumentException(msg, "url");
|
throw new ArgumentException(msg, "url");
|
||||||
|
|
||||||
|
_uri = uri;
|
||||||
_protocols = protocols.ToString(", ");
|
_protocols = protocols.ToString(", ");
|
||||||
_isClient = true;
|
_isClient = true;
|
||||||
_isSecure = _uri.Scheme == "wss" ? true : false;
|
_isSecure = uri.Scheme == "wss" ? true : false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -215,7 +208,17 @@ namespace WebSocketSharp {
|
|||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Properties
|
#region Internal Property
|
||||||
|
|
||||||
|
internal NameValueCollection QueryString {
|
||||||
|
get {
|
||||||
|
return _queryString;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region Public Properties
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the amount of untransmitted data.
|
/// Gets the amount of untransmitted data.
|
||||||
@ -320,10 +323,8 @@ namespace WebSocketSharp {
|
|||||||
public Uri Url {
|
public Uri Url {
|
||||||
get { return _uri; }
|
get { return _uri; }
|
||||||
set {
|
set {
|
||||||
if (_readyState != WsState.CONNECTING || _isClient)
|
if (_readyState == WsState.CONNECTING && !_isClient)
|
||||||
return;
|
_uri = value;
|
||||||
|
|
||||||
_uri = value;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -355,6 +356,7 @@ namespace WebSocketSharp {
|
|||||||
|
|
||||||
#region Private Methods
|
#region Private Methods
|
||||||
|
|
||||||
|
// As Server
|
||||||
private void acceptHandshake()
|
private void acceptHandshake()
|
||||||
{
|
{
|
||||||
var req = receiveOpeningHandshake();
|
var req = receiveOpeningHandshake();
|
||||||
@ -439,7 +441,7 @@ namespace WebSocketSharp {
|
|||||||
private void close(ushort code, string reason)
|
private void close(ushort code, string reason)
|
||||||
{
|
{
|
||||||
var data = new List<byte>(code.ToBytes(ByteOrder.BIG));
|
var data = new List<byte>(code.ToBytes(ByteOrder.BIG));
|
||||||
if (!String.IsNullOrEmpty(reason))
|
if (!reason.IsNullOrEmpty())
|
||||||
{
|
{
|
||||||
var buffer = Encoding.UTF8.GetBytes(reason);
|
var buffer = Encoding.UTF8.GetBytes(reason);
|
||||||
data.AddRange(buffer);
|
data.AddRange(buffer);
|
||||||
@ -462,20 +464,20 @@ namespace WebSocketSharp {
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (_baseContext != null)
|
if (!_baseContext.IsNull())
|
||||||
{
|
{
|
||||||
_baseContext.Response.Close();
|
_baseContext.Response.Close();
|
||||||
_wsStream = null;
|
_wsStream = null;
|
||||||
_baseContext = null;
|
_baseContext = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_wsStream != null)
|
if (!_wsStream.IsNull())
|
||||||
{
|
{
|
||||||
_wsStream.Dispose();
|
_wsStream.Dispose();
|
||||||
_wsStream = null;
|
_wsStream = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_tcpClient != null)
|
if (!_tcpClient.IsNull())
|
||||||
{
|
{
|
||||||
_tcpClient.Close();
|
_tcpClient.Close();
|
||||||
_tcpClient = null;
|
_tcpClient = null;
|
||||||
@ -495,12 +497,13 @@ namespace WebSocketSharp {
|
|||||||
var args = new CloseEventArgs(data);
|
var args = new CloseEventArgs(data);
|
||||||
var frame = createFrame(Fin.FINAL, Opcode.CLOSE, data);
|
var frame = createFrame(Fin.FINAL, Opcode.CLOSE, data);
|
||||||
if (send(frame) && !Thread.CurrentThread.IsBackground)
|
if (send(frame) && !Thread.CurrentThread.IsBackground)
|
||||||
if (_exitMessageLoop != null)
|
if (!_exitMessageLoop.IsNull())
|
||||||
_exitMessageLoop.WaitOne(5 * 1000);
|
_exitMessageLoop.WaitOne(5 * 1000);
|
||||||
|
|
||||||
onClose(args);
|
onClose(args);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Client
|
||||||
private void createClientStream()
|
private void createClientStream()
|
||||||
{
|
{
|
||||||
var host = _uri.DnsSafeHost;
|
var host = _uri.DnsSafeHost;
|
||||||
@ -529,6 +532,7 @@ namespace WebSocketSharp {
|
|||||||
: new WsFrame(fin, opcode, Mask.UNMASK, payloadData);
|
: new WsFrame(fin, opcode, Mask.UNMASK, payloadData);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Client
|
||||||
private RequestHandshake createOpeningHandshake()
|
private RequestHandshake createOpeningHandshake()
|
||||||
{
|
{
|
||||||
var path = _uri.PathAndQuery;
|
var path = _uri.PathAndQuery;
|
||||||
@ -545,13 +549,14 @@ namespace WebSocketSharp {
|
|||||||
var req = new RequestHandshake(path);
|
var req = new RequestHandshake(path);
|
||||||
req.AddHeader("Host", host);
|
req.AddHeader("Host", host);
|
||||||
req.AddHeader("Sec-WebSocket-Key", _base64key);
|
req.AddHeader("Sec-WebSocket-Key", _base64key);
|
||||||
if (!String.IsNullOrEmpty(_protocols))
|
if (!_protocols.IsNullOrEmpty())
|
||||||
req.AddHeader("Sec-WebSocket-Protocol", _protocols);
|
req.AddHeader("Sec-WebSocket-Protocol", _protocols);
|
||||||
req.AddHeader("Sec-WebSocket-Version", _version);
|
req.AddHeader("Sec-WebSocket-Version", _version);
|
||||||
|
|
||||||
return req;
|
return req;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Server
|
||||||
private ResponseHandshake createResponseHandshake()
|
private ResponseHandshake createResponseHandshake()
|
||||||
{
|
{
|
||||||
var res = new ResponseHandshake();
|
var res = new ResponseHandshake();
|
||||||
@ -560,6 +565,7 @@ namespace WebSocketSharp {
|
|||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Server
|
||||||
private ResponseHandshake createResponseHandshake(HttpStatusCode code)
|
private ResponseHandshake createResponseHandshake(HttpStatusCode code)
|
||||||
{
|
{
|
||||||
var res = ResponseHandshake.CreateCloseResponse(code);
|
var res = ResponseHandshake.CreateCloseResponse(code);
|
||||||
@ -568,6 +574,7 @@ namespace WebSocketSharp {
|
|||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Client
|
||||||
private void doHandshake()
|
private void doHandshake()
|
||||||
{
|
{
|
||||||
var res = sendOpeningHandshake();
|
var res = sendOpeningHandshake();
|
||||||
@ -601,6 +608,7 @@ namespace WebSocketSharp {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Server
|
||||||
private bool isValidRequest(RequestHandshake request, out string message)
|
private bool isValidRequest(RequestHandshake request, out string message)
|
||||||
{
|
{
|
||||||
Func<string, Func<string, string, string>> func = s =>
|
Func<string, Func<string, string, string>> func = s =>
|
||||||
@ -617,9 +625,6 @@ namespace WebSocketSharp {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isValidRequestUri(request.RequestUri, func("Request URI"), out message))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (_uri.IsAbsoluteUri)
|
if (_uri.IsAbsoluteUri)
|
||||||
if (!isValidRequestHost(request.GetHeaderValues("Host")[0], func("Host"), out message))
|
if (!isValidRequestHost(request.GetHeaderValues("Host")[0], func("Host"), out message))
|
||||||
return false;
|
return false;
|
||||||
@ -638,10 +643,13 @@ namespace WebSocketSharp {
|
|||||||
if (request.HeaderExists("Sec-WebSocket-Extensions"))
|
if (request.HeaderExists("Sec-WebSocket-Extensions"))
|
||||||
_extensions = request.Headers["Sec-WebSocket-Extensions"];
|
_extensions = request.Headers["Sec-WebSocket-Extensions"];
|
||||||
|
|
||||||
|
_queryString = request.QueryString;
|
||||||
|
|
||||||
message = String.Empty;
|
message = String.Empty;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Server
|
||||||
private bool isValidRequestHost(string value, Func<string, string, string> func, out string message)
|
private bool isValidRequestHost(string value, Func<string, string, string> func, out string message)
|
||||||
{
|
{
|
||||||
var host = _uri.DnsSafeHost;
|
var host = _uri.DnsSafeHost;
|
||||||
@ -669,28 +677,7 @@ namespace WebSocketSharp {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
private bool isValidRequestUri(Uri requestUri, Func<string, string, string> func, out string message)
|
// As Client
|
||||||
{
|
|
||||||
if (_uri.IsAbsoluteUri && requestUri.IsAbsoluteUri)
|
|
||||||
if (_uri.ToString().NotEqualsDo(requestUri.ToString(), func, out message, false))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (_uri.IsAbsoluteUri && !requestUri.IsAbsoluteUri)
|
|
||||||
if (_uri.PathAndQuery.NotEqualsDo(requestUri.ToString(), func, out message, false))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (!_uri.IsAbsoluteUri && requestUri.IsAbsoluteUri)
|
|
||||||
if (_uri.ToString().NotEqualsDo(requestUri.PathAndQuery, func, out message, false))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (!_uri.IsAbsoluteUri && !requestUri.IsAbsoluteUri)
|
|
||||||
if (_uri.ToString().NotEqualsDo(requestUri.ToString(), func, out message, false))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
message = String.Empty;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
private bool isValidResponse(ResponseHandshake response, out string message)
|
private bool isValidResponse(ResponseHandshake response, out string message)
|
||||||
{
|
{
|
||||||
if (!response.IsWebSocketResponse)
|
if (!response.IsWebSocketResponse)
|
||||||
@ -724,24 +711,6 @@ namespace WebSocketSharp {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
private bool isValidUrl(string url, out string message)
|
|
||||||
{
|
|
||||||
if (url == String.Empty)
|
|
||||||
{
|
|
||||||
message = "'url' is empty.";
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
var uri = url.ToUri();
|
|
||||||
if (!uri.IsValidWebSocketUri(out message))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
_uri = uri;
|
|
||||||
message = String.Empty;
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void message()
|
private void message()
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
@ -791,7 +760,7 @@ namespace WebSocketSharp {
|
|||||||
|
|
||||||
private void onMessage(MessageEventArgs eventArgs)
|
private void onMessage(MessageEventArgs eventArgs)
|
||||||
{
|
{
|
||||||
if (eventArgs != null)
|
if (!eventArgs.IsNull())
|
||||||
OnMessage.Emit(this, eventArgs);
|
OnMessage.Emit(this, eventArgs);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -802,9 +771,9 @@ namespace WebSocketSharp {
|
|||||||
OnOpen.Emit(this, EventArgs.Empty);
|
OnOpen.Emit(this, EventArgs.Empty);
|
||||||
}
|
}
|
||||||
|
|
||||||
private bool ping(string data, int millisecondsTimeout)
|
private bool ping(string message, int millisecondsTimeout)
|
||||||
{
|
{
|
||||||
var buffer = Encoding.UTF8.GetBytes(data);
|
var buffer = Encoding.UTF8.GetBytes(message);
|
||||||
if (buffer.Length > 125)
|
if (buffer.Length > 125)
|
||||||
{
|
{
|
||||||
var msg = "Ping frame must have a payload length of 125 bytes or less.";
|
var msg = "Ping frame must have a payload length of 125 bytes or less.";
|
||||||
@ -833,7 +802,7 @@ namespace WebSocketSharp {
|
|||||||
private WsFrame readFrame()
|
private WsFrame readFrame()
|
||||||
{
|
{
|
||||||
var frame = _wsStream.ReadFrame();
|
var frame = _wsStream.ReadFrame();
|
||||||
if (frame == null)
|
if (frame.IsNull())
|
||||||
{
|
{
|
||||||
var msg = "WebSocket data frame can not be read from network stream.";
|
var msg = "WebSocket data frame can not be read from network stream.";
|
||||||
close(CloseStatusCode.ABNORMAL, msg);
|
close(CloseStatusCode.ABNORMAL, msg);
|
||||||
@ -862,7 +831,7 @@ namespace WebSocketSharp {
|
|||||||
private MessageEventArgs receive()
|
private MessageEventArgs receive()
|
||||||
{
|
{
|
||||||
var frame = _isClient ? readFrame() : readFrameWithTimeout(1 * 100);
|
var frame = _isClient ? readFrame() : readFrameWithTimeout(1 * 100);
|
||||||
if (frame == null)
|
if (frame.IsNull())
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
if ((frame.Fin == Fin.FINAL && frame.Opcode == Opcode.CONT) ||
|
if ((frame.Fin == Fin.FINAL && frame.Opcode == Opcode.CONT) ||
|
||||||
@ -872,7 +841,7 @@ namespace WebSocketSharp {
|
|||||||
if (frame.Fin == Fin.MORE)
|
if (frame.Fin == Fin.MORE)
|
||||||
{// MORE
|
{// MORE
|
||||||
var merged = receiveFragmented(frame);
|
var merged = receiveFragmented(frame);
|
||||||
if (merged == null)
|
if (merged.IsNull())
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
return new MessageEventArgs(frame.Opcode, new PayloadData(merged));
|
return new MessageEventArgs(frame.Opcode, new PayloadData(merged));
|
||||||
@ -912,7 +881,7 @@ namespace WebSocketSharp {
|
|||||||
while (true)
|
while (true)
|
||||||
{
|
{
|
||||||
var frame = readFrame();
|
var frame = readFrame();
|
||||||
if (frame == null)
|
if (frame.IsNull())
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
if (frame.Fin == Fin.MORE)
|
if (frame.Fin == Fin.MORE)
|
||||||
@ -972,11 +941,10 @@ namespace WebSocketSharp {
|
|||||||
return buffer.ToArray();
|
return buffer.ToArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Server
|
||||||
private RequestHandshake receiveOpeningHandshake()
|
private RequestHandshake receiveOpeningHandshake()
|
||||||
{
|
{
|
||||||
var req = _context != null
|
var req = RequestHandshake.Parse(_context);
|
||||||
? RequestHandshake.Parse(_context)
|
|
||||||
: RequestHandshake.Parse(readHandshake());
|
|
||||||
#if DEBUG
|
#if DEBUG
|
||||||
Console.WriteLine("WS: Info@receiveOpeningHandshake: Opening handshake from client:\n");
|
Console.WriteLine("WS: Info@receiveOpeningHandshake: Opening handshake from client:\n");
|
||||||
Console.WriteLine(req.ToString());
|
Console.WriteLine(req.ToString());
|
||||||
@ -984,6 +952,7 @@ namespace WebSocketSharp {
|
|||||||
return req;
|
return req;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Client
|
||||||
private ResponseHandshake receiveResponseHandshake()
|
private ResponseHandshake receiveResponseHandshake()
|
||||||
{
|
{
|
||||||
var res = ResponseHandshake.Parse(readHandshake());
|
var res = ResponseHandshake.Parse(readHandshake());
|
||||||
@ -1008,7 +977,7 @@ namespace WebSocketSharp {
|
|||||||
{
|
{
|
||||||
if (_unTransmittedBuffer.Count == 0)
|
if (_unTransmittedBuffer.Count == 0)
|
||||||
{
|
{
|
||||||
if (_wsStream != null)
|
if (!_wsStream.IsNull())
|
||||||
{
|
{
|
||||||
_wsStream.WriteFrame(frame);
|
_wsStream.WriteFrame(frame);
|
||||||
return true;
|
return true;
|
||||||
@ -1097,13 +1066,16 @@ namespace WebSocketSharp {
|
|||||||
return readLen;
|
return readLen;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Client
|
||||||
private ResponseHandshake sendOpeningHandshake()
|
private ResponseHandshake sendOpeningHandshake()
|
||||||
{
|
{
|
||||||
var req = createOpeningHandshake();
|
var req = createOpeningHandshake();
|
||||||
sendOpeningHandshake(req);
|
sendOpeningHandshake(req);
|
||||||
|
|
||||||
return receiveResponseHandshake();
|
return receiveResponseHandshake();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Client
|
||||||
private void sendOpeningHandshake(RequestHandshake request)
|
private void sendOpeningHandshake(RequestHandshake request)
|
||||||
{
|
{
|
||||||
#if DEBUG
|
#if DEBUG
|
||||||
@ -1113,18 +1085,21 @@ namespace WebSocketSharp {
|
|||||||
writeHandshake(request);
|
writeHandshake(request);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Server
|
||||||
private void sendResponseHandshake()
|
private void sendResponseHandshake()
|
||||||
{
|
{
|
||||||
var res = createResponseHandshake();
|
var res = createResponseHandshake();
|
||||||
sendResponseHandshake(res);
|
sendResponseHandshake(res);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Server
|
||||||
private void sendResponseHandshake(HttpStatusCode code)
|
private void sendResponseHandshake(HttpStatusCode code)
|
||||||
{
|
{
|
||||||
var res = createResponseHandshake(code);
|
var res = createResponseHandshake(code);
|
||||||
sendResponseHandshake(res);
|
sendResponseHandshake(res);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// As Server
|
||||||
private void sendResponseHandshake(ResponseHandshake response)
|
private void sendResponseHandshake(ResponseHandshake response)
|
||||||
{
|
{
|
||||||
#if DEBUG
|
#if DEBUG
|
||||||
@ -1147,6 +1122,11 @@ namespace WebSocketSharp {
|
|||||||
messageInvoker.BeginInvoke(messageLoopCallback, messageInvoker);
|
messageInvoker.BeginInvoke(messageLoopCallback, messageInvoker);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private bool tryCreateUri(string uriString, out Uri result, out string message)
|
||||||
|
{
|
||||||
|
return uriString.TryCreateWebSocketUri(out result, out message);
|
||||||
|
}
|
||||||
|
|
||||||
private void writeHandshake(Handshake handshake)
|
private void writeHandshake(Handshake handshake)
|
||||||
{
|
{
|
||||||
_wsStream.WriteHandshake(handshake);
|
_wsStream.WriteHandshake(handshake);
|
||||||
@ -1156,6 +1136,7 @@ namespace WebSocketSharp {
|
|||||||
|
|
||||||
#region Internal Method
|
#region Internal Method
|
||||||
|
|
||||||
|
// As Server
|
||||||
internal void Close(HttpStatusCode code)
|
internal void Close(HttpStatusCode code)
|
||||||
{
|
{
|
||||||
close(code);
|
close(code);
|
||||||
@ -1290,15 +1271,18 @@ namespace WebSocketSharp {
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// Sends a Ping frame with a message using the connection.
|
/// Sends a Ping frame with a message using the connection.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="data">
|
/// <param name="message">
|
||||||
/// A <see cref="string"/> that contains the message data to be sent.
|
/// A <see cref="string"/> that contains the message to be sent.
|
||||||
/// </param>
|
/// </param>
|
||||||
/// <returns>
|
/// <returns>
|
||||||
/// <c>true</c> if the WebSocket receives a Pong frame in a time; otherwise, <c>false</c>.
|
/// <c>true</c> if the WebSocket receives a Pong frame in a time; otherwise, <c>false</c>.
|
||||||
/// </returns>
|
/// </returns>
|
||||||
public bool Ping(string data)
|
public bool Ping(string message)
|
||||||
{
|
{
|
||||||
return ping(data, 5 * 1000);
|
if (message.IsNull())
|
||||||
|
message = String.Empty;
|
||||||
|
|
||||||
|
return ping(message, 5 * 1000);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -1309,6 +1293,12 @@ namespace WebSocketSharp {
|
|||||||
/// </param>
|
/// </param>
|
||||||
public void Send(string data)
|
public void Send(string data)
|
||||||
{
|
{
|
||||||
|
if (data.IsNull())
|
||||||
|
{
|
||||||
|
onError("'data' must not be null.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
var buffer = Encoding.UTF8.GetBytes(data);
|
var buffer = Encoding.UTF8.GetBytes(data);
|
||||||
send(Opcode.TEXT, buffer);
|
send(Opcode.TEXT, buffer);
|
||||||
}
|
}
|
||||||
@ -1321,6 +1311,12 @@ namespace WebSocketSharp {
|
|||||||
/// </param>
|
/// </param>
|
||||||
public void Send(byte[] data)
|
public void Send(byte[] data)
|
||||||
{
|
{
|
||||||
|
if (data.IsNull())
|
||||||
|
{
|
||||||
|
onError("'data' must not be null.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
send(Opcode.BINARY, data);
|
send(Opcode.BINARY, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1332,6 +1328,12 @@ namespace WebSocketSharp {
|
|||||||
/// </param>
|
/// </param>
|
||||||
public void Send(FileInfo file)
|
public void Send(FileInfo file)
|
||||||
{
|
{
|
||||||
|
if (file.IsNull())
|
||||||
|
{
|
||||||
|
onError("'file' must not be null.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
using (FileStream fs = file.OpenRead())
|
using (FileStream fs = file.OpenRead())
|
||||||
{
|
{
|
||||||
send(Opcode.BINARY, fs);
|
send(Opcode.BINARY, fs);
|
||||||
|
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.
Loading…
Reference in New Issue
Block a user