2 * ========================LICENSE_START=================================
4 * ======================================================================
5 * Copyright (C) 2020 Nordix Foundation. All rights reserved.
6 * ======================================================================
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this file except in compliance with the License.
9 * You may obtain a copy of the License at
11 * http://www.apache.org/licenses/LICENSE-2.0
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
18 * ========================LICENSE_END===================================
21 package org.onap.ccsdk.oran.a1policymanagementservice.dmaap;
23 import static ch.qos.logback.classic.Level.WARN;
24 import static org.assertj.core.api.Assertions.assertThat;
25 import static org.junit.jupiter.api.Assertions.assertNotNull;
26 import static org.junit.jupiter.api.Assertions.assertThrows;
27 import static org.junit.jupiter.api.Assertions.assertTrue;
28 import static org.mockito.ArgumentMatchers.any;
29 import static org.mockito.Mockito.doNothing;
30 import static org.mockito.Mockito.doReturn;
31 import static org.mockito.Mockito.inOrder;
32 import static org.mockito.Mockito.spy;
33 import static org.mockito.Mockito.verify;
34 import static org.mockito.Mockito.verifyNoMoreInteractions;
35 import static org.mockito.Mockito.when;
37 import ch.qos.logback.classic.spi.ILoggingEvent;
38 import ch.qos.logback.core.read.ListAppender;
40 import com.google.gson.Gson;
41 import com.google.gson.GsonBuilder;
42 import com.google.gson.JsonObject;
44 import java.time.Duration;
45 import java.util.LinkedList;
46 import java.util.List;
48 import org.junit.jupiter.api.AfterEach;
49 import org.junit.jupiter.api.Test;
50 import org.junit.jupiter.api.extension.ExtendWith;
51 import org.mockito.ArgumentCaptor;
52 import org.mockito.InOrder;
53 import org.mockito.Mock;
54 import org.mockito.junit.jupiter.MockitoExtension;
55 import org.onap.ccsdk.oran.a1policymanagementservice.clients.AsyncRestClient;
56 import org.onap.ccsdk.oran.a1policymanagementservice.configuration.ApplicationConfig;
57 import org.onap.ccsdk.oran.a1policymanagementservice.dmaap.DmaapRequestMessage.Operation;
58 import org.onap.ccsdk.oran.a1policymanagementservice.exceptions.ServiceException;
59 import org.onap.ccsdk.oran.a1policymanagementservice.utils.LoggingUtils;
60 import org.springframework.http.HttpStatus;
61 import org.springframework.http.ResponseEntity;
62 import reactor.core.publisher.Mono;
64 @ExtendWith(MockitoExtension.class)
65 class DmaapMessageConsumerTest {
67 private ApplicationConfig applicationConfigMock;
69 private AsyncRestClient messageRouterConsumerMock;
71 private DmaapMessageHandler messageHandlerMock;
73 private DmaapMessageConsumer messageConsumerUnderTest;
75 private Gson gson = new GsonBuilder().create();
79 LoggingUtils.getLogListAppender(DmaapMessageConsumer.class);
83 void dmaapNotConfigured_thenSleepAndRetryUntilConfig() throws Exception {
84 messageConsumerUnderTest = spy(new DmaapMessageConsumer(applicationConfigMock));
86 doNothing().when(messageConsumerUnderTest).sleep(any(Duration.class));
87 doReturn(false, false, false, true).when(messageConsumerUnderTest).isStopped();
88 doReturn(false, true, true).when(messageConsumerUnderTest).isDmaapConfigured();
89 doReturn(new LinkedList<>()).when(messageConsumerUnderTest).fetchAllMessages();
91 messageConsumerUnderTest.start().join();
93 InOrder orderVerifier = inOrder(messageConsumerUnderTest);
94 orderVerifier.verify(messageConsumerUnderTest).sleep(DmaapMessageConsumer.TIME_BETWEEN_DMAAP_RETRIES);
95 orderVerifier.verify(messageConsumerUnderTest).fetchAllMessages();
99 void dmaapConfigurationRemoved_thenStopPollingDmaapSleepAndRetry() throws Exception {
100 messageConsumerUnderTest = spy(new DmaapMessageConsumer(applicationConfigMock));
102 doNothing().when(messageConsumerUnderTest).sleep(any(Duration.class));
103 doReturn(false, false, false, false, true).when(messageConsumerUnderTest).isStopped();
104 doReturn(true, true, false).when(messageConsumerUnderTest).isDmaapConfigured();
105 doReturn(new LinkedList<>()).when(messageConsumerUnderTest).fetchAllMessages();
107 messageConsumerUnderTest.start().join();
109 InOrder orderVerifier = inOrder(messageConsumerUnderTest);
110 orderVerifier.verify(messageConsumerUnderTest).fetchAllMessages();
111 orderVerifier.verify(messageConsumerUnderTest).sleep(DmaapMessageConsumer.TIME_BETWEEN_DMAAP_RETRIES);
115 void dmaapConfiguredAndNoMessages_thenPollOnce() throws Exception {
118 messageConsumerUnderTest = spy(new DmaapMessageConsumer(applicationConfigMock));
120 Mono<ResponseEntity<String>> response = Mono.empty();
122 doReturn(false, true).when(messageConsumerUnderTest).isStopped();
123 doReturn(messageRouterConsumerMock).when(messageConsumerUnderTest).getMessageRouterConsumer();
124 doReturn(response).when(messageRouterConsumerMock).getForEntity(any());
126 messageConsumerUnderTest.start().join();
128 verify(messageRouterConsumerMock).getForEntity(any());
129 verifyNoMoreInteractions(messageRouterConsumerMock);
133 void dmaapConfiguredAndErrorGettingMessages_thenLogWarningAndSleep() throws Exception {
136 messageConsumerUnderTest = spy(new DmaapMessageConsumer(applicationConfigMock));
138 doNothing().when(messageConsumerUnderTest).sleep(any(Duration.class));
139 doReturn(false, true).when(messageConsumerUnderTest).isStopped();
140 doReturn(messageRouterConsumerMock).when(messageConsumerUnderTest).getMessageRouterConsumer();
142 Mono<ResponseEntity<String>> response = Mono.just(new ResponseEntity<>("Error", HttpStatus.BAD_REQUEST));
143 when(messageRouterConsumerMock.getForEntity(any())).thenReturn(response);
145 final ListAppender<ILoggingEvent> logAppender =
146 LoggingUtils.getLogListAppender(DmaapMessageConsumer.class, WARN);
148 messageConsumerUnderTest.start().join();
150 assertThat(logAppender.list.get(0).getFormattedMessage())
151 .isEqualTo("Cannot fetch because of Error respons: 400 BAD_REQUEST Error");
153 verify(messageConsumerUnderTest).sleep(DmaapMessageConsumer.TIME_BETWEEN_DMAAP_RETRIES);
157 void dmaapConfiguredAndOneMessage_thenPollOnceAndProcessMessage() throws Exception {
158 // The message from MR is here an array of Json objects
160 messageConsumerUnderTest = spy(new DmaapMessageConsumer(applicationConfigMock));
162 String messages = jsonArray(gson.toJson(dmaapRequestMessage(Operation.PUT)));
164 doReturn(false, true).when(messageConsumerUnderTest).isStopped();
165 doReturn(messageRouterConsumerMock).when(messageConsumerUnderTest).getMessageRouterConsumer();
167 Mono<ResponseEntity<String>> response = Mono.just(new ResponseEntity<>(messages, HttpStatus.OK));
168 when(messageRouterConsumerMock.getForEntity(any())).thenReturn(response);
170 doReturn(messageHandlerMock).when(messageConsumerUnderTest).getDmaapMessageHandler();
172 messageConsumerUnderTest.start().join();
174 ArgumentCaptor<DmaapRequestMessage> captor = ArgumentCaptor.forClass(DmaapRequestMessage.class);
175 verify(messageHandlerMock).handleDmaapMsg(captor.capture());
176 DmaapRequestMessage messageAfterJsonParsing = captor.getValue();
177 assertThat(messageAfterJsonParsing.apiVersion()).isNotEmpty();
179 verifyNoMoreInteractions(messageHandlerMock);
183 void testMessageParsing() throws ServiceException {
184 messageConsumerUnderTest = new DmaapMessageConsumer(applicationConfigMock);
185 String json = gson.toJson(dmaapRequestMessage(Operation.PUT));
187 String jsonArrayOfObject = jsonArray(json);
188 List<DmaapRequestMessage> parsedMessage = messageConsumerUnderTest.parseMessages(jsonArrayOfObject);
189 assertNotNull(parsedMessage);
190 assertTrue(parsedMessage.get(0).payload().isPresent());
193 String jsonArrayOfString = jsonArray(quote(json));
194 List<DmaapRequestMessage> parsedMessage = messageConsumerUnderTest.parseMessages(jsonArrayOfString);
195 assertNotNull(parsedMessage);
196 assertTrue(parsedMessage.get(0).payload().isPresent());
202 void incomingUnparsableRequest_thenSendResponse() throws Exception {
203 messageConsumerUnderTest = spy(new DmaapMessageConsumer(applicationConfigMock));
204 doReturn(messageHandlerMock).when(messageConsumerUnderTest).getDmaapMessageHandler();
205 doReturn(Mono.just("OK")).when(messageHandlerMock).sendDmaapResponse(any(), any(), any());
206 Exception actualException =
207 assertThrows(ServiceException.class, () -> messageConsumerUnderTest.parseMessages("[\"abc:\"def\"]"));
208 assertThat(actualException.getMessage())
209 .contains("Could not parse incomming request. Reason :com.google.gson.stream.MalformedJsonException");
211 verify(messageHandlerMock).sendDmaapResponse(any(), any(), any());
215 void incomingUnparsableRequest_thenSendingResponseFailed() throws Exception {
216 messageConsumerUnderTest = spy(new DmaapMessageConsumer(applicationConfigMock));
217 doReturn(messageHandlerMock).when(messageConsumerUnderTest).getDmaapMessageHandler();
218 doReturn(Mono.error(new Exception("Sending response failed"))).when(messageHandlerMock).sendDmaapResponse(any(),
220 Exception actualException =
221 assertThrows(Exception.class, () -> messageConsumerUnderTest.parseMessages("[\"abc:\"def\"]"));
222 assertThat(actualException.getMessage()).contains("Sending response failed");
224 verify(messageHandlerMock).sendDmaapResponse(any(), any(), any());
227 private void setUpMrConfig() {
228 when(applicationConfigMock.getDmaapConsumerTopicUrl()).thenReturn("url");
229 when(applicationConfigMock.getDmaapProducerTopicUrl()).thenReturn("url");
232 private String jsonArray(String s) {
233 return "[" + s + "]";
236 private String quote(String s) {
237 return "\"" + s.replace("\"", "\\\"") + "\"";
240 private DmaapRequestMessage dmaapRequestMessage(Operation operation) {
241 return ImmutableDmaapRequestMessage.builder() //
242 .apiVersion("apiVersion") //
243 .correlationId("correlationId") //
244 .operation(operation) //
245 .originatorId("originatorId") //
246 .payload(new JsonObject()) //
247 .requestId("requestId") //
249 .timestamp("timestamp") //