e8ee6b3c4b0e7935df6bc2502fe0ed300d31f9bd
[sdc.git] / openecomp-be / api / openecomp-sdc-rest-webapp / vendor-software-products-rest / vendor-software-products-rest-services / src / main / java / org / openecomp / sdcrests / vsp / rest / services / OrchestrationTemplateCandidateImpl.java
1 /*
2  * Copyright © 2016-2018 European Support Limited
3  * Copyright © 2021 Nokia
4  * Copyright © 2021 Nordix Foundation
5  *
6  * Licensed under the Apache License, Version 2.0 (the "License");
7  * you may not use this file except in compliance with the License.
8  * You may obtain a copy of the License at
9  *
10  *      http://www.apache.org/licenses/LICENSE-2.0
11  *
12  * Unless required by applicable law or agreed to in writing, software
13  * distributed under the License is distributed on an "AS IS" BASIS,
14  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15  * See the License for the specific language governing permissions and
16  * limitations under the License.
17  * ============LICENSE_END=========================================================
18  * Modifications copyright (c) 2019 Nokia
19  * Modifications copyright (c) 2021 Nordix Foundation
20  * ================================================================================
21  */
22 package org.openecomp.sdcrests.vsp.rest.services;
23
24 import static javax.ws.rs.core.Response.Status.EXPECTATION_FAILED;
25 import static javax.ws.rs.core.Response.Status.INTERNAL_SERVER_ERROR;
26 import static javax.ws.rs.core.Response.Status.NOT_ACCEPTABLE;
27 import static javax.ws.rs.core.Response.Status.NOT_FOUND;
28 import static org.openecomp.core.validation.errors.ErrorMessagesFormatBuilder.getErrorWithParameters;
29 import static org.openecomp.sdc.common.errors.Messages.ERROR_HAS_OCCURRED_WHILE_PERSISTING_THE_ARTIFACT;
30 import static org.openecomp.sdc.common.errors.Messages.ERROR_HAS_OCCURRED_WHILE_REDUCING_THE_ARTIFACT_SIZE;
31 import static org.openecomp.sdc.common.errors.Messages.NO_FILE_WAS_UPLOADED_OR_FILE_NOT_EXIST;
32 import static org.openecomp.sdc.common.errors.Messages.PACKAGE_PROCESS_ERROR;
33 import static org.openecomp.sdc.common.errors.Messages.UNEXPECTED_PROBLEM_HAPPENED_WHILE_GETTING;
34
35 import java.io.ByteArrayInputStream;
36 import java.io.FileOutputStream;
37 import java.io.IOException;
38 import java.io.InputStream;
39 import java.nio.file.Files;
40 import java.nio.file.Path;
41 import java.util.ArrayList;
42 import java.util.Collections;
43 import java.util.HashMap;
44 import java.util.List;
45 import java.util.Map;
46 import java.util.Optional;
47 import java.util.UUID;
48 import javax.activation.DataHandler;
49 import javax.inject.Named;
50 import javax.ws.rs.core.Response;
51 import org.apache.commons.lang3.tuple.Pair;
52 import org.apache.cxf.jaxrs.ext.multipart.Attachment;
53 import org.openecomp.sdc.activitylog.ActivityLogManager;
54 import org.openecomp.sdc.activitylog.ActivityLogManagerFactory;
55 import org.openecomp.sdc.activitylog.dao.type.ActivityLogEntity;
56 import org.openecomp.sdc.activitylog.dao.type.ActivityType;
57 import org.openecomp.sdc.be.csar.storage.ArtifactInfo;
58 import org.openecomp.sdc.be.csar.storage.ArtifactStorageConfig;
59 import org.openecomp.sdc.be.csar.storage.ArtifactStorageManager;
60 import org.openecomp.sdc.be.csar.storage.PackageSizeReducer;
61 import org.openecomp.sdc.be.csar.storage.StorageFactory;
62 import org.openecomp.sdc.common.util.ValidationUtils;
63 import org.openecomp.sdc.common.utils.SdcCommon;
64 import org.openecomp.sdc.datatypes.error.ErrorLevel;
65 import org.openecomp.sdc.datatypes.error.ErrorMessage;
66 import org.openecomp.sdc.logging.api.Logger;
67 import org.openecomp.sdc.logging.api.LoggerFactory;
68 import org.openecomp.sdc.vendorsoftwareproduct.OrchestrationTemplateCandidateManager;
69 import org.openecomp.sdc.vendorsoftwareproduct.OrchestrationTemplateCandidateManagerFactory;
70 import org.openecomp.sdc.vendorsoftwareproduct.VendorSoftwareProductManager;
71 import org.openecomp.sdc.vendorsoftwareproduct.VspManagerFactory;
72 import org.openecomp.sdc.vendorsoftwareproduct.dao.type.VspDetails;
73 import org.openecomp.sdc.vendorsoftwareproduct.impl.onboarding.OnboardingPackageProcessor;
74 import org.openecomp.sdc.vendorsoftwareproduct.impl.onboarding.validation.CnfPackageValidator;
75 import org.openecomp.sdc.vendorsoftwareproduct.types.OnboardPackageInfo;
76 import org.openecomp.sdc.vendorsoftwareproduct.types.OrchestrationTemplateActionResponse;
77 import org.openecomp.sdc.vendorsoftwareproduct.types.UploadFileResponse;
78 import org.openecomp.sdc.vendorsoftwareproduct.types.ValidationResponse;
79 import org.openecomp.sdc.vendorsoftwareproduct.types.candidateheat.FilesDataStructure;
80 import org.openecomp.sdc.versioning.dao.types.Version;
81 import org.openecomp.sdcrests.vendorsoftwareproducts.types.FileDataStructureDto;
82 import org.openecomp.sdcrests.vendorsoftwareproducts.types.OrchestrationTemplateActionResponseDto;
83 import org.openecomp.sdcrests.vendorsoftwareproducts.types.UploadFileResponseDto;
84 import org.openecomp.sdcrests.vendorsoftwareproducts.types.ValidationResponseDto;
85 import org.openecomp.sdcrests.vsp.rest.OrchestrationTemplateCandidate;
86 import org.openecomp.sdcrests.vsp.rest.mapping.MapFilesDataStructureToDto;
87 import org.openecomp.sdcrests.vsp.rest.mapping.MapUploadFileResponseToUploadFileResponseDto;
88 import org.openecomp.sdcrests.vsp.rest.mapping.MapValidationResponseToDto;
89 import org.springframework.context.annotation.Scope;
90 import org.springframework.stereotype.Service;
91
92 @Named
93 @Service("orchestrationTemplateCandidate")
94 @Scope(value = "prototype")
95 public class OrchestrationTemplateCandidateImpl implements OrchestrationTemplateCandidate {
96
97     private static final Logger LOGGER = LoggerFactory.getLogger(OrchestrationTemplateCandidateImpl.class);
98     private final OrchestrationTemplateCandidateManager candidateManager;
99     private final VendorSoftwareProductManager vendorSoftwareProductManager;
100     private final ActivityLogManager activityLogManager;
101     private final ArtifactStorageManager artifactStorageManager;
102     private final PackageSizeReducer packageSizeReducer;
103
104     public OrchestrationTemplateCandidateImpl() {
105         this.candidateManager = OrchestrationTemplateCandidateManagerFactory.getInstance().createInterface();
106         this.vendorSoftwareProductManager = VspManagerFactory.getInstance().createInterface();
107         this.activityLogManager = ActivityLogManagerFactory.getInstance().createInterface();
108         LOGGER.info("Instantiating artifactStorageManager");
109         final StorageFactory storageFactory = new StorageFactory();
110         this.artifactStorageManager = storageFactory.createArtifactStorageManager();
111         LOGGER.info("Instantiating packageSizeReducer");
112         this.packageSizeReducer = storageFactory.createPackageSizeReducer().orElse(null);
113     }
114
115     // Constructor used in test to avoid mock static
116     public OrchestrationTemplateCandidateImpl(final OrchestrationTemplateCandidateManager candidateManager,
117                                               final VendorSoftwareProductManager vendorSoftwareProductManager,
118                                               final ActivityLogManager activityLogManager,
119                                               final ArtifactStorageManager artifactStorageManager,
120                                               final PackageSizeReducer packageSizeReducer) {
121         this.candidateManager = candidateManager;
122         this.vendorSoftwareProductManager = vendorSoftwareProductManager;
123         this.activityLogManager = activityLogManager;
124         this.artifactStorageManager = artifactStorageManager;
125         this.packageSizeReducer = packageSizeReducer;
126     }
127
128     @Override
129     public Response upload(String vspId, String versionId, final Attachment fileToUpload, final String user) {
130         vspId = ValidationUtils.sanitizeInputString(vspId);
131         versionId = ValidationUtils.sanitizeInputString(versionId);
132         final byte[] fileToUploadBytes;
133         final DataHandler dataHandler = fileToUpload.getDataHandler();
134         final var filename = ValidationUtils.sanitizeInputString(dataHandler.getName());
135         ArtifactInfo artifactInfo = null;
136         if (artifactStorageManager.isEnabled()) {
137             final Path tempArtifactPath;
138             try {
139                 final ArtifactStorageConfig storageConfiguration = artifactStorageManager.getStorageConfiguration();
140
141                 final Path folder = Path.of(storageConfiguration.getTempPath()).resolve(vspId).resolve(versionId);
142                 tempArtifactPath = folder.resolve(UUID.randomUUID().toString());
143                 Files.createDirectories(folder);
144                 try (final InputStream packageInputStream = dataHandler.getInputStream();
145                     final var fileOutputStream = new FileOutputStream(tempArtifactPath.toFile())) {
146                     packageInputStream.transferTo(fileOutputStream);
147                 }
148             } catch (final Exception e) {
149                 return Response.status(INTERNAL_SERVER_ERROR).entity(buildUploadResponseWithError(
150                     new ErrorMessage(ErrorLevel.ERROR, UNEXPECTED_PROBLEM_HAPPENED_WHILE_GETTING.formatMessage(filename)))).build();
151             }
152             try (final InputStream inputStream = Files.newInputStream(tempArtifactPath)) {
153                 artifactInfo = artifactStorageManager.upload(vspId, versionId, inputStream);
154             } catch (final Exception e) {
155                 LOGGER.error("Package Size Reducer not configured", e);
156                 return Response.status(INTERNAL_SERVER_ERROR).entity(buildUploadResponseWithError(
157                     new ErrorMessage(ErrorLevel.ERROR, ERROR_HAS_OCCURRED_WHILE_PERSISTING_THE_ARTIFACT.formatMessage(filename)))).build();
158             }
159             try {
160                 fileToUploadBytes = packageSizeReducer.reduce(tempArtifactPath);
161                 Files.delete(tempArtifactPath);
162             } catch (final Exception e) {
163                 LOGGER.error("Package Size Reducer not configured", e);
164                 return Response.status(INTERNAL_SERVER_ERROR).entity(buildUploadResponseWithError(
165                     new ErrorMessage(ErrorLevel.ERROR,
166                         ERROR_HAS_OCCURRED_WHILE_REDUCING_THE_ARTIFACT_SIZE.formatMessage(tempArtifactPath.toString())))).build();
167             }
168         } else {
169             fileToUploadBytes = fileToUpload.getObject(byte[].class);
170         }
171
172         final var onboardingPackageProcessor = new OnboardingPackageProcessor(filename, fileToUploadBytes, new CnfPackageValidator(), artifactInfo);
173         final ErrorMessage[] errorMessages = onboardingPackageProcessor.getErrorMessages().toArray(new ErrorMessage[0]);
174         if (onboardingPackageProcessor.hasErrors()) {
175             return Response.status(NOT_ACCEPTABLE).entity(buildUploadResponseWithError(errorMessages)).build();
176         }
177         final var onboardPackageInfo = onboardingPackageProcessor.getOnboardPackageInfo().orElse(null);
178         if (onboardPackageInfo == null) {
179             final UploadFileResponseDto uploadFileResponseDto = buildUploadResponseWithError(
180                 new ErrorMessage(ErrorLevel.ERROR, PACKAGE_PROCESS_ERROR.formatMessage(filename)));
181             return Response.ok(uploadFileResponseDto).build();
182         }
183         final var version = new Version(versionId);
184         final var vspDetails = vendorSoftwareProductManager.getVsp(vspId, version);
185         final Response response = processOnboardPackage(onboardPackageInfo, vspDetails, errorMessages);
186         final UploadFileResponseDto entity = (UploadFileResponseDto) response.getEntity();
187         if (artifactStorageManager.isEnabled()) {
188             if (!entity.getErrors().isEmpty()) {
189                 artifactStorageManager.delete(artifactInfo);
190             } else {
191                 artifactStorageManager.put(vspId, versionId + ".reduced", new ByteArrayInputStream(fileToUploadBytes));
192             }
193         }
194         return response;
195     }
196
197     private Response processOnboardPackage(final OnboardPackageInfo onboardPackageInfo, final VspDetails vspDetails,
198                                            final ErrorMessage... errorMessages) {
199         final UploadFileResponse uploadFileResponse = candidateManager.upload(vspDetails, onboardPackageInfo);
200         final UploadFileResponseDto uploadFileResponseDto = new MapUploadFileResponseToUploadFileResponseDto()
201             .applyMapping(uploadFileResponse, UploadFileResponseDto.class);
202         if (errorMessages.length > 0) {
203             uploadFileResponseDto.setErrors(getErrorMap(errorMessages));
204         }
205         return Response.ok(uploadFileResponseDto).build();
206     }
207
208     private Map<String, List<ErrorMessage>> getErrorMap(ErrorMessage[] errorMessages) {
209         final Map<String, List<ErrorMessage>> errorMap = new HashMap<>();
210         final List<ErrorMessage> errorMessageList = new ArrayList<>();
211         Collections.addAll(errorMessageList, errorMessages);
212         errorMap.put(SdcCommon.UPLOAD_FILE, errorMessageList);
213         return errorMap;
214     }
215
216     private UploadFileResponseDto buildUploadResponseWithError(final ErrorMessage... errorMessages) {
217         final UploadFileResponseDto uploadFileResponseDto = new UploadFileResponseDto();
218         uploadFileResponseDto.setErrors(getErrorMap(errorMessages));
219         return uploadFileResponseDto;
220     }
221
222     @Override
223     public Response get(String vspId, String versionId, String user) throws IOException {
224         Optional<Pair<String, byte[]>> zipFile = candidateManager.get(vspId, new Version(versionId));
225         String fileName;
226         if (zipFile.isPresent()) {
227             fileName = "Candidate." + zipFile.get().getLeft();
228         } else {
229             zipFile = vendorSoftwareProductManager.get(vspId, new Version((versionId)));
230             if (!zipFile.isPresent()) {
231                 ErrorMessage errorMessage = new ErrorMessage(ErrorLevel.ERROR,
232                     getErrorWithParameters(NO_FILE_WAS_UPLOADED_OR_FILE_NOT_EXIST.getErrorMessage(), ""));
233                 LOGGER.error(errorMessage.getMessage());
234                 return Response.status(NOT_FOUND).build();
235             }
236             fileName = "Processed." + zipFile.get().getLeft();
237         }
238         Response.ResponseBuilder response = Response.ok(zipFile.get().getRight());
239         response.header("Content-Disposition", "attachment; filename=" + fileName);
240         return response.build();
241     }
242
243     @Override
244     public Response abort(String vspId, String versionId) {
245         candidateManager.abort(vspId, new Version(versionId));
246         return Response.ok().build();
247     }
248
249     @Override
250     public Response process(String vspId, String versionId, String user) {
251         Version version = new Version(versionId);
252         OrchestrationTemplateActionResponse response = candidateManager.process(vspId, version);
253         activityLogManager.logActivity(new ActivityLogEntity(vspId, version, ActivityType.Upload_Network_Package, user, true, "", ""));
254         OrchestrationTemplateActionResponseDto responseDto = copyOrchestrationTemplateActionResponseToDto(response);
255         return Response.ok(responseDto).build();
256     }
257
258     @Override
259     public Response updateFilesDataStructure(String vspId, String versionId, FileDataStructureDto fileDataStructureDto, String user) {
260         FilesDataStructure fileDataStructure = copyFilesDataStructureDtoToFilesDataStructure(fileDataStructureDto);
261         ValidationResponse response = candidateManager.updateFilesDataStructure(vspId, new Version(versionId), fileDataStructure);
262         if (!response.isValid()) {
263             return Response.status(EXPECTATION_FAILED)
264                 .entity(new MapValidationResponseToDto().applyMapping(response, ValidationResponseDto.class))
265                 .build();
266         }
267         return Response.ok(fileDataStructureDto).build();
268     }
269
270     @Override
271     public Response getFilesDataStructure(String vspId, String versionId, String user) {
272         Optional<FilesDataStructure> filesDataStructure = candidateManager.getFilesDataStructure(vspId, new Version(versionId));
273         if (!filesDataStructure.isPresent()) {
274             filesDataStructure = vendorSoftwareProductManager.getOrchestrationTemplateStructure(vspId, new Version(versionId));
275         }
276         FileDataStructureDto fileDataStructureDto = filesDataStructure
277             .map(dataStructure -> new MapFilesDataStructureToDto().applyMapping(dataStructure, FileDataStructureDto.class))
278             .orElse(new FileDataStructureDto());
279         return Response.ok(fileDataStructureDto).build();
280     }
281
282     private OrchestrationTemplateActionResponseDto copyOrchestrationTemplateActionResponseToDto(OrchestrationTemplateActionResponse response) {
283         OrchestrationTemplateActionResponseDto result = new OrchestrationTemplateActionResponseDto();
284         result.setErrors(response.getErrors());
285         result.setFileNames(response.getFileNames());
286         result.setStatus(response.getStatus());
287         return result;
288     }
289
290     private FilesDataStructure copyFilesDataStructureDtoToFilesDataStructure(FileDataStructureDto fileDataStructureDto) {
291         FilesDataStructure filesDataStructure = new FilesDataStructure();
292         filesDataStructure.setArtifacts(fileDataStructureDto.getArtifacts());
293         filesDataStructure.setModules(fileDataStructureDto.getModules());
294         filesDataStructure.setNested(fileDataStructureDto.getNested());
295         filesDataStructure.setUnassigned(fileDataStructureDto.getUnassigned());
296         return filesDataStructure;
297     }
298 }