RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 1588909
Accepted
Dev18
Dev18
Asked:2024-07-30 21:34:47 +0000 UTC2024-07-30 21:34:47 +0000 UTC 2024-07-30 21:34:47 +0000 UTC

如何管理 Web 应用程序会话以通过 API 挂钩更新两个会话?

  • 772

任务:根据 API 请求更新所有打开会话中的页面。当 API 发送请求时,两个会话都必须更新。例如,如果 Alice 和 Bob 打开页面,则两个页面都应该更新。

问题:目前,如果您打开第二个会话并且它离开页面 X,则第一个会话将失去其订阅并且不再接收更新。

状况:

  1. 第一个 Web 应用程序是在 Blazor Server 中编写的。
  2. 第二个是一个简单的 API 挂钩。

如何确保两个会话保持签名并接收更新,而不管用户在另一个会话中的操作如何?

挂钩API

public static readonly ConcurrentDictionary<string, string> Subscribers = new();

[HttpPost("subscribe")]
public IActionResult Subscribe([FromBody] SubscriptionRequest request)
{
    Subscribers[request.Url] = request.Url;
    return Ok(new { status = "subscribed" });
}

[HttpPost("unsubscribe")]
public IActionResult Unsubscribe([FromBody] SubscriptionRequest request)
{
    Subscribers.TryRemove(request.Url, out _);
    return Ok(new { status = "unsubscribed" });
}

private async Task<IActionResult> SendWebhookToSubscribers(IEnumerable<SimpleDataForHookTest> payload)
{
    var client = _httpClientFactory.CreateClient();
    foreach (var subscriber in Subscribers.Values)
    {
        await client.PostAsJsonAsync(subscriber, payload);
    }
    return Ok(new { status = "webhook sent" });
}

布拉佐尔

protected override async Task OnInitializedAsync()
{
    await SubscriptionService.Subscribe("https://localhost:7052/api/TestHook/TestWebHook");
    HookService.Register(ReceivePlanningData);
}

private void ReceivePlanningData(IEnumerable<SimpleDataForHookTest> planningListDto)
{
    MyPropertyTestHook = planningListDto;
    InvokeAsync(StateHasChanged);
}

public void Dispose()
{
    HookService.UnRegister(ReceivePlanningData);
    SubscriptionService.Unsubscribe("https://localhost:7052/api/TestHook/TestWebHook").GetAwaiter().GetResult();
}

完整的代码位于 GitHub 上(这里我分享了控制器和 Razor 页面的主要代码)。如果您运行这两个应用程序,则 15 秒后页面 1 会更新,但如果您并行打开页面 2 并保留它,则第一个页面将不再更新。

我试过:

a) Blazor 端的计时器每 n 分钟发出 GET 请求。这是昂贵的并且会在用户工作时破坏页面。

b) SignalR,虽然订阅成功了,但没有起作用,也不刷新页面。在 main 上有一个带有 SignalR 的版本,我无法修改它。

GitHub 上的工作分支:

  • blazor-hookWithoutSession

  • 钩钩无会话

(我通常使用Azure DevOps,我不知道是否需要在GitHub上创建拉取请求,或者我是否可以克隆。我通过Visual Studio创建了一个新分支并提交了它,但我找不到该选项在 GitHub 上创建拉取请求)。

也许有人有自己的想法,可以告诉我如何从 API 更新数据。数据如何到达 API 并不重要,但 API 将数据发送给订阅者(在此示例中为 Blazor Web)。

c#
  • 1 1 个回答
  • 39 Views

