Cha*_*lie 3 c# websocket signalr asp.net-web-api asp.net-web-api2
目标:通过在数据库中存储新消息(发布到 Web API)并将其推送到单个监听客户端 (SignalR),促进 2 个用户之间的聊天。
以下是我的担忧:
我想过尝试一下:
也许我的应用程序太简单,不需要更好的设计?
这是我到目前为止所拥有的:
public class chatPush
{
public chatPush(){}
public push(string msg)
{
try{
string jsonMsg = JsonConvert.SerializeObject(msg);
await Clients.User(userId).send(jsonMsg);
}
catch
{
string errorMsg = //error with msg
await Clients.User(userId).send(errorMsg);
}
}
}
async Task<IHttpActionResult> PostMessage(Message msg)
db.Messages.Add(msg);
try
{
await db.SaveChangesAsync();
}
catch
{
//return error in JSON
}
//if update succeeded push it out
_chatPush.push(jsonMsg);
return //Success or Error code
Run Code Online (Sandbox Code Playgroud)
小智 5
我认为您不需要使用任何复杂的设计模式,而您的需求就这么简单。SignalR 仅在 POST/等中工作得很好。API 和 MVC 的处理程序。
如果您需要扩展到多个前端服务器,更复杂的设计会对您有所帮助,因为每个用户只会连接到您的一台 SignalR 主机,因此您需要担心通过背板或数据库的持久用户连接映射。但当你需要时,SignalR 有很好的文档。
就基本设计而言,我有一些建议:
1)如果服务器端有MessageHub,创建一个MessageHubHelper可以轻松地将数据推送给正确的用户。
[Authorize]
public class MessageHub : Hub
{
}
public static class MessageHubHelper
{
private static readonly IHubContext _hubContext =
GlobalHost.ConnectionManager.GetHubContext<MessageHub>();
public static IHubContext Context
{
get { return _hubContext; }
}
public static void Send(User user, Message message)
{
_hubContext.Clients.User(GetUserId(user)).Send(message);
}
public static void DoSomethingElse(User user, Message message)
{
_hubContext.Clients.User(GetUserId(user)).DoSomethingElse(message);
}
private static string GetUserId(User user)
{
return user.IdentityUser.UserName;
}
}
Run Code Online (Sandbox Code Playgroud)
2) 确保仅在数据库提交成功时才推送数据(例如,在 try {} 中的 SaveChangesAsync 之后,除非 catch 会提前抛出/返回)。
3)您不需要手动处理JSON。只需向下推对象,C# SignalR 库就会为您处理编码和解码。
4)根据我的经验,SignalR 的重新连接可能有点不稳定,尤其是 IUserIdProvider(我见过客户端和服务器都认为它们已重新连接但实际上没有推送的情况)。因此,我推荐类似https://github.com/DamianEdwards/NDCLondon2013/tree/master/UserPresence的内容,为您提供更可靠(或至少更可调试)的用户状态管理。