如何使用StackExchange.Redis发布到ServiceStack.Redis消息队列?

时间:2020-03-24 20:32:10

标签: c# redis stackexchange.redis servicestack.redis

我有现有的ServiceStack服务,想一次切换到StackExchange.Redis。这涉及交换发送者,最终交换接收者。这个问题是关于从StackExchange.Redis发布到ServiceStack.Redis。

这是我在控制台应用程序中组合在一起以测试概念的简单发布者。

namespace SEMQSender
{
    public class MessagePublisher
    {
        IConnectionMultiplexer _connectionMultiplexer;
        public MessagePublisher()
        {
            _connectionMultiplexer = ConnectionMultiplexer.Connect(new ConfigurationOptions()
            {
                EndPoints = {
                        {
                        "MyRedisServer"
                        }
                },
                DefaultDatabase = 0,
                AllowAdmin = true,
                SyncTimeout = 100000
            });
        }

        public void Run()
        {
            var request = new MyRequest()
            {
                Id = 27
            };
            PushServiceStackRequest(request);
        }

        public void PushServiceStackRequest<T>(T request)
        {
            var messageText = SerializeRequestAsServiceStackMessage(request);
            Push($"mq:{request.GetType().Name}.inq", messageText);
        }

        public string SerializeRequestAsServiceStackMessage<T>(T request)
        {
            var requestJson = JsonSerializer.Serialize(request);
            requestJson.Remove(0, 1);
            var serviceStackMessage = new ServiceStackMessage()
            {
                Id = Guid.NewGuid(),
                CreatedDate = DateTimeOffset.Now,
                Options = 1,
                Priority = 0,
                RetryAttempts = 0
            };
            var messageJson = JsonSerializer.Serialize(serviceStackMessage);
            var requestType = request.GetType();
            var sBuilder = new StringBuilder();
            sBuilder.AppendJoin('.', requestType.Namespace.Split('.').Take(2));
            var ns = sBuilder.ToString();
            var result = $"{messageJson.Remove(messageJson.Length - 1, 1)}, \"Body\":{{\"__type\":\"{requestType.FullName}, {ns}\",{requestJson.Remove(0, 1)}}}";
            return result;
        }

        public void Push(RedisKey queueName, RedisValue value)
        {
            _connectionMultiplexer.GetDatabase().ListRightPush(queueName, value);
        }
    }

    public class ServiceStackRedisMessage
    {
        public Guid Id { get; set; }
        public DateTimeOffset CreatedDate { get; set; }
        public int Priority { get; set; }
        public int RetryAttempts { get; set; }
        public int Options { get; set; }
    }
}

namespace MyServiceStackService.ServiceModel.MyService
{
    public class MyRequest
    {
        public int Id { get; set; }
    }
}

这是我们的ServiceStack服务如何订阅Redis消息的示例

    container.Register<IRedisClientsManager>(c => new RedisManagerPool(ConfigurationManager.AppSettings["Redis"]));
    container.Register<ICacheClient>(c => container.Resolve<IRedisClientsManager>().GetCacheClient());
    var mqHost = new RedisMqServer(container.Resolve<IRedisClientsManager>(), retryCount: 2);
    container.Register<IMessageService>(c => mqHost);

    mqHost.RegisterHandler<MyRequest>(this.ServiceController.ExecuteMessage);

    mqHost.Start();

据我所知,Redis的键和值与使用ServiceStack发布消息时生成的值相同,但是在订户端发生了一些奇怪的事情。仅在服务首次启动时才从队列中提取消息。之后放置在队列中的所有消息都将留在原处,直到重新启动服务为止。拾取的邮件在反序列化的对象上具有所有预期的数据。

希望对StackExchange.Redis或ServiceStack.Redis有更多了解的人可以提供帮助。以防万一有人好奇:我们正在切换到StackExchange.Redis,以便我们可以对Redis进行异步调用,而ServiceStack.Redis不支持该Redis。

1 个答案:

答案 0 :(得分:1)

如果您想知道客户端要发送什么命令,可以使用redis-cli中的Redis MONITOR debugging command,这将使您实时查看发布给redis服务器的所有命令。

要模仿Redis MQ客户端,您还需要将队列的名称发布到Redis发布/子主题QueueNames.TopicInmq:topic:in),以通知Redis MQ Server消息已经被发​​送。发布到该MQ。

相关问题