1 个回答

  • Voted
  1. Best Answer
    Dev18
    2024-07-30T23:29:48Z2024-07-30T23:29:48Z

    如果在实现 SignalR 时向应用程序地址本身添加标识符,则可以获得每个会话的唯一链接,从而使每个连接都是唯一的。

    在此输入图像描述

    因此,使用SignalR,可以实现所有会话的更新,无论其中之一是否已离开订阅。

    挂钩API

    PlanningHub.cs

    using System.Collections.Concurrent;
    using Microsoft.AspNetCore.SignalR;
    
    namespace TestHookApiSimpleTest.Models
    {
        public class PlanningHub : Hub
        {
            public static readonly ConcurrentDictionary<string, string> Subscribers = new();
    
            public override Task OnConnectedAsync()
            {
                Console.WriteLine($"Client connected: {Context.ConnectionId}");
                return base.OnConnectedAsync();
            }
    
            public override Task OnDisconnectedAsync(Exception? exception)
            {
                Subscribers.TryRemove(Context.ConnectionId, out _);
                Console.WriteLine($"Client disconnected: {Context.ConnectionId}");
                return base.OnDisconnectedAsync(exception);
            }
    
            public Task Subscribe(string url)
            {
                Subscribers[Context.ConnectionId] = url;
                Console.WriteLine($"Client subscribed: {Context.ConnectionId} with URL: {url}");
                return Task.CompletedTask;
            }
    
            public Task Unsubscribe()
            {
                Subscribers.TryRemove(Context.ConnectionId, out _);
                Console.WriteLine($"Client unsubscribed: {Context.ConnectionId}");
                return Task.CompletedTask;
            }
    
            public static IReadOnlyCollection<string> GetSubscribers()
            {
                return Subscribers.Values.ToList();
            }
        }
    }
    

    EventController.cs

    using Microsoft.AspNetCore.Mvc;
    using Microsoft.AspNetCore.SignalR;
    using System.Collections.Generic;
    using System.Net.Http;
    using System.Threading.Tasks;
    using TestHookApiSimpleTest.Models;
    
    namespace TestHookApiSimpleTest.Controllers
    {
        [Route("api/[controller]")]
        [ApiController]
        public class EventController : ControllerBase
        {
            private readonly IHttpClientFactory _httpClientFactory;
            private readonly IHubContext<PlanningHub> _hubContext;
    
            public EventController(IHttpClientFactory httpClientFactory, IHubContext<PlanningHub> hubContext)
            {
                _httpClientFactory = httpClientFactory;
                _hubContext = hubContext;
            }
    
            [HttpPost("planning")]
            public async Task<IActionResult> SendWebhook([FromBody] IEnumerable<SimpleDataForHookTest> payload)
            {
                return await SendWebhookToSubscribers(payload);
            }
    
            [HttpPost("subscribe")]
            public async Task<IActionResult> Subscribe([FromBody] SubscriptionRequest request)
            {
                await _hubContext.Clients.All.SendAsync("Subscribe", request.Url);
                return Ok(new { status = "subscribed" });
            }
    
            [HttpPost("unsubscribe")]
            public async Task<IActionResult> Unsubscribe([FromBody] SubscriptionRequest request)
            {
                await _hubContext.Clients.All.SendAsync("Unsubscribe");
                return Ok(new { status = "unsubscribed" });
            }
    
            private async Task<IActionResult> SendWebhookToSubscribers(IEnumerable<SimpleDataForHookTest> payload)
            {
                try
                {
                    var subscribers = PlanningHub.GetSubscribers();
    
                    var client = _httpClientFactory.CreateClient();
                    foreach (var subscriber in subscribers)
                    {
                        var response = await client.PostAsJsonAsync(subscriber, payload);
                        response.EnsureSuccessStatusCode();
                    }
    
                    return Ok(new { status = "webhook sent" });
                }
                catch (Exception ex)
                {
                    return StatusCode(500, new { error = ex.Message });
                }
            }
        }
    }
    

    布拉佐尔

    UpdateHub.cs

    using Microsoft.AspNetCore.SignalR;
    
    namespace TestHook.Data
    {
        public class UpdateHub : Hub
        {
            public async Task SendUpdate(IEnumerable<SimpleDataForHookTest> data)
            {
                await Clients.All.SendAsync("ReceiveUpdate", data);
            }
        }
    }
    

    PageRazorExample.razor.cs

    using Microsoft.AspNetCore.Components;
    using Microsoft.AspNetCore.SignalR.Client;
    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Threading.Tasks;
    using TestHook.Data;
    using TestHook.Services;
    
    namespace TestHook.Pages
    {
        public partial class PageRazorExample : IAsyncDisposable
        {
            private List<SimpleDataForHookTest> updates;
            private HubConnection hubConnection;
    
            [Inject]
            public ISubscriptionService SubscriptionService { get; set; }
    
            [Inject]
            public IHookService HookService { get; set; }
    
            protected override async Task OnInitializedAsync()
            {
                hubConnection = new HubConnectionBuilder()
                    .WithUrl(Navigation.ToAbsoluteUri("https://localhost:7006/planninghub"))
                    .Build();
    
                hubConnection.On<IEnumerable<SimpleDataForHookTest>>("ReceiveUpdate", (data) =>
                {
                    updates = data.ToList();
                    InvokeAsync(StateHasChanged);
                });
    
                await hubConnection.StartAsync();
                await hubConnection.SendAsync("Subscribe", "https://localhost:7052/api/TestHook/TestWebHook");
    
                HookService.Register(ReceivePlanningData);
            }
    
            IEnumerable<SimpleDataForHookTest> MyPropertyTestHook { get; set; }
    
            private void ReceivePlanningData(IEnumerable<SimpleDataForHookTest> planningListDto)
            {
                if (planningListDto != null && planningListDto.Any())
                {
                    MyPropertyTestHook = planningListDto;
                    Console.WriteLine("Received planning data.");
                    InvokeAsync(StateHasChanged);
                }
                else
                {
                    Console.WriteLine("Not send data from hook");
                }
            }
    
            public async ValueTask DisposeAsync()
            {
                await hubConnection.SendAsync("Unsubscribe");
                HookService.UnRegister(ReceivePlanningData);
                await hubConnection.DisposeAsync();
            }
        }
    }
    

    Program.cs

    ....
    // Add SignalR services
    builder.Services.AddSignalR();
    ...
    app.MapHub<UpdateHub>("/updatehub");
    

    已测试:在多个会话中工作(Chrome 中的 1 个页面 + 通过 Ctrl+Shift+N 的 2 个页面)。刷新两个页面以重置 HTML,然后退出页面 2,页面 1 将继续刷新。

    • 0

