Browse Source

Merge develop

release/3.x.x
Christian Kratky 7 years ago
parent
commit
1ee48a978f
100 changed files with 727 additions and 442 deletions
  1. +3
    -0
      .gitignore
  2. +3
    -3
      Build/MQTTnet.AspNetCore.nuspec
  3. +47
    -0
      Build/MQTTnet.Extensions.Rpc.nuspec
  4. +22
    -9
      Build/MQTTnet.nuspec
  5. +10
    -0
      Build/build.ps1
  6. +30
    -0
      Extensions/MQTTnet.Extensions.Rpc/MQTTnet.Extensions.Rpc.csproj
  7. +101
    -0
      Extensions/MQTTnet.Extensions.Rpc/MqttRpcClient.cs
  8. +1
    -0
      Extensions/MQTTnet.Extensions.Rpc/SampleCCode.c
  9. +1
    -1
      Frameworks/MQTTnet.AspnetCore/ApplicationBuilderExtensions.cs
  10. +11
    -2
      Frameworks/MQTTnet.AspnetCore/MQTTnet.AspnetCore.csproj
  11. +5
    -5
      Frameworks/MQTTnet.AspnetCore/MqttHostedServer.cs
  12. +6
    -6
      Frameworks/MQTTnet.AspnetCore/MqttWebSocketServerAdapter.cs
  13. +1
    -1
      Frameworks/MQTTnet.AspnetCore/MqttWebSocketServerChannel.cs
  14. +4
    -4
      Frameworks/MQTTnet.AspnetCore/ServiceCollectionExtensions.cs
  15. +3
    -3
      Frameworks/MQTTnet.NetStandard/Adapter/IMqttChannelAdapter.cs
  16. +3
    -3
      Frameworks/MQTTnet.NetStandard/Adapter/IMqttServerAdapter.cs
  17. +58
    -47
      Frameworks/MQTTnet.NetStandard/Adapter/MqttChannelAdapter.cs
  18. +2
    -2
      Frameworks/MQTTnet.NetStandard/Adapter/MqttChannelAdapterExtensions.cs
  19. +3
    -3
      Frameworks/MQTTnet.NetStandard/Adapter/MqttConnectingFailedException.cs
  20. +1
    -1
      Frameworks/MQTTnet.NetStandard/Adapter/MqttServerAdapterClientAcceptedEventArgs.cs
  21. +18
    -0
      Frameworks/MQTTnet.NetStandard/Adapter/ReceivedMqttPacket.cs
  22. +1
    -1
      Frameworks/MQTTnet.NetStandard/ApplicationMessagePublisherExtensions.cs
  23. +3
    -3
      Frameworks/MQTTnet.NetStandard/Channel/IMqttChannel.cs
  24. +2
    -3
      Frameworks/MQTTnet.NetStandard/Client/IMqttClient.cs
  25. +3
    -3
      Frameworks/MQTTnet.NetStandard/Client/IMqttClientAdapterFactory.cs
  26. +1
    -1
      Frameworks/MQTTnet.NetStandard/Client/IMqttClientChannelOptions.cs
  27. +1
    -1
      Frameworks/MQTTnet.NetStandard/Client/IMqttClientCredentials.cs
  28. +3
    -3
      Frameworks/MQTTnet.NetStandard/Client/IMqttClientFactory.cs
  29. +2
    -2
      Frameworks/MQTTnet.NetStandard/Client/IMqttClientOptions.cs
  30. +36
    -50
      Frameworks/MQTTnet.NetStandard/Client/MqttClient.cs
  31. +1
    -1
      Frameworks/MQTTnet.NetStandard/Client/MqttClientConnectResult.cs
  32. +1
    -1
      Frameworks/MQTTnet.NetStandard/Client/MqttClientConnectedEventArgs.cs
  33. +1
    -1
      Frameworks/MQTTnet.NetStandard/Client/MqttClientCredentials.cs
  34. +1
    -1
      Frameworks/MQTTnet.NetStandard/Client/MqttClientDisconnectedEventArgs.cs
  35. +10
    -1
      Frameworks/MQTTnet.NetStandard/Client/MqttClientExtensions.cs
  36. +2
    -2
      Frameworks/MQTTnet.NetStandard/Client/MqttClientOptions.cs
  37. +5
    -5
      Frameworks/MQTTnet.NetStandard/Client/MqttClientOptionsBuilder.cs
  38. +1
    -1
      Frameworks/MQTTnet.NetStandard/Client/MqttClientTcpOptions.cs
  39. +1
    -1
      Frameworks/MQTTnet.NetStandard/Client/MqttClientTcpOptionsExtensions.cs
  40. +1
    -1
      Frameworks/MQTTnet.NetStandard/Client/MqttClientTlsOptions.cs
  41. +1
    -1
      Frameworks/MQTTnet.NetStandard/Client/MqttClientWebSocketOptions.cs
  42. +5
    -5
      Frameworks/MQTTnet.NetStandard/Client/MqttPacketDispatcher.cs
  43. +2
    -3
      Frameworks/MQTTnet.NetStandard/Client/MqttSubscribeResult.cs
  44. +1
    -1
      Frameworks/MQTTnet.NetStandard/Diagnostics/IMqttNetLogger.cs
  45. +0
    -50
      Frameworks/MQTTnet.NetStandard/Diagnostics/MqttNetConsoleLogger.cs
  46. +2
    -2
      Frameworks/MQTTnet.NetStandard/Diagnostics/MqttNetGlobalLogger.cs
  47. +1
    -1
      Frameworks/MQTTnet.NetStandard/Diagnostics/MqttNetLogLevel.cs
  48. +1
    -1
      Frameworks/MQTTnet.NetStandard/Diagnostics/MqttNetLogMessage.cs
  49. +1
    -1
      Frameworks/MQTTnet.NetStandard/Diagnostics/MqttNetLogMessagePublishedEventArgs.cs
  50. +4
    -4
      Frameworks/MQTTnet.NetStandard/Diagnostics/MqttNetLogger.cs
  51. +1
    -1
      Frameworks/MQTTnet.NetStandard/Diagnostics/TargetFrameworkInfoProvider.cs
  52. +1
    -1
      Frameworks/MQTTnet.NetStandard/Exceptions/MqttCommunicationException.cs
  53. +1
    -1
      Frameworks/MQTTnet.NetStandard/Exceptions/MqttCommunicationTimedOutException.cs
  54. +1
    -1
      Frameworks/MQTTnet.NetStandard/Exceptions/MqttProtocolViolationException.cs
  55. +1
    -1
      Frameworks/MQTTnet.NetStandard/IApplicationMessagePublisher.cs
  56. +1
    -2
      Frameworks/MQTTnet.NetStandard/IApplicationMessageReceiver.cs
  57. +4
    -4
      Frameworks/MQTTnet.NetStandard/Implementations/MqttClientAdapterFactory.cs
  58. +5
    -5
      Frameworks/MQTTnet.NetStandard/Implementations/MqttServerAdapter.Uwp.cs
  59. +43
    -12
      Frameworks/MQTTnet.NetStandard/Implementations/MqttServerAdapter.cs
  60. +2
    -2
      Frameworks/MQTTnet.NetStandard/Implementations/MqttTcpChannel.Uwp.cs
  61. +6
    -9
      Frameworks/MQTTnet.NetStandard/Implementations/MqttTcpChannel.cs
  62. +3
    -3
      Frameworks/MQTTnet.NetStandard/Implementations/MqttWebSocketChannel.cs
  63. +1
    -1
      Frameworks/MQTTnet.NetStandard/Implementations/WebSocketStream.cs
  64. +2
    -2
      Frameworks/MQTTnet.NetStandard/Internal/MqttApplicationMessageExtensions.cs
  65. +13
    -9
      Frameworks/MQTTnet.NetStandard/Internal/TaskExtensions.cs
  66. +10
    -7
      Frameworks/MQTTnet.NetStandard/MQTTnet.Netstandard.csproj
  67. +3
    -3
      Frameworks/MQTTnet.NetStandard/ManagedClient/IManagedMqttClient.cs
  68. +2
    -2
      Frameworks/MQTTnet.NetStandard/ManagedClient/IManagedMqttClientOptions.cs
  69. +1
    -1
      Frameworks/MQTTnet.NetStandard/ManagedClient/IManagedMqttClientStorage.cs
  70. +133
    -69
      Frameworks/MQTTnet.NetStandard/ManagedClient/ManagedMqttClient.cs
  71. +11
    -2
      Frameworks/MQTTnet.NetStandard/ManagedClient/ManagedMqttClientExtensions.cs
  72. +2
    -2
      Frameworks/MQTTnet.NetStandard/ManagedClient/ManagedMqttClientOptions.cs
  73. +2
    -2
      Frameworks/MQTTnet.NetStandard/ManagedClient/ManagedMqttClientOptionsBuilder.cs
  74. +13
    -21
      Frameworks/MQTTnet.NetStandard/ManagedClient/ManagedMqttClientStorageManager.cs
  75. +1
    -1
      Frameworks/MQTTnet.NetStandard/ManagedClient/ReconnectionResult.cs
  76. +2
    -2
      Frameworks/MQTTnet.NetStandard/MqttApplicationMessage.cs
  77. +3
    -3
      Frameworks/MQTTnet.NetStandard/MqttApplicationMessageBuilder.cs
  78. +1
    -1
      Frameworks/MQTTnet.NetStandard/MqttApplicationMessageReceivedEventArgs.cs
  79. +5
    -5
      Frameworks/MQTTnet.NetStandard/MqttFactory.cs
  80. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/IMqttPacketWithIdentifier.cs
  81. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/MqttBasePacket.cs
  82. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/MqttBasePublishPacket.cs
  83. +2
    -2
      Frameworks/MQTTnet.NetStandard/Packets/MqttConnAckPacket.cs
  84. +2
    -2
      Frameworks/MQTTnet.NetStandard/Packets/MqttConnectPacket.cs
  85. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/MqttDisconnectPacket.cs
  86. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/MqttPacketExtensions.cs
  87. +2
    -2
      Frameworks/MQTTnet.NetStandard/Packets/MqttPacketHeader.cs
  88. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/MqttPingReqPacket.cs
  89. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/MqttPingRespPacket.cs
  90. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/MqttPubAckPacket.cs
  91. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/MqttPubCompPacket.cs
  92. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/MqttPubRecPacket.cs
  93. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/MqttPubRelPacket.cs
  94. +2
    -2
      Frameworks/MQTTnet.NetStandard/Packets/MqttPublishPacket.cs
  95. +2
    -2
      Frameworks/MQTTnet.NetStandard/Packets/MqttSubAckPacket.cs
  96. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/MqttSubscribePacket.cs
  97. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/MqttUnsubAckPacket.cs
  98. +1
    -1
      Frameworks/MQTTnet.NetStandard/Packets/MqttUnsubscribe.cs
  99. +1
    -1
      Frameworks/MQTTnet.NetStandard/Protocol/MqttConnectReturnCode.cs
  100. +1
    -1
      Frameworks/MQTTnet.NetStandard/Protocol/MqttControlPacketType.cs

+ 3
- 0
.gitignore View File

@@ -291,3 +291,6 @@ __pycache__/
Build/nuget.exe Build/nuget.exe
*.js *.js
*.map *.map

*codeSigningKey.pfx
/Tests/MQTTnet.TestApp.NetCore/RetainedMessages.json

+ 3
- 3
Build/MQTTnet.AspNetCore.nuspec View File

@@ -2,7 +2,7 @@
<package > <package >
<metadata> <metadata>
<id>MQTTnet.AspNetCore</id> <id>MQTTnet.AspNetCore</id>
<version>2.5.3</version>
<version>2.6.0</version>
<authors>Christian Kratky</authors> <authors>Christian Kratky</authors>
<owners>Christian Kratky</owners> <owners>Christian Kratky</owners>
<licenseUrl>https://github.com/chkr1011/MQTTnet/blob/master/LICENSE</licenseUrl> <licenseUrl>https://github.com/chkr1011/MQTTnet/blob/master/LICENSE</licenseUrl>
@@ -10,13 +10,13 @@
<iconUrl>https://raw.githubusercontent.com/chkr1011/MQTTnet/master/Images/Logo_128x128.png</iconUrl> <iconUrl>https://raw.githubusercontent.com/chkr1011/MQTTnet/master/Images/Logo_128x128.png</iconUrl>
<requireLicenseAcceptance>false</requireLicenseAcceptance> <requireLicenseAcceptance>false</requireLicenseAcceptance>
<description>This is a support library to integrate MQTTnet into AspNetCore.</description> <description>This is a support library to integrate MQTTnet into AspNetCore.</description>
<releaseNotes>* Updated to MQTTnet 2.5.3.
<releaseNotes>* Updated to MQTTnet 2.6.0.
</releaseNotes> </releaseNotes>
<copyright>Copyright Christian Kratky 2016-2017</copyright> <copyright>Copyright Christian Kratky 2016-2017</copyright>
<tags>MQTT Message Queue Telemetry Transport MQTTClient MQTTServer Server MQTTBroker Broker NETStandard IoT InternetOfThings Messaging Hardware Arduino Sensor Actuator M2M ESP Smart Home Cities Automation Xamarin</tags> <tags>MQTT Message Queue Telemetry Transport MQTTClient MQTTServer Server MQTTBroker Broker NETStandard IoT InternetOfThings Messaging Hardware Arduino Sensor Actuator M2M ESP Smart Home Cities Automation Xamarin</tags>
<dependencies> <dependencies>
<group targetFramework="netstandard2.0"> <group targetFramework="netstandard2.0">
<dependency id="MQTTnet" version="2.5.3" />
<dependency id="MQTTnet" version="2.6.0" />
</group> </group>
</dependencies> </dependencies>
</metadata> </metadata>


+ 47
- 0
Build/MQTTnet.Extensions.Rpc.nuspec View File

@@ -0,0 +1,47 @@
<?xml version="1.0"?>
<package >
<metadata>
<id>MQTTnet.Extensions.Rpc</id>
<version>2.6.0</version>
<authors>Christian Kratky</authors>
<owners>Christian Kratky</owners>
<licenseUrl>https://github.com/chkr1011/MQTTnet/blob/master/LICENSE</licenseUrl>
<projectUrl>https://github.com/chkr1011/MQTTnet</projectUrl>
<iconUrl>https://raw.githubusercontent.com/chkr1011/MQTTnet/master/Images/Logo_128x128.png</iconUrl>
<requireLicenseAcceptance>false</requireLicenseAcceptance>
<description>This is a extension library which allows executing synchronous device calls including a response using MQTTnet.</description>
<releaseNotes>* Initial version.
</releaseNotes>
<copyright>Copyright Christian Kratky 2016-2017</copyright>
<tags>MQTT Message Queue Telemetry Transport MQTTClient MQTTServer Server MQTTBroker Broker NETStandard IoT InternetOfThings Messaging Hardware Arduino Sensor Actuator M2M ESP Smart Home Cities Automation Xamarin</tags>
<dependencies>
<group targetFramework="netstandard2.0">
<dependency id="MQTTnet" version="2.6.0-rc1" />
</group>
<group targetFramework="netstandard1.3">
<dependency id="MQTTnet" version="2.6.0-rc1" />
</group>

<group targetFramework="netstandard2.0">
<dependency id="MQTTnet" version="2.6.0-rc1" />
</group>
<group targetFramework="uap10.0">
<dependency id="MQTTnet" version="2.6.0-rc1" />
</group>
<group targetFramework="net452">
<dependency id="MQTTnet" version="2.6.0-rc1" />
</group>

