SignalR 在协商请求期间发生错误

3

我有一个使用 SignalR 和 AspNet 的项目。我试图连接我的客户端(它是 cors),第一个请求返回 200 状态码,但我在客户端端收到这个错误:

Error during negotiation request.

我的 SignalR 服务器端类:

public class Startup1
{
    public void Configuration(IAppBuilder app)
    {
        // Branch the pipeline here for requests that start with "/signalr"
        app.Map("/signalr", map =>
        {
            // Setup the CORS middleware to run before SignalR.
            // By default this will allow all origins. You can 
            // configure the set of origins and/or http verbs by
            // providing a cors options with a different policy.
            map.UseCors(CorsOptions.AllowAll);
            var hubConfiguration = new HubConfiguration
            {
                // You can enable JSONP by uncommenting line below.
                // JSONP requests are insecure but some older browsers (and some
                // versions of IE) require JSONP to work cross domain
                EnableJSONP = true
            };
            // Run the SignalR pipeline. We're not using MapSignalR
            // since this branch already runs under the "/signalr"
            // path.
            map.RunSignalR(hubConfiguration);
        });
    }
}

我的客户端js代码:

<script src="@Arbor.CVC.Common.Common.BuildServerFilePath("inc/js/jquery.signalR-2.2.3.min.js")">
</script>
<script type="text/javascript">
        $(document).ready(function () {
               var username = "";
               var id = "";
               var connection = $.hubConnection();
               var contosoChatHubProxy = connection.createHubProxy('Chat');

               connection.url = 'http://localhost:64585/signalr';
               connection.start().done(function () { 
                         console.error('Now connected, connection ID=' + connection.id); }).fail(function (e) { 
                         console.error('Could not connect ' + e); });
 </script>

请求返回如下结果:

Url /signalr
ConnectionToken BwSsXO+oHqBNh7kqklTWTawIR7/Do3Rc4N+48KrCNzZLB37PlP0V+DnCYgW9EguJsYcjUAf6lhqz3LNd1hqJNxGJHHWbssn4YZEZQBNqeOPC8Ex7ndJfEvEfGslEvCDI
ConnectionId    352c6a53-64b9-4b45-85ce-ae7d20b33ba9
KeepAliveTimeout    20
DisconnectTimeout   30
ConnectionTimeout   110
TryWebSockets   true
ProtocolVersion 1.4
TransportConnectTimeout 5
LongPollDelay   0

但我仍然遇到了协商错误。
1个回答

4

我从启动项中删除了CORS:

public class Startup1
{
    public void Configuration(IAppBuilder app)
    {
        // Branch the pipeline here for requests that start with "/signalr"
        app.Map("/signalr", map =>
        {
            // Setup the CORS middleware to run before SignalR.
            // By default this will allow all origins. You can 
            // configure the set of origins and/or http verbs by
            // providing a cors options with a different policy.
            //map.UseCors(CorsOptions.AllowAll);
            var hubConfiguration = new HubConfiguration
            {
                // You can enable JSONP by uncommenting line below.
                // JSONP requests are insecure but some older browsers (and some
                // versions of IE) require JSONP to work cross domain
                EnableJSONP = true,
                EnableJavaScriptProxies = true,
                EnableDetailedErrors = true
            };
            // Run the SignalR pipeline. We're not using MapSignalR
            // since this branch already runs under the "/signalr"
            // path.
            map.RunSignalR(hubConfiguration);
        });
        app.MapSignalR();
    }
}

我在我的Chat.cs类中添加了标签[HubName("Chat")]

另外,我需要定义来源,不能使用*

<system.webServer>
    <httpProtocol>
      <customHeaders>
          <add name="Access-Control-Allow-Origin" value="http://localhost:27947" />
          <add name="Access-Control-Allow-Methods" value="*" />
          <add name="Access-Control-Allow-Credentials" value="true" />
       </customHeaders>
    </httpProtocol>
</system.webServer>

在JS中:

var connection = $.hubConnection();
var contosoChatHubProxy = connection.createHubProxy('Chat');

connection.url = 'http://localhost:64585/signalr';
connection.start({ transport: ['webSockets', 'longPolling'] }).done(function () {console.log('Now connected, connection ID=' + connection.id);}).fail(function (e) { console.error('Could not connect ' + e); });

如果需要允许多个来源,请使用以下代码在 web.config (IIS) 中进行配置:
<system.webServer>
    <httpProtocol>
      <customHeaders>    
        <add name="Access-Control-Allow-Methods" value="*" />
        <add name="Access-Control-Allow-Credentials" value="true" />
      </customHeaders>
    </httpProtocol>
    <rewrite>            
        <outboundRules>
            <clear />                
            <rule name="AddCrossDomainHeader">
                <match serverVariable="RESPONSE_Access_Control_Allow_Origin" pattern=".*" />
                <conditions logicalGrouping="MatchAll" trackAllCaptures="true">
                    <add input="{HTTP_ORIGIN}" pattern="(http(s)?://((.+\.)?localhost:27947|(.+\.)?localhost:26928))" />
                </conditions>
                <action type="Rewrite" value="{C:0}" />
            </rule>           
        </outboundRules>
    </rewrite>
  </system.webServer>

嗨@patricia,system.webServer配置是在客户端还是中心端?谢谢!好帖子! - Yaron
此外,我无法弄清楚这些端口指的是什么。中心似乎在64585端口运行,但规则适用于27947、26928等端口。 - Yaron
@Yaron,web.config 是针对 hub 端的。端口是为了连接到 hub 的不同应用程序,因此规则是针对您想要连接到 hub 的 URL。 - patricia
为我解决了问题的是添加web.config参数。我只有Access-Control-Allow-Origin参数。 - Chase Ernst

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接