diff --git a/Discord.Net.sln b/Discord.Net.sln
index 206c5b54a..9663b21dc 100644
--- a/Discord.Net.sln
+++ b/Discord.Net.sln
@@ -1,11 +1,10 @@
Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio 15
-VisualStudioVersion = 15.0.25914.0
+VisualStudioVersion = 15.0.26014.0
MinimumVisualStudioVersion = 10.0.40219.1
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{F7F3E124-93C7-4846-AE87-9CE12BD82859}"
ProjectSection(SolutionItems) = preProject
- global.json = global.json
README.md = README.md
EndProjectSection
EndProject
@@ -23,6 +22,10 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Discord.Net.Commands", "src
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Discord.Net.WebSocket", "src\Discord.Net.WebSocket\Discord.Net.WebSocket.csproj", "{688FD1D8-7F01-4539-B2E9-F473C5D699C7}"
EndProject
+Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Providers", "Providers", "{B0657AAE-DCC5-4FBF-8E5D-1FB578CF3012}"
+EndProject
+Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Discord.Net.Providers.WS4Net", "src\Discord.Net.Providers.WS4Net\Discord.Net.Providers.WS4Net.csproj", "{6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7}"
+EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
@@ -39,8 +42,8 @@ Global
{496DB20A-A455-4D01-B6BC-90FE6D7C6B81}.Debug|x64.Build.0 = Debug|Any CPU
{496DB20A-A455-4D01-B6BC-90FE6D7C6B81}.Debug|x86.ActiveCfg = Debug|Any CPU
{496DB20A-A455-4D01-B6BC-90FE6D7C6B81}.Debug|x86.Build.0 = Debug|Any CPU
- {496DB20A-A455-4D01-B6BC-90FE6D7C6B81}.Release|Any CPU.ActiveCfg = Debug|Any CPU
- {496DB20A-A455-4D01-B6BC-90FE6D7C6B81}.Release|Any CPU.Build.0 = Debug|Any CPU
+ {496DB20A-A455-4D01-B6BC-90FE6D7C6B81}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {496DB20A-A455-4D01-B6BC-90FE6D7C6B81}.Release|Any CPU.Build.0 = Release|Any CPU
{496DB20A-A455-4D01-B6BC-90FE6D7C6B81}.Release|x64.ActiveCfg = Debug|Any CPU
{496DB20A-A455-4D01-B6BC-90FE6D7C6B81}.Release|x64.Build.0 = Debug|Any CPU
{496DB20A-A455-4D01-B6BC-90FE6D7C6B81}.Release|x86.ActiveCfg = Debug|Any CPU
@@ -51,8 +54,8 @@ Global
{91E9E7BD-75C9-4E98-84AA-2C271922E5C2}.Debug|x64.Build.0 = Debug|x64
{91E9E7BD-75C9-4E98-84AA-2C271922E5C2}.Debug|x86.ActiveCfg = Debug|x86
{91E9E7BD-75C9-4E98-84AA-2C271922E5C2}.Debug|x86.Build.0 = Debug|x86
- {91E9E7BD-75C9-4E98-84AA-2C271922E5C2}.Release|Any CPU.ActiveCfg = Debug|Any CPU
- {91E9E7BD-75C9-4E98-84AA-2C271922E5C2}.Release|Any CPU.Build.0 = Debug|Any CPU
+ {91E9E7BD-75C9-4E98-84AA-2C271922E5C2}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {91E9E7BD-75C9-4E98-84AA-2C271922E5C2}.Release|Any CPU.Build.0 = Release|Any CPU
{91E9E7BD-75C9-4E98-84AA-2C271922E5C2}.Release|x64.ActiveCfg = Release|x64
{91E9E7BD-75C9-4E98-84AA-2C271922E5C2}.Release|x64.Build.0 = Release|x64
{91E9E7BD-75C9-4E98-84AA-2C271922E5C2}.Release|x86.ActiveCfg = Release|x86
@@ -63,8 +66,8 @@ Global
{BFC6DC28-0351-4573-926A-D4124244C04F}.Debug|x64.Build.0 = Debug|Any CPU
{BFC6DC28-0351-4573-926A-D4124244C04F}.Debug|x86.ActiveCfg = Debug|Any CPU
{BFC6DC28-0351-4573-926A-D4124244C04F}.Debug|x86.Build.0 = Debug|Any CPU
- {BFC6DC28-0351-4573-926A-D4124244C04F}.Release|Any CPU.ActiveCfg = Debug|Any CPU
- {BFC6DC28-0351-4573-926A-D4124244C04F}.Release|Any CPU.Build.0 = Debug|Any CPU
+ {BFC6DC28-0351-4573-926A-D4124244C04F}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {BFC6DC28-0351-4573-926A-D4124244C04F}.Release|Any CPU.Build.0 = Release|Any CPU
{BFC6DC28-0351-4573-926A-D4124244C04F}.Release|x64.ActiveCfg = Debug|Any CPU
{BFC6DC28-0351-4573-926A-D4124244C04F}.Release|x64.Build.0 = Debug|Any CPU
{BFC6DC28-0351-4573-926A-D4124244C04F}.Release|x86.ActiveCfg = Debug|Any CPU
@@ -75,8 +78,8 @@ Global
{5688A353-121E-40A1-8BFA-B17B91FB48FB}.Debug|x64.Build.0 = Debug|Any CPU
{5688A353-121E-40A1-8BFA-B17B91FB48FB}.Debug|x86.ActiveCfg = Debug|Any CPU
{5688A353-121E-40A1-8BFA-B17B91FB48FB}.Debug|x86.Build.0 = Debug|Any CPU
- {5688A353-121E-40A1-8BFA-B17B91FB48FB}.Release|Any CPU.ActiveCfg = Debug|Any CPU
- {5688A353-121E-40A1-8BFA-B17B91FB48FB}.Release|Any CPU.Build.0 = Debug|Any CPU
+ {5688A353-121E-40A1-8BFA-B17B91FB48FB}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {5688A353-121E-40A1-8BFA-B17B91FB48FB}.Release|Any CPU.Build.0 = Release|Any CPU
{5688A353-121E-40A1-8BFA-B17B91FB48FB}.Release|x64.ActiveCfg = Debug|Any CPU
{5688A353-121E-40A1-8BFA-B17B91FB48FB}.Release|x64.Build.0 = Debug|Any CPU
{5688A353-121E-40A1-8BFA-B17B91FB48FB}.Release|x86.ActiveCfg = Debug|Any CPU
@@ -87,8 +90,8 @@ Global
{078DD7E6-943D-4D09-AFC2-D2BA58B76C9C}.Debug|x64.Build.0 = Debug|Any CPU
{078DD7E6-943D-4D09-AFC2-D2BA58B76C9C}.Debug|x86.ActiveCfg = Debug|Any CPU
{078DD7E6-943D-4D09-AFC2-D2BA58B76C9C}.Debug|x86.Build.0 = Debug|Any CPU
- {078DD7E6-943D-4D09-AFC2-D2BA58B76C9C}.Release|Any CPU.ActiveCfg = Debug|Any CPU
- {078DD7E6-943D-4D09-AFC2-D2BA58B76C9C}.Release|Any CPU.Build.0 = Debug|Any CPU
+ {078DD7E6-943D-4D09-AFC2-D2BA58B76C9C}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {078DD7E6-943D-4D09-AFC2-D2BA58B76C9C}.Release|Any CPU.Build.0 = Release|Any CPU
{078DD7E6-943D-4D09-AFC2-D2BA58B76C9C}.Release|x64.ActiveCfg = Debug|Any CPU
{078DD7E6-943D-4D09-AFC2-D2BA58B76C9C}.Release|x64.Build.0 = Debug|Any CPU
{078DD7E6-943D-4D09-AFC2-D2BA58B76C9C}.Release|x86.ActiveCfg = Debug|Any CPU
@@ -105,6 +108,18 @@ Global
{688FD1D8-7F01-4539-B2E9-F473C5D699C7}.Release|x64.Build.0 = Release|x64
{688FD1D8-7F01-4539-B2E9-F473C5D699C7}.Release|x86.ActiveCfg = Release|x86
{688FD1D8-7F01-4539-B2E9-F473C5D699C7}.Release|x86.Build.0 = Release|x86
+ {6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
+ {6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7}.Debug|x64.ActiveCfg = Debug|x64
+ {6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7}.Debug|x64.Build.0 = Debug|x64
+ {6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7}.Debug|x86.ActiveCfg = Debug|x86
+ {6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7}.Debug|x86.Build.0 = Debug|x86
+ {6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7}.Release|Any CPU.Build.0 = Release|Any CPU
+ {6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7}.Release|x64.ActiveCfg = Release|x64
+ {6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7}.Release|x64.Build.0 = Release|x64
+ {6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7}.Release|x86.ActiveCfg = Release|x86
+ {6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7}.Release|x86.Build.0 = Release|x86
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
@@ -113,5 +128,6 @@ Global
{BFC6DC28-0351-4573-926A-D4124244C04F} = {288C363D-A636-4EAE-9AC1-4698B641B26E}
{5688A353-121E-40A1-8BFA-B17B91FB48FB} = {288C363D-A636-4EAE-9AC1-4698B641B26E}
{688FD1D8-7F01-4539-B2E9-F473C5D699C7} = {288C363D-A636-4EAE-9AC1-4698B641B26E}
+ {6BDEEC08-417B-459F-9CA3-FF8BAB18CAC7} = {B0657AAE-DCC5-4FBF-8E5D-1FB578CF3012}
EndGlobalSection
EndGlobal
diff --git a/src/Discord.Net.Providers.WS4Net/Discord.Net.Providers.WS4Net.csproj b/src/Discord.Net.Providers.WS4Net/Discord.Net.Providers.WS4Net.csproj
new file mode 100644
index 000000000..963de8269
--- /dev/null
+++ b/src/Discord.Net.Providers.WS4Net/Discord.Net.Providers.WS4Net.csproj
@@ -0,0 +1,29 @@
+
+
+ An optional WebSocket client provider for Discord.Net using WebSocket4Net
+ 1.0.0-beta2
+ net45
+ true
+ Discord.Net.Providers.WS4Net
+ discord;discordapp
+ https://github.com/RogueException/Discord.Net
+ http://opensource.org/licenses/MIT
+ git
+ git://github.com/RogueException/Discord.Net
+
+
+
+
+
+
+
+
+
+
+
+
+ $(NoWarn);CS1573;CS1591
+ true
+ true
+
+
\ No newline at end of file
diff --git a/src/Discord.Net.Providers.WS4Net/WS4NetClient.cs b/src/Discord.Net.Providers.WS4Net/WS4NetClient.cs
new file mode 100644
index 000000000..7ab8685ba
--- /dev/null
+++ b/src/Discord.Net.Providers.WS4Net/WS4NetClient.cs
@@ -0,0 +1,166 @@
+using Discord.Net.WebSockets;
+using System;
+using System.Text;
+using System.Threading;
+using System.Threading.Tasks;
+using System.Collections.Generic;
+using System.Linq;
+using WebSocket4Net;
+using WS4NetSocket = WebSocket4Net.WebSocket;
+
+namespace Discord.Net.Providers.WS4Net
+{
+ internal class WS4NetClient : IWebSocketClient, IDisposable
+ {
+ public event Func BinaryMessage;
+ public event Func TextMessage;
+ public event Func Closed;
+
+ private readonly SemaphoreSlim _lock;
+ private readonly Dictionary _headers;
+ private WS4NetSocket _client;
+ private CancellationTokenSource _cancelTokenSource;
+ private CancellationToken _cancelToken, _parentToken;
+ private ManualResetEventSlim _waitUntilConnect;
+ private bool _isDisposed;
+
+ public WS4NetClient()
+ {
+ _headers = new Dictionary();
+ _lock = new SemaphoreSlim(1, 1);
+ _cancelTokenSource = new CancellationTokenSource();
+ _cancelToken = CancellationToken.None;
+ _parentToken = CancellationToken.None;
+ _waitUntilConnect = new ManualResetEventSlim();
+ }
+ private void Dispose(bool disposing)
+ {
+ if (!_isDisposed)
+ {
+ if (disposing)
+ DisconnectInternalAsync(true).GetAwaiter().GetResult();
+ _isDisposed = true;
+ }
+ }
+ public void Dispose()
+ {
+ Dispose(true);
+ }
+
+ public async Task ConnectAsync(string host)
+ {
+ await _lock.WaitAsync().ConfigureAwait(false);
+ try
+ {
+ await ConnectInternalAsync(host).ConfigureAwait(false);
+ }
+ finally
+ {
+ _lock.Release();
+ }
+ }
+ private async Task ConnectInternalAsync(string host)
+ {
+ await DisconnectInternalAsync().ConfigureAwait(false);
+
+ _cancelTokenSource = new CancellationTokenSource();
+ _cancelToken = CancellationTokenSource.CreateLinkedTokenSource(_parentToken, _cancelTokenSource.Token).Token;
+
+ _client = new WS4NetSocket(host, customHeaderItems: _headers.ToList())
+ {
+ EnableAutoSendPing = false,
+ NoDelay = true,
+ Proxy = null
+ };
+
+ _client.MessageReceived += OnTextMessage;
+ _client.DataReceived += OnBinaryMessage;
+ _client.Opened += OnConnected;
+ _client.Closed += OnClosed;
+
+ _client.Open();
+ _waitUntilConnect.Wait(_cancelToken);
+ }
+
+ public async Task DisconnectAsync()
+ {
+ await _lock.WaitAsync().ConfigureAwait(false);
+ try
+ {
+ await DisconnectInternalAsync().ConfigureAwait(false);
+ }
+ finally
+ {
+ _lock.Release();
+ }
+ }
+ private Task DisconnectInternalAsync(bool isDisposing = false)
+ {
+ _cancelTokenSource.Cancel();
+ if (_client == null)
+ return Task.Delay(0);
+
+ if (_client.State == WebSocketState.Open)
+ {
+ try { _client.Close(1000, ""); }
+ catch { }
+ }
+
+ _client.MessageReceived -= OnTextMessage;
+ _client.DataReceived -= OnBinaryMessage;
+ _client.Opened -= OnConnected;
+ _client.Closed -= OnClosed;
+
+ try { _client.Dispose(); }
+ catch { }
+ _client = null;
+
+ _waitUntilConnect.Reset();
+ return Task.Delay(0);
+ }
+
+ public void SetHeader(string key, string value)
+ {
+ _headers[key] = value;
+ }
+ public void SetCancelToken(CancellationToken cancelToken)
+ {
+ _parentToken = cancelToken;
+ _cancelToken = CancellationTokenSource.CreateLinkedTokenSource(_parentToken, _cancelTokenSource.Token).Token;
+ }
+
+ public async Task SendAsync(byte[] data, int index, int count, bool isText)
+ {
+ await _lock.WaitAsync(_cancelToken).ConfigureAwait(false);
+ try
+ {
+ if (isText)
+ _client.Send(Encoding.UTF8.GetString(data, index, count));
+ else
+ _client.Send(data, index, count);
+ }
+ finally
+ {
+ _lock.Release();
+ }
+ }
+
+ private void OnTextMessage(object sender, MessageReceivedEventArgs e)
+ {
+ TextMessage(e.Message).GetAwaiter().GetResult();
+ }
+ private void OnBinaryMessage(object sender, DataReceivedEventArgs e)
+ {
+ BinaryMessage(e.Data, 0, e.Data.Count()).GetAwaiter().GetResult();
+ }
+ private void OnConnected(object sender, object e)
+ {
+ _waitUntilConnect.Set();
+ }
+ private void OnClosed(object sender, object e)
+ {
+ var ex = (e as SuperSocket.ClientEngine.ErrorEventArgs)?.Exception ?? new Exception("Unexpected close");
+ Closed(ex).GetAwaiter().GetResult();
+ }
+ }
+}
\ No newline at end of file
diff --git a/src/Discord.Net.Providers.WS4Net/WS4NetProvider.cs b/src/Discord.Net.Providers.WS4Net/WS4NetProvider.cs
new file mode 100644
index 000000000..166e767d0
--- /dev/null
+++ b/src/Discord.Net.Providers.WS4Net/WS4NetProvider.cs
@@ -0,0 +1,9 @@
+using Discord.Net.WebSockets;
+
+namespace Discord.Net.Providers.WS4Net
+{
+ public static class WS4NetProvider
+ {
+ public static readonly WebSocketProvider Instance = () => new WS4NetClient();
+ }
+}
diff --git a/src/Discord.Net.Providers.WS4Net/project.json b/src/Discord.Net.Providers.WS4Net/project.json
new file mode 100644
index 000000000..4e35d6bbb
--- /dev/null
+++ b/src/Discord.Net.Providers.WS4Net/project.json
@@ -0,0 +1,38 @@
+{
+ "version": "1.0.0-*",
+ "description": "An optional WebSocket client provider for Discord.Net using WebSocket4Net.",
+ "authors": [ "RogueException", "foxbot" ],
+
+ "packOptions": {
+ "tags": [ "discord", "discordapp" ],
+ "licenseUrl": "http://opensource.org/licenses/MIT",
+ "projectUrl": "https://github.com/RogueException/Discord.Net",
+ "repository": {
+ "type": "git",
+ "url": "git://github.com/RogueException/Discord.Net"
+ }
+ },
+
+ "configurations": {
+ "Release": {
+ "buildOptions": {
+ "define": [ "RELEASE" ],
+ "nowarn": [ "CS1573", "CS1591" ],
+ "optimize": true,
+ "warningsAsErrors": true,
+ "xmlDoc": true
+ }
+ }
+ },
+
+ "dependencies": {
+ "Discord.Net.Core": {
+ "target": "project"
+ },
+ "WebSocket4Net": "0.14.1"
+ },
+
+ "frameworks": {
+ "net45": {}
+ }
+}
\ No newline at end of file