/// <summary> /// 定义一个队列 /// </summary> /// <param name="subscribeModel"></param> /// <returns></returns> public bool DeclareQueueOnServer(RMQSubscribeModel subscribeModel) { var connection = RabbitMQBaseHelper.CreateMQConnectionInPoolNew(); using (var channel = connection.CreateModel())//建立通讯信道 { SubscribeHelper.DefineQueue(connection, subscribeModel); RabbitMQBaseHelper.ResetMQConnectionToFree(connection); } return(true); }
public static IModel DefineQueue(IConnection connection, RMQSubscribeModel subscribeModel) { var exchangeTypeString = RabbitMQBaseHelper.GetExchangeTypeString(subscribeModel.ExchangeType); string routingKey = subscribeModel.RoutingKey.Trim(); string queueName = subscribeModel.QueueName.Trim(); string exChangeName = subscribeModel.ExchangeName.Trim(); if (string.IsNullOrEmpty(routingKey)) { routingKey = queueName; } var channel = connection.CreateModel(); //死信交换器 Dictionary <string, object> args = new Dictionary <string, object>(); string dlx_exchange = RabbitMQConstant.DLXPrefix + exChangeName; string dlx_queue = RabbitMQConstant.DLXPrefix + queueName; args.Add("x-dead-letter-exchange", dlx_exchange); args.Add("x-dead-letter-routing-key", routingKey); channel.ExchangeDeclare(exChangeName, exchangeTypeString, true, false, null); //死信交换器 if (subscribeModel.IsNeedDLX) { channel.QueueDeclare(queueName, true, false, false, args); } else { channel.QueueDeclare(queueName, true, false, false, null); } channel.QueueBind(queueName, exChangeName, routingKey); if (subscribeModel.IsNeedDLX) { channel.ExchangeDeclare(dlx_exchange, exchangeTypeString, true, false, null); channel.QueueDeclare(dlx_queue, true, false, false, null); channel.QueueBind(dlx_queue, dlx_exchange, routingKey.Trim()); } channel.BasicQos(0, subscribeModel.SendMessageCount, false); //分发机制为触发式 return(channel); }
/// <summary> /// 重新连接 /// </summary> /// <param name="connection"></param> /// <param name="handler"></param> /// <param name="exChangeName"></param> /// <param name="queueName"></param> /// <param name="routingKey"></param> /// <param name="isNeedDLX"></param> /// <returns></returns> private static IModel Consume(IConnection connection, Func <string, Task <ConsumeActionEnum> > handler, RMQSubscribeModel subscribeModel) { var channel = DefineQueue(connection, subscribeModel); var consumer = new EventingBasicConsumer(channel); ConsumeActionEnum consumeResult = ConsumeActionEnum.RETRY; Task <ConsumeActionEnum> result; consumer.Received += async(ch, ea) => { var body = ea.Body; var message = Encoding.UTF8.GetString(body); result = handler(message); consumeResult = await result; if (consumeResult == ConsumeActionEnum.ACCEPT) { channel.BasicAck(ea.DeliveryTag, false); //消息从队列中删除 } else if (consumeResult == ConsumeActionEnum.RETRY) { channel.BasicNack(ea.DeliveryTag, false, true); //消息重回队列 } else { channel.BasicNack(ea.DeliveryTag, false, false); //消息直接丢弃 } }; channel.BasicConsume(subscribeModel.QueueName, false, consumer); return(channel); }
/// <summary> /// 消息消费(fanout) /// </summary>` /// <typeparam name="T">消息类型</typeparam> /// <param name="handler">回调</param> /// <param name="exChangeName">交换器名</param> /// <param name="queueName">队列名</param> /// <param name="routingKey">路由名(默认和队列一致)</param> /// <param name="isNeedDLX">是否需要死信队列</param> public static IModel FanoutConsume(Func <string, Task <ConsumeActionEnum> > handler, RMQSubscribeModel subscribeModel) { var connection = RabbitMQBaseHelper.CreateMQConnectionInPoolNew(); try { return(Consume(connection, handler, subscribeModel)); } catch (Exception) { // throw ex; // return null; //重新连接 return(Consume(connection, handler, subscribeModel)); } finally { RabbitMQBaseHelper.ResetMQConnectionToFree(connection); } }
/// <summary> /// 拉模式消费死信队列 /// </summary> /// <returns></returns> public static async Task <IModel> PullMessageConsumeForDLXAsync(Func <string, Task <ConsumeActionEnum> > handler, RMQSubscribeModel subscribeModel) { var connection = RabbitMQBaseHelper.CreateMQConnectionInPoolNew(); // var channel = connection.CreateModel(); var channel = DefineQueue(connection, subscribeModel); BasicGetResult response = channel.BasicGet(RabbitMQConstant.DLXPrefix + subscribeModel.QueueName, false); if (response != null) { string message = Encoding.UTF8.GetString(response.Body); ConsumeActionEnum consumeResult = ConsumeActionEnum.RETRY; Task <ConsumeActionEnum> result = handler(message); consumeResult = await result; if (consumeResult == ConsumeActionEnum.ACCEPT) { channel.BasicAck(response.DeliveryTag, false); //消息从队列中删除 } else if (consumeResult == ConsumeActionEnum.RETRY) { channel.BasicNack(response.DeliveryTag, false, true); //消息重回队列 } else { channel.BasicNack(response.DeliveryTag, false, false); //消息直接丢弃 } } else { Console.WriteLine($"{RabbitMQConstant.DLXPrefix + subscribeModel.QueueName}暂无可消费信息"); } RabbitMQBaseHelper.ResetMQConnectionToFree(connection); return(channel); }