相关问题

  • 使用嵌套类导出 xml 文件

  • 分层数据模板 [WPF]

  • 如何在 WPF 中为 ListView 手动创建列?

  • 在 2D 空间中,Collider 2D 挂在玩家身上,它对敌人的重量相同,我需要它这样当它们碰撞时,它们不会飞向不同的方向。统一

  • 如何在 c# 中使用 python 神经网络来创建语音合成?

  • 如何知道类中的方法是否属于接口?

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    我看不懂措辞

    • 1 个回答
  • Marko Smith

    请求的模块“del”不提供名为“default”的导出

    • 3 个回答
  • Marko Smith

    "!+tab" 在 HTML 的 vs 代码中不起作用

    • 5 个回答
  • Marko Smith

    我正在尝试解决“猜词”的问题。Python

    • 2 个回答
  • Marko Smith

    可以使用哪些命令将当前指针移动到指定的提交而不更改工作目录中的文件?

    • 1 个回答
  • Marko Smith

    Python解析野莓

    • 1 个回答
  • Marko Smith

    问题:“警告:检查最新版本的 pip 时出错。”

    • 2 个回答
  • Marko Smith

    帮助编写一个用值填充变量的循环。解决这个问题

    • 2 个回答
  • Marko Smith

    尽管依赖数组为空,但在渲染上调用了 2 次 useEffect

    • 2 个回答
  • Marko Smith

    数据不通过 Telegram.WebApp.sendData 发送

    • 1 个回答
  • Martin Hope
    Alexandr_TT 2020年新年大赛! 2020-12-20 18:20:21 +0000 UTC
  • Martin Hope
    Alexandr_TT 圣诞树动画 2020-12-23 00:38:08 +0000 UTC
  • Martin Hope
    Air 究竟是什么标识了网站访问者? 2020-11-03 15:49:20 +0000 UTC
  • Martin Hope
    Qwertiy 号码显示 9223372036854775807 2020-07-11 18:16:49 +0000 UTC
  • Martin Hope
    user216109 如何为黑客设下陷阱,或充分击退攻击? 2020-05-10 02:22:52 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5