2 * ============LICENSE_START======================================================================
3 * Copyright (C) 2018 NOKIA Intellectual Property, 2018-2019 Nordix Foundation. All rights reserved.
4 * ===============================================================================================
5 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
6 * in compliance with the License. You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software distributed under the License
11 * is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
12 * or implied. See the License for the specific language governing permissions and limitations under
14 * ============LICENSE_END========================================================================
17 package org.onap.dcaegen2.collectors.datafile.tasks;
19 import static org.junit.jupiter.api.Assertions.assertEquals;
20 import static org.junit.jupiter.api.Assertions.assertTrue;
21 import static org.mockito.ArgumentMatchers.any;
22 import static org.mockito.Mockito.doReturn;
23 import static org.mockito.Mockito.mock;
24 import static org.mockito.Mockito.spy;
25 import static org.mockito.Mockito.times;
26 import static org.mockito.Mockito.verify;
27 import static org.mockito.Mockito.verifyNoMoreInteractions;
28 import static org.mockito.Mockito.when;
30 import java.io.ByteArrayInputStream;
31 import java.io.InputStream;
33 import java.nio.file.Path;
34 import java.nio.file.Paths;
35 import java.time.Duration;
36 import java.util.ArrayList;
37 import java.util.HashMap;
38 import java.util.List;
41 import org.apache.http.Header;
42 import org.apache.http.HttpResponse;
43 import org.apache.http.StatusLine;
44 import org.apache.http.client.methods.HttpPut;
45 import org.apache.http.client.methods.HttpUriRequest;
46 import org.junit.jupiter.api.BeforeAll;
47 import org.junit.jupiter.api.Test;
48 import org.mockito.ArgumentCaptor;
49 import org.onap.dcaegen2.collectors.datafile.configuration.AppConfig;
50 import org.onap.dcaegen2.collectors.datafile.exceptions.DatafileTaskException;
51 import org.onap.dcaegen2.collectors.datafile.model.FilePublishInformation;
52 import org.onap.dcaegen2.collectors.datafile.model.ImmutableFilePublishInformation;
53 import org.onap.dcaegen2.collectors.datafile.service.producer.DmaapProducerHttpClient;
54 import org.onap.dcaegen2.services.sdk.rest.services.dmaap.client.config.DmaapPublisherConfiguration;
55 import org.springframework.http.HttpStatus;
56 import org.springframework.web.util.DefaultUriBuilderFactory;
57 import org.springframework.web.util.UriBuilder;
59 import reactor.test.StepVerifier;
62 * Tests the DataRouter publisher.
64 * @author <a href="mailto:przemyslaw.wasala@nokia.com">Przemysław Wąsala</a> on 5/17/18
65 * @author <a href="mailto:henrik.b.andersson@est.tech">Henrik Andersson</a>
67 class DataRouterPublisherTest {
68 private static final String PRODUCT_NAME = "NrRadio";
69 private static final String VENDOR_NAME = "Ericsson";
70 private static final String LAST_EPOCH_MICROSEC = "8745745764578";
71 private static final String SOURCE_NAME = "oteNB5309";
72 private static final String START_EPOCH_MICROSEC = "8745745764578";
73 private static final String TIME_ZONE_OFFSET = "UTC+05:00";
74 private static final String PM_FILE_NAME = "A20161224.1030-1045.bin.gz";
75 private static final String FTPES_ADDRESS = "ftpes://192.168.0.101:22/ftp/rop/" + PM_FILE_NAME;
77 private static final String COMPRESSION = "gzip";
78 private static final String FILE_FORMAT_TYPE = "org.3GPP.32.435#measCollec";
79 private static final String FILE_FORMAT_VERSION = "V10";
80 private static final String X_DMAAP_DR_META = "X-DMAAP-DR-META";
82 private static final String HOST = "54.45.33.2";
83 private static final String HTTPS_SCHEME = "https";
84 private static final int PORT = 1234;
85 private static final String APPLICATION_OCTET_STREAM_CONTENT_TYPE = "application/octet-stream";
86 private static final String PUBLISH_TOPIC = "publish";
87 private static final String FEED_ID = "1";
88 private static final String FILE_CONTENT = "Just a string.";
90 private static FilePublishInformation filePublishInformation;
91 private static DmaapProducerHttpClient httpClientMock;
92 private static AppConfig appConfig;
93 private static DmaapPublisherConfiguration publisherConfigurationMock = mock(DmaapPublisherConfiguration.class);
94 private static Map<String, String> context = new HashMap<>();
95 private static DataRouterPublisher publisherTaskUnderTestSpy;
98 public static void setUp() {
99 when(publisherConfigurationMock.dmaapHostName()).thenReturn(HOST);
100 when(publisherConfigurationMock.dmaapProtocol()).thenReturn(HTTPS_SCHEME);
101 when(publisherConfigurationMock.dmaapPortNumber()).thenReturn(PORT);
103 filePublishInformation = ImmutableFilePublishInformation.builder() //
104 .productName(PRODUCT_NAME) //
105 .vendorName(VENDOR_NAME) //
106 .lastEpochMicrosec(LAST_EPOCH_MICROSEC) //
107 .sourceName(SOURCE_NAME) //
108 .startEpochMicrosec(START_EPOCH_MICROSEC) //
109 .timeZoneOffset(TIME_ZONE_OFFSET) //
110 .name(PM_FILE_NAME) //
111 .location(FTPES_ADDRESS) //
112 .internalLocation(Paths.get("target/" + PM_FILE_NAME)) //
113 .compression("gzip") //
114 .fileFormatType(FILE_FORMAT_TYPE) //
115 .fileFormatVersion(FILE_FORMAT_VERSION) //
118 appConfig = mock(AppConfig.class);
119 publisherTaskUnderTestSpy = spy(new DataRouterPublisher(appConfig));
123 public void whenPassedObjectFits_ReturnsCorrectStatus() throws Exception {
124 prepareMocksForTests(null, Integer.valueOf(HttpStatus.OK.value()));
126 .create(publisherTaskUnderTestSpy.publishFile(filePublishInformation, 1, Duration.ofSeconds(0)))
127 .expectNext(filePublishInformation) //
130 ArgumentCaptor<HttpUriRequest> requestCaptor = ArgumentCaptor.forClass(HttpUriRequest.class);
131 verify(httpClientMock).getBaseUri();
132 verify(httpClientMock).addUserCredentialsToHead(any(HttpUriRequest.class));
133 verify(httpClientMock).getDmaapProducerResponseWithRedirect(requestCaptor.capture(), any());
134 verifyNoMoreInteractions(httpClientMock);
136 HttpPut actualPut = (HttpPut) requestCaptor.getValue();
137 URI actualUri = actualPut.getURI();
138 assertEquals(HTTPS_SCHEME, actualUri.getScheme());
139 assertEquals(HOST, actualUri.getHost());
140 assertEquals(PORT, actualUri.getPort());
141 Path actualPath = Paths.get(actualUri.getPath());
142 assertTrue(PUBLISH_TOPIC.equals(actualPath.getName(0).toString()));
143 assertTrue(FEED_ID.equals(actualPath.getName(1).toString()));
144 assertTrue(PM_FILE_NAME.equals(actualPath.getName(2).toString()));
146 Header[] contentHeaders = actualPut.getHeaders("content-type");
147 assertEquals(APPLICATION_OCTET_STREAM_CONTENT_TYPE, contentHeaders[0].getValue());
149 Header[] metaHeaders = actualPut.getHeaders(X_DMAAP_DR_META);
150 Map<String, String> metaHash = getMetaDataAsMap(metaHeaders);
152 assertEquals(PRODUCT_NAME, metaHash.get("productName"));
153 assertEquals(VENDOR_NAME, metaHash.get("vendorName"));
154 assertEquals(LAST_EPOCH_MICROSEC, metaHash.get("lastEpochMicrosec"));
155 assertEquals(SOURCE_NAME, metaHash.get("sourceName"));
156 assertEquals(START_EPOCH_MICROSEC, metaHash.get("startEpochMicrosec"));
157 assertEquals(TIME_ZONE_OFFSET, metaHash.get("timeZoneOffset"));
158 assertEquals(COMPRESSION, metaHash.get("compression"));
159 assertEquals(FTPES_ADDRESS, metaHash.get("location"));
160 assertEquals(FILE_FORMAT_TYPE, metaHash.get("fileFormatType"));
161 assertEquals(FILE_FORMAT_VERSION, metaHash.get("fileFormatVersion"));
163 // Note that the following line checks the number of properties that are sent to the data router.
164 // This should be 10 unless the API is updated (which is the fields checked above)
165 assertEquals(10, metaHash.size());
169 void whenPassedObjectFits_firstFailsWithExceptionThenSucceeds() throws Exception {
170 prepareMocksForTests(new DatafileTaskException("Error"), HttpStatus.OK.value());
173 .create(publisherTaskUnderTestSpy.publishFile(filePublishInformation, 2, Duration.ofSeconds(0)))
174 .expectNext(filePublishInformation) //
179 public void whenPassedObjectFits_firstFailsThenSucceeds() throws Exception {
180 prepareMocksForTests(null, Integer.valueOf(HttpStatus.BAD_GATEWAY.value()),
181 Integer.valueOf(HttpStatus.OK.value()));
184 .create(publisherTaskUnderTestSpy.publishFile(filePublishInformation, 1, Duration.ofSeconds(0)))
185 .expectNext(filePublishInformation) //
188 verify(httpClientMock, times(2)).getBaseUri();
189 verify(httpClientMock, times(2)).addUserCredentialsToHead(any(HttpUriRequest.class));
190 verify(httpClientMock, times(2)).getDmaapProducerResponseWithRedirect(any(HttpUriRequest.class), any());
191 verifyNoMoreInteractions(httpClientMock);
195 public void whenPassedObjectFits_firstFailsThenFails() throws Exception {
196 prepareMocksForTests(null, Integer.valueOf(HttpStatus.BAD_GATEWAY.value()),
197 Integer.valueOf((HttpStatus.BAD_GATEWAY.value())));
200 .create(publisherTaskUnderTestSpy.publishFile(filePublishInformation, 1, Duration.ofSeconds(0)))
201 .expectErrorMessage("Retries exhausted: 1/1") //
204 verify(httpClientMock, times(2)).getBaseUri();
205 verify(httpClientMock, times(2)).addUserCredentialsToHead(any(HttpUriRequest.class));
206 verify(httpClientMock, times(2)).getDmaapProducerResponseWithRedirect(any(HttpUriRequest.class), any());
207 verifyNoMoreInteractions(httpClientMock);
211 final void prepareMocksForTests(Exception exception, Integer firstResponse, Integer... nextHttpResponses)
213 httpClientMock = mock(DmaapProducerHttpClient.class);
214 when(appConfig.getDmaapPublisherConfiguration()).thenReturn(publisherConfigurationMock);
215 doReturn(publisherConfigurationMock).when(publisherTaskUnderTestSpy).resolveConfiguration();
216 doReturn(httpClientMock).when(publisherTaskUnderTestSpy).resolveClient();
218 UriBuilder uriBuilder = new DefaultUriBuilderFactory().builder().scheme(HTTPS_SCHEME).host(HOST).port(PORT);
219 when(httpClientMock.getBaseUri()).thenReturn(uriBuilder);
221 HttpResponse httpResponseMock = mock(HttpResponse.class);
222 if (exception == null) {
223 when(httpClientMock.getDmaapProducerResponseWithRedirect(any(HttpUriRequest.class), any()))
224 .thenReturn(httpResponseMock);
226 when(httpClientMock.getDmaapProducerResponseWithRedirect(any(HttpUriRequest.class), any()))
227 .thenThrow(exception).thenReturn(httpResponseMock);
229 StatusLine statusLineMock = mock(StatusLine.class);
230 when(httpResponseMock.getStatusLine()).thenReturn(statusLineMock);
231 when(statusLineMock.getStatusCode()).thenReturn(firstResponse, nextHttpResponses);
233 InputStream fileStream = new ByteArrayInputStream(FILE_CONTENT.getBytes());
234 doReturn(fileStream).when(publisherTaskUnderTestSpy).createInputStream(Paths.get("target", PM_FILE_NAME));
237 private Map<String, String> getMetaDataAsMap(Header[] metaHeaders) {
238 Map<String, String> metaHash = new HashMap<>();
239 String actualMetaData = metaHeaders[0].getValue();
240 actualMetaData = actualMetaData.substring(1, actualMetaData.length() - 1);
241 actualMetaData = actualMetaData.replace("\"", "");
242 String[] commaSplitedMetaData = actualMetaData.split(",");
243 for (int i = 0; i < commaSplitedMetaData.length; i++) {
244 String[] keyValuePair = commaSplitedMetaData[i].split(":");
245 if (keyValuePair.length > 2) {
246 List<String> arrayKeyValuePair = new ArrayList<>(keyValuePair.length);
247 for (int j = 1; j < keyValuePair.length; j++) {
248 arrayKeyValuePair.add(keyValuePair[j]);
250 keyValuePair[1] = String.join(":", arrayKeyValuePair);
252 metaHash.put(keyValuePair[0], keyValuePair[1]);