SignalR:如何从服务器/ C#真正调用集线器的方法

时间:2015-07-01 18:59:22

标签: c# signalr

我正在尝试改进我的应用程序,这需要从C#调用集线器而不是javascript。在我的应用中添加任务的当前工作流程是:

  • 进行API调用以将数据添加到数据库
  • 将新记录返回给AngularJS控制器
  • 从控制器
  • 调用hub的方法
  • hub正确地向客户广播电话

我想做的是绕过从我的AngularJS控制器调用hub的方法,并直接从我的API控制器方法调用它。

这是我的中心目前的样子:

public class TaskHub : Hub
{
    public void InsertTask(TaskViewModel task)
    {
        Clients.Caller.onInsertTask(task, false);
        Clients.Others.onInsertTask(task, true);
    }
}

关于这个主题有很多SO线程,但我读过的所有内容都会让我将以下代码添加到我的API控制器中:

var hubContext = GlobalHost.ConnectionManager.GetHubContext<TaskHub>();
hubContext.Clients.All.onInsertTask(task);

这有很多问题。首先,我希望客户端广播调用存在于单个类中,而不是直接从我的API控制器调用。其次,hubContextIHubContext而不是IHubCallerConnectionContext的实例。这意味着我只能访问所有客户端,无法像我目前所做的那样向CallerOthers广播不同的响应。

有没有办法真正从C#调用集线器的方法,理想情况下,可以访问不同的调用者选项?理想情况下,我可以从我的API控制器(或者更好的DI解决方案)中执行以下操作:

var taskHub = new TaskHub();
taskHub.InsertTask(task);

提前致谢。

对于后人,根据Wasp的建议,我认为我会包含完整的解决方案。

首先,我修改了我的javascript(AngularJS)服务,将SignalR连接ID包含在API调用的自定义请求标头中,在本例中为INSERT:

var addTask = function (task) {
    var config = { 
        headers: { 'ConnectionId': connection.id } 
    };
    return $http.post('/api/tasks', task, config);
};

然后,在执行适用的CRUD操作后,我从API控制器中的请求中检索了连接ID,然后调用了我的集线器:

public HttpResponseMessage Post(HttpRequestMessage request, [FromBody]TaskViewModel task)
{
    var viewModel = taskAdapter.AddTask(task);
    var connectionId = request.Headers.GetValues("ConnectionId").FirstOrDefault();
    TaskHub.InsertTask(viewModel, connectionId);
    return request.CreateResponse(HttpStatusCode.OK, viewModel);
}

我的集线器看起来像这样,我现在只使用从我的API控制器调用的静态方法:

public class TaskHub : Hub
{
    private static IHubContext context = GlobalHost.ConnectionManager.GetHubContext<TaskHub>();

    public static void InsertTask(TaskViewModel task, string connectionId)
    {
        if (!String.IsNullOrEmpty(connectionId))
        {
            context.Clients.Client(connectionId).onInsertTask(task, false);
            context.Clients.AllExcept(connectionId).onInsertTask(task, true);
        }
        else
        {
            context.Clients.All.onInsertTask(task, true);
        }
    }
}

正如您所看到的,我在hub方法中有一个条件语句来处理是否未从我的应用程序的客户端部分启动集线器调用。如果外部应用程序/服务调用我的API,那就是这样。在这种情况下,SignalR连接和当然“ConnectionId”标头值将不存在。但就我而言,我仍然希望为所有连接的客户端调用onInsertTask方法,以便通知他们数据更改。这应该永远不会发生,但我只是将其包括在内以便完整。

2 个答案:

答案 0 :(得分:7)

为了真正调用集线器方法,当您调用它时,您必须连接到它,并通过该连接进行调用。通过调用不同的东西(您的API),您无法进行此类呼叫,因此您必须求助于服务器启动的广播功能,这些功能本质上无法了解Caller是什么因为没有SignalR的来电者。

也就是说,如果您的客户端在执行呼叫时已经将呼叫API(无论是Javascript还是C#)连接到集线器,您始终装饰您对API的呼叫集线器连接的connectionId(通过查询字符串,标题,...)。如果您的API收到该信息,则可以使用

模拟 Caller API
Clients.Client(connectionId)

它可以为Others

做同样的事情
Clients.AllExcept(connectionId)

IHubContext个实例上。查看official docs

然后,您可以按照DDan关于以方便的集中方式封装IHubContext用法的建议,或者甚至重新构建它以使其易于符合DI标准。

答案 1 :(得分:3)

我正在使用this回答中解释的方法。

public class NewsFeedHub : Hub 
{
    private static IHubContext hubContext = GlobalHost.ConnectionManager.GetHubContext<NewsFeedHub>();

    // Call this from JS: hub.client.send(channel, content)
    public void Send(string groupName, string content)
    {
        Clients.Group(groupName).addMessage(content);
    }

    // Call this from C#: NewsFeedHub.Static_Send(groupName, content)
    public static void Static_Send(string groupName, string content)
    {
        hubContext.Clients.Group(groupName).addMessage(content);
    }

}

集线器定义并使用其hubContext,因此您可以执行以下操作:

var newsFeedHub = new NewsFeedHub();
var newsFeedHub.Static_Send("ch1", "HELLO");

或者:

var taskHub = new TaskHub();
var taskHub.InsertTask(task);

如果您愿意,请根据您的方法命名。