当前位置: 首页 > news >正文

Asp.Net Core SignalR的协议协商问题

文章目录

  • 前言
  • 一、协议协商的原理
  • 二、常见的协商问题及解决办法
    • 1.跨域资源共享(CORS)问题
    • 2.身份验证和授权问题
    • 3.传输方式不兼容问题
    • 4.路由配置错误
    • 5.代理和负载均衡器问题
    • 6.自定义协商(高级)
  • 总结


前言

在ASP.NET Core SignalR 里,协议协商是客户端和服务器建立连接时的重要步骤

一、协议协商的原理

  • SignalR 客户端和服务器在建立连接时,会经历以下协商步骤:
    • 客户端向服务器发送一个 HTTP GET 请求,请求路径为/negotiate
    • 服务器返回客户端支持的传输方式(像 WebSocketServer-Sent Events长轮询)、访问令牌以及其他配置信息。
    • 客户端依据服务器返回的信息,挑选合适的传输方式来建立连接。

二、常见的协商问题及解决办法

1.跨域资源共享(CORS)问题

  • 问题表现:客户端在协商请求时,浏览器控制台显示类似No ‘Access-Control-Allow-Origin’ header is present的错误。

  • 解决办法
    要保证服务器端已正确配置 CORS
    必须启用 SignalR 特定的 CORS 策略。

    //跨域
    string[] urls = new[] { "http://localhost:5173" };
    builder.Services.AddCors(opt => opt.AddDefaultPolicy(builder => builder.WithOrigins(urls).AllowAnyMethod().AllowAnyHeader().AllowCredentials()));
    app.UseCors();
    app.UseHttpsRedirection();
    

2.身份验证和授权问题

  • 问题表现:协商请求返回 401(未授权)或者 403(禁止访问)错误。

  • 解决办法
    要确保客户端在协商请求中正确传递了认证信息。
    对服务器端的授权策略进行检查。

    // 对Hub添加授权要求
    [Authorize]
    public class MyHubService: Hub
    {// ...
    }
    

3.传输方式不兼容问题

  • 问题表现:客户端和服务器没有共同支持的传输方式。
  • 解决办法
    检查客户端和服务器是否都支持相同的传输方式。
    可以在客户端限制传输方式。
    // 创建新连接state.connection = new signalR.HubConnectionBuilder().withUrl(state.serverUrl, {//skipNegotiation: true, // 尝试跳过协商步骤transport: signalR.HttpTransportType.WebSockets // 限制传输方式,强制使用 WebSockets}).withAutomaticReconnect({nextRetryDelayInMilliseconds: retryContext => {state.retryCount = retryContext.previousRetryCount + 1;return Math.min(1000 * Math.pow(2, state.retryCount), 30000);}}).configureLogging(signalR.LogLevel.Debug) // 启用详细调试日志.build();
    

4.路由配置错误

  • 问题表现:协商请求返回 404(未找到)错误。
  • 解决办法
    确认 Hub 的路由配置无误。
    // 配置路由
    app.MapHub<MyHubService>("/Hubs/MyHubService");// SignalR 终结点
    

5.代理和负载均衡器问题

  • 问题表现:在 Kubernetes/IIS ARR 后出现 WebSocket is closed 或协商超时。
  • 解决办法
    • 粘性会话(Sticky Sessions):
      若使用 WebSocket,需配置负载均衡器保持会话亲和性。
    • 转发头配置
      app.UseForwardedHeaders(new ForwardedHeadersOptions {ForwardedHeaders = ForwardedHeaders.XForwardedFor | ForwardedHeaders.XForwardedProto
      });
      
    • WebSocket 支持
      IIS 需启用 WebSocket 模块,Azure App Service 配置 webSockets: true。

6.自定义协商(高级)

  • 禁用协商(仅限 WebSocket
    服务器必须启用 WebSocket 并开放正确端口(通常 80/443)。
    // 创建新连接state.connection = new signalR.HubConnectionBuilder().withUrl(state.serverUrl, {skipNegotiation: true, // 尝试跳过协商步骤transport: signalR.HttpTransportType.WebSockets // 强制使用 WebSockets}).withAutomaticReconnect({nextRetryDelayInMilliseconds: retryContext => {state.retryCount = retryContext.previousRetryCount + 1;return Math.min(1000 * Math.pow(2, state.retryCount), 30000);}}).configureLogging(signalR.LogLevel.Debug) // 启用详细调试日志.build();
    

总结

  • 检查终结点是否注册 MapHub

  • 验证 CORS 策略(特别是 AllowCredentials())

  • 确保身份认证信息正确传递

  • 核对客户端/服务端库版本

  • 检查网络层(防火墙、代理、负载均衡)

  • 启用详细日志分析协商过程

相关文章:

  • 2.1HarmonyOS NEXT开发工具链进阶:DevEco Studio深度实践
  • 自动驾驶系列—Monocular 3D Lane Detection for Autonomous Driving
  • “粽”览全局:分布式系统架构与实践深度解析(端午特别版)
  • 打卡day41
  • 2025-05-31 Python深度学习9——网络模型的加载与保存
  • Mybatis-Plus简单介绍
  • 深入探讨redis:主从复制
  • Flutter - 原生交互 - 相机Camera - 01
  • 快速掌握 GO 之 RabbitMQ
  • 【iOS】方法交换
  • c/c++的opencv车牌识别
  • MATLAB实现井字棋
  • 可灵2.1 vs Veo 3:AI视频生成谁更胜一筹?
  • matlab/simulink TLC语法基础练习实例
  • Java 数据处理 - 数值转不同进制的字符串(数值转十进制字符串、数值转二进制字符串、数值转八进制字符串、数值转十六进制字符串)
  • C++23 已移除特性解析
  • CQF预备知识:一、微积分 -- 1.8.1 链式法则 I 详解
  • 电子电路:怎么理解时钟脉冲上升沿这句话?
  • PostgreSQL性能监控双雄:深入解析pg_stat_statements与pg_statsinfo
  • 深度学习驱动的超高清图修复技术——综述
  • 介绍政府网站建设经验/疫情最新消息今天公布
  • 项目计划书ai生成/临沂seo顾问
  • 网站安全检测产品优势/某网站搜索引擎优化
  • 手机如何建设网站/互联网广告联盟
  • 简单大气的网站模板/品牌营销策略案例
  • 如何自己做音频网站/中国十大广告公司排行榜