Skip to content

Commit

Permalink
feat(Release):Upload Source Code Attachment to Releases through a Sch…
Browse files Browse the repository at this point in the history
…eduled Service

Signed-off-by: afsahsyeda <afsah.syeda@siemens-healthineers.com>
  • Loading branch information
afsahsyeda committed Oct 31, 2023
1 parent c0dbccf commit 00772ab
Show file tree
Hide file tree
Showing 17 changed files with 299 additions and 28 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
import com.google.common.collect.*;

import org.eclipse.sw360.common.utils.BackendUtils;
import org.eclipse.sw360.commonIO.AttachmentFrontendUtils;
import org.eclipse.sw360.components.summary.SummaryType;
import org.eclipse.sw360.datahandler.cloudantclient.DatabaseConnectorCloudant;
import org.eclipse.sw360.datahandler.common.CommonUtils;
Expand Down Expand Up @@ -68,11 +69,16 @@
import org.eclipse.sw360.spdx.SpdxBOMImporterSink;
import org.jetbrains.annotations.NotNull;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import org.spdx.library.InvalidSPDXAnalysisException;

import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
import java.nio.ByteBuffer;
import java.nio.file.*;
import java.util.*;
import java.util.concurrent.TimeUnit;
import java.util.function.Function;
Expand Down Expand Up @@ -112,6 +118,7 @@ public class ComponentDatabaseHandler extends AttachmentAwareDatabaseHandler {
private static final String NO_RELEASE = "Don't have Release created!";
private static final List<String> listComponentName = new ArrayList<>();
private static final Map<String, String> mapReleaseName = new HashMap<>();
public static final List<String> formats = new ArrayList<>(Arrays.asList(SW360Constants.URL_FORMATS.split(",")));

/**
* Connection to the couchDB database
Expand All @@ -124,7 +131,7 @@ public class ComponentDatabaseHandler extends AttachmentAwareDatabaseHandler {
private final PackageRepository packageRepository;
private DatabaseHandlerUtil dbHandlerUtil;
private BulkDeleteUtil bulkDeleteUtil;

private final AttachmentConnector attachmentConnector;
private SvmConnector svmConnector;
private final SpdxDocumentDatabaseHandler spdxDocumentDatabaseHandler;
Expand Down Expand Up @@ -179,7 +186,7 @@ public ComponentDatabaseHandler(Supplier<CloudantClient> httpClient, String dbNa
attachmentConnector = new AttachmentConnector(httpClient, attachmentDbName, durationOf(30, TimeUnit.SECONDS));
DatabaseConnectorCloudant dbChangeLogs = new DatabaseConnectorCloudant(httpClient, DatabaseSettings.COUCH_DB_CHANGE_LOGS);
this.dbHandlerUtil = new DatabaseHandlerUtil(dbChangeLogs);

this.bulkDeleteUtil = new BulkDeleteUtil(this, componentRepository, releaseRepository, projectRepository, moderator, releaseModerator,
attachmentConnector, attachmentDatabaseHandler, dbHandlerUtil);

Expand Down Expand Up @@ -393,6 +400,11 @@ public Component getAccessibleComponent(String id, User user) throws SW360Except
return component;
}

//Used by scheduled upload service to get releases of a component without user info
public Release getRelease(String id) {
return releaseRepository.get(id);
}

public Release getRelease(String id, User user) throws SW360Exception {
return getRelease(id, user, null);
}
Expand Down Expand Up @@ -1443,11 +1455,11 @@ public void recomputeReleaseDependentFields(Component component, String skipThis
updateReleaseDependentFieldsForComponent(component, containedRelease);
}
}

public BulkOperationNode deleteBulkRelease(String releaseId, User user, boolean isPreview) throws SW360Exception {
return bulkDeleteUtil.deleteBulkRelease(releaseId, user, isPreview);
}

public BulkDeleteUtil getBulkDeleteUtil() {
return bulkDeleteUtil;
}
Expand Down Expand Up @@ -1983,6 +1995,11 @@ public Map<String, Component> getAllComponentsIdMap() {
return ThriftUtils.getIdMap(components);
}

public List<Component> getAllComponentsWithVCS() {
final List<Component> components = componentRepository.getComponentsByVCS();
return components;
}

@NotNull
private List<ReleaseLink> iterateReleaseRelationShips(Map<String, ?> relations, String parentNodeId, Deque<String> visitedIds) {
List<ReleaseLink> out = new ArrayList<>();
Expand Down Expand Up @@ -3022,4 +3039,134 @@ private ReleaseNode getReleaseNodes(ReleaseNode releaseNode, User user) {
}
return releaseNode;
}

public RequestStatus uploadSourceCodeAttachmentToReleases() {
List<Component> components = getAllComponentsWithVCS();
Set<String> releasesWithoutSRC = new HashSet<>();
Set<String> updateReleases = new HashSet<>();
log.info(String.format("SRC Upload: Found %d components with VCS", components.size()));

components.forEach(c -> {
String VCS = c.getVcs();
log.info(String.format("SRC Upload: %s %s", c.getId(), VCS));
if (isValidURL(VCS)) {
for (String r_id : c.getReleaseIds()) {
boolean isUploaded = false;
Release r = getRelease(r_id);

if (r.getClearingState() == ClearingState.NEW_CLEARING) {
List<Attachment> sourceAttachments = (r.getAttachments() != null) ? r.getAttachments().stream()
.filter(attachment -> AttachmentType.SOURCE.equals(attachment.getAttachmentType()))
.collect(Collectors.toList()) : Collections.emptyList();

if (sourceAttachments.size() == 0) {
releasesWithoutSRC.add(r.getId());
String version = r.getVersion();
Release originalReleaseData = r.deepCopy();

for (String format : formats) {
String downloadURL = String.format(format, c.getVcs(), version);
if (isValidURL(downloadURL)) {
try {
String destinationDirectory = SW360Constants.SRC_ATTACHMENT_DOWNLOAD_LOCATION;
File file = downloadFile(downloadURL, destinationDirectory);
Attachment attachment = new Attachment()
.setAttachmentType(AttachmentType.SOURCE);
Set<Attachment> src_attachment = new HashSet<>();
src_attachment.add(uploadAttachment(file, attachment));
r.setAttachments(src_attachment);
r.setSourceCodeDownloadurl(downloadURL);
releaseRepository.update(r);
isUploaded = true;
updateReleases.add(r.getId());
// Delete the SRC zip file after the release is updated
file.delete();
break;
} catch (IOException | TException e) {
log.error(
"SRC Upload: Error while downloading the source code zip file for release:"
+ r.getId() + " " + e);
}
}
}
if (isUploaded) {
dbHandlerUtil.addChangeLogs(r, originalReleaseData,
SW360Constants.SRC_ATTACHMENT_UPLOADER_EMAIL, Operation.UPDATE,
attachmentConnector, Lists.newArrayList(), null, null);
}
}
}
}
}
});
if (updateReleases.size() == releasesWithoutSRC.size()) {
log.info(String.format("SRC Upload: updated %d releases", updateReleases.size()));
return RequestStatus.SUCCESS;
} else {
log.error("SRC Upload: Failed to upload SRC attachments for releases: "
+ Sets.difference(releasesWithoutSRC, updateReleases));
return RequestStatus.FAILURE;
}
}

private boolean isValidURL(String url) {
try {
URL urlObj = new URL(url);
HttpURLConnection connection = (HttpURLConnection) urlObj.openConnection();
connection.setRequestMethod("HEAD");
int responseCode = connection.getResponseCode();

if (responseCode == HttpURLConnection.HTTP_OK) {
return true;
} else {
return false;
}
} catch (IOException e) {
log.error("Error while checking the validity of the URL " + url, e);
return false;
}
}

public File downloadFile(String url, String destinationDirectory) throws IOException {
URL fileUrl = new URL(url);
String regex = ".*/([^/]+)/archive/refs/tags/(?:v)?([\\d.]+)\\.zip$";
String fileName = url.replaceAll(regex, "$1-$2.zip");
Path destinationPath = Paths.get(destinationDirectory, fileName);

try (InputStream in = fileUrl.openStream()) {
Files.copy(in, destinationPath, StandardCopyOption.REPLACE_EXISTING);
}
return destinationPath.toFile();
}

