Использование SignalR с Redis messagebus отработка отказа с использованием ConnectionUtils BookSleeve.Соединять()
я пытаюсь создать сценарий аварийного переключения шины сообщений Redis с приложением SignalR.
сначала мы попробовали простой аппаратный отказоустойчивый балансировщик нагрузки, который просто контролировал два сервера Redis. Приложение SignalR указывало на единственную конечную точку HLB. Затем я потерпел неудачу на одном сервере, но не смог успешно получить какие-либо сообщения на втором сервере Redis без повторного использования пула приложений SignalR. Предположительно это связано с тем, что ему необходимо выполнить команды установки для новые автобусные сообщения в Redis.
по состоянию на SignalR RC1,Microsoft.AspNet.SignalR.Redis.RedisMessageBus
использует Booksleeve RedisConnection()
для подключения к одному Redis для pub / sub.
я создал новый класс RedisMessageBusCluster()
который использует Booksleeve ConnectionUtils.Connect()
для подключения к одному в кластере серверов Redis.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using BookSleeve;
using Microsoft.AspNet.SignalR.Infrastructure;
namespace Microsoft.AspNet.SignalR.Redis
{
/// <summary>
/// WIP: Getting scaleout for Redis working
/// </summary>
public class RedisMessageBusCluster : ScaleoutMessageBus
{
private readonly int _db;
private readonly string[] _keys;
private RedisConnection _connection;
private RedisSubscriberConnection _channel;
private Task _connectTask;
private readonly TaskQueue _publishQueue = new TaskQueue();
public RedisMessageBusCluster(string serverList, int db, IEnumerable<string> keys, IDependencyResolver resolver)
: base(resolver)
{
_db = db;
_keys = keys.ToArray();
// uses a list of connections
_connection = ConnectionUtils.Connect(serverList);
//_connection = new RedisConnection(host: server, port: port, password: password);
_connection.Closed += OnConnectionClosed;
_connection.Error += OnConnectionError;
// Start the connection - TODO: can remove this Open as the connection is already opened, but there's the _connectTask is used later on
_connectTask = _connection.Open().Then(() =>
{
// Create a subscription channel in redis
_channel = _connection.GetOpenSubscriberChannel();
// Subscribe to the registered connections
_channel.Subscribe(_keys, OnMessage);
// Dirty hack but it seems like subscribe returns before the actual
// subscription is properly setup in some cases
while (_channel.SubscriptionCount == 0)
{
Thread.Sleep(500);
}
});
}
protected override Task Send(Message[] messages)
{
return _connectTask.Then(msgs =>
{
var taskCompletionSource = new TaskCompletionSource<object>();
// Group messages by source (connection id)
var messagesBySource = msgs.GroupBy(m => m.Source);
SendImpl(messagesBySource.GetEnumerator(), taskCompletionSource);
return taskCompletionSource.Task;
},
messages);
}
private void SendImpl(IEnumerator<IGrouping<string, Message>> enumerator, TaskCompletionSource<object> taskCompletionSource)
{
if (!enumerator.MoveNext())
{
taskCompletionSource.TrySetResult(null);
}
else
{
IGrouping<string, Message> group = enumerator.Current;
// Get the channel index we're going to use for this message
int index = Math.Abs(group.Key.GetHashCode()) % _keys.Length;
string key = _keys[index];
// Increment the channel number
_connection.Strings.Increment(_db, key)
.Then((id, k) =>
{
var message = new RedisMessage(id, group.ToArray());
return _connection.Publish(k, message.GetBytes());
}, key)
.Then((enumer, tcs) => SendImpl(enumer, tcs), enumerator, taskCompletionSource)
.ContinueWithNotComplete(taskCompletionSource);
}
}
private void OnConnectionClosed(object sender, EventArgs e)
{
// Should we auto reconnect?
if (true)
{
;
}
}
private void OnConnectionError(object sender, BookSleeve.ErrorEventArgs e)
{
// How do we bubble errors?
if (true)
{
;
}
}
private void OnMessage(string key, byte[] data)
{
// The key is the stream id (channel)
var message = RedisMessage.Deserialize(data);
_publishQueue.Enqueue(() => OnReceived(key, (ulong)message.Id, message.Messages));
}
protected override void Dispose(bool disposing)
{
if (disposing)
{
if (_channel != null)
{
_channel.Unsubscribe(_keys);
_channel.Close(abort: true);
}
if (_connection != null)
{
_connection.Close(abort: true);
}
}
base.Dispose(disposing);
}
}
}
Booksleeve имеет свой собственный механизм для определения мастера, и автоматически переключится на другой сервер, и теперь я тестирую это с SignalR.Chat
.
на web.config
, Я установил список из доступных серверов:
<add key="redis.serverList" value="dbcache1.local:6379,dbcache2.local:6379"/>
затем в Application_Start()
:
// Redis cluster server list
string redisServerlist = ConfigurationManager.AppSettings["redis.serverList"];
List<string> eventKeys = new List<string>();
eventKeys.Add("SignalR.Redis.FailoverTest");
GlobalHost.DependencyResolver.UseRedisCluster(redisServerlist, eventKeys);
я добавил два дополнительных метода для Microsoft.AspNet.SignalR.Redis.DependencyResolverExtensions
:
public static IDependencyResolver UseRedisCluster(this IDependencyResolver resolver, string serverList, IEnumerable<string> eventKeys)
{
return UseRedisCluster(resolver, serverList, db: 0, eventKeys: eventKeys);
}
public static IDependencyResolver UseRedisCluster(this IDependencyResolver resolver, string serverList, int db, IEnumerable<string> eventKeys)
{
var bus = new Lazy<RedisMessageBusCluster>(() => new RedisMessageBusCluster(serverList, db, eventKeys, resolver));
resolver.Register(typeof(IMessageBus), () => bus.Value);
return resolver;
}
теперь проблема в том, что когда у меня есть несколько точек останова включен, до тех пор, пока имя пользователя не было добавлено, а затем отключить все точки останова, приложение работает, как ожидалось. Однако с отключенными с самого начала точками останова, похоже, существует некоторое состояние гонки, которое может быть неудачным во время соединения процесс.
таким образом, в RedisMessageCluster()
:
// Start the connection
_connectTask = _connection.Open().Then(() =>
{
// Create a subscription channel in redis
_channel = _connection.GetOpenSubscriberChannel();
// Subscribe to the registered connections
_channel.Subscribe(_keys, OnMessage);
// Dirty hack but it seems like subscribe returns before the actual
// subscription is properly setup in some cases
while (_channel.SubscriptionCount == 0)
{
Thread.Sleep(500);
}
});
я попытался добавить оба a Task.Wait
, и еще дополнительно Sleep()
(не показано выше) - которые ждали/etc, но все еще получали ошибки.
повторяющаяся ошибка, кажется, в Booksleeve.MessageQueue.cs
~ln 71:
A first chance exception of type 'System.InvalidOperationException' occurred in BookSleeve.dll
iisexpress.exe Error: 0 : SignalR exception thrown by Task: System.AggregateException: One or more errors occurred. ---> System.InvalidOperationException: The queue is closed
at BookSleeve.MessageQueue.Enqueue(RedisMessage item, Boolean highPri) in c:ProjectsFrameworksBookSleeve-1.2.0.5BookSleeveMessageQueue.cs:line 71
at BookSleeve.RedisConnectionBase.EnqueueMessage(RedisMessage message, Boolean queueJump) in c:ProjectsFrameworksBookSleeve-1.2.0.5BookSleeveRedisConnectionBase.cs:line 910
at BookSleeve.RedisConnectionBase.ExecuteInt64(RedisMessage message, Boolean queueJump) in c:ProjectsFrameworksBookSleeve-1.2.0.5BookSleeveRedisConnectionBase.cs:line 826
at BookSleeve.RedisConnection.IncrementImpl(Int32 db, String key, Int64 value, Boolean queueJump) in c:ProjectsFrameworksBookSleeve-1.2.0.5BookSleeveIStringCommands.cs:line 277
at BookSleeve.RedisConnection.BookSleeve.IStringCommands.Increment(Int32 db, String key, Int64 value, Boolean queueJump) in c:ProjectsFrameworksBookSleeve-1.2.0.5BookSleeveIStringCommands.cs:line 270
at Microsoft.AspNet.SignalR.Redis.RedisMessageBusCluster.SendImpl(IEnumerator`1 enumerator, TaskCompletionSource`1 taskCompletionSource) in c:ProjectsFrameworksSignalRSignalR.1.0RC1SignalRsrcMicrosoft.AspNet.SignalR.RedisRedisMessageBusCluster.cs:line 90
at Microsoft.AspNet.SignalR.Redis.RedisMessageBusCluster.<Send>b__2(Message[] msgs) in c:ProjectsFrameworksSignalRSignalR.1.0RC1SignalRsrcMicrosoft.AspNet.SignalR.RedisRedisMessageBusCluster.cs:line 67
at Microsoft.AspNet.SignalR.TaskAsyncHelper.GenericDelegates`4.<>c__DisplayClass57.<ThenWithArgs>b__56() in c:ProjectsFrameworksSignalRSignalR.1.0RC1SignalRsrcMicrosoft.AspNet.SignalR.CoreTaskAsyncHelper.cs:line 893
at Microsoft.AspNet.SignalR.TaskAsyncHelper.TaskRunners`2.<>c__DisplayClass42.<RunTask>b__41(Task t) in c:ProjectsFrameworksSignalRSignalR.1.0RC1SignalRsrcMicrosoft.AspNet.SignalR.CoreTaskAsyncHelper.cs:line 821
--- End of inner exception stack trace ---
---> (Inner Exception #0) System.InvalidOperationException: The queue is closed
at BookSleeve.MessageQueue.Enqueue(RedisMessage item, Boolean highPri) in c:ProjectsFrameworksBookSleeve-1.2.0.5BookSleeveMessageQueue.cs:line 71
at BookSleeve.RedisConnectionBase.EnqueueMessage(RedisMessage message, Boolean queueJump) in c:ProjectsFrameworksBookSleeve-1.2.0.5BookSleeveRedisConnectionBase.cs:line 910
at BookSleeve.RedisConnectionBase.ExecuteInt64(RedisMessage message, Boolean queueJump) in c:ProjectsFrameworksBookSleeve-1.2.0.5BookSleeveRedisConnectionBase.cs:line 826
at BookSleeve.RedisConnection.IncrementImpl(Int32 db, String key, Int64 value, Boolean queueJump) in c:ProjectsFrameworksBookSleeve-1.2.0.5BookSleeveIStringCommands.cs:line 277
at BookSleeve.RedisConnection.BookSleeve.IStringCommands.Increment(Int32 db, String key, Int64 value, Boolean queueJump) in c:ProjectsFrameworksBookSleeve-1.2.0.5BookSleeveIStringCommands.cs:line 270
at Microsoft.AspNet.SignalR.Redis.RedisMessageBusCluster.SendImpl(IEnumerator`1 enumerator, TaskCompletionSource`1 taskCompletionSource) in c:ProjectsFrameworksSignalRSignalR.1.0RC1SignalRsrcMicrosoft.AspNet.SignalR.RedisRedisMessageBusCluster.cs:line 90
at Microsoft.AspNet.SignalR.Redis.RedisMessageBusCluster.<Send>b__2(Message[] msgs) in c:ProjectsFrameworksSignalRSignalR.1.0RC1SignalRsrcMicrosoft.AspNet.SignalR.RedisRedisMessageBusCluster.cs:line 67
at Microsoft.AspNet.SignalR.TaskAsyncHelper.GenericDelegates`4.<>c__DisplayClass57.<ThenWithArgs>b__56() in c:ProjectsFrameworksSignalRSignalR.1.0RC1SignalRsrcMicrosoft.AspNet.SignalR.CoreTaskAsyncHelper.cs:line 893
at Microsoft.AspNet.SignalR.TaskAsyncHelper.TaskRunners`2.<>c__DisplayClass42.<RunTask>b__41(Task t) in c:ProjectsFrameworksSignalRSignalR.1.0RC1SignalRsrcMicrosoft.AspNet.SignalR.CoreTaskAsyncHelper.cs:line 821<---
public void Enqueue(RedisMessage item, bool highPri)
{
lock (stdPriority)
{
if (closed)
{
throw new InvalidOperationException("The queue is closed");
}
где создается исключение закрытой очереди.
я предвижу еще одну проблему: поскольку соединение Redis выполняется в Application_Start()
там могут быть некоторые проблемы в "переподключения" на другой сервер. Однако, я думаю, что это справедливо при использовании единственного числа RedisConnection()
, где есть только одно соединение на выбор. Однако, с введением ConnectionUtils.Connect()
Я хотел бы услышать от @dfowler
или другие ребята SignalR в том, как этот сценарий обрабатывается в SignalR.
1 ответ:
команда SignalR теперь реализовала поддержку фабрики пользовательских соединений с StackExchange.Редис, преемник BookSleeve, который поддерживает избыточные соединения Redis через ConnectionMultiplexer.
первоначальная проблема заключалась в том, что, несмотря на создание моих собственных методов расширения в BookSleeve для принятия коллекции серверов, сбой был невозможен.
теперь, с эволюцией BookSleeve в StackExchange.Редис, теперь мы можем настроить коллекция серверов / портов прямо в
Connect
инициализации.новая реализация намного проще, чем дорога, по которой я шел, в создании
UseRedisCluster
метод, а фоновое оперение теперь поддерживает true fail-over:var conn = ConnectionMultiplexer.Connect("redisServer1:6380,redisServer2:6380,redisServer3:6380,allowAdmin=true");
StackExchange.Redis также обеспечивает дополнительную ручную настройку, как описано в документация:
ConfigurationOptions config = new ConfigurationOptions { EndPoints = { { "redis0", 6379 }, { "redis1", 6380 } }, CommandMap = CommandMap.Create(new HashSet<string> { // EXCLUDE a few commands "INFO", "CONFIG", "CLUSTER", "PING", "ECHO", "CLIENT" }, available: false), KeepAlive = 180, DefaultVersion = new Version(2, 8, 8), Password = "changeme" };
по сути, возможность инициализации нашего SignalR масштабируемая среда с коллекцией серверов теперь решает начальную проблему.