diff --git a/RELEASE_NOTES.md b/RELEASE_NOTES.md index bd78fb19ce..0d2cb2c38e 100644 --- a/RELEASE_NOTES.md +++ b/RELEASE_NOTES.md @@ -6,6 +6,25 @@ If you are reading this in the browser, then you can quickly jump to specific ve ## 5.0.0 (under development) +### support for PDE Api Tools Annotations + +Tycho now supports PDE Api Tools Annotations to be added to the project automatically. + +To enable this add + +```xml + + org.eclipse.tycho + tycho-apitools-plugin + ${tycho-version} + +``` + +to your project and make sure it has the `org.eclipse.pde.api.tools.apiAnalysisNature` nature enabled in the `.project` file, for details how to use these see: + +- https://help.eclipse.org/latest/topic/org.eclipse.pde.doc.user/reference/api-tooling/api_javadoc_tags.htm +- https://help.eclipse.org/latest/topic/org.eclipse.pde.doc.user/reference/api/org/eclipse/pde/api/tools/annotations/package-summary.html + ### new tycho-repository-plugin Tycho now contains a new `tycho-repository-plugin` that can be used to package OSGi repositories. diff --git a/tycho-apitools-plugin/src/main/java/org/eclipse/tycho/apitools/ApiAnalysisMojo.java b/tycho-apitools-plugin/src/main/java/org/eclipse/tycho/apitools/ApiAnalysisMojo.java index 8b0cc866f4..3e83767c99 100644 --- a/tycho-apitools-plugin/src/main/java/org/eclipse/tycho/apitools/ApiAnalysisMojo.java +++ b/tycho-apitools-plugin/src/main/java/org/eclipse/tycho/apitools/ApiAnalysisMojo.java @@ -138,7 +138,7 @@ public void execute() throws MojoExecutionException, MojoFailureException { } Optional eclipseProject = projectManager.getEclipseProject(project); if (eclipseProject.isEmpty() - || !eclipseProject.get().hasNature("org.eclipse.pde.api.tools.apiAnalysisNature")) { + || !eclipseProject.get().hasNature(ApiPlugin.NATURE_ID)) { return; } diff --git a/tycho-apitools-plugin/src/main/java/org/eclipse/tycho/apitools/ApiAnnotationsClasspathContributor.java b/tycho-apitools-plugin/src/main/java/org/eclipse/tycho/apitools/ApiAnnotationsClasspathContributor.java new file mode 100644 index 0000000000..e1560017db --- /dev/null +++ b/tycho-apitools-plugin/src/main/java/org/eclipse/tycho/apitools/ApiAnnotationsClasspathContributor.java @@ -0,0 +1,60 @@ +/******************************************************************************* + * Copyright (c) 2023 Christoph Läubrich and others. + * This program and the accompanying materials + * are made available under the terms of the Eclipse Public License 2.0 + * which accompanies this distribution, and is available at + * https://www.eclipse.org/legal/epl-2.0/ + * + * SPDX-License-Identifier: EPL-2.0 + * + * Contributors: + * Christoph Läubrich - initial API and implementation + *******************************************************************************/ +package org.eclipse.tycho.apitools; + +import java.util.Optional; + +import javax.inject.Inject; + +import org.apache.maven.SessionScoped; +import org.apache.maven.execution.MavenSession; +import org.apache.maven.project.MavenProject; +import org.codehaus.plexus.component.annotations.Component; +import org.eclipse.pde.api.tools.internal.provisional.ApiPlugin; +import org.eclipse.tycho.classpath.ClasspathContributor; +import org.eclipse.tycho.core.TychoProjectManager; +import org.eclipse.tycho.core.osgitools.AbstractSpecificationClasspathContributor; +import org.eclipse.tycho.model.project.EclipseProject; +import org.osgi.framework.VersionRange; + +@Component(role = ClasspathContributor.class, hint = "apitools-annotations") +@SessionScoped +public class ApiAnnotationsClasspathContributor extends AbstractSpecificationClasspathContributor { + + private static final String PACKAGE_NAME = "org.eclipse.pde.api.tools.annotations"; + private static final String GROUP_ID = "org.eclipse.pde"; + private static final String ARTIFACT_ID = "org.eclipse.pde.api.tools.annotations"; + private static final VersionRange VERSION = new VersionRange("[1,2)"); + private TychoProjectManager projectManager; + + @Inject + public ApiAnnotationsClasspathContributor(MavenSession session, TychoProjectManager projectManager) { + super(session, PACKAGE_NAME, GROUP_ID, ARTIFACT_ID); + this.projectManager = projectManager; + } + + @Override + protected VersionRange getSpecificationVersion(MavenProject project) { + return VERSION; + } + + @Override + protected boolean isValidProject(MavenProject project) { + Optional eclipseProject = projectManager.getEclipseProject(project); + if (eclipseProject.isPresent()) { + return eclipseProject.get().hasNature(ApiPlugin.NATURE_ID); + } + return false; + } + +} diff --git a/tycho-core/src/main/java/org/eclipse/tycho/core/osgitools/AbstractSpecificationClasspathContributor.java b/tycho-core/src/main/java/org/eclipse/tycho/core/osgitools/AbstractSpecificationClasspathContributor.java index e89bcce6da..0dd99077d8 100644 --- a/tycho-core/src/main/java/org/eclipse/tycho/core/osgitools/AbstractSpecificationClasspathContributor.java +++ b/tycho-core/src/main/java/org/eclipse/tycho/core/osgitools/AbstractSpecificationClasspathContributor.java @@ -66,18 +66,24 @@ protected AbstractSpecificationClasspathContributor(MavenSession session, String @Override public final List getAdditionalClasspathEntries(MavenProject project, String scope) { - VersionRange specificationVersion = getSpecificationVersion(project); - Optional mavenBundle = findBundle(project, specificationVersion); - if (mavenBundle.isPresent()) { - ResolvedArtifactKey resolved = mavenBundle.get(); - logger.debug("Resolved " + packageName + " to " + resolved.getId() + " " + resolved.getVersion() + " @ " - + resolved.getLocation()); - return List.of(new DefaultClasspathEntry(resolved, List.of(accessRule))); + if (isValidProject(project)) { + VersionRange specificationVersion = getSpecificationVersion(project); + Optional mavenBundle = findBundle(project, specificationVersion); + if (mavenBundle.isPresent()) { + ResolvedArtifactKey resolved = mavenBundle.get(); + logger.debug("Resolved " + packageName + " to " + resolved.getId() + " " + resolved.getVersion() + " @ " + + resolved.getLocation()); + return List.of(new DefaultClasspathEntry(resolved, List.of(accessRule))); + } + logger.warn("Cannot resolve specification package " + packageName + ", classpath might be incomplete"); } - logger.warn("Cannot resolve specification package " + packageName + ", classpath might be incomplete"); return Collections.emptyList(); } + protected boolean isValidProject(MavenProject project) { + return true; + } + protected Optional findBundle(MavenProject project, VersionRange specificationVersion) { Optional mavenBundle = mavenBundleResolver.resolveMavenBundle(project, session, MavenArtifactKey.of(PublisherHelper.CAPABILITY_NS_JAVA_PACKAGE, packageName, diff --git a/tycho-its/projects/api-tools/annotations/bundle2/.classpath b/tycho-its/projects/api-tools/annotations/bundle2/.classpath new file mode 100644 index 0000000000..8af7df3db4 --- /dev/null +++ b/tycho-its/projects/api-tools/annotations/bundle2/.classpath @@ -0,0 +1,7 @@ + + + + + + + diff --git a/tycho-its/projects/api-tools/annotations/bundle2/.project b/tycho-its/projects/api-tools/annotations/bundle2/.project new file mode 100644 index 0000000000..df2962ed9f --- /dev/null +++ b/tycho-its/projects/api-tools/annotations/bundle2/.project @@ -0,0 +1,34 @@ + + + api-bundle-2 + + + + + + org.eclipse.jdt.core.javabuilder + + + + + org.eclipse.pde.ManifestBuilder + + + + + org.eclipse.pde.SchemaBuilder + + + + + org.eclipse.pde.api.tools.apiAnalysisBuilder + + + + + + org.eclipse.jdt.core.javanature + org.eclipse.pde.PluginNature + org.eclipse.pde.api.tools.apiAnalysisNature + + diff --git a/tycho-its/projects/api-tools/annotations/bundle2/META-INF/MANIFEST.MF b/tycho-its/projects/api-tools/annotations/bundle2/META-INF/MANIFEST.MF new file mode 100644 index 0000000000..79329a422c --- /dev/null +++ b/tycho-its/projects/api-tools/annotations/bundle2/META-INF/MANIFEST.MF @@ -0,0 +1,8 @@ +Manifest-Version: 1.0 +Bundle-ManifestVersion: 2 +Bundle-Name: Plugin with API +Bundle-SymbolicName: api-bundle-2 +Bundle-Version: 0.0.1.qualifier +Require-Capability: osgi.ee;filter:="(&(osgi.ee=JavaSE)(version=11))" +Export-Package: bundle +Automatic-Module-Name: bundle diff --git a/tycho-its/projects/api-tools/annotations/bundle2/build.properties b/tycho-its/projects/api-tools/annotations/bundle2/build.properties new file mode 100644 index 0000000000..12b49e3215 --- /dev/null +++ b/tycho-its/projects/api-tools/annotations/bundle2/build.properties @@ -0,0 +1,4 @@ +source.. = src/ +output.. = bin/ +bin.includes = .,\ + META-INF/ diff --git a/tycho-its/projects/api-tools/annotations/bundle2/pom.xml b/tycho-its/projects/api-tools/annotations/bundle2/pom.xml new file mode 100644 index 0000000000..455407d576 --- /dev/null +++ b/tycho-its/projects/api-tools/annotations/bundle2/pom.xml @@ -0,0 +1,23 @@ + + + 4.0.0 + org.eclipse.tycho.tycho-its + api-bundle-2 + 0.0.1-SNAPSHOT + eclipse-plugin + + + + org.eclipse.tycho + tycho-maven-plugin + ${tycho-version} + true + + + org.eclipse.tycho + tycho-apitools-plugin + ${tycho-version} + + + + diff --git a/tycho-its/projects/api-tools/annotations/bundle2/src/bundle/ApiInterface2.java b/tycho-its/projects/api-tools/annotations/bundle2/src/bundle/ApiInterface2.java new file mode 100644 index 0000000000..83e6639045 --- /dev/null +++ b/tycho-its/projects/api-tools/annotations/bundle2/src/bundle/ApiInterface2.java @@ -0,0 +1,10 @@ +package bundle; + +import org.eclipse.pde.api.tools.annotations.NoExtend; + +@NoExtend +public interface ApiInterface2 { + + void sayHello(); + +} diff --git a/tycho-its/src/test/java/org/eclipse/tycho/test/apitools/ApiToolsTest.java b/tycho-its/src/test/java/org/eclipse/tycho/test/apitools/ApiToolsTest.java index 9546933741..3b8f936e33 100644 --- a/tycho-its/src/test/java/org/eclipse/tycho/test/apitools/ApiToolsTest.java +++ b/tycho-its/src/test/java/org/eclipse/tycho/test/apitools/ApiToolsTest.java @@ -142,4 +142,11 @@ public void testSingleJar() throws Exception { verifier.executeGoals(List.of("clean", "verify")); verifier.verifyErrorFreeLog(); } + + @Test + public void testAnnotations() throws Exception { + Verifier verifier = getVerifier("api-tools/annotations", true, true); + verifier.executeGoals(List.of("clean", "verify")); + verifier.verifyErrorFreeLog(); + } }