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 to specify target locations in the pom configuration #3396

Merged
merged 1 commit into from
Jan 22, 2024
Merged
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
40 changes: 37 additions & 3 deletions RELEASE_NOTES.md
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,40 @@ If you are reading this in the browser, then you can quickly jump to specific ve

## 5.0.0 (under development)

### support for embedded target locations

You can already define target definition files in various ways, e.g. as maven artifact or file reference,
now it is also possible to define them as an embedded part of the target platform configuration,
all locations are handled as if they are part of a single target file:

```xml
<plugin>
<groupId>org.eclipse.tycho</groupId>
<artifactId>target-platform-configuration</artifactId>
<version>${tycho-version}</version>
<configuration>
<target>
<!-- one or more location elements like in a target file -->
<location includeDependencyDepth="infinite" includeDependencyScopes="compile" includeSource="true" missingManifest="generate" type="Maven">
<dependencies>
<dependency>
...
</dependency>
</dependencies>
</location>
<location includeAllPlatforms="true" includeMode="slicer" type="InstallableUnit">
<unit id="org.eclipse.license.feature.group" version="2.0.2.v20181016-2210"/>
...
<repository location="https://download.eclipse.org/cbi/updates/license/2.0.2.v20181016-2210"/>
</location>
...
</target>
</configuration>
</plugin>
```

this is especially useful if you need some content only for the build but not in the IDE.

### using javac as the compiler for Tycho

You can now use `javac` as the compiler backend for Tycho by adding the following configuration:
Expand All @@ -19,7 +53,7 @@ You can now use `javac` as the compiler backend for Tycho by adding the followin
<compilerId>javac</compilerId>
</configuration>
</plugin>
```
```


### new `mirror-target-platform`
Expand All @@ -39,7 +73,7 @@ There is a new `mirror-target-platform` that allows to mirror the current target
</execution>
</executions>
</plugin>
```
```

the most usual use-case for this is to transform an existing target-file into a standalone repository.

Expand Down Expand Up @@ -68,7 +102,7 @@ This mojo can be used in two ways:
</execution>
</executions>
</plugin>
```
```


### new `tycho-eclipse-plugin`
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
import java.io.File;
import java.net.URI;

import org.codehaus.plexus.configuration.PlexusConfiguration;
import org.eclipse.tycho.p2.repository.GAV;

/**
Expand All @@ -23,4 +24,5 @@ public class TargetParameterObject {
public GAV target;
public File file;
public URI uri;
public PlexusConfiguration location;
}
Original file line number Diff line number Diff line change
Expand Up @@ -89,6 +89,7 @@ public class TargetPlatformConfigurationMojo extends AbstractMojo {
* <li><code>&lt;file></code> to define a file local to the build</li>
* <li><code>&lt;uri></code> to define a (remote) URI that specifies a target, currently only
* URIs that can be converted to URLs are supported (e.g. file:/.... http://..., )</li>
* <li>{@code <locations>} to define target locations inline</li>
* </ul>
*/
@Parameter(name = DefaultTargetPlatformConfigurationReader.TARGET)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,9 @@

import java.io.File;
import java.net.URI;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.Base64;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
Expand All @@ -28,6 +30,7 @@
import java.util.Set;
import java.util.function.Supplier;

import org.codehaus.plexus.util.xml.Xpp3Dom;
import org.eclipse.equinox.p2.metadata.IRequirement;
import org.eclipse.tycho.ArtifactKey;
import org.eclipse.tycho.OptionalResolutionAction;
Expand Down Expand Up @@ -121,6 +124,8 @@ public enum InjectP2MavenMetadataHandling {

private ReferencedRepositoryMode referencedRepositoryMode = ReferencedRepositoryMode.include;

private List<Xpp3Dom> xmlFragments = new ArrayList<>();

/**
* Returns the list of configured target environments, or the running environment if no
* environments have been specified explicitly.
Expand All @@ -141,6 +146,18 @@ public synchronized List<TargetDefinitionFile> getTargets() {
targets.add(supplier.get().toURI());
iterator.remove();
}
if (!xmlFragments.isEmpty()) {
Xpp3Dom target = new Xpp3Dom("target");
Xpp3Dom locations = new Xpp3Dom(TargetDefinitionFile.ELEMENT_LOCATIONS);
target.addChild(locations);
for (Xpp3Dom location : xmlFragments) {
locations.addChild(new Xpp3Dom(location));
}
String collect = target.toString();
targets.add(URI.create("data:" + TargetDefinitionFile.APPLICATION_TARGET + ";base64,"
+ Base64.getEncoder().encodeToString(collect.getBytes(StandardCharsets.UTF_8))));
xmlFragments.clear();
}
return targets.stream().map(TargetDefinitionFile::read).toList();
}

Expand Down Expand Up @@ -319,4 +336,8 @@ public void setReferencedRepositoryMode(ReferencedRepositoryMode referencedRepos
this.referencedRepositoryMode = referencedRepositoryMode;
}

public void addTargetLocation(Xpp3Dom locationDom) {
xmlFragments.add(locationDom);
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -331,10 +331,10 @@ private void setBREEHeaderSelectionPolicy(TargetPlatformConfiguration result, Xp

/**
* Take the constraints of the configured execution environment into account when resolving
* dependencies or target definitions. These constraints include the list of system packages and the
* <tt>Bundle-RequiredExecutionEnvironment</tt> header. When set to <code>true</code>, the
* dependency resolution verifies that the bundle and all required bundles can be used in an OSGi
* container with the configured execution environment.
* dependencies or target definitions. These constraints include the list of system packages and
* the <tt>Bundle-RequiredExecutionEnvironment</tt> header. When set to <code>true</code>, the
* dependency resolution verifies that the bundle and all required bundles can be used in an
* OSGi container with the configured execution environment.
*/
private void setResolveWithEEContraints(TargetPlatformConfiguration result, Xpp3Dom resolverDom) {
String value = getStringValue(resolverDom.getChild(RESOLVE_WITH_EXECUTION_ENVIRONMENT_CONSTRAINTS));
Expand Down Expand Up @@ -462,6 +462,13 @@ private void setTarget(TargetPlatformConfiguration result, MavenSession session,
}
}
}
Xpp3Dom[] locationsArray = targetDom.getChildren("location");
if (locationsArray != null && locationsArray.length > 0) {
for (Xpp3Dom locationDom : locationsArray) {
result.addTargetLocation(locationDom);
}
}

}

