ASP.NET SignalR 1.0 实现的一个特性HubPipeline -实现任何消息incoming和outgoing的拦截。SignalR HubPipeline功能对应的ASP.NET Web API和ASP.NET MVC的 ActionFilter。
下面的方法是作为一个实体在IHubPipelineModule定义:
public interface IHubPipelineModule{ Func> BuildIncoming(Func > invoke); Func BuildOutgoing(Func send); Func BuildConnect(Func connect); Func BuildReconnect(Func reconnect); Func BuildDisconnect(Func disconnect); Func BuildAuthorizeConnect(Func authorizeConnect); Func , IList > BuildRejoiningGroups(Func , IList > rejoiningGroups);}
是不是感觉有非常复杂的委托。不用急, HubPipelineModule 类已经为我们实现了大部分的功能,大部分情况下已经够用了,可以继承这个雷重写里面方法就可以了.
public abstract class HubPipelineModule : IHubPipelineModule{ protected virtual bool OnBeforeAuthorizeConnect(HubDescriptor hubDescriptor, IRequest request); protected virtual bool OnBeforeConnect(IHub hub); protected virtual void OnAfterConnect(IHub hub); protected virtual bool OnBeforeReconnect(IHub hub); protected virtual void OnAfterReconnect(IHub hub); protected virtual bool OnBeforeOutgoing(IHubOutgoingInvokerContext context); protected virtual void OnAfterOutgoing(IHubOutgoingInvokerContext context); protected virtual bool OnBeforeDisconnect(IHub hub); protected virtual void OnAfterDisconnect(IHub hub); protected virtual bool OnBeforeIncoming(IHubIncomingInvokerContext context); protected virtual object OnAfterIncoming(object result, IHubIncomingInvokerContext context); protected virtual void OnIncomingError(Exception ex, IHubIncomingInvokerContext context);}
这代码看起来是不是和ASP.NET Web API和ASP.NET MVC的 ActionFilter 很相似呢。如果其中称为OnBefore的方法返回false,它会立即结束。
public class AntiClickModule : HubPipelineModule{ public AntiClickModule() { Interval = 1000; } public int Interval { get; set; } private readonly ConcurrentDictionary_connections = new ConcurrentDictionary (); protected override void OnAfterDisconnect(IHub hub) { DateTime lastDateTime; _connections.TryRemove(hub.Context.ConnectionId, out lastDateTime); } protected override bool OnBeforeIncoming(IHubIncomingInvokerContext context) { var now = DateTime.Now; var connectionId = context.Hub.Context.ConnectionId; DateTime lastDateTime; if (_connections.TryGetValue(connectionId, out lastDateTime)) { var span = now - lastDateTime; if (span.TotalMilliseconds < Interval) { return false; } } _connections.AddOrUpdate(connectionId, now, (_, __) => now); return true; }}
记录下连接到每个ID所请求的时间,我们简单地检查下一个请求的时间间隔。 如果2个请求之间的时间差小于我们设定的时间间隔,直接返回。
您可以把它添加到使用GlobalHost的模块Global.asax.cs
protected void Application_Start(object sender, EventArgs e){ GlobalHost.HubPipeline.AddModule(new AntiClickModule());}