How can I read messages from a queue in parallel?

前端 未结 5 1372
孤独总比滥情好
孤独总比滥情好 2021-01-03 12:36

Situation

We have one message queue. We would like to process messages in parallel and limit the number of simultaneously processed messages.

Our trial code

5条回答
  •  挽巷
    挽巷 (楼主)
    2021-01-03 13:25

    EDIT

    I spent a lot of time thinking about reliability of the pump - specifically if a message is received from the MessageQueue, cancellation becomes tricky - so I provided two ways to terminate the queue:

    • Signaling the CancellationToken stops the pipeline as quickly as possible and will likely result in dropped messages.
    • Calling MessagePump.Stop() terminates the pump but allows all messages which have already been taken from the queue to be fully processed before the MessagePump.Completion task transitions to RanToCompletion.

    The solution uses TPL Dataflow (NuGet: Microsoft.Tpl.Dataflow).

    Full implementation:

    using System;
    using System.Messaging;
    using System.Threading;
    using System.Threading.Tasks;
    using System.Threading.Tasks.Dataflow;
    
    namespace StackOverflow.Q34437298
    {
        /// 
        /// Pumps the message queue and processes messages in parallel.
        /// 
        public sealed class MessagePump
        {
            /// 
            /// Creates a  and immediately starts pumping.
            /// 
            public static MessagePump Run(
                MessageQueue messageQueue,
                Func processMessage,
                int maxDegreeOfParallelism,
                CancellationToken ct = default(CancellationToken))
            {
                if (messageQueue == null) throw new ArgumentNullException(nameof(messageQueue));
                if (processMessage == null) throw new ArgumentNullException(nameof(processMessage));
                if (maxDegreeOfParallelism <= 0) throw new ArgumentOutOfRangeException(nameof(maxDegreeOfParallelism));
    
                ct.ThrowIfCancellationRequested();
    
                return new MessagePump(messageQueue, processMessage, maxDegreeOfParallelism, ct);
            }
    
            private readonly TaskCompletionSource _stop = new TaskCompletionSource();
    
            /// 
            ///  which completes when this instance
            /// stops due to a  or cancellation request.
            /// 
            public Task Completion { get; }
    
            /// 
            /// Maximum number of parallel message processors.
            /// 
            public int MaxDegreeOfParallelism { get; }
    
            /// 
            ///  that is pumped by this instance.
            /// 
            public MessageQueue MessageQueue { get; }
    
            /// 
            /// Creates a new  instance.
            /// 
            private MessagePump(MessageQueue messageQueue, Func processMessage, int maxDegreeOfParallelism, CancellationToken ct)
            {
                MessageQueue = messageQueue;
                MaxDegreeOfParallelism = maxDegreeOfParallelism;
    
                // Kick off the loop.
                Completion = RunAsync(processMessage, ct);
            }
    
            /// 
            /// Soft-terminates the pump so that no more messages will be pumped.
            /// Any messages already removed from the message queue will be
            /// processed before this instance fully completes.
            /// 
            public void Stop()
            {
                // Multiple calls to Stop are fine.
                _stop.TrySetResult(true);
            }
    
            /// 
            /// Pump implementation.
            /// 
            private async Task RunAsync(Func processMessage, CancellationToken ct = default(CancellationToken))
            {
                using (CancellationTokenSource producerCTS = ct.CanBeCanceled
                    ? CancellationTokenSource.CreateLinkedTokenSource(ct)
                    : new CancellationTokenSource())
                {
                    // This CancellationToken will either be signaled
                    // externally, or if our consumer errors.
                    ct = producerCTS.Token;
    
                    // Handover between producer and consumer.
                    DataflowBlockOptions bufferOptions = new DataflowBlockOptions {
                        // There is no point in dequeuing more messages than we can process,
                        // so we'll throttle the producer by limiting the buffer capacity.
                        BoundedCapacity = MaxDegreeOfParallelism,
                        CancellationToken = ct
                    };
    
                    BufferBlock buffer = new BufferBlock(bufferOptions);
    
                    Task producer = Task.Run(async () =>
                    {
                        try
                        {
                            while (_stop.Task.Status != TaskStatus.RanToCompletion)
                            {
                                // This line and next line are the *only* two cancellation
                                // points which will not cause dropped messages.
                                ct.ThrowIfCancellationRequested();
    
                                Task peekTask = WithCancellation(PeekAsync(MessageQueue), ct);
    
                                if (await Task.WhenAny(peekTask, _stop.Task).ConfigureAwait(false) == _stop.Task)
                                {
                                    // Stop was signaled before PeekAsync returned. Wind down the producer gracefully
                                    // by breaking out and propagating completion to the consumer blocks.
                                    break;
                                }
    
                                await peekTask.ConfigureAwait(false); // Observe Peek exceptions.
    
                                ct.ThrowIfCancellationRequested();
    
                                // Zero timeout means that we will error if someone else snatches the
                                // peeked message from the queue before we get to it (due to a race).
                                // I deemed this better than getting stuck waiting for a message which
                                // may never arrive, or, worse yet, let this ReceiveAsync run onobserved
                                // due to a cancellation (if we choose to abandon it like we do PeekAsync).
                                // You will have to restart the pump if this throws.
                                // Omit timeout if this behaviour is undesired.
                                Message message = await ReceiveAsync(MessageQueue, timeout: TimeSpan.Zero).ConfigureAwait(false);
    
                                await buffer.SendAsync(message, ct).ConfigureAwait(false);
                            }
                        }
                        finally
                        {
                            buffer.Complete();
                        }
                    },
                    ct);
    
                    // Wire up the parallel consumers.
                    ExecutionDataflowBlockOptions executionOptions = new ExecutionDataflowBlockOptions {
                        CancellationToken = ct,
                        MaxDegreeOfParallelism = MaxDegreeOfParallelism,
                        SingleProducerConstrained = true, // We don't require thread safety guarantees.
                        BoundedCapacity = MaxDegreeOfParallelism,
                    };
    
                    ActionBlock consumer = new ActionBlock(async message =>
                    {
                        ct.ThrowIfCancellationRequested();
    
                        await processMessage(message).ConfigureAwait(false);
                    },
                    executionOptions);
    
                    buffer.LinkTo(consumer, new DataflowLinkOptions { PropagateCompletion = true });
    
                    if (await Task.WhenAny(producer, consumer.Completion).ConfigureAwait(false) == consumer.Completion)
                    {
                        // If we got here, consumer probably errored. Stop the producer
                        // before we throw so we don't go dequeuing more messages.
                        producerCTS.Cancel();
                    }
    
                    // Task.WhenAll checks faulted tasks before checking any
                    // canceled tasks, so if our consumer threw a legitimate
                    // execption, that's what will be rethrown, not the OCE.
                    await Task.WhenAll(producer, consumer.Completion).ConfigureAwait(false);
                }
            }
    
            /// 
            /// APM -> TAP conversion for MessageQueue.Begin/EndPeek.
            /// 
            private static Task PeekAsync(MessageQueue messageQueue)
            {
                return Task.Factory.FromAsync(messageQueue.BeginPeek(), messageQueue.EndPeek);
            }
    
            /// 
            /// APM -> TAP conversion for MessageQueue.Begin/EndReceive.
            /// 
            private static Task ReceiveAsync(MessageQueue messageQueue, TimeSpan timeout)
            {
                return Task.Factory.FromAsync(messageQueue.BeginReceive(timeout), messageQueue.EndPeek);
            }
    
            /// 
            /// Allows abandoning tasks which do not natively
            /// support cancellation. Use with caution.
            /// 
            private static async Task WithCancellation(Task task, CancellationToken ct)
            {
                ct.ThrowIfCancellationRequested();
    
                TaskCompletionSource tcs = new TaskCompletionSource();
    
                using (ct.Register(s => ((TaskCompletionSource)s).TrySetResult(true), tcs, false))
                {
                    if (task != await Task.WhenAny(task, tcs.Task).ConfigureAwait(false))
                    {
                        // Cancellation task completed first.
                        // We are abandoning the original task.
                        throw new OperationCanceledException(ct);
                    }
                }
    
                // Task completed: synchronously return result or propagate exceptions.
                return await task.ConfigureAwait(false);
            }
        }
    }
    

    Usage:

    using (MessageQueue msMq = GetQueue())
    {
        MessagePump pump = MessagePump.Run(
            msMq,
            async message =>
            {
                await Task.Delay(50);
                Console.WriteLine($"Finished processing message {message.Id}");
            },
            maxDegreeOfParallelism: 4
        );
    
        for (int i = 0; i < 100; i++)
        {
            msMq.Send(new Message());
    
            Thread.Sleep(25);
        }
    
        pump.Stop();
    
        await pump.Completion;
    }
    

    Untidy but functional unit tests:

    https://gist.github.com/KirillShlenskiy/7f3e2c4b28b9f940c3da

    ORIGINAL ANSWER

    As mentioned in my comment, there are established producer/consumer patterns in .NET, one of which is pipeline. An excellent example of such can be found in "Patterns of Parallel Programming" by Microsoft's own Stephen Toub (full text here: https://www.microsoft.com/en-au/download/details.aspx?id=19222, page 55).

    The idea is simple: producers continuously throw stuff in a queue, and consumers pull it out and process (in parallel to producers and possibly one another).

    Here's an example of a message pipeline where the consumer uses synchronous, blocking methods to process the items as they arrive (I've parallelised the consumer to suit your scenario):

    void MessageQueueWithBlockingCollection()
    {
        // If your processing is continuous and never stops throughout the lifetime of
        // your application, you can ignore the fact that BlockingCollection is IDisposable.
        using (BlockingCollection messages = new BlockingCollection())
        {
            Task producer = Task.Run(() =>
            {
                try
                {
                    for (int i = 0; i < 10; i++)
                    {
                        // Hand over the message to the consumer.
                        messages.Add(new Message());
    
                        // Simulated arrival delay for the next message.
                        Thread.Sleep(10);
                    }
                }
                finally
                {
                    // Notify consumer that there is no more data.
                    messages.CompleteAdding();
                }
            });
    
            Task consumer = Task.Run(() =>
            {
                ParallelOptions options = new ParallelOptions {
                    MaxDegreeOfParallelism = 4
                };
    
                Parallel.ForEach(messages.GetConsumingEnumerable(), options, message => {
                    ProcessMessage(message);
                });
            });
    
            Task.WaitAll(producer, consumer);
        }
    }
    
    void ProcessMessage(Message message)
    {
        Thread.Sleep(40);
    }
    

    The above code completes in approx 130-140 ms, which is exactly what you would expect given the parallelisation of the consumers.

    Now, in your scenario you are using Tasks and async/await better suited to TPL Dataflow (official Microsoft supported library tailored to parallel and asynchronous sequence processing).

    Here's a little demo showing the different types of TPL Dataflow processing blocks that you would use for the job:

    async Task MessageQueueWithTPLDataflow()
    {
        // Set up our queue.
        BufferBlock queue = new BufferBlock();
    
        // Set up our processing stage (consumer).
        ExecutionDataflowBlockOptions options = new ExecutionDataflowBlockOptions {
            CancellationToken = CancellationToken.None, // Plug in your own in case you need to support cancellation.
            MaxDegreeOfParallelism = 4
        };
    
        ActionBlock consumer = new ActionBlock(m => ProcessMessageAsync(m), options);
    
        // Link the queue to the consumer.
        queue.LinkTo(consumer, new DataflowLinkOptions { PropagateCompletion = true });
    
        // Wire up our producer.
        Task producer = Task.Run(async () =>
        {
            try
            {
                for (int i = 0; i < 10; i++)
                {
                    queue.Post(new Message());
    
                    await Task.Delay(10).ConfigureAwait(false);
                }
            }
            finally
            {
                // Signal to the consumer that there are no more items.
                queue.Complete();
            }
        });
    
        await consumer.Completion.ConfigureAwait(false);
    }
    
    Task ProcessMessageAsync(Message message)
    {
        return Task.Delay(40);
    }
    

    It's not hard to adapt the above to use your MessageQueue and you can be sure that the end result will be free of threading issues. I'll do just that if I get a bit more time today/tomorrow.

提交回复
热议问题