protected void addTargetArtifact(TargetPlatformConfiguration result, MavenSession session, MavenProject project,
Expand Down Expand Up @@ -592,9 +599,9 @@ private static String getStringValue(Xpp3Dom element) {
*
* @param project
* @param targetFile
* the target file to check
* the target file to check
* @param otherTargetFiles
* other target files to take into account
* other target files to take into account
* @return <code>true</code> if the target file is the primary artifact, <code>false</code>
* otherwise
*/
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,14 +21,17 @@
package org.eclipse.tycho.targetplatform;

import java.io.BufferedOutputStream;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.StringReader;
import java.net.MalformedURLException;
import java.net.URI;
import java.text.MessageFormat;
import java.util.ArrayList;
import java.util.Base64;
import java.util.Collection;
import java.util.Collections;
import java.util.LinkedHashSet;
Expand Down Expand Up @@ -65,7 +68,8 @@

public final class TargetDefinitionFile implements TargetDefinition {

private static final Map<URI, TargetDefinitionFile> FILE_CACHE = new ConcurrentHashMap<>();
public static final String ELEMENT_LOCATIONS = "locations";
private static final Map<URI, TargetDefinitionFile> FILE_CACHE = new ConcurrentHashMap<>();
//just for information purpose
private final String origin;

Expand All @@ -75,6 +79,7 @@ public final class TargetDefinitionFile implements TargetDefinition {

private String targetEE;
public static final String FILE_EXTENSION = ".target";
public static final String APPLICATION_TARGET = "application/target";

private abstract static class AbstractPathLocation implements TargetDefinition.PathLocation {
private String path;
Expand Down Expand Up @@ -554,8 +559,8 @@ public static TargetDefinitionFile read(URI uri) {
try {
return FILE_CACHE.computeIfAbsent(uri.normalize(), key -> {
try {
try (InputStream input = uri.toURL().openStream()) {
return parse(parseDocument(input), uri.toASCIIString());
try (InputStream input = openTargetStream(uri)) {
return parse(parseDocument(input), getOrigin(uri));
} catch (ParserConfigurationException e) {
throw new TargetDefinitionSyntaxException("No valid XML parser: " + e.getMessage(), e);
} catch (SAXException e) {
Expand All @@ -573,6 +578,31 @@ public static TargetDefinitionFile read(URI uri) {
}
}

private static String getOrigin(URI uri) {
if (isDataUrl(uri)) {
return "<embedded>";
}
return uri.toASCIIString();
}

private static InputStream openTargetStream(URI uri) throws IOException, MalformedURLException {
if (isDataUrl(uri)) {
String rawPath = uri.toASCIIString();
int indexOf = rawPath.indexOf(',');
if (indexOf > -1) {
String data = rawPath.substring(indexOf + 1);
return new ByteArrayInputStream(Base64.getDecoder().decode(data));
} else {
throw new MalformedURLException("invalid data url!");
}
}
return uri.toURL().openStream();
}

private static boolean isDataUrl(URI uri) {
return "data".equals(uri.getScheme());
}

public static TargetDefinitionFile parse(Document document, String origin) {
return new TargetDefinitionFile(document, origin);
}
Expand Down Expand Up @@ -618,7 +648,7 @@ public static boolean isTargetFile(File file) {

private static List<? extends TargetDefinition.Location> parseLocations(Element dom) {
ArrayList<TargetDefinition.Location> locations = new ArrayList<>();
Element locationsDom = getChild(dom, "locations");
Element locationsDom = getChild(dom, ELEMENT_LOCATIONS);
if (locationsDom != null) {
for (Element locationDom : getChildren(locationsDom, "location")) {
String type = locationDom.getAttribute("type");
Expand Down
Loading