<group targetFramework="net461">
<dependency id="MQTTnet" version="2.6.0-rc1" />
</group>
</dependencies>
</metadata>

<files>
<!-- .NET Standard 2.0 -->
<file src="..\Frameworks\MQTTnet.AspNetCore\bin\Release\netstandard2.0\MQTTnet.AspNetCore.*" target="lib\netstandard2.0\"/>
</files>
</package>

+ 22
- 9
Build/MQTTnet.nuspec View File

@@ -2,7 +2,7 @@
<package > <package >
<metadata> <metadata>
<id>MQTTnet</id> <id>MQTTnet</id>
<version>2.5.3</version>
<version>2.6.0</version>
<authors>Christian Kratky</authors> <authors>Christian Kratky</authors>
<owners>Christian Kratky</owners> <owners>Christian Kratky</owners>
<licenseUrl>https://github.com/chkr1011/MQTTnet/blob/master/LICENSE</licenseUrl> <licenseUrl>https://github.com/chkr1011/MQTTnet/blob/master/LICENSE</licenseUrl>
@@ -10,9 +10,26 @@
<iconUrl>https://raw.githubusercontent.com/chkr1011/MQTTnet/master/Images/Logo_128x128.png</iconUrl> <iconUrl>https://raw.githubusercontent.com/chkr1011/MQTTnet/master/Images/Logo_128x128.png</iconUrl>
<requireLicenseAcceptance>false</requireLicenseAcceptance> <requireLicenseAcceptance>false</requireLicenseAcceptance>
<description>MQTTnet is a high performance .NET library for MQTT based communication. It provides a MQTT client and a MQTT server (broker).</description> <description>MQTTnet is a high performance .NET library for MQTT based communication. It provides a MQTT client and a MQTT server (broker).</description>
<releaseNotes>* [Core] Removed all dependencies to other libraries (BREAKING CHANGE!).
* [Core] Updated SDK libraries.
* [Client] Fixed broken support for WebSocketSecure connections (Thanks to @StAI).
<releaseNotes>* [Core] Merged projects (BREAKING CHANGE! But only namespace changes).
* [Core] Added a strong name for the assembly.
* [Core] Performance optimizations.
* [Core] Fixed a logging issue when dealing with IOExceptions.
* [Core] Fixed a typo in the global logger class (BREAKING CHANGE! Please find new example in Wiki).
* [Core] Added support for project compilation under macOS (Thanks to @FourOnes).
* [Client] The certificate for encryption is now loaded as a _X509Certificate2_ which prodives more data (Thanks to @cvellan).
* [Client] Fixed an issue in _ManagedClient_ which can cause the client to stop when publishing subscriptions.
* [Client] Fixed an issue in _ManagedClient_ which prevents changing the QoS of an existing subscription (BREAKING CHANGE!).
* [Client] Fixed an issue in _MqttClientOptionsBuilder_ which prevents adding TLS options to the client options when calling _Build()_. (Thanks to @cvellan).
* [Client] Added an overload for subscribing at both clients which reduces required code.
* [Client] Added the _ClientId_ to the application message interceptor context.
* [Server] The application message interceptor can now delete any received application message.
* [Server] Added a ConnectionValidator context to align with other APIs (BREAKING CHANGE! Please find new example in Wiki).
* [Server] Added an interface for the _MqttServerOptions_.
* [Server] Added packet statistics for the connected clients.
* [Server] Fixed a security issue which sends retained packages to a failed subscription.
* [Server] Fixed the response (MaximumQoS) of a subscription (Thanks to @redbeans2017).
* [Server] The keep alive timeouts are now checked for every client (Thanks to @RainerMueller82).
* [Server] The used IP addresses can be specified now (IPAddress.Any as default).
</releaseNotes> </releaseNotes>
<copyright>Copyright Christian Kratky 2016-2017</copyright> <copyright>Copyright Christian Kratky 2016-2017</copyright>
<tags>MQTT Message Queue Telemetry Transport MQTTClient MQTTServer Server MQTTBroker Broker NETStandard IoT InternetOfThings Messaging Hardware Arduino Sensor Actuator M2M ESP Smart Home Cities Automation Xamarin</tags> <tags>MQTT Message Queue Telemetry Transport MQTTClient MQTTServer Server MQTTBroker Broker NETStandard IoT InternetOfThings Messaging Hardware Arduino Sensor Actuator M2M ESP Smart Home Cities Automation Xamarin</tags>
@@ -46,23 +63,19 @@
</metadata> </metadata>


<files> <files>
<!-- .NET Standard 1.3 --> <!-- .NET Standard 1.3 -->
<file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\netstandard1.3\MQTTnet.Core.*" target="lib\netstandard1.3\"/>
<file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\netstandard1.3\MQTTnet.*" target="lib\netstandard1.3\"/> <file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\netstandard1.3\MQTTnet.*" target="lib\netstandard1.3\"/>
<!-- .NET Standard 2.0 --> <!-- .NET Standard 2.0 -->
<file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\netstandard2.0\MQTTnet.Core.*" target="lib\netstandard2.0\"/>
<file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\netstandard2.0\MQTTnet.*" target="lib\netstandard2.0\"/> <file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\netstandard2.0\MQTTnet.*" target="lib\netstandard2.0\"/>


<!-- Universal Windows --> <!-- Universal Windows -->
<file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\uap10.0\MQTTnet.Core.*" target="lib\uap10.0\"/>
<file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\uap10.0\MQTTnet.*" target="lib\uap10.0\"/> <file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\uap10.0\MQTTnet.*" target="lib\uap10.0\"/>


<!-- .NET Framework --> <!-- .NET Framework -->
<file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\net452\MQTTnet.Core.*" target="lib\net452\"/>
<file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\net452\MQTTnet.*" target="lib\net452\"/> <file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\net452\MQTTnet.*" target="lib\net452\"/>


<file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\net461\MQTTnet.Core.*" target="lib\net461\"/>
<file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\net461\MQTTnet.*" target="lib\net461\"/> <file src="..\Frameworks\MQTTnet.Netstandard\bin\Release\net461\MQTTnet.*" target="lib\net461\"/>
</files> </files>

+ 10
- 0
Build/build.ps1 View File

@@ -8,16 +8,26 @@ $path = &$vswhere -latest -products * -requires Microsoft.Component.MSBuild -pro
if ($path) { if ($path) {
$msbuild = join-path $path 'MSBuild\15.0\Bin\MSBuild.exe' $msbuild = join-path $path 'MSBuild\15.0\Bin\MSBuild.exe'


# Build the core library
&$msbuild ..\Frameworks\MQTTnet.Netstandard\MQTTnet.Netstandard.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="net452" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m &$msbuild ..\Frameworks\MQTTnet.Netstandard\MQTTnet.Netstandard.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="net452" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m
&$msbuild ..\Frameworks\MQTTnet.Netstandard\MQTTnet.Netstandard.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="net461" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m &$msbuild ..\Frameworks\MQTTnet.Netstandard\MQTTnet.Netstandard.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="net461" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m
&$msbuild ..\Frameworks\MQTTnet.Netstandard\MQTTnet.Netstandard.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="netstandard1.3" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m &$msbuild ..\Frameworks\MQTTnet.Netstandard\MQTTnet.Netstandard.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="netstandard1.3" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m
&$msbuild ..\Frameworks\MQTTnet.Netstandard\MQTTnet.Netstandard.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="netstandard2.0" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m &$msbuild ..\Frameworks\MQTTnet.Netstandard\MQTTnet.Netstandard.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="netstandard2.0" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m
&$msbuild ..\Frameworks\MQTTnet.Netstandard\MQTTnet.Netstandard.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="uap10.0" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m &$msbuild ..\Frameworks\MQTTnet.Netstandard\MQTTnet.Netstandard.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="uap10.0" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m
# Build the ASP.NET Core 2.0 extension
&$msbuild ..\Frameworks\MQTTnet.AspNetCore\MQTTnet.AspNetCore.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="netstandard2.0" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m &$msbuild ..\Frameworks\MQTTnet.AspNetCore\MQTTnet.AspNetCore.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="netstandard2.0" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m


# Build the RPC extension
&$msbuild ..\Extensions\MQTTnet.Extensions.Rpc\MQTTnet.Extensions.Rpc.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="net452" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m
&$msbuild ..\Extensions\MQTTnet.Extensions.Rpc\MQTTnet.Extensions.Rpc.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="net461" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m
&$msbuild ..\Extensions\MQTTnet.Extensions.Rpc\MQTTnet.Extensions.Rpc.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="netstandard1.3" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m
&$msbuild ..\Extensions\MQTTnet.Extensions.Rpc\MQTTnet.Extensions.Rpc.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="netstandard2.0" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m
&$msbuild ..\Extensions\MQTTnet.Extensions.Rpc\MQTTnet.Extensions.Rpc.csproj /t:Build /p:Configuration="Release" /p:TargetFramework="uap10.0" /p:FileVersion=$assemblyVersion /p:AssemblyVersion=$assemblyVersion /verbosity:m

Remove-Item .\NuGet -Force -Recurse Remove-Item .\NuGet -Force -Recurse
New-Item -ItemType Directory -Force -Path .\NuGet New-Item -ItemType Directory -Force -Path .\NuGet
.\NuGet.exe pack MQTTnet.nuspec -Verbosity detailed -Symbols -OutputDir "NuGet" -Version $nugetVersion .\NuGet.exe pack MQTTnet.nuspec -Verbosity detailed -Symbols -OutputDir "NuGet" -Version $nugetVersion
.\NuGet.exe pack MQTTnet.AspNetCore.nuspec -Verbosity detailed -Symbols -OutputDir "NuGet" -Version $nugetVersion .\NuGet.exe pack MQTTnet.AspNetCore.nuspec -Verbosity detailed -Symbols -OutputDir "NuGet" -Version $nugetVersion
.\NuGet.exe pack MQTTnet.Extensions.Rpc.nuspec -Verbosity detailed -Symbols -OutputDir "NuGet" -Version $nugetVersion
} }

+ 30
- 0
Extensions/MQTTnet.Extensions.Rpc/MQTTnet.Extensions.Rpc.csproj View File

@@ -0,0 +1,30 @@
<Project Sdk="Microsoft.NET.Sdk">

<PropertyGroup>
<TargetFrameworks>netstandard1.3;netstandard2.0;net452;net461;uap10.0</TargetFrameworks>
<AssemblyVersion>0.0.0.0</AssemblyVersion>
<FileVersion>0.0.0.0</FileVersion>
<Product />
<Company />
<Authors />
<PackageId />
<Version>0.0.0.0</Version>
</PropertyGroup>

<PropertyGroup Condition="'$(TargetFramework)' == 'uap10.0'">
<CopyLocalLockFileAssemblies>false</CopyLocalLockFileAssemblies>
<NugetTargetMoniker>UAP,Version=v10.0</NugetTargetMoniker>
<TargetPlatformIdentifier>UAP</TargetPlatformIdentifier>
<TargetPlatformVersion>10.0.16299.0</TargetPlatformVersion>
<TargetPlatformMinVersion>10.0.10240.0</TargetPlatformMinVersion>
<TargetFrameworkIdentifier>.NETCore</TargetFrameworkIdentifier>
<TargetFrameworkVersion>v5.0</TargetFrameworkVersion>
<DefineConstants>$(DefineConstants);WINDOWS_UWP</DefineConstants>
<LanguageTargets>$(MSBuildExtensionsPath)\Microsoft\WindowsXaml\v$(VisualStudioVersion)\Microsoft.Windows.UI.Xaml.CSharp.targets</LanguageTargets>
</PropertyGroup>

<ItemGroup>
<ProjectReference Include="..\..\Frameworks\MQTTnet.Netstandard\MQTTnet.NetStandard.csproj" />
</ItemGroup>

</Project>

+ 101
- 0
Extensions/MQTTnet.Extensions.Rpc/MqttRpcClient.cs View File

@@ -0,0 +1,101 @@
using System;
using System.Collections.Concurrent;
using System.Threading.Tasks;
using MQTTnet.Client;
using MQTTnet.Internal;
using MQTTnet.Protocol;

namespace MQTTnet.Extensions.Rpc
{
public sealed class MqttRpcClient : IDisposable
{
private const string ResponseTopic = "$MQTTnet.RPC/+/+/response";
private readonly ConcurrentDictionary<string, TaskCompletionSource<byte[]>> _waitingCalls = new ConcurrentDictionary<string, TaskCompletionSource<byte[]>>();
private readonly IMqttClient _mqttClient;
private bool _isEnabled;

public MqttRpcClient(IMqttClient mqttClient)
{
_mqttClient = mqttClient ?? throw new ArgumentNullException(nameof(mqttClient));

_mqttClient.ApplicationMessageReceived += OnApplicationMessageReceived;
}

public async Task EnableAsync()
{
await _mqttClient.SubscribeAsync(new TopicFilterBuilder().WithTopic(ResponseTopic).WithAtLeastOnceQoS().Build());
_isEnabled = true;
}

public async Task DisableAsync()
{
await _mqttClient.UnsubscribeAsync(ResponseTopic);
_isEnabled = false;
}

public async Task<byte[]> ExecuteAsync(TimeSpan timeout, string methodName, byte[] payload, MqttQualityOfServiceLevel qualityOfServiceLevel)
{
if (methodName == null) throw new ArgumentNullException(nameof(methodName));

if (methodName.Contains("/") || methodName.Contains("+") || methodName.Contains("#"))
{
throw new ArgumentException("The method name cannot contain /, + or #.");
}

if (!_isEnabled)
{
throw new InvalidOperationException("The RPC client is not enabled.");
}

var requestTopic = $"$MQTTnet.RPC/{Guid.NewGuid():N}/{methodName}";
var responseTopic = requestTopic + "/response";

var requestMessage = new MqttApplicationMessageBuilder()
.WithTopic(requestTopic)
.WithPayload(payload)
.WithQualityOfServiceLevel(qualityOfServiceLevel)
.Build();

try
{
var tcs = new TaskCompletionSource<byte[]>();
if (!_waitingCalls.TryAdd(responseTopic, tcs))
{
throw new InvalidOperationException();
}

await _mqttClient.PublishAsync(requestMessage);
return await tcs.Task.TimeoutAfter(timeout);
}
finally
{
_waitingCalls.TryRemove(responseTopic, out _);
}
}

private void OnApplicationMessageReceived(object sender, MqttApplicationMessageReceivedEventArgs eventArgs)
{
if (!_waitingCalls.TryRemove(eventArgs.ApplicationMessage.Topic, out TaskCompletionSource<byte[]> tcs))
{
return;
}

if (tcs.Task.IsCompleted || tcs.Task.IsCanceled)
{
return;
}

tcs.TrySetResult(eventArgs.ApplicationMessage.Payload);
}

public void Dispose()
{
foreach (var tcs in _waitingCalls)
{
tcs.Value.SetCanceled();
}

_waitingCalls.Clear();
}
}
}

+ 1
- 0
Extensions/MQTTnet.Extensions.Rpc/SampleCCode.c View File

@@ -0,0 +1 @@


+ 1
- 1
Frameworks/MQTTnet.AspnetCore/ApplicationBuilderExtensions.cs View File

@@ -1,8 +1,8 @@
using System; using System;
using Microsoft.AspNetCore.Builder; using Microsoft.AspNetCore.Builder;
using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.DependencyInjection;
using MQTTnet.Core.Server;
using System.Linq; using System.Linq;
using MQTTnet.Server;


