common: simplify wamp logic

previously we were opening a new channel for each topic which was unnecessary
This commit is contained in:
Abdullah Atta
2025-12-22 13:38:58 +05:00
parent 265b456c46
commit c7bb053cea
5 changed files with 103 additions and 39 deletions

View 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;
}
}

View File

@@ -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)

View File

@@ -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));

View File

@@ -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)

View File

@@ -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
{
}
} }