/spring-data-mongodb/src/test/java/org/springframework/data/mongodb/core/messaging/DefaultMessageListenerContainerUnitTests.java

http://github.com/SpringSource/spring-data-mongodb · Java · 297 lines · 197 code · 80 blank · 20 comment · 5 complexity · f5626eda5791006648cad48e887731f2 MD5 · raw file

  1. /*
  2. * Copyright 2018-2021 the original author or authors.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * https://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. package org.springframework.data.mongodb.core.messaging;
  17. import static edu.umd.cs.mtc.TestFramework.*;
  18. import static org.assertj.core.api.Assertions.*;
  19. import edu.umd.cs.mtc.MultithreadedTestCase;
  20. import java.time.Duration;
  21. import org.junit.jupiter.api.BeforeEach;
  22. import org.junit.jupiter.api.Test;
  23. import org.junit.jupiter.api.extension.ExtendWith;
  24. import org.mockito.Mock;
  25. import org.mockito.junit.jupiter.MockitoExtension;
  26. import org.springframework.dao.DataAccessResourceFailureException;
  27. import org.springframework.data.mongodb.core.MongoTemplate;
  28. import org.springframework.util.ErrorHandler;
  29. /**
  30. * Unit tests for {@link DefaultMessageListenerContainer}.
  31. *
  32. * @author Christoph Strobl
  33. */
  34. @ExtendWith(MockitoExtension.class)
  35. class DefaultMessageListenerContainerUnitTests {
  36. @Mock MongoTemplate template;
  37. @Mock ErrorHandler errorHandler;
  38. private DefaultMessageListenerContainer container;
  39. @BeforeEach
  40. void setUp() {
  41. container = new DefaultMessageListenerContainer(template);
  42. }
  43. @Test // DATAMONGO-1803
  44. void throwsErrorOnNullTemplate() {
  45. assertThatIllegalArgumentException().isThrownBy(() -> new DefaultMessageListenerContainer(null));
  46. }
  47. @Test // DATAMONGO-1803
  48. void startStopContainer() throws Throwable {
  49. runOnce(new MultithreadedStartStopContainer(container));
  50. }
  51. @Test // DATAMONGO-1803
  52. void subscribeToContainerBeforeStartup() throws Throwable {
  53. runOnce(new MultithreadedSubscribeBeforeStartup(container));
  54. }
  55. @Test // DATAMONGO-1803
  56. void subscribeToContainerAfterStartup() throws Throwable {
  57. runOnce(new MultithreadedSubscribeAfterStartup(container));
  58. }
  59. @Test // DATAMONGO-1803
  60. void stopSubscriptionWhileRunning() throws Throwable {
  61. runOnce(new StopSubscriptionWhileRunning(container));
  62. }
  63. @Test // DATAMONGO-1803
  64. void removeSubscriptionWhileRunning() throws Throwable {
  65. runOnce(new RemoveSubscriptionWhileRunning(container));
  66. }
  67. private static class RemoveSubscriptionWhileRunning extends MultithreadedTestCase {
  68. DefaultMessageListenerContainer container;
  69. Subscription subscription;
  70. RemoveSubscriptionWhileRunning(DefaultMessageListenerContainer container) {
  71. this.container = container;
  72. subscription = container.register(new MockSubscriptionRequest(), new MockTask());
  73. }
  74. public void thread1() {
  75. assertTick(0);
  76. container.start();
  77. waitForTick(2);
  78. assertThat(container.isRunning());
  79. container.stop();
  80. }
  81. public void thread2() throws InterruptedException {
  82. waitForTick(1);
  83. assertThat(subscription.isActive()).isTrue();
  84. container.remove(subscription);
  85. assertThat(subscription.isActive()).isFalse();
  86. }
  87. }
  88. private static class StopSubscriptionWhileRunning extends MultithreadedTestCase {
  89. DefaultMessageListenerContainer container;
  90. Subscription subscription;
  91. StopSubscriptionWhileRunning(DefaultMessageListenerContainer container) {
  92. this.container = container;
  93. subscription = container.register(new MockSubscriptionRequest(), new MockTask());
  94. }
  95. public void thread1() {
  96. assertTick(0);
  97. container.start();
  98. waitForTick(2);
  99. assertThat(container.isRunning());
  100. container.stop();
  101. }
  102. public void thread2() throws InterruptedException {
  103. waitForTick(1);
  104. assertThat(subscription.isActive()).isTrue();
  105. subscription.cancel();
  106. assertThat(subscription.isActive()).isFalse();
  107. }
  108. }
  109. private static class MultithreadedSubscribeAfterStartup extends MultithreadedTestCase {
  110. DefaultMessageListenerContainer container;
  111. MultithreadedSubscribeAfterStartup(DefaultMessageListenerContainer container) {
  112. this.container = container;
  113. }
  114. public void thread1() {
  115. assertTick(0);
  116. container.start();
  117. waitForTick(2);
  118. container.stop();
  119. }
  120. public void thread2() throws InterruptedException {
  121. waitForTick(1);
  122. Subscription subscription = container.register(new MockSubscriptionRequest(), new MockTask());
  123. Thread.sleep(10);
  124. assertThat(subscription.isActive()).isTrue();
  125. waitForTick(3);
  126. assertThat(subscription.isActive()).isFalse();
  127. }
  128. }
  129. private static class MultithreadedSubscribeBeforeStartup extends MultithreadedTestCase {
  130. DefaultMessageListenerContainer container;
  131. MultithreadedSubscribeBeforeStartup(DefaultMessageListenerContainer container) {
  132. this.container = container;
  133. }
  134. public void thread1() {
  135. assertTick(0);
  136. Subscription subscription = container.register(new MockSubscriptionRequest(), new MockTask());
  137. assertThat(subscription.isActive()).isFalse();
  138. waitForTick(2);
  139. assertThat(subscription.isActive()).isTrue();
  140. waitForTick(4);
  141. assertThat(subscription.isActive()).isFalse();
  142. }
  143. public void thread2() {
  144. waitForTick(1);
  145. container.start();
  146. waitForTick(3);
  147. container.stop();
  148. }
  149. }
  150. private static class MultithreadedStartStopContainer extends MultithreadedTestCase {
  151. DefaultMessageListenerContainer container;
  152. MultithreadedStartStopContainer(DefaultMessageListenerContainer container) {
  153. this.container = container;
  154. }
  155. public void thread1() {
  156. assertTick(0);
  157. container.start();
  158. waitForTick(2);
  159. assertThat(container.isRunning()).isFalse();
  160. }
  161. public void thread2() {
  162. waitForTick(1);
  163. assertThat(container.isRunning()).isTrue();
  164. container.stop();
  165. }
  166. }
  167. static class MockTask implements Task {
  168. volatile State state;
  169. volatile RuntimeException error;
  170. @Override
  171. public void cancel() throws DataAccessResourceFailureException {
  172. state = State.CANCELLED;
  173. }
  174. @Override
  175. public boolean isLongLived() {
  176. return true;
  177. }
  178. @Override
  179. public State getState() {
  180. return state;
  181. }
  182. @Override
  183. public void run() {
  184. state = State.RUNNING;
  185. while (isActive()) {
  186. if (error != null) {
  187. throw error;
  188. }
  189. try {
  190. Thread.sleep(10);
  191. } catch (InterruptedException e) {
  192. Thread.interrupted();
  193. }
  194. }
  195. }
  196. void emitError(RuntimeException error) {
  197. this.error = error;
  198. }
  199. @Override
  200. public boolean awaitStart(Duration timeout) throws InterruptedException {
  201. while (getState() == State.STARTING) {
  202. Thread.sleep(10);
  203. }
  204. return true;
  205. }
  206. }
  207. static class MockSubscriptionRequest implements SubscriptionRequest {
  208. @Override
  209. public MessageListener getMessageListener() {
  210. return message -> {};
  211. }
  212. @Override
  213. public RequestOptions getRequestOptions() {
  214. return () -> "foo";
  215. }
  216. }
  217. }