Skip to content
This repository has been archived by the owner on Feb 5, 2020. It is now read-only.

[WIP] Server-side blazor support #12

Open
wants to merge 19 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 4 commits
Commits
Show all changes
19 commits
Select commit Hold shift + click to select a range
1fff4b1
Moved wwwroot to server project
CatoLeanTruetschel Mar 13, 2019
a694437
Prepared project for running server-side
CatoLeanTruetschel Mar 15, 2019
7d1876d
Added razor components prototype
CatoLeanTruetschel Mar 15, 2019
6db616f
Replaced obsolete IHostingEnvironment with IWebHostEnvironment
CatoLeanTruetschel Mar 15, 2019
363995c
Cleanup and code style
CatoLeanTruetschel Mar 16, 2019
e39849e
Fixed typo
CatoLeanTruetschel Mar 16, 2019
0836843
Merge remote-tracking branch 'upstream/master' into master
CatoLeanTruetschel May 18, 2019
5bd330a
Reenable pre-rendering
CatoLeanTruetschel May 18, 2019
7f1a495
Re-enable client-side logging
CatoLeanTruetschel May 18, 2019
4f94347
Add missing JSRuntime service to HubConnectionBuilder services
CatoLeanTruetschel May 18, 2019
31aa5dc
Implement client-side HttpMessageHandler that is usable server-side
CatoLeanTruetschel May 19, 2019
e4bbeea
Replace server-side http dispatch with new client-side HttpMessageHan…
CatoLeanTruetschel May 19, 2019
726d626
Replace synchronous JS->.Net invokation with async alternative
CatoLeanTruetschel May 19, 2019
be4dba0
Add new server-side test project
CatoLeanTruetschel May 19, 2019
20b72ca
Ensure backwards compatibility
CatoLeanTruetschel May 19, 2019
f7c9680
Revert temporary changes
CatoLeanTruetschel May 19, 2019
e2b4d9e
Await the JS->.Net invokations in event handlers
CatoLeanTruetschel May 23, 2019
1d33089
Revert "Ensure backwards compatibility"
CatoLeanTruetschel May 23, 2019
e1d6ccf
Add comment to explain pre-rendering short-circuit
CatoLeanTruetschel May 23, 2019
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
19 changes: 17 additions & 2 deletions BlazorSignalR.sln
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@

Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio 15
VisualStudioVersion = 15.0.26124.0
# Visual Studio Version 16
VisualStudioVersion = 16.0.28711.60
MinimumVisualStudioVersion = 15.0.26124.0
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "src", "src", "{B286BCBD-DAD8-4DE7-9334-3DE18DF233AF}"
EndProject
Expand All @@ -20,6 +20,8 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "BlazorSignalR.Test.Client",
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "BlazorSignalR.Test.Server", "test\BlazorSignalR.Test.Server\BlazorSignalR.Test.Server.csproj", "{8952910D-6251-48A0-8A00-369553370547}"
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "BlazorSignalR.Test.RazorComponents", "test\BlazorSignalR.Test.RazorComponents\BlazorSignalR.Test.RazorComponents.csproj", "{38EF5314-698C-41A1-BC4E-E6595ABA33A6}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Expand Down Expand Up @@ -78,6 +80,18 @@ Global
{8952910D-6251-48A0-8A00-369553370547}.Release|x64.Build.0 = Release|Any CPU
{8952910D-6251-48A0-8A00-369553370547}.Release|x86.ActiveCfg = Release|Any CPU
{8952910D-6251-48A0-8A00-369553370547}.Release|x86.Build.0 = Release|Any CPU
{38EF5314-698C-41A1-BC4E-E6595ABA33A6}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{38EF5314-698C-41A1-BC4E-E6595ABA33A6}.Debug|Any CPU.Build.0 = Debug|Any CPU
{38EF5314-698C-41A1-BC4E-E6595ABA33A6}.Debug|x64.ActiveCfg = Debug|Any CPU
{38EF5314-698C-41A1-BC4E-E6595ABA33A6}.Debug|x64.Build.0 = Debug|Any CPU
{38EF5314-698C-41A1-BC4E-E6595ABA33A6}.Debug|x86.ActiveCfg = Debug|Any CPU
{38EF5314-698C-41A1-BC4E-E6595ABA33A6}.Debug|x86.Build.0 = Debug|Any CPU
{38EF5314-698C-41A1-BC4E-E6595ABA33A6}.Release|Any CPU.ActiveCfg = Release|Any CPU
{38EF5314-698C-41A1-BC4E-E6595ABA33A6}.Release|Any CPU.Build.0 = Release|Any CPU
{38EF5314-698C-41A1-BC4E-E6595ABA33A6}.Release|x64.ActiveCfg = Release|Any CPU
{38EF5314-698C-41A1-BC4E-E6595ABA33A6}.Release|x64.Build.0 = Release|Any CPU
{38EF5314-698C-41A1-BC4E-E6595ABA33A6}.Release|x86.ActiveCfg = Release|Any CPU
{38EF5314-698C-41A1-BC4E-E6595ABA33A6}.Release|x86.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
Expand All @@ -87,6 +101,7 @@ Global
{1C49147F-7C73-4962-A71C-6A193970D058} = {B286BCBD-DAD8-4DE7-9334-3DE18DF233AF}
{752EC32C-EC70-4E7E-B2C3-610B0893249F} = {20DAA632-F8AD-4C5F-9E5F-FC82B7CB56A7}
{8952910D-6251-48A0-8A00-369553370547} = {20DAA632-F8AD-4C5F-9E5F-FC82B7CB56A7}
{38EF5314-698C-41A1-BC4E-E6595ABA33A6} = {20DAA632-F8AD-4C5F-9E5F-FC82B7CB56A7}
EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {A97C0A4B-E309-4485-BB76-898B37BFBFFF}
Expand Down
29 changes: 21 additions & 8 deletions src/BlazorSignalR/BlazorSignalRExtensions.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
using System;
using BlazorSignalR.Internal;
using Microsoft.AspNetCore.Components.Services;
using Microsoft.AspNetCore.Http.Connections;
using Microsoft.AspNetCore.SignalR.Client;
using Microsoft.Extensions.DependencyInjection;
Expand All @@ -9,14 +10,24 @@ namespace BlazorSignalR
{
public static class BlazorSignalRExtensions
{
public static IHubConnectionBuilder WithUrlBlazor(this IHubConnectionBuilder hubConnectionBuilder, string url, IJSRuntime jsRuntime,
HttpTransportType? transports = null, Action<BlazorHttpConnectionOptions> options = null)
public static IHubConnectionBuilder WithUrlBlazor(
this IHubConnectionBuilder hubConnectionBuilder,
string url,
IJSRuntime jsRuntime,
IUriHelper uriHelper,
HttpTransportType? transports = null,
Action<BlazorHttpConnectionOptions> options = null)
{
return WithUrlBlazor(hubConnectionBuilder, new Uri(url), jsRuntime, transports, options);
return WithUrlBlazor(hubConnectionBuilder, new Uri(url, UriKind.Relative), jsRuntime, uriHelper, transports, options);
}

public static IHubConnectionBuilder WithUrlBlazor(this IHubConnectionBuilder hubConnectionBuilder, Uri url, IJSRuntime jsRuntime,
HttpTransportType? transports = null, Action<BlazorHttpConnectionOptions> options = null)
public static IHubConnectionBuilder WithUrlBlazor(
this IHubConnectionBuilder hubConnectionBuilder,
Uri url,
IJSRuntime jsRuntime,
IUriHelper uriHelper,
HttpTransportType? transports = null,
Action<BlazorHttpConnectionOptions> options = null)
{
if (hubConnectionBuilder == null)
throw new ArgumentNullException(nameof(hubConnectionBuilder));
Expand All @@ -33,15 +44,17 @@ public static IHubConnectionBuilder WithUrlBlazor(this IHubConnectionBuilder hub
});
if (options != null)
hubConnectionBuilder.Services.Configure(options);
hubConnectionBuilder.Services.AddSingleton(provider => BuildBlazorHttpConnectionFactory(provider, jsRuntime));
hubConnectionBuilder.Services.AddSingleton(provider => BuildBlazorHttpConnectionFactory(provider, jsRuntime, uriHelper));
return hubConnectionBuilder;
}

private static IConnectionFactory BuildBlazorHttpConnectionFactory(IServiceProvider provider, IJSRuntime jsRuntime)
private static IConnectionFactory BuildBlazorHttpConnectionFactory(
IServiceProvider provider, IJSRuntime jsRuntime, IUriHelper uriHelper)
{
return ActivatorUtilities.CreateInstance<BlazorHttpConnectionFactory>(
provider,
jsRuntime);
jsRuntime,
uriHelper);
}
}
}
32 changes: 27 additions & 5 deletions src/BlazorSignalR/Internal/BlazorHttpConnection.cs
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
using System.Threading.Tasks;
using Microsoft.AspNetCore.Blazor.Http;
using Microsoft.AspNetCore.Blazor.Services;
using Microsoft.AspNetCore.Components.Services;
using Microsoft.AspNetCore.Connections;
using Microsoft.AspNetCore.Connections.Features;
using Microsoft.AspNetCore.Http.Connections;
Expand All @@ -23,6 +24,7 @@ internal class BlazorHttpConnection : ConnectionContext, IConnectionInherentKeep
{
private static readonly int _maxRedirects = 100;
private static readonly Task<string> NoAccessToken = Task.FromResult<string>((string)null);
private readonly bool _isServerSide;

public override string ConnectionId
{
Expand Down Expand Up @@ -51,6 +53,7 @@ public override IDuplexPipe Transport

private readonly BlazorHttpConnectionOptions _options;
private readonly IJSRuntime _jsRuntime;
private readonly IUriHelper _uriHelper;
private readonly ILoggerFactory _loggerFactory;
private readonly ILogger<BlazorHttpConnection> _logger;
private readonly HttpClient _httpClient;
Expand All @@ -64,20 +67,29 @@ public override IDuplexPipe Transport
private bool _hasInherentKeepAlive;
private Func<Task<string>> _accessTokenProvider;

public BlazorHttpConnection(BlazorHttpConnectionOptions options, IJSRuntime jsRuntime, ILoggerFactory loggerFactory)
public BlazorHttpConnection(
BlazorHttpConnectionOptions options,
IJSRuntime jsRuntime,
IUriHelper uriHelper,
bool isServerSide,
ILoggerFactory loggerFactory)
{
if (jsRuntime == null)
throw new ArgumentNullException(nameof(jsRuntime));

if (uriHelper == null)
throw new ArgumentNullException(nameof(uriHelper));

_options = options;
_jsRuntime = jsRuntime;
_uriHelper = uriHelper;
_isServerSide = isServerSide;
_loggerFactory = loggerFactory ?? NullLoggerFactory.Instance;
_logger = _loggerFactory.CreateLogger<BlazorHttpConnection>();
_httpClient = CreateHttpClient();
Features.Set<IConnectionInherentKeepAliveFeature>(this);
}


public async Task StartAsync(TransferFormat transferFormat)
{
CheckDisposed();
Expand Down Expand Up @@ -117,7 +129,7 @@ private async Task SelectAndStartTransport(TransferFormat transferFormat)
// Fix relative url paths
if (!uri.IsAbsoluteUri || uri.Scheme == Uri.UriSchemeFile && uri.OriginalString.StartsWith("/", StringComparison.Ordinal))
{
Uri baseUrl = new Uri(WebAssemblyUriHelper.Instance.GetBaseUri());
Uri baseUrl = new Uri(_uriHelper.GetBaseUri());
uri = new Uri(baseUrl, uri);
}

Expand Down Expand Up @@ -320,7 +332,17 @@ private static Uri CreateConnectUrl(Uri url, string connectionId)

private HttpClient CreateHttpClient()
{
HttpMessageHandler handler = new WebAssemblyHttpMessageHandler();
HttpMessageHandler handler;

if (_isServerSide)
{
handler = new HttpClientHandler();
csnewman marked this conversation as resolved.
Show resolved Hide resolved
}
else
{
handler = new WebAssemblyHttpMessageHandler();
}

if (_options.HttpMessageHandlerFactory != null)
{
handler = _options.HttpMessageHandlerFactory(handler);
Expand All @@ -333,7 +355,7 @@ private HttpClient CreateHttpClient()
HttpClient httpClient =
new HttpClient(new LoggingHttpMessageHandler(handler, _loggerFactory))
{
BaseAddress = new Uri(WebAssemblyUriHelper.Instance.GetBaseUri()),
BaseAddress = new Uri(_uriHelper.GetBaseUri()),
Timeout = HttpClientTimeout
};
// httpClient.DefaultRequestHeaders.UserAgent.Add(Constants.UserAgentHeader);
Expand Down
19 changes: 17 additions & 2 deletions src/BlazorSignalR/Internal/BlazorHttpConnectionFactory.cs
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
using System;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.AspNetCore.Components.Services;
using Microsoft.AspNetCore.Connections;
using Microsoft.AspNetCore.SignalR.Client;
using Microsoft.Extensions.Logging;
Expand All @@ -13,22 +15,35 @@ internal class BlazorHttpConnectionFactory : IConnectionFactory
{
private readonly BlazorHttpConnectionOptions _options;
private readonly IJSRuntime _jsRuntime;
private readonly IUriHelper _uriHelper;
private readonly bool _isServerSide;
private readonly ILoggerFactory _loggerFactory;

public BlazorHttpConnectionFactory(IOptions<BlazorHttpConnectionOptions> options, IJSRuntime jsRuntime, ILoggerFactory loggerFactory)
public BlazorHttpConnectionFactory(
IOptions<BlazorHttpConnectionOptions> options,
IJSRuntime jsRuntime,
IUriHelper uriHelper,
ILoggerFactory loggerFactory)
{
if (jsRuntime == null)
throw new ArgumentNullException(nameof(jsRuntime));

if (uriHelper == null)
throw new ArgumentNullException(nameof(uriHelper));

_options = options.Value;
_jsRuntime = jsRuntime;
_uriHelper = uriHelper;
_loggerFactory = loggerFactory;

// TODO: Is there a better/cleaner way?
_isServerSide = AppDomain.CurrentDomain.GetAssemblies().Any(p => p.GetName().Name == "Microsoft.AspNetCore.Mvc");
CatoLeanTruetschel marked this conversation as resolved.
Show resolved Hide resolved
}

public async Task<ConnectionContext> ConnectAsync(TransferFormat transferFormat,
CancellationToken cancellationToken = new CancellationToken())
{
BlazorHttpConnection connection = new BlazorHttpConnection(_options, _jsRuntime, _loggerFactory);
BlazorHttpConnection connection = new BlazorHttpConnection(_options, _jsRuntime, _uriHelper, _isServerSide, _loggerFactory);

try
{
Expand Down
17 changes: 12 additions & 5 deletions test/BlazorSignalR.Test.Client/Pages/ChatComponent.cs
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
using System.Threading.Tasks;
//using Blazor.Extensions.Logging;
using Microsoft.AspNetCore.Components;
using Microsoft.AspNetCore.Components.Services;
using Microsoft.AspNetCore.SignalR.Client;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
Expand All @@ -16,6 +17,7 @@ public class ChatComponent : ComponentBase
[Inject] private HttpClient _http { get; set; }
[Inject] private ILogger<ChatComponent> _logger { get; set; }
[Inject] private IJSRuntime _jsRuntime { get; set; }
[Inject] private IUriHelper _uriHelper { get; set; }
internal string _toEverybody { get; set; }
internal string _toConnection { get; set; }
internal string _connectionId { get; set; }
Expand All @@ -30,19 +32,24 @@ public class ChatComponent : ComponentBase

protected override async Task OnInitAsync()
{
// https://github.com/aspnet/AspNetCore/issues/8327
// https://github.com/aspnet/AspNetCore/issues/8404
// if(Prednering)
// { return; }

HubConnectionBuilder factory = new HubConnectionBuilder();

factory.Services.AddLogging(builder => builder
//.AddBrowserConsole() // Add Blazor.Extensions.Logging.BrowserConsoleLogger // This is not yet available for Blazor 0.8.0 https://github.com/BlazorExtensions/Logging/pull/22
.SetMinimumLevel(LogLevel.Trace)
);

factory.WithUrlBlazor("/chathub", _jsRuntime, options: opt =>
factory.WithUrlBlazor("/chathub", _jsRuntime, _uriHelper, options: opt =>
{
// opt.Transports = HttpTransportType.WebSockets;
// opt.SkipNegotiation = true;
opt.AccessTokenProvider = async () =>
{
// opt.Transports = HttpTransportType.WebSockets;
// opt.SkipNegotiation = true;
opt.AccessTokenProvider = async () =>
csnewman marked this conversation as resolved.
Show resolved Hide resolved
{
var token = await this.GetJwtToken("DemoUser");
this._logger.LogInformation($"Access Token: {token}");
return token;
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
<Project Sdk="Microsoft.NET.Sdk.Web">

<PropertyGroup>
<TargetFramework>netcoreapp3.0</TargetFramework>
<LangVersion>7.3</LangVersion>
<_RazorComponentInclude>Components\**\*.cshtml</_RazorComponentInclude>
</PropertyGroup>

<ItemGroup>
<PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="3.0.0-preview3-19153-02" />
<PackageReference Include="Microsoft.AspNetCore.Components.Server" Version="3.0.0-preview3-19153-02" />
<PackageReference Include="Microsoft.AspNetCore.Mvc.NewtonsoftJson" Version="3.0.0-preview3-19153-02" />
</ItemGroup>

<ItemGroup>
<ProjectReference Include="..\BlazorSignalR.Test.Client\BlazorSignalR.Test.Client.csproj" />
</ItemGroup>

</Project>
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
using Microsoft.AspNetCore.Mvc;
using Microsoft.IdentityModel.Tokens;
using System;
using System.IdentityModel.Tokens.Jwt;
using System.Security.Claims;

namespace BlazorSignalR.Test.RazorComponents.Controllers
{
[Route("generatetoken")]
public class TokenController : Controller
{
[HttpGet]
public string GenerateToken()
{
var claims = new[] { new Claim(ClaimTypes.NameIdentifier, this.Request.Query["user"]) };
csnewman marked this conversation as resolved.
Show resolved Hide resolved
var credentials = new SigningCredentials(Startup.SecurityKey, SecurityAlgorithms.HmacSha256); // Too lazy to inject the key as a service
var token = new JwtSecurityToken("SignalRTestServer", "SignalRTests", claims, expires: DateTime.UtcNow.AddSeconds(30), signingCredentials: credentials);
return Startup.JwtTokenHandler.WriteToken(token); // Even more lazy here
}
}
}
Loading