public Attachment uploadAttachment(File file, Attachment newAttachment) throws IOException, TException {
String fileName = file.getName();
String contentType = "application/zip";
final AttachmentContent attachmentContent = makeAttachmentContent(fileName, contentType);
FileInputStream inputStream = new FileInputStream(file);
Attachment attachment = new AttachmentFrontendUtils().uploadAttachmentContent(attachmentContent, inputStream, null);

attachment.setSha1(attachmentConnector.getSha1FromAttachmentContentId(attachmentContent.getId()));
attachment.setAttachmentType(AttachmentType.SOURCE);
attachment.setCheckStatus(CheckStatus.NOTCHECKED);
attachment.setCreatedComment("Uploaded by the SW360 scheduled service based on the VCS url of the component");
attachment.setCreatedBy(SW360Constants.SRC_ATTACHMENT_UPLOADER_EMAIL);
return attachment;
}

private AttachmentContent makeAttachmentContent(String filename, String contentType) {
AttachmentContent attachment = new AttachmentContent()
.setContentType(contentType)
.setFilename(filename)
.setOnlyRemote(false);
return makeAttachmentContent(attachment);
}

private AttachmentContent makeAttachmentContent(AttachmentContent content) {
try {
return new AttachmentFrontendUtils().makeAttachmentContent(content);
} catch (TException e) {
throw new RuntimeException(e);
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -147,6 +147,8 @@ public class ComponentRepository extends SummaryAwareRepository<Component> {
" }" +
"}";

private static final String BY_VCS = "function(doc) { if (doc.type == 'component' && doc.vcs) emit(null, doc._id) }";

public ComponentRepository(DatabaseConnectorCloudant db, ReleaseRepository releaseRepository, VendorRepository vendorRepository) {
super(Component.class, db, new ComponentSummary(releaseRepository, vendorRepository));
Map<String, MapReduce> views = new HashMap<String, MapReduce>();
Expand All @@ -165,6 +167,7 @@ public ComponentRepository(DatabaseConnectorCloudant db, ReleaseRepository relea
views.put("bynamelowercase", createMapReduce(BY_NAME_LOWERCASE, null));
views.put("bymainlicense", createMapReduce(BY_MAIN_LICENSE, null));
views.put("byvendor", createMapReduce(BY_VENDOR, null));
views.put("byVCS", createMapReduce(BY_VCS, null));
initStandardDesignDocument(views, db);
}

Expand Down Expand Up @@ -309,4 +312,8 @@ public Map<PaginationData, List<Component>> getRecentComponentsSummary(User user
result.put(pageData, components);
return result;
}

public List<Component> getComponentsByVCS() {
return queryView("byVCS");
}
}
Loading

0 comments on commit 00772ab

Please sign in to comment.