|
31 | 31 |
|
32 | 32 | using System.Threading.Tasks; |
33 | 33 | using RabbitMQ.Client; |
| 34 | +using RabbitMQ.Client.Events; |
34 | 35 | using RabbitMQ.Client.Exceptions; |
| 36 | +using RabbitMQ.Client.Framing; |
35 | 37 | using Xunit; |
36 | 38 | using Xunit.Abstractions; |
37 | 39 |
|
@@ -150,5 +152,72 @@ public async Task TestThatDeletedQueuesDontReappearOnRecovery() |
150 | 152 | AssertShutdownError(e.ShutdownReason, 404); |
151 | 153 | } |
152 | 154 | } |
| 155 | + |
| 156 | + [Fact] |
| 157 | + public async Task TestAutoDeleteQueueBindingsRemovedWhenConsumerCancelled() |
| 158 | + { |
| 159 | + // See rabbitmq/rabbitmq-dotnet-client#1905. |
| 160 | + // |
| 161 | + // When the last consumer on an auto-delete queue is cancelled, the queue |
| 162 | + // and its bindings must be removed from recorded topology so that recovery |
| 163 | + // does not try to restore them. |
| 164 | + string exchangeName = GenerateExchangeName(); |
| 165 | + await _channel.ExchangeDeclareAsync(exchangeName, ExchangeType.Direct, autoDelete: true); |
| 166 | + |
| 167 | + var queueDeclareOk = await _channel.QueueDeclareAsync("", false, false, autoDelete: true); |
| 168 | + string queueName = queueDeclareOk.QueueName; |
| 169 | + await _channel.QueueBindAsync(queueName, exchangeName, routingKey: "key"); |
| 170 | + |
| 171 | + var autorecoveringConn = (AutorecoveringConnection)_conn; |
| 172 | + Assert.Equal(1, autorecoveringConn.RecordedExchangesCount); |
| 173 | + Assert.Equal(1, autorecoveringConn.RecordedQueuesCount); |
| 174 | + Assert.Equal(1, autorecoveringConn.RecordedBindingsCount); |
| 175 | + |
| 176 | + var consumer = new AsyncEventingBasicConsumer(_channel); |
| 177 | + string consumerTag = await _channel.BasicConsumeAsync(queueName, true, consumer); |
| 178 | + await _channel.BasicCancelAsync(consumerTag); |
| 179 | + |
| 180 | + Assert.Equal(0, autorecoveringConn.RecordedExchangesCount); |
| 181 | + Assert.Equal(0, autorecoveringConn.RecordedQueuesCount); |
| 182 | + Assert.Equal(0, autorecoveringConn.RecordedBindingsCount); |
| 183 | + |
| 184 | + await CloseAndWaitForRecoveryAsync(); |
| 185 | + Assert.True(_channel.IsOpen); |
| 186 | + } |
| 187 | + |
| 188 | + [Fact] |
| 189 | + public async Task TestAutoDeleteQueueBindingsRemovedWhenChannelClosed() |
| 190 | + { |
| 191 | + // See rabbitmq/rabbitmq-dotnet-client#1905. |
| 192 | + // |
| 193 | + // Same as above but uses channel closure as the trigger. |
| 194 | + string exchangeName = GenerateExchangeName(); |
| 195 | + var autorecoveringConn = (AutorecoveringConnection)_conn; |
| 196 | + |
| 197 | + IChannel ch = await _conn.CreateChannelAsync(_createChannelOptions); |
| 198 | + await using (ch.ConfigureAwait(false)) |
| 199 | + { |
| 200 | + await ch.ExchangeDeclareAsync(exchangeName, ExchangeType.Direct, autoDelete: true); |
| 201 | + |
| 202 | + var queueDeclareOk = await ch.QueueDeclareAsync("", false, false, autoDelete: true); |
| 203 | + string queueName = queueDeclareOk.QueueName; |
| 204 | + await ch.QueueBindAsync(queueName, exchangeName, routingKey: "key"); |
| 205 | + |
| 206 | + Assert.Equal(1, autorecoveringConn.RecordedExchangesCount); |
| 207 | + Assert.Equal(1, autorecoveringConn.RecordedQueuesCount); |
| 208 | + Assert.Equal(1, autorecoveringConn.RecordedBindingsCount); |
| 209 | + |
| 210 | + var consumer = new AsyncEventingBasicConsumer(ch); |
| 211 | + await ch.BasicConsumeAsync(queueName, true, consumer); |
| 212 | + await ch.CloseAsync(); |
| 213 | + } |
| 214 | + |
| 215 | + Assert.Equal(0, autorecoveringConn.RecordedExchangesCount); |
| 216 | + Assert.Equal(0, autorecoveringConn.RecordedQueuesCount); |
| 217 | + Assert.Equal(0, autorecoveringConn.RecordedBindingsCount); |
| 218 | + |
| 219 | + await CloseAndWaitForRecoveryAsync(); |
| 220 | + Assert.True(_channel.IsOpen); |
| 221 | + } |
153 | 222 | } |
154 | 223 | } |
0 commit comments