From ae9010e541318dfb08ed6810a20bcf681914a1a0 Mon Sep 17 00:00:00 2001 From: Nikesh kumar Date: Mon, 12 Feb 2024 12:36:12 +0530 Subject: [PATCH] feat(rest): create new enpoint to upload component csv file. Signed-off-by: Nikesh kumar --- .../src/docs/asciidoc/api-guide.adoc | 1 + .../src/docs/asciidoc/importExport.adoc | 60 +++++++- .../importexport/ImportExportController.java | 119 +++++++++------- .../Sw360ImportExportService.java | 131 ++++++++++++++++-- .../restdocs/ImportExportSpec.java | 109 +++++++++++++++ 5 files changed, 360 insertions(+), 60 deletions(-) create mode 100644 rest/resource-server/src/test/java/org/eclipse/sw360/rest/resourceserver/restdocs/ImportExportSpec.java diff --git a/rest/resource-server/src/docs/asciidoc/api-guide.adoc b/rest/resource-server/src/docs/asciidoc/api-guide.adoc index 17fafca60d..cee04dd6e9 100644 --- a/rest/resource-server/src/docs/asciidoc/api-guide.adoc +++ b/rest/resource-server/src/docs/asciidoc/api-guide.adoc @@ -311,3 +311,4 @@ include::ecc.adoc[] include::importExport.adoc[] include::attachmentCleanUp.adoc[] include::databaseSanitation.adoc[] +include::importExport.adoc[] diff --git a/rest/resource-server/src/docs/asciidoc/importExport.adoc b/rest/resource-server/src/docs/asciidoc/importExport.adoc index db22a1689b..bf5079cd92 100644 --- a/rest/resource-server/src/docs/asciidoc/importExport.adoc +++ b/rest/resource-server/src/docs/asciidoc/importExport.adoc @@ -1,4 +1,3 @@ -// // Copyright Siemens AG, 2024. Part of the SW360 Portal Project. // // This program and the accompanying materials are made @@ -77,4 +76,61 @@ A `GET` request to download the component information in csv format. include::{snippets}/should_document_get_download_component_details/curl-request.adoc[] ===== Example response -include::{snippets}/should_document_get_download_component_details/http-response.adoc[] \ No newline at end of file +include::{snippets}/should_document_get_download_component_details/http-response.adoc[] +======= +[[upload-component]] +==== Upload component csv file. + +A `POST` request help to upload the component csv file. + +[red]#Request parameter# +|=== +|Parameter |Description + +|componentFile +|Upload the component csv file. +|=== + +===== Example request +include::{snippets}/should_document_upload_component_file/curl-request.adoc[] + +===== Example response +include::{snippets}/should_document_upload_component_file/http-response.adoc[] + +[[upload-release]] +==== Upload release link csv file. + +A `POST` request help to upload the release csv file. + +[red]#Request parameter# +|=== +|Parameter |Description + +|releaseFile +|Upload the release link csv file. +|=== + +===== Example request +include::{snippets}/should_document_upload_release_link_file/curl-request.adoc[] + +===== Example response +include::{snippets}/should_document_upload_release_link_file/http-response.adoc[] + +[[upload-component-attachment]] +==== Upload component attachment file. + +A `POST` request help to upload the component attachment file. + +[red]#Request parameter# +|=== +|Parameter |Description + +|component +|Upload the component attachment file. +|=== + +===== Example request +include::{snippets}/should_document_upload_component_attachment_file/curl-request.adoc[] + +===== Example response +include::{snippets}/should_document_upload_component_attachment_file/http-response.adoc[] diff --git a/rest/resource-server/src/main/java/org/eclipse/sw360/rest/resourceserver/importexport/ImportExportController.java b/rest/resource-server/src/main/java/org/eclipse/sw360/rest/resourceserver/importexport/ImportExportController.java index ac06747003..bdaa6f90ad 100644 --- a/rest/resource-server/src/main/java/org/eclipse/sw360/rest/resourceserver/importexport/ImportExportController.java +++ b/rest/resource-server/src/main/java/org/eclipse/sw360/rest/resourceserver/importexport/ImportExportController.java @@ -16,10 +16,15 @@ import java.io.IOException; import org.apache.thrift.TException; +import org.apache.thrift.transport.TTransportException; import org.eclipse.sw360.datahandler.thrift.users.User; import org.eclipse.sw360.rest.resourceserver.core.RestControllerHelper; import org.slf4j.Logger; import org.slf4j.LoggerFactory; +import org.eclipse.sw360.datahandler.common.CommonUtils; +import org.eclipse.sw360.datahandler.thrift.RequestStatus; +import org.eclipse.sw360.datahandler.thrift.RequestSummary; +import org.eclipse.sw360.rest.resourceserver.project.ProjectController; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.rest.webmvc.BasePathAwareController; import org.springframework.data.rest.webmvc.RepositoryLinksResource; @@ -38,6 +43,13 @@ import io.swagger.v3.oas.annotations.Parameter; import io.swagger.v3.oas.annotations.enums.ParameterIn; import io.swagger.v3.oas.annotations.security.SecurityRequirement; +import org.springframework.http.HttpStatus.Series; +import org.springframework.http.MediaType; +import org.springframework.web.HttpMediaTypeNotAcceptableException; +import org.springframework.web.bind.annotation.RequestParam; +import org.springframework.web.multipart.MultipartFile; + +import jakarta.servlet.ServletException; import jakarta.servlet.http.HttpServletRequest; import jakarta.servlet.http.HttpServletResponse; import lombok.NonNull; @@ -47,11 +59,13 @@ @RequiredArgsConstructor(onConstructor = @__(@Autowired)) @RestController @SecurityRequirement(name = "tokenAuth") -public class ImportExportController implements RepresentationModelProcessor { +public class ImportExportController implements RepresentationModelProcessor { private static final Logger LOGGER = LoggerFactory.getLogger(ImportExportController.class); public static final String IMPORTEXPORT_URL = "/importExport"; + private static final MediaType form = null; + @NonNull private final RestControllerHelper restControllerHelper; @@ -64,14 +78,8 @@ public RepositoryLinksResource process(RepositoryLinksResource resource) { return resource; } - @Operation( - summary = "Download csv component template.", - description = "Download csv component template.", - tags = {"ImportExport"}, - parameters = { - @Parameter(name = "Accept", in = ParameterIn.HEADER, required = true), - } - ) + @Operation(summary = "Download csv component template.", description = "Download csv component template.", tags = { + "ImportExport" }, parameters = { @Parameter(name = "Accept", in = ParameterIn.HEADER, required = true), }) @PreAuthorize("hasAuthority('WRITE')") @GetMapping(value = IMPORTEXPORT_URL + "/downloadComponentTemplate") public void downloadComponentTemplate(HttpServletResponse response) { @@ -84,14 +92,8 @@ public void downloadComponentTemplate(HttpServletResponse response) { } } - @Operation( - summary = "Download csv attachment sample info.", - description = "Download csv attachment sample information.", - tags = {"ImportExport"}, - parameters = { - @Parameter(name = "Accept", in = ParameterIn.HEADER, required = true), - } - ) + @Operation(summary = "Download csv attachment sample info.", description = "Download csv attachment sample information.", tags = { + "ImportExport" }, parameters = { @Parameter(name = "Accept", in = ParameterIn.HEADER, required = true), }) @PreAuthorize("hasAuthority('WRITE')") @GetMapping(value = IMPORTEXPORT_URL + "/downloadAttachmentSample") public void downloadAttachmentSample(HttpServletResponse response) { @@ -104,17 +106,11 @@ public void downloadAttachmentSample(HttpServletResponse response) { } } - @Operation( - summary = "Download csv attachment information.", - description = "Download csv attachment information.", - tags = {"ImportExport"}, - parameters = { - @Parameter(name = "Accept", in = ParameterIn.HEADER, required = true), - } - ) + @Operation(summary = "Download csv attachment information.", description = "Download csv attachment information.", tags = { + "ImportExport" }, parameters = { @Parameter(name = "Accept", in = ParameterIn.HEADER, required = true), }) @PreAuthorize("hasAuthority('WRITE')") @GetMapping(value = IMPORTEXPORT_URL + "/downloadAttachmentInfo") - public void downloadAttachmentInfo(HttpServletResponse response) { + public void downloadAttachmentInfo(HttpServletResponse response) throws TTransportException{ try { User sw360User = restControllerHelper.getSw360UserFromAuthentication(); importExportService.getDownloadAttachmentInfo(sw360User, response); @@ -124,14 +120,8 @@ public void downloadAttachmentInfo(HttpServletResponse response) { } } - @Operation( - summary = "Download csv release sample.", - description = "Download csv release link sample.", - tags = {"ImportExport"}, - parameters = { - @Parameter(name = "Accept", in = ParameterIn.HEADER, required = true), - } - ) + @Operation(summary = "Download csv release sample.", description = "Download csv release link sample.", tags = { + "ImportExport" }, parameters = { @Parameter(name = "Accept", in = ParameterIn.HEADER, required = true), }) @PreAuthorize("hasAuthority('WRITE')") @GetMapping(value = IMPORTEXPORT_URL + "/downloadReleaseSample") public void downloadReleaseSample(HttpServletResponse response) { @@ -144,14 +134,8 @@ public void downloadReleaseSample(HttpServletResponse response) { } } - @Operation( - summary = "Download csv release link.", - description = "Download csv release link information.", - tags = {"ImportExport"}, - parameters = { - @Parameter(name = "Accept", in = ParameterIn.HEADER, required = true), - } - ) + @Operation(summary = "Download csv release link.", description = "Download csv release link information.", tags = { + "ImportExport" }, parameters = { @Parameter(name = "Accept", in = ParameterIn.HEADER, required = true), }) @PreAuthorize("hasAuthority('WRITE')") @GetMapping(value = IMPORTEXPORT_URL + "/downloadReleaseLink") public void downloadReleaseLink(HttpServletResponse response) { @@ -164,14 +148,8 @@ public void downloadReleaseLink(HttpServletResponse response) { } } - @Operation( - summary = "Download component in csv format.", - description = "Download component.", - tags = {"ImportExport"}, - parameters = { - @Parameter(name = "Accept", in = ParameterIn.HEADER, required = true), - } - ) + @Operation(summary = "Download component in csv format.", description = "Download component.", tags = { + "ImportExport" }, parameters = { @Parameter(name = "Accept", in = ParameterIn.HEADER, required = true), }) @PreAuthorize("hasAuthority('WRITE')") @GetMapping(value = IMPORTEXPORT_URL + "/downloadComponent") public void downloadComponent(HttpServletResponse response) { @@ -198,4 +176,45 @@ public ResponseEntity handleGlobalException(Exception e) { LOGGER.error("Unhandled exception: {}", e.getMessage(), e); return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body(e.getMessage()); } + + @Operation(summary = "Upload component csv file.", description = "Upload component csv file.", tags = { + "Component" }) + @RequestMapping(value = IMPORTEXPORT_URL + "/uploadComponent", method = RequestMethod.POST, consumes = { + MediaType.MULTIPART_MIXED_VALUE, + MediaType.MULTIPART_FORM_DATA_VALUE }, produces = { MediaType.APPLICATION_JSON_VALUE }) + public ResponseEntity uploadComponentCsv( + @Parameter(description = "The component csv file to be uploaded.") @RequestParam("componentFile") MultipartFile file, + HttpServletRequest request, HttpServletResponse response) throws TException, IOException, ServletException { + + User sw360User = restControllerHelper.getSw360UserFromAuthentication(); + RequestSummary requestSummary = importExportService.uploadComponent(sw360User, file, request, response); + return ResponseEntity.ok(requestSummary); + } + + @Operation(summary = "release link file.", description = "release link file.", tags = { "Release" }) + @RequestMapping(value = IMPORTEXPORT_URL + "/uploadRelease", method = RequestMethod.POST, consumes = { + MediaType.MULTIPART_MIXED_VALUE, + MediaType.MULTIPART_FORM_DATA_VALUE }, produces = { MediaType.APPLICATION_JSON_VALUE }) + public ResponseEntity uploadReleaseCsv( + @Parameter(description = "The release csv file to be uploaded.") @RequestParam("releaseFile") MultipartFile file, + HttpServletRequest request, HttpServletResponse response) throws TException, IOException, ServletException { + + User sw360User = restControllerHelper.getSw360UserFromAuthentication(); + RequestSummary requestSummary = importExportService.uploadReleaseLink(sw360User, file, request); + return ResponseEntity.ok(requestSummary); + } + + @Operation(summary = "component attachment file.", description = "component attachment file.", tags = { + "Component" }) + @RequestMapping(value = IMPORTEXPORT_URL + "/componentAttachment", method = RequestMethod.POST, consumes = { + MediaType.MULTIPART_MIXED_VALUE, + MediaType.MULTIPART_FORM_DATA_VALUE }, produces = { MediaType.APPLICATION_JSON_VALUE }) + public ResponseEntity uploadComponentAttachment( + @Parameter(description = "The component attachment csv file to be uploaded.") @RequestParam("attachmentFile") MultipartFile file, + HttpServletRequest request, HttpServletResponse response) throws TException, IOException, ServletException { + + User sw360User = restControllerHelper.getSw360UserFromAuthentication(); + RequestSummary requestSummary = importExportService.uploadComponentAttachment(sw360User, file, request); + return ResponseEntity.ok(requestSummary); + } } diff --git a/rest/resource-server/src/main/java/org/eclipse/sw360/rest/resourceserver/importexport/Sw360ImportExportService.java b/rest/resource-server/src/main/java/org/eclipse/sw360/rest/resourceserver/importexport/Sw360ImportExportService.java index 3ce26ca151..3d06095d6d 100644 --- a/rest/resource-server/src/main/java/org/eclipse/sw360/rest/resourceserver/importexport/Sw360ImportExportService.java +++ b/rest/resource-server/src/main/java/org/eclipse/sw360/rest/resourceserver/importexport/Sw360ImportExportService.java @@ -13,28 +13,53 @@ import static org.eclipse.sw360.importer.ComponentImportUtils.getFlattenedView; import static org.eclipse.sw360.importer.ComponentImportUtils.getReleasesById; +import static org.eclipse.sw360.datahandler.common.ImportCSV.readAsCSVRecords; +import static org.eclipse.sw360.importer.ComponentImportUtils.convertCSVRecordsToCompCSVRecords; +import static org.eclipse.sw360.importer.ComponentImportUtils.convertCSVRecordsToComponentAttachmentCSVRecords; +import static org.eclipse.sw360.importer.ComponentImportUtils.convertCSVRecordsToReleaseLinkCSVRecords; +import static org.eclipse.sw360.importer.ComponentImportUtils.writeAttachmentsToDatabase; +import static org.eclipse.sw360.importer.ComponentImportUtils.writeReleaseLinksToDatabase; +import static org.eclipse.sw360.importer.ComponentImportUtils.writeToDatabase; import java.io.ByteArrayInputStream; +import java.io.File; +import java.io.FileInputStream; import java.io.IOException; +import java.io.InputStream; +import java.io.InputStreamReader; import java.util.ArrayList; +import java.util.Collection; import java.util.List; import java.util.Map; import java.util.Set; import java.util.function.Consumer; +import org.apache.commons.csv.CSVFormat; +import org.apache.commons.csv.CSVParser; +import org.apache.commons.csv.CSVRecord; import org.apache.thrift.TException; +import org.apache.thrift.protocol.TCompactProtocol; +import org.apache.thrift.protocol.TProtocol; +import org.apache.thrift.transport.THttpClient; +import org.apache.thrift.transport.TTransportException; +import org.apache.tomcat.util.http.fileupload.FileItem; +import org.apache.tomcat.util.http.fileupload.RequestContext; +import org.apache.tomcat.util.http.fileupload.disk.DiskFileItemFactory; import org.eclipse.sw360.datahandler.common.SW360Utils; import org.eclipse.sw360.datahandler.permissions.PermissionUtils; import org.eclipse.sw360.datahandler.thrift.ReleaseRelationship; +import org.eclipse.sw360.datahandler.thrift.RequestSummary; import org.eclipse.sw360.datahandler.thrift.ThriftClients; import org.eclipse.sw360.datahandler.thrift.ThriftUtils; import org.eclipse.sw360.datahandler.thrift.attachments.Attachment; +import org.eclipse.sw360.datahandler.thrift.attachments.AttachmentService; import org.eclipse.sw360.datahandler.thrift.components.Component; import org.eclipse.sw360.datahandler.thrift.components.ComponentService; import org.eclipse.sw360.datahandler.thrift.components.ComponentService.Iface; import org.eclipse.sw360.datahandler.thrift.components.Release; import org.eclipse.sw360.datahandler.thrift.users.User; import org.eclipse.sw360.datahandler.thrift.users.UserGroup; +import org.eclipse.sw360.datahandler.thrift.vendors.VendorService; import org.eclipse.sw360.exporter.CSVExport; import org.eclipse.sw360.importer.ComponentAttachmentCSVRecord; import org.eclipse.sw360.importer.ComponentAttachmentCSVRecordBuilder; @@ -44,18 +69,30 @@ import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Value; import org.springframework.security.access.AccessDeniedException; import org.springframework.stereotype.Service; import org.springframework.util.FileCopyUtils; +import org.springframework.web.multipart.MultipartFile; +import com.fasterxml.jackson.annotation.JsonInclude; +import com.google.common.base.Joiner; +import com.google.common.collect.FluentIterable; import com.google.common.collect.ImmutableList; +import jakarta.servlet.ServletException; +import jakarta.servlet.http.HttpServletRequest; import jakarta.servlet.http.HttpServletResponse; +import jakarta.servlet.http.Part; + import lombok.RequiredArgsConstructor; + @Service @RequiredArgsConstructor(onConstructor = @__(@Autowired)) public class Sw360ImportExportService { + @Value("${sw360.thrift-server-url:http://localhost:8080}") + private String thriftServerUrl; private final Logger log = LoggerFactory.getLogger(this.getClass()); private static final String CONTENT_DISPOSITION = "Content-Disposition"; @@ -86,7 +123,7 @@ public void getDownloadAttachmentTemplate(User sw360User, HttpServletResponse re FileCopyUtils.copy(byteArrayInputStream, response.getOutputStream()); } - public void getDownloadAttachmentInfo(User sw360User, HttpServletResponse response) throws IOException { + public void getDownloadAttachmentInfo(User sw360User, HttpServletResponse response) throws IOException, TTransportException { List> csvRows = new ArrayList<>(); if (!PermissionUtils.isUserAtLeast(UserGroup.ADMIN, sw360User)) { throw new AccessDeniedException("User is not admin"); @@ -139,7 +176,7 @@ private void printAttachments(Set attachments, List } } - private List getComponentDetailedSummaryForExport() { + private List getComponentDetailedSummaryForExport() throws TTransportException{ final ComponentService.Iface componentClient = getThriftComponentClient(); final List componentDetailedSummaryForExport; @@ -153,10 +190,6 @@ private List getComponentDetailedSummaryForExport() { return componentDetailedSummaryForExport; } - private Iface getThriftComponentClient() { - return new ThriftClients().makeComponentClient(); - } - public void getDownloadReleaseSample(User sw360User, HttpServletResponse response) throws TException, IOException { if (!PermissionUtils.isUserAtLeast(UserGroup.ADMIN, sw360User)) { throw new AccessDeniedException("User is not admin"); @@ -185,7 +218,8 @@ public void getDownloadReleaseLink(User sw360User, HttpServletResponse response) } } - ByteArrayInputStream byteArrayInputStream = CSVExport.createCSV(ReleaseLinkCSVRecord.getCSVHeaderIterable(), csvRows); + ByteArrayInputStream byteArrayInputStream = CSVExport.createCSV(ReleaseLinkCSVRecord.getCSVHeaderIterable(), + csvRows); String filename = String.format("ReleaseLinkInfo_%s.csv", SW360Utils.getCreatedOn()); response.setHeader(CONTENT_DISPOSITION, String.format("Release; filename=\"%s\"", filename)); FileCopyUtils.copy(byteArrayInputStream, response.getOutputStream()); @@ -231,7 +265,8 @@ private void printReleaseLinkEntry(Component component, Release release, Release csvRows.add(releaseLinkCSVRecordBuilder.build().getCSVIterable()); } - public void getComponentDetailedExport(User sw360User, HttpServletResponse response) throws TException, IOException { + public void getComponentDetailedExport(User sw360User, HttpServletResponse response) + throws TException, IOException { if (!PermissionUtils.isUserAtLeast(UserGroup.ADMIN, sw360User)) { throw new AccessDeniedException("User is not admin"); } @@ -244,5 +279,85 @@ public void getComponentDetailedExport(User sw360User, HttpServletResponse respo String filename = String.format("ComponentsReleasesVendors_%s.csv", SW360Utils.getCreatedOn()); response.setHeader(CONTENT_DISPOSITION, String.format("Components; filename=\"%s\"", filename)); FileCopyUtils.copy(byteArrayInputStream, response.getOutputStream()); + + } + + @JsonInclude + public RequestSummary uploadComponent(User sw360User, MultipartFile file, HttpServletRequest request, + HttpServletResponse response) throws IOException, TException, ServletException { + if (!PermissionUtils.isUserAtLeast(UserGroup.ADMIN, sw360User)) { + throw new AccessDeniedException("Unable to upload component csv file. User is not admin"); + } + List releaseRecords = getCSVFromRequest(request, "file"); + FluentIterable compCSVRecords = convertCSVRecordsToCompCSVRecords(releaseRecords); + ComponentService.Iface sw360ComponentClient = getThriftComponentClient(); + VendorService.Iface sw360VendorClient = getThriftVendorClient(); + AttachmentService.Iface sw360AttachmentClient = getThriftAttachmentClient(); + RequestSummary requestSummary = writeToDatabase(compCSVRecords, sw360ComponentClient, sw360VendorClient, + sw360AttachmentClient, sw360User); + return requestSummary; + } + + private List getCSVFromRequest(HttpServletRequest request, String fileUploadFormId) + throws IOException, TException, ServletException { + final InputStream stream = getInputStreamFromRequest(request, fileUploadFormId); + return readAsCSVRecords(stream); + } + + private InputStream getInputStreamFromRequest(HttpServletRequest request, String fileUploadFormId) + throws IOException, ServletException { + Collection parts = request.getParts(); + + for (Part part : parts) { + if (!part.getName().equals(fileUploadFormId)) { + return part.getInputStream(); + } + } + throw new IOException("File not found in the request with the specified field name."); + } + + private ComponentService.Iface getThriftComponentClient() throws TTransportException { + THttpClient thriftClient = new THttpClient(thriftServerUrl + "/components/thrift"); + TProtocol protocol = new TCompactProtocol(thriftClient); + return new ComponentService.Client(protocol); + } + + private VendorService.Iface getThriftVendorClient() throws TTransportException { + THttpClient thriftClient = new THttpClient(thriftServerUrl + "/vendors/thrift"); + TProtocol protocol = new TCompactProtocol(thriftClient); + return new VendorService.Client(protocol); + } + + private AttachmentService.Iface getThriftAttachmentClient() throws TTransportException { + THttpClient thriftClient = new THttpClient(thriftServerUrl + "/attachments/thrift"); + TProtocol protocol = new TCompactProtocol(thriftClient); + return new AttachmentService.Client(protocol); + } + + public RequestSummary uploadReleaseLink(User sw360User, MultipartFile file, HttpServletRequest request) + throws IOException, TException, ServletException { + if (!PermissionUtils.isUserAtLeast(UserGroup.ADMIN, sw360User)) { + throw new AccessDeniedException("Unable to upload component csv file. User is not admin"); + } + List releaseLinkRecords = getCSVFromRequest(request, "file"); + FluentIterable csvRecords = convertCSVRecordsToReleaseLinkCSVRecords(releaseLinkRecords); + ComponentService.Iface sw360ComponentClient = getThriftComponentClient(); + final RequestSummary requestSummary = writeReleaseLinksToDatabase(csvRecords, sw360ComponentClient, sw360User); + return requestSummary; + } + + public RequestSummary uploadComponentAttachment(User sw360User, MultipartFile file, HttpServletRequest request) + throws IOException, TException, ServletException { + if (!PermissionUtils.isUserAtLeast(UserGroup.ADMIN, sw360User)) { + throw new AccessDeniedException("Unable to upload component attachment csv file. User is not admin"); + } + List attachmentRecords = getCSVFromRequest(request, "file"); + FluentIterable compCSVRecords = convertCSVRecordsToComponentAttachmentCSVRecords( + attachmentRecords); + ComponentService.Iface sw360ComponentClient = getThriftComponentClient(); + AttachmentService.Iface sw360AttachmentClient = getThriftAttachmentClient(); + final RequestSummary requestSummary = writeAttachmentsToDatabase(compCSVRecords, sw360User, + sw360ComponentClient, sw360AttachmentClient); + return requestSummary; } } diff --git a/rest/resource-server/src/test/java/org/eclipse/sw360/rest/resourceserver/restdocs/ImportExportSpec.java b/rest/resource-server/src/test/java/org/eclipse/sw360/rest/resourceserver/restdocs/ImportExportSpec.java new file mode 100644 index 0000000000..258170ddb3 --- /dev/null +++ b/rest/resource-server/src/test/java/org/eclipse/sw360/rest/resourceserver/restdocs/ImportExportSpec.java @@ -0,0 +1,109 @@ +/* + * Copyright Siemens AG, 2024-2025. + * Part of the SW360 Portal Project. + * + * This program and the accompanying materials are made + * available under the terms of the Eclipse Public License 2.0 + * which is available at https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + */ + + +package org.eclipse.sw360.rest.resourceserver.restdocs; + +import static org.mockito.ArgumentMatchers.any; +import static org.mockito.BDDMockito.given; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; + +import java.io.IOException; + +import org.apache.thrift.TException; +import org.eclipse.sw360.datahandler.thrift.RequestStatus; +import org.eclipse.sw360.datahandler.thrift.RequestSummary; +import org.eclipse.sw360.datahandler.thrift.licenses.LicenseType; +import org.eclipse.sw360.datahandler.thrift.users.User; +import org.eclipse.sw360.rest.resourceserver.TestHelper; +import org.eclipse.sw360.rest.resourceserver.importexport.Sw360ImportExportService; +import org.eclipse.sw360.rest.resourceserver.user.Sw360UserService; +import org.junit.Before; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.springframework.beans.factory.annotation.Value; +import org.springframework.boot.test.mock.mockito.MockBean; +import org.springframework.mock.web.MockMultipartFile; +import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; +import org.springframework.test.web.servlet.request.MockHttpServletRequestBuilder; +import org.springframework.test.web.servlet.request.MockMvcRequestBuilders; + +import jakarta.servlet.ServletException; + +@RunWith(SpringJUnit4ClassRunner.class) +public class ImportExportSpec extends TestRestDocsSpecBase { + + @Value("${sw360.test-user-id}") + private String testUserId; + + @Value("${sw360.test-user-password}") + private String testUserPassword; + + @MockBean + private Sw360ImportExportService importExportService; + + private RequestSummary requestSummary = new RequestSummary(); + + @Before + @SuppressWarnings("unchecked") + public void before() throws TException, IOException, ServletException { + User sw360User = new User(); + sw360User.setId("123456789"); + sw360User.setEmail("admin@sw360.org"); + sw360User.setFullname("John Doe"); + + requestSummary.setRequestStatus(RequestStatus.SUCCESS); + LicenseType licensetype = new LicenseType(); + licensetype.setId("1234"); + licensetype.setLicenseType("wer"); + licensetype.setLicenseTypeId(123); + licensetype.setType("xyz"); + + given(this.importExportService.uploadComponent(any(), any(), any(),any())).willReturn(requestSummary); + given(this.userServiceMock.getUserByEmailOrExternalId("admin@sw360.org")).willReturn(sw360User); + given(this.importExportService.uploadReleaseLink(any(), any(), any())).willReturn(requestSummary); + given(this.importExportService.uploadComponentAttachment(any(), any(), any())).willReturn(requestSummary); + + } + + @Test + public void should_document_upload_component_file() throws Exception { + MockMultipartFile file = new MockMultipartFile("componentFile","file=@/bom.spdx.rdf".getBytes()); + String accessToken = TestHelper.generateAuthHeader(testUserId, testUserPassword); + MockHttpServletRequestBuilder builder = MockMvcRequestBuilders.multipart("/api/importExport/uploadComponent") + .file(file) + .header("Authorization", accessToken) + .queryParam("componentFile", "Must need to attach file."); + this.mockMvc.perform(builder).andExpect(status().isOk()).andDo(this.documentationHandler.document()); + } + + @Test + public void should_document_upload_release_link_file() throws Exception { + MockMultipartFile file = new MockMultipartFile("releaseFile","file=@/bom.spdx.rdf".getBytes()); + String accessToken = TestHelper.generateAuthHeader(testUserId, testUserPassword); + MockHttpServletRequestBuilder builder = MockMvcRequestBuilders.multipart("/api/importExport/uploadRelease") + .file(file) + .header("Authorization", accessToken) + .queryParam("releaseFile", "Must need to attach file."); + this.mockMvc.perform(builder).andExpect(status().isOk()).andDo(this.documentationHandler.document()); + } + + @Test + public void should_document_upload_component_attachment_file() throws Exception { + MockMultipartFile file = new MockMultipartFile("attachmentFile","file=@/bom.spdx.rdf".getBytes()); + String accessToken = TestHelper.generateAuthHeader(testUserId, testUserPassword); + MockHttpServletRequestBuilder builder = MockMvcRequestBuilders.multipart("/api/importExport/componentAttachment") + .file(file) + .header("Authorization", accessToken) + .queryParam("attachmentFile", "Must need to attach file."); + this.mockMvc.perform(builder).andExpect(status().isOk()).andDo(this.documentationHandler.document()); + } +}