Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for Windows artifacts to Deliverables Analyzer #4137

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,6 @@
import com.github.packageurl.MalformedPackageURLException;
import com.github.packageurl.PackageURL;
import com.github.packageurl.PackageURLBuilder;
import lombok.Value;
import lombok.extern.slf4j.Slf4j;
import org.jboss.pnc.api.deliverablesanalyzer.dto.Artifact;
import org.jboss.pnc.api.deliverablesanalyzer.dto.ArtifactType;
Expand All @@ -29,6 +28,7 @@
import org.jboss.pnc.api.deliverablesanalyzer.dto.FinderResult;
import org.jboss.pnc.api.deliverablesanalyzer.dto.LicenseInfo;
import org.jboss.pnc.api.deliverablesanalyzer.dto.MavenArtifact;
import org.jboss.pnc.api.deliverablesanalyzer.dto.WindowsArtifact;
import org.jboss.pnc.api.dto.Request;
import org.jboss.pnc.api.enums.DeliverableAnalyzerReportLabel;
import org.jboss.pnc.api.enums.LabelOperation;
Expand All @@ -48,7 +48,6 @@
import org.jboss.pnc.enums.RepositoryType;
import org.jboss.pnc.facade.OperationsManager;
import org.jboss.pnc.facade.deliverables.api.AnalysisResult;
import org.jboss.pnc.facade.util.UserService;
import org.jboss.pnc.mapper.api.ArtifactMapper;
import org.jboss.pnc.mapper.api.DeliverableAnalyzerOperationMapper;
import org.jboss.pnc.model.Base32LongID;
Expand Down Expand Up @@ -95,6 +94,7 @@
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
Expand Down Expand Up @@ -543,9 +543,8 @@ private org.jboss.pnc.model.Artifact findOrCreateNotFoundArtifact(
private org.jboss.pnc.model.Artifact createArtifact(
org.jboss.pnc.model.Artifact artifact,
TargetRepository targetRepo) {

artifact.setTargetRepository(targetRepo);
artifact.setPurl(createGenericPurl(artifact.getFilename().toString(), artifact.getSha256()));
artifact.setPurl(createGenericPurl(artifact.getFilename(), artifact.getSha256()));
org.jboss.pnc.model.Artifact savedArtifact = artifactRepository.save(artifact);
targetRepo.getArtifacts().add(savedArtifact);
return savedArtifact;
Expand All @@ -567,17 +566,12 @@ private org.jboss.pnc.model.Artifact mapBrewArtifact(
String nvr,
TargetRepository targetRepository,
User user) {
if (artifact.getArtifactType() != ArtifactType.MAVEN) {
throw new UnsupportedOperationException("Brew artifact " + artifact + " is not Maven!");
}
MavenArtifact mavenArtifact = (MavenArtifact) artifact;

org.jboss.pnc.model.Artifact.Builder builder = mapArtifact(mavenArtifact, user);
builder.identifier(createIdentifier(mavenArtifact));
builder.filename(createFileName(mavenArtifact));
builder.deployPath(createDeployPath(mavenArtifact));
builder.originUrl(createBrewOriginURL(mavenArtifact, nvr));
builder.purl(createPURL(mavenArtifact));
org.jboss.pnc.model.Artifact.Builder builder = mapArtifact(artifact, user);
builder.identifier(createIdentifier(artifact));
builder.filename(createFileName(artifact));
builder.deployPath(createDeployPath(artifact));
builder.originUrl(createBrewOriginURL(artifact, nvr));
builder.purl(createPURL(artifact));
builder.targetRepository(targetRepository);

return builder.build();
Expand All @@ -603,49 +597,82 @@ private org.jboss.pnc.model.Artifact.Builder mapArtifact(Artifact artifact, User
return builder;
}

private static String createDeployPath(MavenArtifact mavenArt) {
String filename = createFileName(mavenArt);
String deployPath = "/" + mavenArt.getGroupId().replace('.', '/') + "/" + mavenArt.getArtifactId() + "/"
+ mavenArt.getVersion() + "/" + filename;
private static String createDeployPath(Artifact artifact) {
String filename = createFileName(artifact);
String deployPath;

if (artifact instanceof MavenArtifact) {
MavenArtifact mavenArtifact = (MavenArtifact) artifact;
deployPath = "/" + mavenArtifact.getGroupId().replace('.', '/') + "/" + mavenArtifact.getArtifactId() + "/"
+ mavenArtifact.getVersion() + "/" + filename;
} else if (artifact instanceof WindowsArtifact) {
deployPath = "/" + filename;
Copy link
Contributor

@janinko janinko Dec 17, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@pkocandr I am not sure how the deploy path would be if we got that artifact via brew pull (is it even possible?) and if this matches.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As I understand the code, this isn't the Maven repository path, e.g., /maven2. This is the original Brew path (see createBrewOriginURL()).

Maven artifacts start with /maven and Windows artifacts start with /win, so I think it just uses the artifact type to start the path. For Windows, since there is no group identifier, etc., the rest of the path is just the filename itself.

Copy link
Contributor

@pkocandr pkocandr Jan 8, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We only have support to pull Maven dependecies from Brew, so it would only be possible if a Windows "artifact" was deployed as Maven artifact in a Maven repo probably as a result of a native build. The deploy path would then normally forllow the groupId artifactId coordinates as any other Maven artifact.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

But, the whole point of this code was is add support for Windows artifacts in addition to Maven artifacts. So, when you say that we only support pulling Maven dependencies from Brew I am not sure that I understand.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am under the impression that this change only applies to Deliverables Analyzer and that it shouldn't matter if other parts of PNC support Maven only.

} else {
throw new IllegalArgumentException("Unsupported artifact type: " + artifact.getArtifactType());
}

return deployPath;
}

private static String createFileName(MavenArtifact mavenArt) {
String filename = mavenArt.getArtifactId() + "-" + mavenArt.getVersion();
if (!Strings.isEmpty(mavenArt.getClassifier())) {
filename += "-" + mavenArt.getClassifier();
private static String createFileName(Artifact artifact) {
String filename;

if (artifact instanceof MavenArtifact) {
MavenArtifact mavenArtifact = (MavenArtifact) artifact;
filename = mavenArtifact.getArtifactId() + "-" + mavenArtifact.getVersion();
if (!Strings.isEmpty(mavenArtifact.getClassifier())) {
filename += "-" + mavenArtifact.getClassifier();
}
filename += "." + mavenArtifact.getType();
} else if (artifact instanceof WindowsArtifact) {
WindowsArtifact windowsArtifact = (WindowsArtifact) artifact;
filename = windowsArtifact.getFilename();
} else {
throw new IllegalArgumentException("Unsupported artifact type: " + artifact.getArtifactType());
}
filename += "." + mavenArt.getType();

return filename;
}

private String createBrewOriginURL(MavenArtifact mavenArt, String nvr) {
private String createBrewOriginURL(Artifact artifact, String nvr) {
String brewContentUrl = globalConfig.getBrewContentUrl();

Matcher matcher = NVR_PATTERN.matcher(nvr);
if (!matcher.matches()) {
throw new IllegalArgumentException("NVR " + nvr + " does not match expected format.");
}
String name = matcher.group(1);
String version = matcher.group(2);
String release = matcher.group(3);

return brewContentUrl + "/" + name + "/" + version + "/" + release + "/maven" + createDeployPath(mavenArt);
return brewContentUrl + "/" + name + "/" + version + "/" + release + "/"
+ artifact.getArtifactType().name().toLowerCase(Locale.ENGLISH) + createDeployPath(artifact);
}

private String createPURL(MavenArtifact mavenArtifact) {
private String createPURL(Artifact artifact) {
try {
PackageURLBuilder purlBuilder = PackageURLBuilder.aPackageURL()
.withType(PackageURL.StandardTypes.MAVEN)
.withNamespace(mavenArtifact.getGroupId())
.withName(mavenArtifact.getArtifactId())
.withVersion(mavenArtifact.getVersion())
.withQualifier(
"type",
StringUtils.isEmpty(mavenArtifact.getType()) ? "jar" : mavenArtifact.getType());

if (!StringUtils.isEmpty(mavenArtifact.getClassifier())) {
purlBuilder.withQualifier("classifier", mavenArtifact.getClassifier());
PackageURLBuilder purlBuilder;
if (artifact instanceof MavenArtifact) {
MavenArtifact mavenArtifact = (MavenArtifact) artifact;
purlBuilder = PackageURLBuilder.aPackageURL()
.withType(PackageURL.StandardTypes.MAVEN)
.withNamespace(mavenArtifact.getGroupId())
.withName(mavenArtifact.getArtifactId())
.withVersion(mavenArtifact.getVersion())
.withQualifier(
"type",
StringUtils.isEmpty(mavenArtifact.getType()) ? "jar" : mavenArtifact.getType());

if (!StringUtils.isEmpty(mavenArtifact.getClassifier())) {
purlBuilder.withQualifier("classifier", mavenArtifact.getClassifier());
}

} else if (artifact instanceof WindowsArtifact) {
WindowsArtifact windowsArtifact = (WindowsArtifact) artifact;
purlBuilder = PackageURLBuilder.aPackageURL()
.withType(PackageURL.StandardTypes.GENERIC)
.withName(windowsArtifact.getName())
.withVersion(windowsArtifact.getVersion());
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Namespace?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@pkocandr Should the namespace be empty? Type is GENERIC as there is no WINDOWS type.

} else {
throw new IllegalArgumentException("Unsupported artifact type: " + artifact.getArtifactType());
}
return purlBuilder.build().toString();
} catch (MalformedPackageURLException e) {
Expand Down Expand Up @@ -673,17 +700,25 @@ private String createGenericPurl(String filename, String sha256) {
}
}

private String createIdentifier(MavenArtifact mavenArtifact) {
return Arrays
.asList(
mavenArtifact.getGroupId(),
mavenArtifact.getArtifactId(),
mavenArtifact.getType(),
mavenArtifact.getVersion(),
mavenArtifact.getClassifier())
.stream()
.filter(Objects::nonNull)
.collect(Collectors.joining(":"));
private String createIdentifier(Artifact artifact) {
if (artifact instanceof MavenArtifact) {
MavenArtifact mavenArtifact = (MavenArtifact) artifact;
return Arrays
.asList(
mavenArtifact.getGroupId(),
mavenArtifact.getArtifactId(),
mavenArtifact.getType(),
mavenArtifact.getVersion(),
mavenArtifact.getClassifier())
.stream()
.filter(Objects::nonNull)
.collect(Collectors.joining(":"));
} else if (artifact instanceof WindowsArtifact) {
WindowsArtifact windowsArtifact = (WindowsArtifact) artifact;
return String.join("-", windowsArtifact.getName(), windowsArtifact.getVersion());
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not sure.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@pkocandr For Windows, I just chose <name>-<version> as the identifier. I wonder if it's similar for NPM. It obviously will not match the Maven identifier format.

} else {
throw new IllegalArgumentException("Unsupported artifact type: " + artifact.getArtifactType());
}
}

private TargetRepository getDistributionRepository(String distURL) {
Expand Down
3 changes: 2 additions & 1 deletion pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -108,7 +108,7 @@
<version.lombok>1.18.22</version.lombok>
<version.mapstruct>1.5.3.Final</version.mapstruct>
<version.pncmetrics>1.1.3</version.pncmetrics>
<version.pnc-api>3.0.5</version.pnc-api>
<version.pnc-api>3.0.6-SNAPSHOT</version.pnc-api>
<version.pnc-common>3.0.1</version.pnc-common>
<version.rex-api>1.0.1</version.rex-api>
<bifrost-client.version>3.0.0</bifrost-client.version>
Expand Down Expand Up @@ -1671,6 +1671,7 @@
<version>2.12.2</version>
<configuration>
<configFile>../eclipse-codeStyle.xml</configFile>
<lineEnding>LF</lineEnding>
</configuration>
<executions>
<execution>
Expand Down