mirror of
https://github.com/streetwriters/notesnook-sync-server.git
synced 2026-02-12 11:12:44 +00:00
common: simplify wamp logic
previously we were opening a new channel for each topic which was unnecessary
This commit is contained in:
48
Streetwriters.Common/Accessors/WampServiceAccessor.cs
Normal file
48
Streetwriters.Common/Accessors/WampServiceAccessor.cs
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
/*
|
||||||
|
This file is part of the Notesnook Sync Server project (https://notesnook.com/)
|
||||||
|
|
||||||
|
Copyright (C) 2023 Streetwriters (Private) Limited
|
||||||
|
|
||||||
|
This program is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the Affero GNU General Public License as published by
|
||||||
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
Affero GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the Affero GNU General Public License
|
||||||
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Microsoft.Extensions.Hosting;
|
||||||
|
using Streetwriters.Common.Interfaces;
|
||||||
|
|
||||||
|
namespace Streetwriters.Common.Accessors
|
||||||
|
{
|
||||||
|
public class WampServiceAccessor(Server server) : IHostedService
|
||||||
|
{
|
||||||
|
public IUserAccountService UserAccountService { get; set; }
|
||||||
|
public IUserSubscriptionService? UserSubscriptionService { get; set; }
|
||||||
|
|
||||||
|
public async Task StartAsync(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
await InitAsync();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task InitAsync()
|
||||||
|
{
|
||||||
|
this.UserAccountService = await WampServers.IdentityServer.GetServiceAsync<IUserAccountService>(InitAsync);
|
||||||
|
if (!Constants.IS_SELF_HOSTED && server != Servers.SubscriptionServer)
|
||||||
|
{
|
||||||
|
this.UserSubscriptionService = await WampServers.SubscriptionServer.GetServiceAsync<IUserSubscriptionService>(InitAsync);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public Task StopAsync(CancellationToken cancellationToken) => Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -26,11 +26,11 @@ using Microsoft.AspNetCore.Hosting;
|
|||||||
using Microsoft.AspNetCore.Http;
|
using Microsoft.AspNetCore.Http;
|
||||||
using Microsoft.AspNetCore.HttpOverrides;
|
using Microsoft.AspNetCore.HttpOverrides;
|
||||||
using Microsoft.Extensions.DependencyInjection;
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
|
using Microsoft.Extensions.Hosting;
|
||||||
using WampSharp.AspNetCore.WebSockets.Server;
|
using WampSharp.AspNetCore.WebSockets.Server;
|
||||||
using WampSharp.Binding;
|
using WampSharp.Binding;
|
||||||
using WampSharp.V2;
|
using WampSharp.V2;
|
||||||
using WampSharp.V2.Realm;
|
using WampSharp.V2.Realm;
|
||||||
using Microsoft.Extensions.Hosting;
|
|
||||||
|
|
||||||
namespace Streetwriters.Common.Extensions
|
namespace Streetwriters.Common.Extensions
|
||||||
{
|
{
|
||||||
@@ -55,9 +55,9 @@ namespace Streetwriters.Common.Extensions
|
|||||||
return app;
|
return app;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static IApplicationBuilder UseWamp<T>(this IApplicationBuilder app, WampServer<T> server, Action<IWampHostedRealm, WampServer<T>> action) where T : new()
|
public static IApplicationBuilder UseWamp(this IApplicationBuilder app, WampServer server, Action<IWampHostedRealm, WampServer> action)
|
||||||
{
|
{
|
||||||
WampHost host = new WampHost();
|
WampHost host = new();
|
||||||
|
|
||||||
app.Map(server.Endpoint, builder =>
|
app.Map(server.Endpoint, builder =>
|
||||||
{
|
{
|
||||||
@@ -81,10 +81,8 @@ namespace Streetwriters.Common.Extensions
|
|||||||
|
|
||||||
public static T GetScopedService<T>(this IApplicationBuilder app) where T : notnull
|
public static T GetScopedService<T>(this IApplicationBuilder app) where T : notnull
|
||||||
{
|
{
|
||||||
using (var scope = app.ApplicationServices.CreateScope())
|
using var scope = app.ApplicationServices.CreateScope();
|
||||||
{
|
return scope.ServiceProvider.GetRequiredService<T>();
|
||||||
return scope.ServiceProvider.GetRequiredService<T>();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static IApplicationBuilder UseForwardedHeadersWithKnownProxies(this IApplicationBuilder app, IWebHostEnvironment env, string forwardedForHeaderName = null)
|
public static IApplicationBuilder UseForwardedHeadersWithKnownProxies(this IApplicationBuilder app, IWebHostEnvironment env, string forwardedForHeaderName = null)
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
using Microsoft.Extensions.DependencyInjection;
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
|
using Streetwriters.Common.Accessors;
|
||||||
using Streetwriters.Data.DbContexts;
|
using Streetwriters.Data.DbContexts;
|
||||||
using Streetwriters.Data.Repositories;
|
using Streetwriters.Data.Repositories;
|
||||||
|
|
||||||
@@ -25,6 +26,13 @@ namespace Streetwriters.Common.Extensions
|
|||||||
{
|
{
|
||||||
public static class ServiceCollectionServiceExtensions
|
public static class ServiceCollectionServiceExtensions
|
||||||
{
|
{
|
||||||
|
public static IServiceCollection AddWampServiceAccessor(this IServiceCollection services, Server server)
|
||||||
|
{
|
||||||
|
services.AddSingleton<WampServiceAccessor>((provider) => new WampServiceAccessor(server));
|
||||||
|
services.AddHostedService(provider => provider.GetRequiredService<WampServiceAccessor>());
|
||||||
|
return services;
|
||||||
|
}
|
||||||
|
|
||||||
public static IServiceCollection AddRepository<T>(this IServiceCollection services, string collectionName, string database) where T : class
|
public static IServiceCollection AddRepository<T>(this IServiceCollection services, string collectionName, string database) where T : class
|
||||||
{
|
{
|
||||||
services.AddSingleton((provider) => MongoDbContext.GetMongoCollection<T>(provider.GetRequiredService<MongoDB.Driver.IMongoClient>(), database, collectionName));
|
services.AddSingleton((provider) => MongoDbContext.GetMongoCollection<T>(provider.GetRequiredService<MongoDB.Driver.IMongoClient>(), database, collectionName));
|
||||||
|
|||||||
@@ -28,15 +28,28 @@ namespace Streetwriters.Common.Helpers
|
|||||||
{
|
{
|
||||||
public class WampHelper
|
public class WampHelper
|
||||||
{
|
{
|
||||||
public static async Task<IWampRealmProxy> OpenWampChannelAsync(string server, string realmName)
|
public static async Task<IWampChannel> OpenWampChannelAsync(string server, string realmName)
|
||||||
{
|
{
|
||||||
DefaultWampChannelFactory channelFactory = new();
|
DefaultWampChannelFactory channelFactory = new();
|
||||||
|
|
||||||
IWampChannel channel = channelFactory.CreateJsonChannel(server, realmName);
|
IWampChannel channel = channelFactory.CreateJsonChannel(server, realmName);
|
||||||
|
|
||||||
await channel.Open();
|
var isConnected = false;
|
||||||
|
while (!isConnected)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await channel.Open();
|
||||||
|
isConnected = true;
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
await Task.Delay(5000);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return channel.RealmProxy;
|
return channel;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void PublishMessage<T>(IWampRealmProxy realm, string topicName, T message)
|
public static void PublishMessage<T>(IWampRealmProxy realm, string topicName, T message)
|
||||||
|
|||||||
@@ -24,72 +24,74 @@ using System.Reactive.Subjects;
|
|||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Streetwriters.Common.Helpers;
|
using Streetwriters.Common.Helpers;
|
||||||
using Streetwriters.Common.Interfaces;
|
using Streetwriters.Common.Interfaces;
|
||||||
|
using WampSharp.V2;
|
||||||
using WampSharp.V2.Client;
|
using WampSharp.V2.Client;
|
||||||
|
|
||||||
namespace Streetwriters.Common
|
namespace Streetwriters.Common
|
||||||
{
|
{
|
||||||
public class WampServer<T> where T : new()
|
public class WampServer
|
||||||
{
|
{
|
||||||
private readonly ConcurrentDictionary<string, IWampRealmProxy> Channels = new();
|
|
||||||
|
|
||||||
public required string Endpoint { get; set; }
|
public required string Endpoint { get; set; }
|
||||||
public required string Address { get; set; }
|
public required string Address { get; set; }
|
||||||
public T Topics { get; set; } = new T();
|
|
||||||
public required string Realm { get; set; }
|
public required string Realm { get; set; }
|
||||||
|
|
||||||
private async Task<IWampRealmProxy> GetChannelAsync(string topic)
|
private IWampChannel? channel = null;
|
||||||
|
private async Task<IWampChannel> GetChannelAsync()
|
||||||
{
|
{
|
||||||
if (!Channels.TryGetValue(topic, out IWampRealmProxy? channel) || channel == null || !channel.Monitor.IsConnected)
|
if (channel != null && channel.RealmProxy.Monitor.IsConnected)
|
||||||
{
|
return channel;
|
||||||
channel = await WampHelper.OpenWampChannelAsync(Address, Realm);
|
channel = await WampHelper.OpenWampChannelAsync(Address, Realm);
|
||||||
Channels.AddOrUpdate(topic, (key) => channel, (key, old) => channel);
|
|
||||||
}
|
|
||||||
return channel;
|
return channel;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<V> GetServiceAsync<V>(string topic) where V : class
|
public async Task<V> GetServiceAsync<V>(Func<Task>? onDisconnect = null) where V : class
|
||||||
{
|
{
|
||||||
var channel = await GetChannelAsync(topic);
|
var channel = await GetChannelAsync();
|
||||||
return channel.Services.GetCalleeProxy<V>();
|
if (onDisconnect != null)
|
||||||
|
{
|
||||||
|
channel.RealmProxy.Monitor.ConnectionBroken += (s, e) => onDisconnect();
|
||||||
|
channel.RealmProxy.Monitor.ConnectionError += (s, e) => onDisconnect();
|
||||||
|
}
|
||||||
|
return channel.RealmProxy.Services.GetCalleeProxy<V>();
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task PublishMessageAsync<V>(string topic, V message)
|
public async Task PublishMessageAsync<V>(string topic, V message)
|
||||||
{
|
{
|
||||||
IWampRealmProxy channel = await GetChannelAsync(topic);
|
IWampChannel channel = await GetChannelAsync();
|
||||||
WampHelper.PublishMessage(channel, topic, message);
|
WampHelper.PublishMessage(channel.RealmProxy, topic, message);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task PublishMessagesAsync<V>(string topic, IEnumerable<V> messages)
|
public async Task PublishMessagesAsync<V>(string topic, IEnumerable<V> messages)
|
||||||
{
|
{
|
||||||
IWampRealmProxy channel = await GetChannelAsync(topic);
|
IWampChannel channel = await GetChannelAsync();
|
||||||
WampHelper.PublishMessages(channel, topic, messages);
|
WampHelper.PublishMessages(channel.RealmProxy, topic, messages);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public class WampServers
|
public class WampServers
|
||||||
{
|
{
|
||||||
public static WampServer<MessengerServerTopics> MessengerServer { get; } = new WampServer<MessengerServerTopics>
|
public static WampServer MessengerServer { get; } = new WampServer
|
||||||
{
|
{
|
||||||
Endpoint = "/wamp",
|
Endpoint = "/wamp",
|
||||||
Address = $"{Servers.MessengerServer.WS()}/wamp",
|
Address = $"{Servers.MessengerServer.WS()}/wamp",
|
||||||
Realm = "messages",
|
Realm = "messages",
|
||||||
};
|
};
|
||||||
|
|
||||||
public static WampServer<SubscriptionServerTopics> SubscriptionServer { get; } = new WampServer<SubscriptionServerTopics>
|
public static WampServer SubscriptionServer { get; } = new WampServer
|
||||||
{
|
{
|
||||||
Endpoint = "/wamp",
|
Endpoint = "/wamp",
|
||||||
Address = $"{Servers.SubscriptionServer.WS()}/wamp",
|
Address = $"{Servers.SubscriptionServer.WS()}/wamp",
|
||||||
Realm = "messages",
|
Realm = "messages",
|
||||||
};
|
};
|
||||||
|
|
||||||
public static WampServer<IdentityServerTopics> IdentityServer { get; } = new WampServer<IdentityServerTopics>
|
public static WampServer IdentityServer { get; } = new WampServer
|
||||||
{
|
{
|
||||||
Endpoint = "/wamp",
|
Endpoint = "/wamp",
|
||||||
Address = $"{Servers.IdentityServer.WS()}/wamp",
|
Address = $"{Servers.IdentityServer.WS()}/wamp",
|
||||||
Realm = "messages",
|
Realm = "messages",
|
||||||
};
|
};
|
||||||
|
|
||||||
public static WampServer<NotesnookServerTopics> NotesnookServer { get; } = new WampServer<NotesnookServerTopics>
|
public static WampServer NotesnookServer { get; } = new WampServer
|
||||||
{
|
{
|
||||||
Endpoint = "/wamp",
|
Endpoint = "/wamp",
|
||||||
Address = $"{Servers.NotesnookAPI.WS()}/wamp",
|
Address = $"{Servers.NotesnookAPI.WS()}/wamp",
|
||||||
@@ -97,28 +99,23 @@ namespace Streetwriters.Common
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
public class MessengerServerTopics
|
public struct MessengerServerTopics
|
||||||
{
|
{
|
||||||
public const string SendSSETopic = "co.streetwriters.sse.send";
|
public const string SendSSETopic = "co.streetwriters.sse.send";
|
||||||
}
|
}
|
||||||
|
|
||||||
public class SubscriptionServerTopics
|
public struct SubscriptionServerTopics
|
||||||
{
|
{
|
||||||
public const string UserSubscriptionServiceTopic = "co.streetwriters.subscriptions.subscriptions";
|
public const string UserSubscriptionServiceTopic = "co.streetwriters.subscriptions.subscriptions";
|
||||||
|
|
||||||
public const string CreateSubscriptionTopic = "co.streetwriters.subscriptions.create";
|
public const string CreateSubscriptionTopic = "co.streetwriters.subscriptions.create";
|
||||||
public const string CreateSubscriptionV2Topic = "co.streetwriters.subscriptions.v2.create";
|
public const string CreateSubscriptionV2Topic = "co.streetwriters.subscriptions.v2.create";
|
||||||
public const string DeleteSubscriptionTopic = "co.streetwriters.subscriptions.delete";
|
public const string DeleteSubscriptionTopic = "co.streetwriters.subscriptions.delete";
|
||||||
}
|
}
|
||||||
|
|
||||||
public class IdentityServerTopics
|
public struct IdentityServerTopics
|
||||||
{
|
{
|
||||||
public const string UserAccountServiceTopic = "co.streetwriters.identity.users";
|
public const string UserAccountServiceTopic = "co.streetwriters.identity.users";
|
||||||
public const string ClearCacheTopic = "co.streetwriters.identity.clear_cache";
|
public const string ClearCacheTopic = "co.streetwriters.identity.clear_cache";
|
||||||
public const string DeleteUserTopic = "co.streetwriters.identity.delete_user";
|
public const string DeleteUserTopic = "co.streetwriters.identity.delete_user";
|
||||||
}
|
}
|
||||||
|
|
||||||
public class NotesnookServerTopics
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
Reference in New Issue
Block a user