namespace MQTTnet.AspNetCore namespace MQTTnet.AspNetCore
{ {


+ 11
- 2
Frameworks/MQTTnet.AspnetCore/MQTTnet.AspnetCore.csproj View File

@@ -2,8 +2,17 @@


<PropertyGroup> <PropertyGroup>
<TargetFramework>netstandard2.0</TargetFramework> <TargetFramework>netstandard2.0</TargetFramework>
<AssemblyVersion>2.5.2.0</AssemblyVersion>
<FileVersion>2.5.2.0</FileVersion>
<AssemblyVersion>0.0.0.0</AssemblyVersion>
<FileVersion>0.0.0.0</FileVersion>
<Version>0.0.0.0</Version>
<Product />
<Company />
<Authors />
<PackageId />
</PropertyGroup>

<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|AnyCPU'">
<DefineConstants>RELEASE;NETSTANDARD2_0</DefineConstants>
</PropertyGroup> </PropertyGroup>


<ItemGroup> <ItemGroup>


+ 5
- 5
Frameworks/MQTTnet.AspnetCore/MqttHostedServer.cs View File

@@ -3,17 +3,17 @@ using System.Collections.Generic;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Hosting;
using MQTTnet.Core.Adapter;
using MQTTnet.Core.Diagnostics;
using MQTTnet.Core.Server;
using MQTTnet.Adapter;
using MQTTnet.Diagnostics;
using MQTTnet.Server;


namespace MQTTnet.AspNetCore namespace MQTTnet.AspNetCore
{ {
public class MqttHostedServer : MqttServer, IHostedService public class MqttHostedServer : MqttServer, IHostedService
{ {
private readonly MqttServerOptions _options;
private readonly IMqttServerOptions _options;


public MqttHostedServer(MqttServerOptions options, IEnumerable<IMqttServerAdapter> adapters, IMqttNetLogger logger) : base(adapters, logger)
public MqttHostedServer(IMqttServerOptions options, IEnumerable<IMqttServerAdapter> adapters, IMqttNetLogger logger) : base(adapters, logger)
{ {
_options = options ?? throw new ArgumentNullException(nameof(options)); _options = options ?? throw new ArgumentNullException(nameof(options));
} }


+ 6
- 6
Frameworks/MQTTnet.AspnetCore/MqttWebSocketServerAdapter.cs View File

@@ -1,18 +1,18 @@
using System; using System;
using System.Net.WebSockets; using System.Net.WebSockets;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Adapter;
using MQTTnet.Core.Diagnostics;
using MQTTnet.Core.Serializer;
using MQTTnet.Core.Server;
using MQTTnet.Adapter;
using MQTTnet.Diagnostics;
using MQTTnet.Serializer;
using MQTTnet.Server;


namespace MQTTnet.AspNetCore namespace MQTTnet.AspNetCore
{ {
public class MqttWebSocketServerAdapter : IMqttServerAdapter, IDisposable
public sealed class MqttWebSocketServerAdapter : IMqttServerAdapter, IDisposable
{ {
public event EventHandler<MqttServerAdapterClientAcceptedEventArgs> ClientAccepted; public event EventHandler<MqttServerAdapterClientAcceptedEventArgs> ClientAccepted;


public Task StartAsync(MqttServerOptions options)
public Task StartAsync(IMqttServerOptions options)
{ {
return Task.CompletedTask; return Task.CompletedTask;
} }


+ 1
- 1
Frameworks/MQTTnet.AspnetCore/MqttWebSocketServerChannel.cs View File

@@ -3,7 +3,7 @@ using System.IO;
using System.Net.WebSockets; using System.Net.WebSockets;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Channel;
using MQTTnet.Channel;
using MQTTnet.Implementations; using MQTTnet.Implementations;


namespace MQTTnet.AspNetCore namespace MQTTnet.AspNetCore


+ 4
- 4
Frameworks/MQTTnet.AspnetCore/ServiceCollectionExtensions.cs View File

@@ -1,15 +1,15 @@
using System; using System;
using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Hosting;
using MQTTnet.Core.Adapter;
using MQTTnet.Core.Diagnostics;
using MQTTnet.Core.Server;
using MQTTnet.Adapter;
using MQTTnet.Diagnostics;
using MQTTnet.Server;


namespace MQTTnet.AspNetCore namespace MQTTnet.AspNetCore
{ {
public static class ServiceCollectionExtensions public static class ServiceCollectionExtensions
{ {
public static IServiceCollection AddHostedMqttServer(this IServiceCollection services, MqttServerOptions options)
public static IServiceCollection AddHostedMqttServer(this IServiceCollection services, IMqttServerOptions options)
{ {
if (options == null) throw new ArgumentNullException(nameof(options)); if (options == null) throw new ArgumentNullException(nameof(options));




MQTTnet.Core/Adapter/IMqttChannelAdapter.cs → Frameworks/MQTTnet.NetStandard/Adapter/IMqttChannelAdapter.cs View File

@@ -2,10 +2,10 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Packets;
using MQTTnet.Core.Serializer;
using MQTTnet.Packets;
using MQTTnet.Serializer;


namespace MQTTnet.Core.Adapter
namespace MQTTnet.Adapter
{ {
public interface IMqttChannelAdapter public interface IMqttChannelAdapter
{ {

MQTTnet.Core/Adapter/IMqttServerAdapter.cs → Frameworks/MQTTnet.NetStandard/Adapter/IMqttServerAdapter.cs View File

@@ -1,14 +1,14 @@
using System; using System;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Server;
using MQTTnet.Server;


namespace MQTTnet.Core.Adapter
namespace MQTTnet.Adapter
{ {
public interface IMqttServerAdapter public interface IMqttServerAdapter
{ {
event EventHandler<MqttServerAdapterClientAcceptedEventArgs> ClientAccepted; event EventHandler<MqttServerAdapterClientAcceptedEventArgs> ClientAccepted;


Task StartAsync(MqttServerOptions options);
Task StartAsync(IMqttServerOptions options);
Task StopAsync(); Task StopAsync();
} }
} }

MQTTnet.Core/Adapter/MqttChannelAdapter.cs → Frameworks/MQTTnet.NetStandard/Adapter/MqttChannelAdapter.cs View File

@@ -1,22 +1,25 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.IO; using System.IO;
using System.Net.Sockets;
using System.Runtime.InteropServices; using System.Runtime.InteropServices;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Channel;
using MQTTnet.Core.Diagnostics;
using MQTTnet.Core.Exceptions;
using MQTTnet.Core.Internal;
using MQTTnet.Core.Packets;
using MQTTnet.Core.Serializer;
namespace MQTTnet.Core.Adapter
using MQTTnet.Channel;
using MQTTnet.Diagnostics;
using MQTTnet.Exceptions;
using MQTTnet.Internal;
using MQTTnet.Packets;
using MQTTnet.Serializer;
namespace MQTTnet.Adapter
{ {
public class MqttChannelAdapter : IMqttChannelAdapter public class MqttChannelAdapter : IMqttChannelAdapter
{ {
private const uint ErrorOperationAborted = 0x800703E3; private const uint ErrorOperationAborted = 0x800703E3;


private static readonly byte[] EmptyBody = new byte[0];

private readonly SemaphoreSlim _semaphore = new SemaphoreSlim(1, 1); private readonly SemaphoreSlim _semaphore = new SemaphoreSlim(1, 1);
private readonly IMqttNetLogger _logger; private readonly IMqttNetLogger _logger;
private readonly IMqttChannel _channel; private readonly IMqttChannel _channel;
@@ -30,23 +33,23 @@ namespace MQTTnet.Core.Adapter


public IMqttPacketSerializer PacketSerializer { get; } public IMqttPacketSerializer PacketSerializer { get; }


public async Task ConnectAsync(TimeSpan timeout)
public Task ConnectAsync(TimeSpan timeout)
{ {
_logger.Info<MqttChannelAdapter>("Connecting [Timeout={0}]", timeout);
_logger.Trace<MqttChannelAdapter>("Connecting [Timeout={0}]", timeout);


await ExecuteAndWrapExceptionAsync(() => _channel.ConnectAsync().TimeoutAfter(timeout));
return ExecuteAndWrapExceptionAsync(() => _channel.ConnectAsync().TimeoutAfter(timeout));
} }


public async Task DisconnectAsync(TimeSpan timeout)
public Task DisconnectAsync(TimeSpan timeout)
{ {
_logger.Info<MqttChannelAdapter>("Disconnecting [Timeout={0}]", timeout);
_logger.Trace<MqttChannelAdapter>("Disconnecting [Timeout={0}]", timeout);


await ExecuteAndWrapExceptionAsync(() => _channel.DisconnectAsync().TimeoutAfter(timeout));
return ExecuteAndWrapExceptionAsync(() => _channel.DisconnectAsync().TimeoutAfter(timeout));
} }


public async Task SendPacketsAsync(TimeSpan timeout, CancellationToken cancellationToken, IEnumerable<MqttBasePacket> packets)
public Task SendPacketsAsync(TimeSpan timeout, CancellationToken cancellationToken, IEnumerable<MqttBasePacket> packets)
{ {
await ExecuteAndWrapExceptionAsync(async () =>
return ExecuteAndWrapExceptionAsync(async () =>
{ {
await _semaphore.WaitAsync(cancellationToken).ConfigureAwait(false); await _semaphore.WaitAsync(cancellationToken).ConfigureAwait(false);
try try
@@ -60,8 +63,11 @@ namespace MQTTnet.Core.Adapter


_logger.Trace<MqttChannelAdapter>("TX >>> {0} [Timeout={1}]", packet, timeout); _logger.Trace<MqttChannelAdapter>("TX >>> {0} [Timeout={1}]", packet, timeout);


var writeBuffer = PacketSerializer.Serialize(packet);
await _channel.SendStream.WriteAsync(writeBuffer, 0, writeBuffer.Length, cancellationToken).ConfigureAwait(false);
var chunks = PacketSerializer.Serialize(packet);
foreach (var chunk in chunks)
{
await _channel.SendStream.WriteAsync(chunk.Array, chunk.Offset, chunk.Count, cancellationToken).ConfigureAwait(false);
}
} }


if (timeout > TimeSpan.Zero) if (timeout > TimeSpan.Zero)
@@ -85,36 +91,29 @@ namespace MQTTnet.Core.Adapter
MqttBasePacket packet = null; MqttBasePacket packet = null;
await ExecuteAndWrapExceptionAsync(async () => await ExecuteAndWrapExceptionAsync(async () =>
{ {
ReceivedMqttPacket receivedMqttPacket = null;
try
ReceivedMqttPacket receivedMqttPacket;
if (timeout > TimeSpan.Zero)
{ {
if (timeout > TimeSpan.Zero)
{
receivedMqttPacket = await ReceiveAsync(_channel.ReceiveStream, cancellationToken).TimeoutAfter(timeout).ConfigureAwait(false);
}
else
{
receivedMqttPacket = await ReceiveAsync(_channel.ReceiveStream, cancellationToken).ConfigureAwait(false);
}

if (receivedMqttPacket == null || cancellationToken.IsCancellationRequested)
{
throw new TaskCanceledException();
}

packet = PacketSerializer.Deserialize(receivedMqttPacket);
if (packet == null)
{
throw new MqttProtocolViolationException("Received malformed packet.");
}
receivedMqttPacket = await ReceiveAsync(_channel.ReceiveStream, cancellationToken).TimeoutAfter(timeout).ConfigureAwait(false);
}
else
{
receivedMqttPacket = await ReceiveAsync(_channel.ReceiveStream, cancellationToken).ConfigureAwait(false);
}


_logger.Trace<MqttChannelAdapter>("RX <<< {0}", packet);
if (receivedMqttPacket == null || cancellationToken.IsCancellationRequested)
{
throw new TaskCanceledException();
} }
finally

packet = PacketSerializer.Deserialize(receivedMqttPacket.Header, receivedMqttPacket.Body);
if (packet == null)
{ {
receivedMqttPacket?.Dispose();
throw new MqttProtocolViolationException("Received malformed packet.");
} }
});

_logger.Trace<MqttChannelAdapter>("RX <<< {0}", packet);
}).ConfigureAwait(false);


return packet; return packet;
} }
@@ -129,7 +128,7 @@ namespace MQTTnet.Core.Adapter


if (header.BodyLength == 0) if (header.BodyLength == 0)
{ {
return new ReceivedMqttPacket(header, new MemoryStream(0));
return new ReceivedMqttPacket(header, EmptyBody);
} }


var body = new byte[header.BodyLength]; var body = new byte[header.BodyLength];
@@ -140,8 +139,8 @@ namespace MQTTnet.Core.Adapter
var readBytesCount = await stream.ReadAsync(body, offset, body.Length - offset, cancellationToken).ConfigureAwait(false); var readBytesCount = await stream.ReadAsync(body, offset, body.Length - offset, cancellationToken).ConfigureAwait(false);
offset += readBytesCount; offset += readBytesCount;
} while (offset < header.BodyLength); } while (offset < header.BodyLength);
return new ReceivedMqttPacket(header, new MemoryStream(body, 0, body.Length));
return new ReceivedMqttPacket(header, body);
} }


private static async Task ExecuteAndWrapExceptionAsync(Func<Task> action) private static async Task ExecuteAndWrapExceptionAsync(Func<Task> action)
@@ -168,13 +167,25 @@ namespace MQTTnet.Core.Adapter
} }
catch (COMException comException) catch (COMException comException)
{ {
if ((uint)comException.HResult == ErrorOperationAborted)
if ((uint) comException.HResult == ErrorOperationAborted)
{ {
throw new OperationCanceledException(); throw new OperationCanceledException();
} }


throw new MqttCommunicationException(comException); throw new MqttCommunicationException(comException);
} }
catch (IOException exception)
{
if (exception.InnerException is SocketException socketException)
{
if (socketException.SocketErrorCode == SocketError.ConnectionAborted)
{
throw new OperationCanceledException();
}
}

throw new MqttCommunicationException(exception);
}
catch (Exception exception) catch (Exception exception)
{ {
throw new MqttCommunicationException(exception); throw new MqttCommunicationException(exception);

MQTTnet.Core/Adapter/MqttChannelAdapterExtensions.cs → Frameworks/MQTTnet.NetStandard/Adapter/MqttChannelAdapterExtensions.cs View File

@@ -1,9 +1,9 @@
using System; using System;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Packets;
using MQTTnet.Packets;


namespace MQTTnet.Core.Adapter
namespace MQTTnet.Adapter
{ {
public static class MqttChannelAdapterExtensions public static class MqttChannelAdapterExtensions
{ {

MQTTnet.Core/Adapter/MqttConnectingFailedException.cs → Frameworks/MQTTnet.NetStandard/Adapter/MqttConnectingFailedException.cs View File

@@ -1,7 +1,7 @@
using MQTTnet.Core.Exceptions;
using MQTTnet.Core.Protocol;
using MQTTnet.Exceptions;
using MQTTnet.Protocol;


namespace MQTTnet.Core.Adapter
namespace MQTTnet.Adapter
{ {
public class MqttConnectingFailedException : MqttCommunicationException public class MqttConnectingFailedException : MqttCommunicationException
{ {

MQTTnet.Core/Adapter/MqttServerAdapterClientAcceptedEventArgs.cs → Frameworks/MQTTnet.NetStandard/Adapter/MqttServerAdapterClientAcceptedEventArgs.cs View File

@@ -1,7 +1,7 @@
using System; using System;
using System.Threading.Tasks; using System.Threading.Tasks;


namespace MQTTnet.Core.Adapter
namespace MQTTnet.Adapter
{ {
public class MqttServerAdapterClientAcceptedEventArgs : EventArgs public class MqttServerAdapterClientAcceptedEventArgs : EventArgs
{ {

+ 18
- 0
Frameworks/MQTTnet.NetStandard/Adapter/ReceivedMqttPacket.cs View File

@@ -0,0 +1,18 @@
using System;
using MQTTnet.Packets;

namespace MQTTnet.Adapter
{
public class ReceivedMqttPacket
{
public ReceivedMqttPacket(MqttPacketHeader header, byte[] body)
{
Header = header ?? throw new ArgumentNullException(nameof(header));
Body = body ?? throw new ArgumentNullException(nameof(body));
}

public MqttPacketHeader Header { get; }

public byte[] Body { get; }
}
}

MQTTnet.Core/ApplicationMessagePublisherExtensions.cs → Frameworks/MQTTnet.NetStandard/ApplicationMessagePublisherExtensions.cs View File

@@ -1,7 +1,7 @@
using System; using System;
using System.Threading.Tasks; using System.Threading.Tasks;


namespace MQTTnet.Core
namespace MQTTnet
{ {
public static class ApplicationMessagePublisherExtensions public static class ApplicationMessagePublisherExtensions
{ {

MQTTnet.Core/Channel/IMqttChannel.cs → Frameworks/MQTTnet.NetStandard/Channel/IMqttChannel.cs View File

@@ -1,7 +1,7 @@
using System.Threading.Tasks;
using System.IO;
using System.IO;
using System.Threading.Tasks;


namespace MQTTnet.Core.Channel
namespace MQTTnet.Channel
{ {
public interface IMqttChannel public interface IMqttChannel
{ {

MQTTnet.Core/Client/IMqttClient.cs → Frameworks/MQTTnet.NetStandard/Client/IMqttClient.cs View File

@@ -1,9 +1,8 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Packets;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public interface IMqttClient : IApplicationMessageReceiver, IApplicationMessagePublisher public interface IMqttClient : IApplicationMessageReceiver, IApplicationMessagePublisher
{ {
@@ -16,6 +15,6 @@ namespace MQTTnet.Core.Client
Task DisconnectAsync(); Task DisconnectAsync();


Task<IList<MqttSubscribeResult>> SubscribeAsync(IEnumerable<TopicFilter> topicFilters); Task<IList<MqttSubscribeResult>> SubscribeAsync(IEnumerable<TopicFilter> topicFilters);
Task UnsubscribeAsync(IEnumerable<string> topicFilters);
Task UnsubscribeAsync(IEnumerable<string> topics);
} }
} }

MQTTnet.Core/Client/IMqttClientAdapterFactory.cs → Frameworks/MQTTnet.NetStandard/Client/IMqttClientAdapterFactory.cs View File

@@ -1,7 +1,7 @@
using MQTTnet.Core.Adapter;
using MQTTnet.Core.Diagnostics;
using MQTTnet.Adapter;
using MQTTnet.Diagnostics;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public interface IMqttClientAdapterFactory public interface IMqttClientAdapterFactory
{ {

MQTTnet.Core/Client/IMqttClientChannelOptions.cs → Frameworks/MQTTnet.NetStandard/Client/IMqttClientChannelOptions.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public interface IMqttClientChannelOptions public interface IMqttClientChannelOptions
{ {

MQTTnet.Core/Client/IMqttClientCredentials.cs → Frameworks/MQTTnet.NetStandard/Client/IMqttClientCredentials.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public interface IMqttClientCredentials public interface IMqttClientCredentials
{ {

MQTTnet.Core/Client/IMqttClientFactory.cs → Frameworks/MQTTnet.NetStandard/Client/IMqttClientFactory.cs View File

@@ -1,7 +1,7 @@
using MQTTnet.Core.Diagnostics;
using MQTTnet.Core.ManagedClient;
using MQTTnet.Diagnostics;
using MQTTnet.ManagedClient;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public interface IMqttClientFactory public interface IMqttClientFactory
{ {

MQTTnet.Core/Client/IMqttClientOptions.cs → Frameworks/MQTTnet.NetStandard/Client/IMqttClientOptions.cs View File

@@ -1,7 +1,7 @@
using System; using System;
using MQTTnet.Core.Serializer;
using MQTTnet.Serializer;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public interface IMqttClientOptions public interface IMqttClientOptions
{ {

MQTTnet.Core/Client/MqttClient.cs → Frameworks/MQTTnet.NetStandard/Client/MqttClient.cs View File

@@ -3,19 +3,17 @@ using System.Collections.Generic;
using System.Linq; using System.Linq;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Adapter;
using MQTTnet.Core.Diagnostics;
using MQTTnet.Core.Exceptions;
using MQTTnet.Core.Internal;
using MQTTnet.Core.Packets;
using MQTTnet.Core.Protocol;
namespace MQTTnet.Core.Client
using MQTTnet.Adapter;
using MQTTnet.Diagnostics;
using MQTTnet.Exceptions;
using MQTTnet.Internal;
using MQTTnet.Packets;
using MQTTnet.Protocol;
namespace MQTTnet.Client
{ {
public class MqttClient : IMqttClient
public class MqttClient : IMqttClient, IDisposable
{ {
private readonly HashSet<ushort> _unacknowledgedPublishPackets = new HashSet<ushort>();

private readonly IMqttClientAdapterFactory _adapterFactory; private readonly IMqttClientAdapterFactory _adapterFactory;
private readonly MqttPacketDispatcher _packetDispatcher; private readonly MqttPacketDispatcher _packetDispatcher;
private readonly IMqttNetLogger _logger; private readonly IMqttNetLogger _logger;
@@ -133,7 +131,7 @@ namespace MQTTnet.Core.Client
TopicFilters = topicFilters.ToList() TopicFilters = topicFilters.ToList()
}; };


await SendAndReceiveAsync<MqttUnsubAckPacket>(unsubscribePacket);
await SendAndReceiveAsync<MqttUnsubAckPacket>(unsubscribePacket).ConfigureAwait(false);
} }


public async Task PublishAsync(IEnumerable<MqttApplicationMessage> applicationMessages) public async Task PublishAsync(IEnumerable<MqttApplicationMessage> applicationMessages)
@@ -141,21 +139,21 @@ namespace MQTTnet.Core.Client
ThrowIfNotConnected(); ThrowIfNotConnected();


var publishPackets = applicationMessages.Select(m => m.ToPublishPacket()); var publishPackets = applicationMessages.Select(m => m.ToPublishPacket());
var packetGroups = publishPackets.GroupBy(p => p.QualityOfServiceLevel).OrderBy(g => g.Key);


foreach (var qosGroup in publishPackets.GroupBy(p => p.QualityOfServiceLevel))
foreach (var qosGroup in packetGroups)
{ {
var qosPackets = qosGroup.ToArray();
switch (qosGroup.Key) switch (qosGroup.Key)
{ {
case MqttQualityOfServiceLevel.AtMostOnce: case MqttQualityOfServiceLevel.AtMostOnce:
{ {
// No packet identifier is used for QoS 0 [3.3.2.2 Packet Identifier] // No packet identifier is used for QoS 0 [3.3.2.2 Packet Identifier]
await _adapter.SendPacketsAsync(_options.CommunicationTimeout, _cancellationTokenSource.Token, qosPackets).ConfigureAwait(false);
await _adapter.SendPacketsAsync(_options.CommunicationTimeout, _cancellationTokenSource.Token, qosGroup).ConfigureAwait(false);
break; break;
} }
case MqttQualityOfServiceLevel.AtLeastOnce: case MqttQualityOfServiceLevel.AtLeastOnce:
{ {
foreach (var publishPacket in qosPackets)
foreach (var publishPacket in qosGroup)
{ {
publishPacket.PacketIdentifier = GetNewPacketIdentifier(); publishPacket.PacketIdentifier = GetNewPacketIdentifier();
await SendAndReceiveAsync<MqttPubAckPacket>(publishPacket).ConfigureAwait(false); await SendAndReceiveAsync<MqttPubAckPacket>(publishPacket).ConfigureAwait(false);
@@ -165,7 +163,7 @@ namespace MQTTnet.Core.Client
} }
case MqttQualityOfServiceLevel.ExactlyOnce: case MqttQualityOfServiceLevel.ExactlyOnce:
{ {
foreach (var publishPacket in qosPackets)
foreach (var publishPacket in qosGroup)
{ {
publishPacket.PacketIdentifier = GetNewPacketIdentifier(); publishPacket.PacketIdentifier = GetNewPacketIdentifier();
var pubRecPacket = await SendAndReceiveAsync<MqttPubRecPacket>(publishPacket).ConfigureAwait(false); var pubRecPacket = await SendAndReceiveAsync<MqttPubRecPacket>(publishPacket).ConfigureAwait(false);
@@ -182,6 +180,11 @@ namespace MQTTnet.Core.Client
} }
} }


public void Dispose()
{
_cancellationTokenSource?.Dispose();
}

private async Task<MqttConnAckPacket> AuthenticateAsync(MqttApplicationMessage willApplicationMessage) private async Task<MqttConnAckPacket> AuthenticateAsync(MqttApplicationMessage willApplicationMessage)
{ {
var connectPacket = new MqttConnectPacket var connectPacket = new MqttConnectPacket
@@ -249,28 +252,28 @@ namespace MQTTnet.Core.Client
try try
{ {
_logger.Info<MqttClient>("Received <<< {0}", packet); _logger.Info<MqttClient>("Received <<< {0}", packet);
if (packet is MqttPingReqPacket)
if (packet is MqttPublishPacket publishPacket)
{ {
await SendAsync(new MqttPingRespPacket());
await ProcessReceivedPublishPacketAsync(publishPacket).ConfigureAwait(false);
return; return;
} }


if (packet is MqttDisconnectPacket)
if (packet is MqttPingReqPacket)
{ {
await DisconnectAsync();
await SendAsync(new MqttPingRespPacket()).ConfigureAwait(false);
return; return;
} }


if (packet is MqttPublishPacket publishPacket)
if (packet is MqttDisconnectPacket)
{ {
await ProcessReceivedPublishPacket(publishPacket);
await DisconnectAsync().ConfigureAwait(false);
return; return;
} }


if (packet is MqttPubRelPacket pubRelPacket) if (packet is MqttPubRelPacket pubRelPacket)
{ {
await ProcessReceivedPubRelPacket(pubRelPacket);
await ProcessReceivedPubRelPacket(pubRelPacket).ConfigureAwait(false);
return; return;
} }


@@ -295,32 +298,25 @@ namespace MQTTnet.Core.Client
} }
} }


private async Task ProcessReceivedPublishPacket(MqttPublishPacket publishPacket)
private Task ProcessReceivedPublishPacketAsync(MqttPublishPacket publishPacket)
{ {
if (publishPacket.QualityOfServiceLevel == MqttQualityOfServiceLevel.AtMostOnce) if (publishPacket.QualityOfServiceLevel == MqttQualityOfServiceLevel.AtMostOnce)
{ {
FireApplicationMessageReceivedEvent(publishPacket); FireApplicationMessageReceivedEvent(publishPacket);
return;
return Task.FromResult(0);
} }


if (publishPacket.QualityOfServiceLevel == MqttQualityOfServiceLevel.AtLeastOnce) if (publishPacket.QualityOfServiceLevel == MqttQualityOfServiceLevel.AtLeastOnce)
{ {
FireApplicationMessageReceivedEvent(publishPacket); FireApplicationMessageReceivedEvent(publishPacket);
await SendAsync(new MqttPubAckPacket { PacketIdentifier = publishPacket.PacketIdentifier });
return;
return SendAsync(new MqttPubAckPacket { PacketIdentifier = publishPacket.PacketIdentifier });
} }


if (publishPacket.QualityOfServiceLevel == MqttQualityOfServiceLevel.ExactlyOnce) if (publishPacket.QualityOfServiceLevel == MqttQualityOfServiceLevel.ExactlyOnce)
{ {
// QoS 2 is implement as method "B" [4.3.3 QoS 2: Exactly once delivery] // QoS 2 is implement as method "B" [4.3.3 QoS 2: Exactly once delivery]
lock (_unacknowledgedPublishPackets)
{
_unacknowledgedPublishPackets.Add(publishPacket.PacketIdentifier);
}

FireApplicationMessageReceivedEvent(publishPacket); FireApplicationMessageReceivedEvent(publishPacket);
await SendAsync(new MqttPubRecPacket { PacketIdentifier = publishPacket.PacketIdentifier });
return;
return SendAsync(new MqttPubRecPacket { PacketIdentifier = publishPacket.PacketIdentifier });
} }


throw new MqttCommunicationException("Received a not supported QoS level."); throw new MqttCommunicationException("Received a not supported QoS level.");
@@ -328,11 +324,6 @@ namespace MQTTnet.Core.Client


private Task ProcessReceivedPubRelPacket(MqttPubRelPacket pubRelPacket) private Task ProcessReceivedPubRelPacket(MqttPubRelPacket pubRelPacket)
{ {
lock (_unacknowledgedPublishPackets)
{
_unacknowledgedPublishPackets.Remove(pubRelPacket.PacketIdentifier);
}

return SendAsync(pubRelPacket.CreateResponse<MqttPubCompPacket>()); return SendAsync(pubRelPacket.CreateResponse<MqttPubCompPacket>());
} }


@@ -361,13 +352,8 @@ namespace MQTTnet.Core.Client
{ {
while (!cancellationToken.IsCancellationRequested) while (!cancellationToken.IsCancellationRequested)
{ {
await Task.Delay(_options.KeepAlivePeriod, cancellationToken).ConfigureAwait(false);
if (cancellationToken.IsCancellationRequested)
{
return;
}

await SendAndReceiveAsync<MqttPingRespPacket>(new MqttPingReqPacket()).ConfigureAwait(false); await SendAndReceiveAsync<MqttPingRespPacket>(new MqttPingReqPacket()).ConfigureAwait(false);
await Task.Delay(_options.KeepAlivePeriod, cancellationToken).ConfigureAwait(false);
} }
} }
catch (OperationCanceledException) catch (OperationCanceledException)
@@ -454,7 +440,7 @@ namespace MQTTnet.Core.Client
{ {
#pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed #pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
Task.Run( Task.Run(
() => ProcessReceivedPacketAsync(packet),
async () => await ProcessReceivedPacketAsync(packet).ConfigureAwait(false),
cancellationToken).ConfigureAwait(false); cancellationToken).ConfigureAwait(false);
#pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed #pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
} }
@@ -465,7 +451,7 @@ namespace MQTTnet.Core.Client


#pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed #pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
Task.Run( Task.Run(
async () => await ReceivePacketsAsync(cancellationToken),
async () => await ReceivePacketsAsync(cancellationToken).ConfigureAwait(false),
cancellationToken).ConfigureAwait(false); cancellationToken).ConfigureAwait(false);
#pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed #pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed


@@ -479,7 +465,7 @@ namespace MQTTnet.Core.Client
{ {
#pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed #pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
Task.Run( Task.Run(
async () => await SendKeepAliveMessagesAsync(cancellationToken),
async () => await SendKeepAliveMessagesAsync(cancellationToken).ConfigureAwait(false),
cancellationToken).ConfigureAwait(false); cancellationToken).ConfigureAwait(false);
#pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed #pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
} }

MQTTnet.Core/Client/MqttClientConnectResult.cs → Frameworks/MQTTnet.NetStandard/Client/MqttClientConnectResult.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public class MqttClientConnectResult public class MqttClientConnectResult
{ {

MQTTnet.Core/Client/MqttClientConnectedEventArgs.cs → Frameworks/MQTTnet.NetStandard/Client/MqttClientConnectedEventArgs.cs View File

@@ -1,6 +1,6 @@
using System; using System;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public class MqttClientConnectedEventArgs : EventArgs public class MqttClientConnectedEventArgs : EventArgs
{ {

MQTTnet.Core/Client/MqttClientCredentials.cs → Frameworks/MQTTnet.NetStandard/Client/MqttClientCredentials.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public class MqttClientCredentials : IMqttClientCredentials public class MqttClientCredentials : IMqttClientCredentials
{ {

MQTTnet.Core/Client/MqttClientDisconnectedEventArgs.cs → Frameworks/MQTTnet.NetStandard/Client/MqttClientDisconnectedEventArgs.cs View File

@@ -1,6 +1,6 @@
using System; using System;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public class MqttClientDisconnectedEventArgs : EventArgs public class MqttClientDisconnectedEventArgs : EventArgs
{ {

MQTTnet.Core/Client/MqttClientExtensions.cs → Frameworks/MQTTnet.NetStandard/Client/MqttClientExtensions.cs View File

@@ -2,8 +2,9 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.Linq; using System.Linq;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Protocol;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public static class MqttClientExtensions public static class MqttClientExtensions
{ {
@@ -15,6 +16,14 @@ namespace MQTTnet.Core.Client
return client.SubscribeAsync(topicFilters.ToList()); return client.SubscribeAsync(topicFilters.ToList());
} }


public static Task<IList<MqttSubscribeResult>> SubscribeAsync(this IMqttClient client, string topic, MqttQualityOfServiceLevel qualityOfServiceLevel)
{
if (client == null) throw new ArgumentNullException(nameof(client));
if (topic == null) throw new ArgumentNullException(nameof(topic));

return client.SubscribeAsync(new TopicFilterBuilder().WithTopic(topic).WithQualityOfServiceLevel(qualityOfServiceLevel).Build());
}

public static Task UnsubscribeAsync(this IMqttClient client, params string[] topicFilters) public static Task UnsubscribeAsync(this IMqttClient client, params string[] topicFilters)
{ {
if (client == null) throw new ArgumentNullException(nameof(client)); if (client == null) throw new ArgumentNullException(nameof(client));

MQTTnet.Core/Client/MqttClientOptions.cs → Frameworks/MQTTnet.NetStandard/Client/MqttClientOptions.cs View File

@@ -1,7 +1,7 @@
using System; using System;
using MQTTnet.Core.Serializer;
using MQTTnet.Serializer;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public class MqttClientOptions : IMqttClientOptions public class MqttClientOptions : IMqttClientOptions
{ {

MQTTnet.Core/Client/MqttClientOptionsBuilder.cs → Frameworks/MQTTnet.NetStandard/Client/MqttClientOptionsBuilder.cs View File

@@ -1,8 +1,8 @@
using System; using System;
using System.Linq; using System.Linq;
using MQTTnet.Core.Serializer;
using MQTTnet.Serializer;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public class MqttClientOptionsBuilder public class MqttClientOptionsBuilder
{ {
@@ -119,11 +119,11 @@ namespace MQTTnet.Core.Client


if (_tcpOptions != null) if (_tcpOptions != null)
{ {
_options.ChannelOptions = _tcpOptions;
_tcpOptions.TlsOptions = _tlsOptions;
} }
else
else if (_webSocketOptions != null)
{ {
_options.ChannelOptions = _webSocketOptions;
_webSocketOptions.TlsOptions = _tlsOptions;
} }
} }



MQTTnet.Core/Client/MqttClientTcpOptions.cs → Frameworks/MQTTnet.NetStandard/Client/MqttClientTcpOptions.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public class MqttClientTcpOptions : IMqttClientChannelOptions public class MqttClientTcpOptions : IMqttClientChannelOptions
{ {

MQTTnet.Core/Client/MqttClientTcpOptionsExtensions.cs → Frameworks/MQTTnet.NetStandard/Client/MqttClientTcpOptionsExtensions.cs View File

@@ -1,6 +1,6 @@
using System; using System;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public static class MqttClientTcpOptionsExtensions public static class MqttClientTcpOptionsExtensions
{ {

MQTTnet.Core/Client/MqttClientTlsOptions.cs → Frameworks/MQTTnet.NetStandard/Client/MqttClientTlsOptions.cs View File

@@ -1,6 +1,6 @@
using System.Collections.Generic; using System.Collections.Generic;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public class MqttClientTlsOptions public class MqttClientTlsOptions
{ {

MQTTnet.Core/Client/MqttClientWebSocketOptions.cs → Frameworks/MQTTnet.NetStandard/Client/MqttClientWebSocketOptions.cs View File

@@ -1,7 +1,7 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.Net; using System.Net;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public class MqttClientWebSocketOptions : IMqttClientChannelOptions public class MqttClientWebSocketOptions : IMqttClientChannelOptions
{ {

MQTTnet.Core/Client/MqttPacketDispatcher.cs → Frameworks/MQTTnet.NetStandard/Client/MqttPacketDispatcher.cs View File

@@ -1,12 +1,12 @@
using System; using System;
using System.Collections.Concurrent; using System.Collections.Concurrent;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Diagnostics;
using MQTTnet.Core.Exceptions;
using MQTTnet.Core.Internal;
using MQTTnet.Core.Packets;
using MQTTnet.Diagnostics;
using MQTTnet.Exceptions;
using MQTTnet.Internal;
using MQTTnet.Packets;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public class MqttPacketDispatcher public class MqttPacketDispatcher
{ {

MQTTnet.Core/Client/MqttSubscribeResult.cs → Frameworks/MQTTnet.NetStandard/Client/MqttSubscribeResult.cs View File

@@ -1,7 +1,6 @@
using MQTTnet.Core.Packets;
using MQTTnet.Core.Protocol;
using MQTTnet.Protocol;


namespace MQTTnet.Core.Client
namespace MQTTnet.Client
{ {
public class MqttSubscribeResult public class MqttSubscribeResult
{ {

MQTTnet.Core/Diagnostics/IMqttNetLogger.cs → Frameworks/MQTTnet.NetStandard/Diagnostics/IMqttNetLogger.cs View File

@@ -1,6 +1,6 @@
using System; using System;


namespace MQTTnet.Core.Diagnostics
namespace MQTTnet.Diagnostics
{ {
public interface IMqttNetLogger public interface IMqttNetLogger
{ {

+ 0
- 50
Frameworks/MQTTnet.NetStandard/Diagnostics/MqttNetConsoleLogger.cs View File

@@ -1,50 +0,0 @@
using System;
using System.Text;
using MQTTnet.Core.Diagnostics;

namespace MQTTnet.Diagnostics
{
public static class MqttNetConsoleLogger
{
private static readonly object Lock = new object();

public static void ForwardToConsole()
{
MqttNetGlobalLog.LogMessagePublished -= PrintToConsole;
MqttNetGlobalLog.LogMessagePublished += PrintToConsole;
}

private static void PrintToConsole(object sender, MqttNetLogMessagePublishedEventArgs e)
{
var output = new StringBuilder();
output.AppendLine($">> [{e.TraceMessage.Timestamp:O}] [{e.TraceMessage.ThreadId}] [{e.TraceMessage.Source}] [{e.TraceMessage.Level}]: {e.TraceMessage.Message}");
if (e.TraceMessage.Exception != null)
{
output.AppendLine(e.TraceMessage.Exception.ToString());
}

lock (Lock)
{
var backupColor = Console.ForegroundColor;
switch (e.TraceMessage.Level)
{
case MqttNetLogLevel.Error:
Console.ForegroundColor = ConsoleColor.Red;
break;
case MqttNetLogLevel.Warning:
Console.ForegroundColor = ConsoleColor.Yellow;
break;
case MqttNetLogLevel.Info:
Console.ForegroundColor = ConsoleColor.Green;
break;
case MqttNetLogLevel.Verbose:
Console.ForegroundColor = ConsoleColor.Gray;
break;
}

Console.Write(output);
Console.ForegroundColor = backupColor;
}
}
}
}

MQTTnet.Core/Diagnostics/MqttNetGlobalLog.cs → Frameworks/MQTTnet.NetStandard/Diagnostics/MqttNetGlobalLogger.cs View File

@@ -1,8 +1,8 @@
using System; using System;


namespace MQTTnet.Core.Diagnostics
namespace MQTTnet.Diagnostics
{ {
public static class MqttNetGlobalLog
public static class MqttNetGlobalLogger
{ {
public static event EventHandler<MqttNetLogMessagePublishedEventArgs> LogMessagePublished; public static event EventHandler<MqttNetLogMessagePublishedEventArgs> LogMessagePublished;



MQTTnet.Core/Diagnostics/MqttNetLogLevel.cs → Frameworks/MQTTnet.NetStandard/Diagnostics/MqttNetLogLevel.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Diagnostics
namespace MQTTnet.Diagnostics
{ {
public enum MqttNetLogLevel public enum MqttNetLogLevel
{ {

MQTTnet.Core/Diagnostics/MqttNetLogMessage.cs → Frameworks/MQTTnet.NetStandard/Diagnostics/MqttNetLogMessage.cs View File

@@ -1,6 +1,6 @@
using System; using System;


namespace MQTTnet.Core.Diagnostics
namespace MQTTnet.Diagnostics
{ {
public sealed class MqttNetLogMessage public sealed class MqttNetLogMessage
{ {

MQTTnet.Core/Diagnostics/MqttNetLogMessagePublishedEventArgs.cs → Frameworks/MQTTnet.NetStandard/Diagnostics/MqttNetLogMessagePublishedEventArgs.cs View File

@@ -1,6 +1,6 @@
using System; using System;


namespace MQTTnet.Core.Diagnostics
namespace MQTTnet.Diagnostics
{ {
public sealed class MqttNetLogMessagePublishedEventArgs : EventArgs public sealed class MqttNetLogMessagePublishedEventArgs : EventArgs
{ {

MQTTnet.Core/Diagnostics/MqttNetLogger.cs → Frameworks/MQTTnet.NetStandard/Diagnostics/MqttNetLogger.cs View File

@@ -1,6 +1,6 @@
using System; using System;


namespace MQTTnet.Core.Diagnostics
namespace MQTTnet.Diagnostics
{ {
public class MqttNetLogger : IMqttNetLogger public class MqttNetLogger : IMqttNetLogger
{ {
@@ -10,7 +10,7 @@ namespace MQTTnet.Core.Diagnostics
{ {
_logId = logId; _logId = logId;
} }
public event EventHandler<MqttNetLogMessagePublishedEventArgs> LogMessagePublished; public event EventHandler<MqttNetLogMessagePublishedEventArgs> LogMessagePublished;


public void Trace<TSource>(string message, params object[] parameters) public void Trace<TSource>(string message, params object[] parameters)
@@ -46,7 +46,7 @@ namespace MQTTnet.Core.Diagnostics
private void Publish<TSource>(MqttNetLogLevel logLevel, Exception exception, string message, object[] parameters) private void Publish<TSource>(MqttNetLogLevel logLevel, Exception exception, string message, object[] parameters)
{ {
var hasLocalListeners = LogMessagePublished != null; var hasLocalListeners = LogMessagePublished != null;
var hasGlobalListeners = MqttNetGlobalLog.HasListeners;
var hasGlobalListeners = MqttNetGlobalLogger.HasListeners;


if (!hasLocalListeners && !hasGlobalListeners) if (!hasLocalListeners && !hasGlobalListeners)
{ {
@@ -62,7 +62,7 @@ namespace MQTTnet.Core.Diagnostics


if (hasGlobalListeners) if (hasGlobalListeners)
{ {
MqttNetGlobalLog.Publish(traceMessage);
MqttNetGlobalLogger.Publish(traceMessage);
} }


if (hasLocalListeners) if (hasLocalListeners)

+ 1
- 1
Frameworks/MQTTnet.NetStandard/Diagnostics/TargetFrameworkInfoProvider.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet
namespace MQTTnet.Diagnostics
{ {
public static class TargetFrameworkInfoProvider public static class TargetFrameworkInfoProvider
{ {


MQTTnet.Core/Exceptions/MqttCommunicationException.cs → Frameworks/MQTTnet.NetStandard/Exceptions/MqttCommunicationException.cs View File

@@ -1,6 +1,6 @@
using System; using System;


namespace MQTTnet.Core.Exceptions
namespace MQTTnet.Exceptions
{ {
public class MqttCommunicationException : Exception public class MqttCommunicationException : Exception
{ {

MQTTnet.Core/Exceptions/MqttCommunicationTimedOutException.cs → Frameworks/MQTTnet.NetStandard/Exceptions/MqttCommunicationTimedOutException.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Exceptions
namespace MQTTnet.Exceptions
{ {
public sealed class MqttCommunicationTimedOutException : MqttCommunicationException public sealed class MqttCommunicationTimedOutException : MqttCommunicationException
{ {

MQTTnet.Core/Exceptions/MqttProtocolViolationException.cs → Frameworks/MQTTnet.NetStandard/Exceptions/MqttProtocolViolationException.cs View File

@@ -1,6 +1,6 @@
using System; using System;


namespace MQTTnet.Core.Exceptions
namespace MQTTnet.Exceptions
{ {
public sealed class MqttProtocolViolationException : Exception public sealed class MqttProtocolViolationException : Exception
{ {

MQTTnet.Core/IApplicationMessagePublisher.cs → Frameworks/MQTTnet.NetStandard/IApplicationMessagePublisher.cs View File

@@ -1,7 +1,7 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.Threading.Tasks; using System.Threading.Tasks;


namespace MQTTnet.Core
namespace MQTTnet
{ {
public interface IApplicationMessagePublisher public interface IApplicationMessagePublisher
{ {

MQTTnet.Core/IApplicationMessageReceiver.cs → Frameworks/MQTTnet.NetStandard/IApplicationMessageReceiver.cs View File

@@ -1,7 +1,6 @@
using System; using System;
using MQTTnet.Core.Client;


namespace MQTTnet.Core
namespace MQTTnet
{ {
public interface IApplicationMessageReceiver public interface IApplicationMessageReceiver
{ {

+ 4
- 4
Frameworks/MQTTnet.NetStandard/Implementations/MqttClientAdapterFactory.cs View File

@@ -1,8 +1,8 @@
using System; using System;
using MQTTnet.Core.Adapter;
using MQTTnet.Core.Client;
using MQTTnet.Core.Diagnostics;
using MQTTnet.Core.Serializer;
using MQTTnet.Adapter;
using MQTTnet.Client;
using MQTTnet.Diagnostics;
using MQTTnet.Serializer;


namespace MQTTnet.Implementations namespace MQTTnet.Implementations
{ {


+ 5
- 5
Frameworks/MQTTnet.NetStandard/Implementations/MqttServerAdapter.Uwp.cs View File

@@ -1,11 +1,11 @@
#if WINDOWS_UWP #if WINDOWS_UWP
using System; using System;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Adapter;
using MQTTnet.Core.Server;
using Windows.Networking.Sockets; using Windows.Networking.Sockets;
using MQTTnet.Core.Diagnostics;
using MQTTnet.Core.Serializer;
using MQTTnet.Adapter;
using MQTTnet.Diagnostics;
using MQTTnet.Serializer;
using MQTTnet.Server;


namespace MQTTnet.Implementations namespace MQTTnet.Implementations
{ {
@@ -21,7 +21,7 @@ namespace MQTTnet.Implementations


public event EventHandler<MqttServerAdapterClientAcceptedEventArgs> ClientAccepted; public event EventHandler<MqttServerAdapterClientAcceptedEventArgs> ClientAccepted;


public async Task StartAsync(MqttServerOptions options)
public async Task StartAsync(IMqttServerOptions options)
{ {
if (options == null) throw new ArgumentNullException(nameof(options)); if (options == null) throw new ArgumentNullException(nameof(options));




+ 43
- 12
Frameworks/MQTTnet.NetStandard/Implementations/MqttServerAdapter.cs View File

@@ -7,10 +7,10 @@ using System.Security.Authentication;
using System.Security.Cryptography.X509Certificates; using System.Security.Cryptography.X509Certificates;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Adapter;
using MQTTnet.Core.Server;
using MQTTnet.Core.Diagnostics;
using MQTTnet.Core.Serializer;
using MQTTnet.Adapter;
using MQTTnet.Diagnostics;
using MQTTnet.Serializer;
using MQTTnet.Server;


namespace MQTTnet.Implementations namespace MQTTnet.Implementations
{ {
@@ -30,7 +30,7 @@ namespace MQTTnet.Implementations


public event EventHandler<MqttServerAdapterClientAcceptedEventArgs> ClientAccepted; public event EventHandler<MqttServerAdapterClientAcceptedEventArgs> ClientAccepted;


public Task StartAsync(MqttServerOptions options)
public Task StartAsync(IMqttServerOptions options)
{ {
if (_cancellationTokenSource != null) throw new InvalidOperationException("Server is already started."); if (_cancellationTokenSource != null) throw new InvalidOperationException("Server is already started.");


@@ -39,10 +39,10 @@ namespace MQTTnet.Implementations
if (options.DefaultEndpointOptions.IsEnabled) if (options.DefaultEndpointOptions.IsEnabled)
{ {
_defaultEndpointSocket = new Socket(SocketType.Stream, ProtocolType.Tcp); _defaultEndpointSocket = new Socket(SocketType.Stream, ProtocolType.Tcp);
_defaultEndpointSocket.Bind(new IPEndPoint(IPAddress.Any, options.GetDefaultEndpointPort()));
_defaultEndpointSocket.Bind(new IPEndPoint(options.DefaultEndpointOptions.BoundIPAddress, options.GetDefaultEndpointPort()));
_defaultEndpointSocket.Listen(options.ConnectionBacklog); _defaultEndpointSocket.Listen(options.ConnectionBacklog);


Task.Run(() => AcceptDefaultEndpointConnectionsAsync(_cancellationTokenSource.Token), _cancellationTokenSource.Token);
Task.Run(async () => await AcceptDefaultEndpointConnectionsAsync(_cancellationTokenSource.Token).ConfigureAwait(false), _cancellationTokenSource.Token).ConfigureAwait(false);
} }


if (options.TlsEndpointOptions.IsEnabled) if (options.TlsEndpointOptions.IsEnabled)
@@ -59,10 +59,10 @@ namespace MQTTnet.Implementations
} }


_tlsEndpointSocket = new Socket(SocketType.Stream, ProtocolType.Tcp); _tlsEndpointSocket = new Socket(SocketType.Stream, ProtocolType.Tcp);
_tlsEndpointSocket.Bind(new IPEndPoint(IPAddress.Any, options.GetTlsEndpointPort()));
_tlsEndpointSocket.Bind(new IPEndPoint(options.TlsEndpointOptions.BoundIPAddress, options.GetTlsEndpointPort()));
_tlsEndpointSocket.Listen(options.ConnectionBacklog); _tlsEndpointSocket.Listen(options.ConnectionBacklog);


Task.Run(() => AcceptTlsEndpointConnectionsAsync(_cancellationTokenSource.Token), _cancellationTokenSource.Token);
Task.Run(async () => await AcceptTlsEndpointConnectionsAsync(_cancellationTokenSource.Token).ConfigureAwait(false), _cancellationTokenSource.Token).ConfigureAwait(false);
} }


return Task.FromResult(0); return Task.FromResult(0);
@@ -102,13 +102,29 @@ namespace MQTTnet.Implementations
#else #else
var clientSocket = await _defaultEndpointSocket.AcceptAsync().ConfigureAwait(false); var clientSocket = await _defaultEndpointSocket.AcceptAsync().ConfigureAwait(false);
#endif #endif

var clientAdapter = new MqttChannelAdapter(new MqttTcpChannel(clientSocket, null), new MqttPacketSerializer(), _logger); var clientAdapter = new MqttChannelAdapter(new MqttTcpChannel(clientSocket, null), new MqttPacketSerializer(), _logger);
ClientAccepted?.Invoke(this, new MqttServerAdapterClientAcceptedEventArgs(clientAdapter)); ClientAccepted?.Invoke(this, new MqttServerAdapterClientAcceptedEventArgs(clientAdapter));
} }
catch (ObjectDisposedException)
{
// It can happen that the listener socket is accessed after the cancellation token is already set and the listener socket is disposed.
}
catch (SocketException exception)
{
if (exception.SocketErrorCode == SocketError.OperationAborted)
{
return;
}

_logger.Error<MqttServerAdapter>(exception, "Error while accepting connection at default endpoint.");
}
catch (Exception exception) catch (Exception exception)
{ {
_logger.Error<MqttServerAdapter>(exception, "Error while accepting connection at default endpoint."); _logger.Error<MqttServerAdapter>(exception, "Error while accepting connection at default endpoint.");

}
finally
{
//excessive CPU consumed if in endless loop of socket errors //excessive CPU consumed if in endless loop of socket errors
await Task.Delay(TimeSpan.FromSeconds(1), cancellationToken).ConfigureAwait(false); await Task.Delay(TimeSpan.FromSeconds(1), cancellationToken).ConfigureAwait(false);
} }
@@ -129,14 +145,29 @@ namespace MQTTnet.Implementations


var sslStream = new SslStream(new NetworkStream(clientSocket)); var sslStream = new SslStream(new NetworkStream(clientSocket));
await sslStream.AuthenticateAsServerAsync(_tlsCertificate, false, SslProtocols.Tls12, false).ConfigureAwait(false); await sslStream.AuthenticateAsServerAsync(_tlsCertificate, false, SslProtocols.Tls12, false).ConfigureAwait(false);
var clientAdapter = new MqttChannelAdapter(new MqttTcpChannel(clientSocket, sslStream), new MqttPacketSerializer(), _logger); var clientAdapter = new MqttChannelAdapter(new MqttTcpChannel(clientSocket, sslStream), new MqttPacketSerializer(), _logger);
ClientAccepted?.Invoke(this, new MqttServerAdapterClientAcceptedEventArgs(clientAdapter)); ClientAccepted?.Invoke(this, new MqttServerAdapterClientAcceptedEventArgs(clientAdapter));
} }
catch (ObjectDisposedException)
{
// It can happen that the listener socket is accessed after the cancellation token is already set and the listener socket is disposed.
}
catch (SocketException exception)
{
if (exception.SocketErrorCode == SocketError.OperationAborted)
{
return;
}

_logger.Error<MqttServerAdapter>(exception, "Error while accepting connection at default endpoint.");
}
catch (Exception exception) catch (Exception exception)
{ {
_logger.Error<MqttServerAdapter>(exception, "Error while accepting connection at TLS endpoint."); _logger.Error<MqttServerAdapter>(exception, "Error while accepting connection at TLS endpoint.");

}
finally
{
//excessive CPU consumed if in endless loop of socket errors //excessive CPU consumed if in endless loop of socket errors
await Task.Delay(TimeSpan.FromSeconds(1), cancellationToken).ConfigureAwait(false); await Task.Delay(TimeSpan.FromSeconds(1), cancellationToken).ConfigureAwait(false);
} }


+ 2
- 2
Frameworks/MQTTnet.NetStandard/Implementations/MqttTcpChannel.Uwp.cs View File

@@ -8,8 +8,8 @@ using System.Threading.Tasks;
using Windows.Networking; using Windows.Networking;
using Windows.Networking.Sockets; using Windows.Networking.Sockets;
using Windows.Security.Cryptography.Certificates; using Windows.Security.Cryptography.Certificates;
using MQTTnet.Core.Channel;
using MQTTnet.Core.Client;
using MQTTnet.Channel;
using MQTTnet.Client;


namespace MQTTnet.Implementations namespace MQTTnet.Implementations
{ {


+ 6
- 9
Frameworks/MQTTnet.NetStandard/Implementations/MqttTcpChannel.cs View File

@@ -5,10 +5,10 @@ using System.Net.Sockets;
using System.Security.Authentication; using System.Security.Authentication;
using System.Security.Cryptography.X509Certificates; using System.Security.Cryptography.X509Certificates;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Channel;
using MQTTnet.Core.Client;
using System.IO; using System.IO;
using System.Linq; using System.Linq;
using MQTTnet.Channel;
using MQTTnet.Client;


namespace MQTTnet.Implementations namespace MQTTnet.Implementations
{ {
@@ -70,7 +70,7 @@ namespace MQTTnet.Implementations
_sslStream = new SslStream(new NetworkStream(_socket, true), false, InternalUserCertificateValidationCallback); _sslStream = new SslStream(new NetworkStream(_socket, true), false, InternalUserCertificateValidationCallback);
await _sslStream.AuthenticateAsClientAsync(_options.Server, LoadCertificates(_options), SslProtocols.Tls12, _options.TlsOptions.IgnoreCertificateRevocationErrors).ConfigureAwait(false); await _sslStream.AuthenticateAsClientAsync(_options.Server, LoadCertificates(_options), SslProtocols.Tls12, _options.TlsOptions.IgnoreCertificateRevocationErrors).ConfigureAwait(false);
} }
CreateStreams(_socket, _sslStream); CreateStreams(_socket, _sslStream);
} }


@@ -130,7 +130,7 @@ namespace MQTTnet.Implementations


foreach (var certificate in options.TlsOptions.Certificates) foreach (var certificate in options.TlsOptions.Certificates)
{ {
certificates.Add(new X509Certificate(certificate));
certificates.Add(new X509Certificate2(certificate));
} }


return certificates; return certificates;
@@ -139,10 +139,7 @@ namespace MQTTnet.Implementations
private void CreateStreams(Socket socket, Stream sslStream) private void CreateStreams(Socket socket, Stream sslStream)
{ {
var stream = sslStream ?? new NetworkStream(socket); var stream = sslStream ?? new NetworkStream(socket);

//cannot use this as default buffering prevents from receiving the first connect message
//need two streams otherwise read and write have to be synchronized

//todo: if branch can be used with min dependency NetStandard1.6 //todo: if branch can be used with min dependency NetStandard1.6
#if NET452 || NET461 #if NET452 || NET461
SendStream = new BufferedStream(stream, BufferSize); SendStream = new BufferedStream(stream, BufferSize);
@@ -155,4 +152,4 @@ namespace MQTTnet.Implementations


} }
} }
#endif
#endif

+ 3
- 3
Frameworks/MQTTnet.NetStandard/Implementations/MqttWebSocketChannel.cs View File

@@ -1,11 +1,11 @@
using MQTTnet.Core.Channel;
using MQTTnet.Core.Client;
using System;
using System;
using System.IO; using System.IO;
using System.Net.WebSockets; using System.Net.WebSockets;
using System.Security.Cryptography.X509Certificates; using System.Security.Cryptography.X509Certificates;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Channel;
using MQTTnet.Client;


namespace MQTTnet.Implementations namespace MQTTnet.Implementations
{ {


+ 1
- 1
Frameworks/MQTTnet.NetStandard/Implementations/WebSocketStream.cs View File

@@ -5,7 +5,7 @@ using System.Linq;
using System.Net.WebSockets; using System.Net.WebSockets;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Exceptions;
using MQTTnet.Exceptions;


namespace MQTTnet.Implementations namespace MQTTnet.Implementations
{ {


MQTTnet.Core/Internal/MqttApplicationMessageExtensions.cs → Frameworks/MQTTnet.NetStandard/Internal/MqttApplicationMessageExtensions.cs View File

@@ -1,6 +1,6 @@
using MQTTnet.Core.Packets;
using MQTTnet.Packets;


namespace MQTTnet.Core.Internal
namespace MQTTnet.Internal
{ {
internal static class MqttApplicationMessageExtensions internal static class MqttApplicationMessageExtensions
{ {

MQTTnet.Core/Internal/TaskExtensions.cs → Frameworks/MQTTnet.NetStandard/Internal/TaskExtensions.cs View File

@@ -1,21 +1,23 @@
using System; using System;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Exceptions;
using MQTTnet.Exceptions;


namespace MQTTnet.Core.Internal
namespace MQTTnet.Internal
{ {
public static class TaskExtensions public static class TaskExtensions
{ {
public static async Task TimeoutAfter(this Task task, TimeSpan timeout) public static async Task TimeoutAfter(this Task task, TimeSpan timeout)
{ {
using (var cancellationTokenSource = new CancellationTokenSource())
if (task == null) throw new ArgumentNullException(nameof(task));

using (var timeoutCts = new CancellationTokenSource())
{ {
try try
{ {
var timeoutTask = Task.Delay(timeout, cancellationTokenSource.Token);
var timeoutTask = Task.Delay(timeout, timeoutCts.Token);
var finishedTask = await Task.WhenAny(timeoutTask, task).ConfigureAwait(false); var finishedTask = await Task.WhenAny(timeoutTask, task).ConfigureAwait(false);
if (finishedTask == timeoutTask) if (finishedTask == timeoutTask)
{ {
throw new MqttCommunicationTimedOutException(); throw new MqttCommunicationTimedOutException();
@@ -33,18 +35,20 @@ namespace MQTTnet.Core.Internal
} }
finally finally
{ {
cancellationTokenSource.Cancel();
timeoutCts.Cancel();
} }
} }
} }


public static async Task<TResult> TimeoutAfter<TResult>(this Task<TResult> task, TimeSpan timeout) public static async Task<TResult> TimeoutAfter<TResult>(this Task<TResult> task, TimeSpan timeout)
{ {
using (var cancellationTokenSource = new CancellationTokenSource())
if (task == null) throw new ArgumentNullException(nameof(task));

using (var timeoutCts = new CancellationTokenSource())
{ {
try try
{ {
var timeoutTask = Task.Delay(timeout, cancellationTokenSource.Token);
var timeoutTask = Task.Delay(timeout, timeoutCts.Token);
var finishedTask = await Task.WhenAny(timeoutTask, task).ConfigureAwait(false); var finishedTask = await Task.WhenAny(timeoutTask, task).ConfigureAwait(false);


if (finishedTask == timeoutTask) if (finishedTask == timeoutTask)
@@ -66,7 +70,7 @@ namespace MQTTnet.Core.Internal
} }
finally finally
{ {
cancellationTokenSource.Cancel();
timeoutCts.Cancel();
} }
} }
} }

+ 10
- 7
Frameworks/MQTTnet.NetStandard/MQTTnet.Netstandard.csproj View File

@@ -1,19 +1,23 @@
<Project Sdk="Microsoft.NET.Sdk"> <Project Sdk="Microsoft.NET.Sdk">


<PropertyGroup> <PropertyGroup>
<TargetFrameworks>netstandard1.3;netstandard2.0;net452;net461;uap10.0</TargetFrameworks>
<TargetFrameworks Condition=" '$(OS)' == 'Windows_NT' ">netstandard1.3;netstandard2.0;net452;net461;uap10.0</TargetFrameworks>
<TargetFrameworks Condition=" '$(OS)' != 'Windows_NT' ">netstandard1.3;netstandard2.0</TargetFrameworks>
<AssemblyName>MQTTnet</AssemblyName> <AssemblyName>MQTTnet</AssemblyName>
<RootNamespace>MQTTnet</RootNamespace> <RootNamespace>MQTTnet</RootNamespace>
<GeneratePackageOnBuild>False</GeneratePackageOnBuild> <GeneratePackageOnBuild>False</GeneratePackageOnBuild>
<DebugType>Full</DebugType> <DebugType>Full</DebugType>
<AssemblyVersion>2.5.3.0</AssemblyVersion>
<FileVersion>2.5.3.0</FileVersion>
<AssemblyVersion>0.0.0.0</AssemblyVersion>
<FileVersion>0.0.0.0</FileVersion>
<Version>0.0.0.0</Version> <Version>0.0.0.0</Version>
<Company /> <Company />
<Product /> <Product />
<Description /> <Description />
<Authors /> <Authors />
<PackageId /> <PackageId />
<SignAssembly>true</SignAssembly>
<AssemblyOriginatorKeyFile>codeSigningKey.pfx</AssemblyOriginatorKeyFile>
<DelaySign>false</DelaySign>
</PropertyGroup> </PropertyGroup>


<PropertyGroup Condition="'$(TargetFramework)' == 'uap10.0'"> <PropertyGroup Condition="'$(TargetFramework)' == 'uap10.0'">
@@ -26,16 +30,15 @@
<TargetFrameworkVersion>v5.0</TargetFrameworkVersion> <TargetFrameworkVersion>v5.0</TargetFrameworkVersion>
<DefineConstants>$(DefineConstants);WINDOWS_UWP</DefineConstants> <DefineConstants>$(DefineConstants);WINDOWS_UWP</DefineConstants>
<LanguageTargets>$(MSBuildExtensionsPath)\Microsoft\WindowsXaml\v$(VisualStudioVersion)\Microsoft.Windows.UI.Xaml.CSharp.targets</LanguageTargets> <LanguageTargets>$(MSBuildExtensionsPath)\Microsoft\WindowsXaml\v$(VisualStudioVersion)\Microsoft.Windows.UI.Xaml.CSharp.targets</LanguageTargets>
</PropertyGroup> </PropertyGroup>


<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x86'" /> <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x86'" />


<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'" /> <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'" />


<ItemGroup>
<ProjectReference Include="..\..\MQTTnet.Core\MQTTnet.Core.csproj" />
</ItemGroup>
<PropertyGroup Condition="'$(Configuration)|$(TargetFramework)|$(Platform)'=='Release|netstandard1.3|AnyCPU'">
<DefineConstants>RELEASE;NETSTANDARD1_3</DefineConstants>
</PropertyGroup>


<ItemGroup Condition="'$(TargetFramework)'=='netstandard2.0'"> <ItemGroup Condition="'$(TargetFramework)'=='netstandard2.0'">
<PackageReference Include="System.Net.Security" Version="4.3.2" /> <PackageReference Include="System.Net.Security" Version="4.3.2" />


MQTTnet.Core/ManagedClient/IManagedMqttClient.cs → Frameworks/MQTTnet.NetStandard/ManagedClient/IManagedMqttClient.cs View File

@@ -1,9 +1,9 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Client;
using MQTTnet.Client;


namespace MQTTnet.Core.ManagedClient
namespace MQTTnet.ManagedClient
{ {
public interface IManagedMqttClient : IApplicationMessageReceiver, IApplicationMessagePublisher public interface IManagedMqttClient : IApplicationMessageReceiver, IApplicationMessagePublisher
{ {
@@ -16,6 +16,6 @@ namespace MQTTnet.Core.ManagedClient
Task StopAsync(); Task StopAsync();


Task SubscribeAsync(IEnumerable<TopicFilter> topicFilters); Task SubscribeAsync(IEnumerable<TopicFilter> topicFilters);
Task UnsubscribeAsync(IEnumerable<TopicFilter> topicFilters);
Task UnsubscribeAsync(IEnumerable<string> topics);
} }
} }

MQTTnet.Core/ManagedClient/IManagedMqttClientOptions.cs → Frameworks/MQTTnet.NetStandard/ManagedClient/IManagedMqttClientOptions.cs View File

@@ -1,7 +1,7 @@
using System; using System;
using MQTTnet.Core.Client;
using MQTTnet.Client;


namespace MQTTnet.Core.ManagedClient
namespace MQTTnet.ManagedClient
{ {
public interface IManagedMqttClientOptions public interface IManagedMqttClientOptions
{ {

MQTTnet.Core/ManagedClient/IManagedMqttClientStorage.cs → Frameworks/MQTTnet.NetStandard/ManagedClient/IManagedMqttClientStorage.cs View File

@@ -1,7 +1,7 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.Threading.Tasks; using System.Threading.Tasks;


namespace MQTTnet.Core.ManagedClient
namespace MQTTnet.ManagedClient
{ {
public interface IManagedMqttClientStorage public interface IManagedMqttClientStorage
{ {

MQTTnet.Core/ManagedClient/ManagedMqttClient.cs → Frameworks/MQTTnet.NetStandard/ManagedClient/ManagedMqttClient.cs View File

@@ -4,18 +4,18 @@ using System.Collections.Generic;
using System.Linq; using System.Linq;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Core.Client;
using MQTTnet.Core.Diagnostics;
using MQTTnet.Core.Exceptions;
using MQTTnet.Core.Protocol;
using MQTTnet.Client;
using MQTTnet.Diagnostics;
using MQTTnet.Exceptions;
using MQTTnet.Protocol;


namespace MQTTnet.Core.ManagedClient
namespace MQTTnet.ManagedClient
{ {
public class ManagedMqttClient : IManagedMqttClient
public class ManagedMqttClient : IManagedMqttClient, IDisposable
{ {
private readonly ManagedMqttClientStorageManager _storageManager = new ManagedMqttClientStorageManager();
private readonly BlockingCollection<MqttApplicationMessage> _messageQueue = new BlockingCollection<MqttApplicationMessage>(); private readonly BlockingCollection<MqttApplicationMessage> _messageQueue = new BlockingCollection<MqttApplicationMessage>();
private readonly HashSet<TopicFilter> _subscriptions = new HashSet<TopicFilter>();
private readonly Dictionary<string, MqttQualityOfServiceLevel> _subscriptions = new Dictionary<string, MqttQualityOfServiceLevel>();
private readonly SemaphoreSlim _subscriptionsSemaphore = new SemaphoreSlim(1, 1);


private readonly IMqttClient _mqttClient; private readonly IMqttClient _mqttClient;
private readonly IMqttNetLogger _logger; private readonly IMqttNetLogger _logger;
@@ -23,7 +23,9 @@ namespace MQTTnet.Core.ManagedClient
private CancellationTokenSource _connectionCancellationToken; private CancellationTokenSource _connectionCancellationToken;
private CancellationTokenSource _publishingCancellationToken; private CancellationTokenSource _publishingCancellationToken;


private ManagedMqttClientStorageManager _storageManager;
private IManagedMqttClientOptions _options; private IManagedMqttClientOptions _options;

private bool _subscriptionsNotPushed; private bool _subscriptionsNotPushed;
public ManagedMqttClient(IMqttClient mqttClient, IMqttNetLogger logger) public ManagedMqttClient(IMqttClient mqttClient, IMqttNetLogger logger)
@@ -55,15 +57,11 @@ namespace MQTTnet.Core.ManagedClient
if (_connectionCancellationToken != null) throw new InvalidOperationException("The managed client is already started."); if (_connectionCancellationToken != null) throw new InvalidOperationException("The managed client is already started.");


_options = options; _options = options;
await _storageManager.SetStorageAsync(_options.Storage).ConfigureAwait(false);

if (_options.Storage != null) if (_options.Storage != null)
{ {
var loadedMessages = await _options.Storage.LoadQueuedMessagesAsync().ConfigureAwait(false);
foreach (var loadedMessage in loadedMessages)
{
_messageQueue.Add(loadedMessage);
}
_storageManager = new ManagedMqttClientStorageManager(_options.Storage);
await _storageManager.LoadQueuedMessagesAsync().ConfigureAwait(false);
} }


_connectionCancellationToken = new CancellationTokenSource(); _connectionCancellationToken = new CancellationTokenSource();
@@ -77,11 +75,8 @@ namespace MQTTnet.Core.ManagedClient


public Task StopAsync() public Task StopAsync()
{ {
_connectionCancellationToken?.Cancel(false);
_connectionCancellationToken = null;

_publishingCancellationToken?.Cancel(false);
_publishingCancellationToken = null;
StopPublishing();
StopMaintainingConnection();


while (_messageQueue.Any()) while (_messageQueue.Any())
{ {
@@ -90,50 +85,66 @@ namespace MQTTnet.Core.ManagedClient


return Task.FromResult(0); return Task.FromResult(0);
} }
public async Task PublishAsync(IEnumerable<MqttApplicationMessage> applicationMessages) public async Task PublishAsync(IEnumerable<MqttApplicationMessage> applicationMessages)
{ {
if (applicationMessages == null) throw new ArgumentNullException(nameof(applicationMessages)); if (applicationMessages == null) throw new ArgumentNullException(nameof(applicationMessages));


foreach (var applicationMessage in applicationMessages) foreach (var applicationMessage in applicationMessages)
{ {
await _storageManager.AddAsync(applicationMessage).ConfigureAwait(false);
if (_storageManager != null)
{
await _storageManager.AddAsync(applicationMessage).ConfigureAwait(false);
}
_messageQueue.Add(applicationMessage); _messageQueue.Add(applicationMessage);
} }
} }


public Task SubscribeAsync(IEnumerable<TopicFilter> topicFilters)
public async Task SubscribeAsync(IEnumerable<TopicFilter> topicFilters)
{ {
if (topicFilters == null) throw new ArgumentNullException(nameof(topicFilters)); if (topicFilters == null) throw new ArgumentNullException(nameof(topicFilters));


lock (_subscriptions)
await _subscriptionsSemaphore.WaitAsync().ConfigureAwait(false);
try
{ {
foreach (var topicFilter in topicFilters) foreach (var topicFilter in topicFilters)
{ {
if (_subscriptions.Add(topicFilter))
{
_subscriptionsNotPushed = true;
}
_subscriptions[topicFilter.Topic] = topicFilter.QualityOfServiceLevel;
_subscriptionsNotPushed = true;
} }
} }

return Task.FromResult(0);
finally
{
_subscriptionsSemaphore.Release();
}
} }


public Task UnsubscribeAsync(IEnumerable<TopicFilter> topicFilters)
public async Task UnsubscribeAsync(IEnumerable<string> topics)
{ {
lock (_subscriptions)
await _subscriptionsSemaphore.WaitAsync().ConfigureAwait(false);
try
{ {
foreach (var topicFilter in topicFilters)
foreach (var topic in topics)
{ {
if (_subscriptions.Remove(topicFilter))
if (_subscriptions.Remove(topic))
{ {
_subscriptionsNotPushed = true; _subscriptionsNotPushed = true;
} }
} }
} }
finally
{
_subscriptionsSemaphore.Release();
}
}


return Task.FromResult(0);
public void Dispose()
{
_messageQueue?.Dispose();
_subscriptionsSemaphore?.Dispose();
_connectionCancellationToken?.Dispose();
_publishingCancellationToken?.Dispose();
} }


private async Task MaintainConnectionAsync(CancellationToken cancellationToken) private async Task MaintainConnectionAsync(CancellationToken cancellationToken)
@@ -142,33 +153,48 @@ namespace MQTTnet.Core.ManagedClient
{ {
while (!cancellationToken.IsCancellationRequested) while (!cancellationToken.IsCancellationRequested)
{ {
var connectionState = await ReconnectIfRequiredAsync().ConfigureAwait(false);
if (connectionState == ReconnectionResult.NotConnected)
{
_publishingCancellationToken?.Cancel(false);
_publishingCancellationToken = null;

await Task.Delay(_options.AutoReconnectDelay, cancellationToken).ConfigureAwait(false);
continue;
}
await TryMaintainConnectionAsync(cancellationToken);
}
}
catch (OperationCanceledException)
{
}
catch (Exception exception)
{
_logger.Error<ManagedMqttClient>(exception, "Unhandled exception while maintaining connection.");
}
finally
{
await _mqttClient.DisconnectAsync().ConfigureAwait(false);
_logger.Info<ManagedMqttClient>("Stopped");
}
}


if (connectionState == ReconnectionResult.Reconnected || _subscriptionsNotPushed)
{
await PushSubscriptionsAsync();
private async Task TryMaintainConnectionAsync(CancellationToken cancellationToken)
{
try
{
var connectionState = await ReconnectIfRequiredAsync().ConfigureAwait(false);
if (connectionState == ReconnectionResult.NotConnected)
{
StopPublishing();
await Task.Delay(_options.AutoReconnectDelay, cancellationToken).ConfigureAwait(false);
return;
}


_publishingCancellationToken = new CancellationTokenSource();
if (connectionState == ReconnectionResult.Reconnected || _subscriptionsNotPushed)
{
await PushSubscriptionsAsync().ConfigureAwait(false);


#pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
Task.Run(async () => await PublishQueuedMessagesAsync(_publishingCancellationToken.Token), _publishingCancellationToken.Token).ConfigureAwait(false);
#pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
StartPublishing();


continue;
}
return;
}


if (connectionState == ReconnectionResult.StillConnected)
{
await Task.Delay(TimeSpan.FromSeconds(1), _connectionCancellationToken.Token).ConfigureAwait(false);
}
if (connectionState == ReconnectionResult.StillConnected)
{
await Task.Delay(TimeSpan.FromSeconds(1), cancellationToken).ConfigureAwait(false);
} }
} }
catch (OperationCanceledException) catch (OperationCanceledException)
@@ -182,13 +208,8 @@ namespace MQTTnet.Core.ManagedClient
{ {
_logger.Error<ManagedMqttClient>(exception, "Unhandled exception while maintaining connection."); _logger.Error<ManagedMqttClient>(exception, "Unhandled exception while maintaining connection.");
} }
finally
{
await _mqttClient.DisconnectAsync().ConfigureAwait(false);
_logger.Info<ManagedMqttClient>("Stopped");
}
} }
private async Task PublishQueuedMessagesAsync(CancellationToken cancellationToken) private async Task PublishQueuedMessagesAsync(CancellationToken cancellationToken)
{ {
try try
@@ -206,16 +227,19 @@ namespace MQTTnet.Core.ManagedClient
continue; continue;
} }


await TryPublishQueuedMessageAsync(message).ConfigureAwait(false);
await _storageManager.RemoveAsync(message).ConfigureAwait(false);
await TryPublishQueuedMessageAsync(message).ConfigureAwait(false);
} }
} }
catch (OperationCanceledException) catch (OperationCanceledException)
{ {
} }
catch (Exception exception)
{
_logger.Error<ManagedMqttClient>(exception, "Unhandled exception while publishing queued application messages.");
}
finally finally
{ {
_logger.Info<ManagedMqttClient>("Stopped publishing messages");
_logger.Trace<ManagedMqttClient>("Stopped publishing messages.");
} }
} }


@@ -224,6 +248,11 @@ namespace MQTTnet.Core.ManagedClient
try try
{ {
await _mqttClient.PublishAsync(message).ConfigureAwait(false); await _mqttClient.PublishAsync(message).ConfigureAwait(false);

if (_storageManager != null)
{
await _storageManager.RemoveAsync(message).ConfigureAwait(false);
}
} }
catch (MqttCommunicationException exception) catch (MqttCommunicationException exception)
{ {
@@ -245,13 +274,18 @@ namespace MQTTnet.Core.ManagedClient
_logger.Info<ManagedMqttClient>(nameof(ManagedMqttClient), "Synchronizing subscriptions"); _logger.Info<ManagedMqttClient>(nameof(ManagedMqttClient), "Synchronizing subscriptions");


List<TopicFilter> subscriptions; List<TopicFilter> subscriptions;
lock (_subscriptions)
await _subscriptionsSemaphore.WaitAsync().ConfigureAwait(false);
try
{ {
subscriptions = _subscriptions.ToList();
subscriptions = _subscriptions.Select(i => new TopicFilter(i.Key, i.Value)).ToList();
_subscriptionsNotPushed = false; _subscriptionsNotPushed = false;
} }
finally
{
_subscriptionsSemaphore.Release();
}


if (!_subscriptions.Any())
if (!subscriptions.Any())
{ {
return; return;
} }
@@ -299,5 +333,35 @@ namespace MQTTnet.Core.ManagedClient
{ {
Connected?.Invoke(this, eventArgs); Connected?.Invoke(this, eventArgs);
} }

private void StartPublishing()
{
if (_publishingCancellationToken != null)
{
StopPublishing();
}

var cts = new CancellationTokenSource();

_publishingCancellationToken = cts;

#pragma warning disable CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
Task.Run(async () => await PublishQueuedMessagesAsync(cts.Token).ConfigureAwait(false), cts.Token).ConfigureAwait(false);
#pragma warning restore CS4014 // Because this call is not awaited, execution of the current method continues before the call is completed
}

private void StopPublishing()
{
_publishingCancellationToken?.Cancel(false);
_publishingCancellationToken?.Dispose();
_publishingCancellationToken = null;
}

private void StopMaintainingConnection()
{
_connectionCancellationToken?.Cancel(false);
_connectionCancellationToken?.Dispose();
_connectionCancellationToken = null;
}
} }
} }

MQTTnet.Core/ManagedClient/ManagedMqttClientExtensions.cs → Frameworks/MQTTnet.NetStandard/ManagedClient/ManagedMqttClientExtensions.cs View File

@@ -1,7 +1,8 @@
using System; using System;
using System.Threading.Tasks; using System.Threading.Tasks;
using MQTTnet.Protocol;


namespace MQTTnet.Core.ManagedClient
namespace MQTTnet.ManagedClient
{ {
public static class ManagedMqttClientExtensions public static class ManagedMqttClientExtensions
{ {
@@ -12,7 +13,15 @@ namespace MQTTnet.Core.ManagedClient
return managedClient.SubscribeAsync(topicFilters); return managedClient.SubscribeAsync(topicFilters);
} }


public static Task UnsubscribeAsync(this IManagedMqttClient managedClient, params TopicFilter[] topicFilters)
public static Task SubscribeAsync(this IManagedMqttClient managedClient, string topic, MqttQualityOfServiceLevel qualityOfServiceLevel)
{
if (managedClient == null) throw new ArgumentNullException(nameof(managedClient));
if (topic == null) throw new ArgumentNullException(nameof(topic));

return managedClient.SubscribeAsync(new TopicFilterBuilder().WithTopic(topic).WithQualityOfServiceLevel(qualityOfServiceLevel).Build());
}

public static Task UnsubscribeAsync(this IManagedMqttClient managedClient, params string[] topicFilters)
{ {
if (managedClient == null) throw new ArgumentNullException(nameof(managedClient)); if (managedClient == null) throw new ArgumentNullException(nameof(managedClient));



MQTTnet.Core/ManagedClient/ManagedMqttClientOptions.cs → Frameworks/MQTTnet.NetStandard/ManagedClient/ManagedMqttClientOptions.cs View File

@@ -1,7 +1,7 @@
using System; using System;
using MQTTnet.Core.Client;
using MQTTnet.Client;


namespace MQTTnet.Core.ManagedClient
namespace MQTTnet.ManagedClient
{ {
public class ManagedMqttClientOptions : IManagedMqttClientOptions public class ManagedMqttClientOptions : IManagedMqttClientOptions
{ {

MQTTnet.Core/ManagedClient/ManagedMqttClientOptionsBuilder.cs → Frameworks/MQTTnet.NetStandard/ManagedClient/ManagedMqttClientOptionsBuilder.cs View File

@@ -1,7 +1,7 @@
using System; using System;
using MQTTnet.Core.Client;
using MQTTnet.Client;


namespace MQTTnet.Core.ManagedClient
namespace MQTTnet.ManagedClient
{ {
public class ManagedMqttClientOptionsBuilder public class ManagedMqttClientOptionsBuilder
{ {

MQTTnet.Core/ManagedClient/ManagedMqttClientStorageManager.cs → Frameworks/MQTTnet.NetStandard/ManagedClient/ManagedMqttClientStorageManager.cs View File

@@ -1,25 +1,27 @@
using System.Collections.Generic;
using System;
using System.Collections.Generic;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;


namespace MQTTnet.Core.ManagedClient
namespace MQTTnet.ManagedClient
{ {
public class ManagedMqttClientStorageManager public class ManagedMqttClientStorageManager
{ {
private readonly List<MqttApplicationMessage> _applicationMessages = new List<MqttApplicationMessage>(); private readonly List<MqttApplicationMessage> _applicationMessages = new List<MqttApplicationMessage>();
private readonly SemaphoreSlim _semaphore = new SemaphoreSlim(1, 1); private readonly SemaphoreSlim _semaphore = new SemaphoreSlim(1, 1);
private IManagedMqttClientStorage _storage;
private readonly IManagedMqttClientStorage _storage;


public async Task SetStorageAsync(IManagedMqttClientStorage storage)
public ManagedMqttClientStorageManager(IManagedMqttClientStorage storage)
{ {
await _semaphore.WaitAsync().ConfigureAwait(false);
try
{
_storage = storage;
}
finally
_storage = storage ?? throw new ArgumentNullException(nameof(storage));
}

public async Task LoadQueuedMessagesAsync()
{
var loadedMessages = await _storage.LoadQueuedMessagesAsync().ConfigureAwait(false);
foreach (var loadedMessage in loadedMessages)
{ {
_semaphore.Release();
_applicationMessages.Add(loadedMessage);
} }
} }


@@ -28,11 +30,6 @@ namespace MQTTnet.Core.ManagedClient
await _semaphore.WaitAsync().ConfigureAwait(false); await _semaphore.WaitAsync().ConfigureAwait(false);
try try
{ {
if (_storage == null)
{
return;
}

_applicationMessages.Add(applicationMessage); _applicationMessages.Add(applicationMessage);
await SaveAsync().ConfigureAwait(false); await SaveAsync().ConfigureAwait(false);
} }
@@ -47,11 +44,6 @@ namespace MQTTnet.Core.ManagedClient
await _semaphore.WaitAsync().ConfigureAwait(false); await _semaphore.WaitAsync().ConfigureAwait(false);
try try
{ {
if (_storage == null)
{
return;
}

var index = _applicationMessages.IndexOf(applicationMessage); var index = _applicationMessages.IndexOf(applicationMessage);
if (index == -1) if (index == -1)
{ {

MQTTnet.Core/ManagedClient/ReconnectionResult.cs → Frameworks/MQTTnet.NetStandard/ManagedClient/ReconnectionResult.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.ManagedClient
namespace MQTTnet.ManagedClient
{ {
public enum ReconnectionResult public enum ReconnectionResult
{ {

MQTTnet.Core/MqttApplicationMessage.cs → Frameworks/MQTTnet.NetStandard/MqttApplicationMessage.cs View File

@@ -1,7 +1,7 @@
using System; using System;
using MQTTnet.Core.Protocol;
using MQTTnet.Protocol;


namespace MQTTnet.Core
namespace MQTTnet
{ {
public sealed class MqttApplicationMessage public sealed class MqttApplicationMessage
{ {

MQTTnet.Core/MqttApplicationMessageBuilder.cs → Frameworks/MQTTnet.NetStandard/MqttApplicationMessageBuilder.cs View File

@@ -2,10 +2,10 @@
using System.IO; using System.IO;
using System.Linq; using System.Linq;
using System.Text; using System.Text;
using MQTTnet.Core.Exceptions;
using MQTTnet.Core.Protocol;
using MQTTnet.Exceptions;
using MQTTnet.Protocol;


namespace MQTTnet.Core
namespace MQTTnet
{ {
public class MqttApplicationMessageBuilder public class MqttApplicationMessageBuilder
{ {

MQTTnet.Core/MqttApplicationMessageReceivedEventArgs.cs → Frameworks/MQTTnet.NetStandard/MqttApplicationMessageReceivedEventArgs.cs View File

@@ -1,6 +1,6 @@
using System; using System;


namespace MQTTnet.Core
namespace MQTTnet
{ {
public sealed class MqttApplicationMessageReceivedEventArgs : EventArgs public sealed class MqttApplicationMessageReceivedEventArgs : EventArgs
{ {

+ 5
- 5
Frameworks/MQTTnet.NetStandard/MqttFactory.cs View File

@@ -1,11 +1,11 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using MQTTnet.Core.Adapter;
using MQTTnet.Core.Client;
using MQTTnet.Adapter;
using MQTTnet.Client;
using MQTTnet.Diagnostics;
using MQTTnet.Implementations; using MQTTnet.Implementations;
using MQTTnet.Core.ManagedClient;
using MQTTnet.Core.Server;
using MQTTnet.Core.Diagnostics;
using MQTTnet.ManagedClient;
using MQTTnet.Server;


namespace MQTTnet namespace MQTTnet
{ {


MQTTnet.Core/Packets/IMqttPacketWithIdentifier.cs → Frameworks/MQTTnet.NetStandard/Packets/IMqttPacketWithIdentifier.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public interface IMqttPacketWithIdentifier public interface IMqttPacketWithIdentifier
{ {

MQTTnet.Core/Packets/MqttBasePacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttBasePacket.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public abstract class MqttBasePacket public abstract class MqttBasePacket
{ {

MQTTnet.Core/Packets/MqttBasePublishPacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttBasePublishPacket.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public class MqttBasePublishPacket : MqttBasePacket, IMqttPacketWithIdentifier public class MqttBasePublishPacket : MqttBasePacket, IMqttPacketWithIdentifier
{ {

MQTTnet.Core/Packets/MqttConnAckPacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttConnAckPacket.cs View File

@@ -1,6 +1,6 @@
using MQTTnet.Core.Protocol;
using MQTTnet.Protocol;


namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttConnAckPacket : MqttBasePacket public sealed class MqttConnAckPacket : MqttBasePacket
{ {

MQTTnet.Core/Packets/MqttConnectPacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttConnectPacket.cs View File

@@ -1,6 +1,6 @@
using MQTTnet.Core.Serializer;
using MQTTnet.Serializer;


namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttConnectPacket : MqttBasePacket public sealed class MqttConnectPacket : MqttBasePacket
{ {

MQTTnet.Core/Packets/MqttDisconnectPacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttDisconnectPacket.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttDisconnectPacket : MqttBasePacket public sealed class MqttDisconnectPacket : MqttBasePacket
{ {

MQTTnet.Core/Packets/MqttPacketExtensions.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttPacketExtensions.cs View File

@@ -1,6 +1,6 @@
using System; using System;


namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public static class MqttPacketExtensions public static class MqttPacketExtensions
{ {

MQTTnet.Core/Packets/MqttPacketHeader.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttPacketHeader.cs View File

@@ -1,6 +1,6 @@
using MQTTnet.Core.Protocol;
using MQTTnet.Protocol;


namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public class MqttPacketHeader public class MqttPacketHeader
{ {

MQTTnet.Core/Packets/MqttPingReqPacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttPingReqPacket.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttPingReqPacket : MqttBasePacket public sealed class MqttPingReqPacket : MqttBasePacket
{ {

MQTTnet.Core/Packets/MqttPingRespPacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttPingRespPacket.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttPingRespPacket : MqttBasePacket public sealed class MqttPingRespPacket : MqttBasePacket
{ {

MQTTnet.Core/Packets/MqttPubAckPacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttPubAckPacket.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttPubAckPacket : MqttBasePublishPacket public sealed class MqttPubAckPacket : MqttBasePublishPacket
{ {

MQTTnet.Core/Packets/MqttPubCompPacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttPubCompPacket.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttPubCompPacket : MqttBasePublishPacket public sealed class MqttPubCompPacket : MqttBasePublishPacket
{ {

MQTTnet.Core/Packets/MqttPubRecPacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttPubRecPacket.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttPubRecPacket : MqttBasePublishPacket public sealed class MqttPubRecPacket : MqttBasePublishPacket
{ {

MQTTnet.Core/Packets/MqttPubRelPacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttPubRelPacket.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttPubRelPacket : MqttBasePublishPacket public sealed class MqttPubRelPacket : MqttBasePublishPacket
{ {

MQTTnet.Core/Packets/MqttPublishPacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttPublishPacket.cs View File

@@ -1,6 +1,6 @@
using MQTTnet.Core.Protocol;
using MQTTnet.Protocol;


namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttPublishPacket : MqttBasePublishPacket public sealed class MqttPublishPacket : MqttBasePublishPacket
{ {

MQTTnet.Core/Packets/MqttSubAckPacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttSubAckPacket.cs View File

@@ -1,8 +1,8 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.Linq; using System.Linq;
using MQTTnet.Core.Protocol;
using MQTTnet.Protocol;


namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttSubAckPacket : MqttBasePacket, IMqttPacketWithIdentifier public sealed class MqttSubAckPacket : MqttBasePacket, IMqttPacketWithIdentifier
{ {

MQTTnet.Core/Packets/MqttSubscribePacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttSubscribePacket.cs View File

@@ -1,7 +1,7 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.Linq; using System.Linq;


namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttSubscribePacket : MqttBasePacket, IMqttPacketWithIdentifier public sealed class MqttSubscribePacket : MqttBasePacket, IMqttPacketWithIdentifier
{ {

MQTTnet.Core/Packets/MqttUnsubAckPacket.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttUnsubAckPacket.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttUnsubAckPacket : MqttBasePacket, IMqttPacketWithIdentifier public sealed class MqttUnsubAckPacket : MqttBasePacket, IMqttPacketWithIdentifier
{ {

MQTTnet.Core/Packets/MqttUnsubscribe.cs → Frameworks/MQTTnet.NetStandard/Packets/MqttUnsubscribe.cs View File

@@ -1,6 +1,6 @@
using System.Collections.Generic; using System.Collections.Generic;


namespace MQTTnet.Core.Packets
namespace MQTTnet.Packets
{ {
public sealed class MqttUnsubscribePacket : MqttBasePacket, IMqttPacketWithIdentifier public sealed class MqttUnsubscribePacket : MqttBasePacket, IMqttPacketWithIdentifier
{ {

MQTTnet.Core/Protocol/MqttConnectReturnCode.cs → Frameworks/MQTTnet.NetStandard/Protocol/MqttConnectReturnCode.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Protocol
namespace MQTTnet.Protocol
{ {
public enum MqttConnectReturnCode public enum MqttConnectReturnCode
{ {

MQTTnet.Core/Protocol/MqttControlPacketType.cs → Frameworks/MQTTnet.NetStandard/Protocol/MqttControlPacketType.cs View File

@@ -1,4 +1,4 @@
namespace MQTTnet.Core.Protocol
namespace MQTTnet.Protocol
{ {
public enum MqttControlPacketType public enum MqttControlPacketType
{ {

Some files were not shown because too many files changed in this diff

Loading…
Cancel
Save