You are viewing a plain text version of this content. The canonical link for it is here.
Posted to issues@maven.apache.org by "Marc Philipp (Jira)" <ji...@apache.org> on 2020/05/15 15:34:00 UTC

[jira] [Created] (MNG-6906) Whether a core extension can access classes exported by another core extension should not depend on how it is registered

Marc Philipp created MNG-6906:
---------------------------------

             Summary: Whether a core extension can access classes exported by another core extension should not depend on how it is registered
                 Key: MNG-6906
                 URL: https://issues.apache.org/jira/browse/MNG-6906
             Project: Maven
          Issue Type: Bug
          Components: Class Loading
            Reporter: Marc Philipp


We have a Maven core extension (https://search.maven.org/artifact/com.gradle/gradle-enterprise-maven-extension/1.5/jar) that declares two packages as exported in its {{META-INF/maven/extension.xml}}:

{code}
<?xml version="1.0" encoding="UTF-8"?>
<extension>
  <exportedPackages>
    <exportedPackage>com.gradle.maven.extension.api.scan</exportedPackage>
    <exportedPackage>com.gradle.maven.mojo</exportedPackage>
  </exportedPackages>
  <exportedArtifacts>
    <exportedArtifact>com.gradle:gradle-enterprise-maven-extension</exportedArtifact>
  </exportedArtifacts>
</extension>
{code}

The first package ({{com.gradle.maven.extension.api.scan}}) contains a {{BuildScanApi}} interface for which the extension registers a component in an {{EventSpy}} at runtime.

We would now like to consume that component in another core extension (let's call it consuming-extension) that has a {{provided}} dependency to {{com.gradle:gradle-enterprise-maven-extension}} like this:

{code}
BuildScanApi buildScan = (BuildScanApi) session.lookup("com.gradle.maven.extension.api.scan.BuildScanApi");
{code}

However, whether that works depends on how the core extensions are registered. Relevant for us are the following registration locations: {{<maven-home>/lib/ext}}, {{-Dmaven.ext.class.path}}, and {{.mvn/extensions.xml}}.

||consuming-extension||gradle-enterprise-maven-extension||Result||
|{{<maven-home>/lib/ext}}|{{<maven-home>/lib/ext}}|(/)|
|{{.mvn/extensions.xml}}|{{<maven-home>/lib/ext}}|(/)|
|{{-Dmaven.ext.class.path}}|{{<maven-home>/lib/ext}}|(/)|
|{{<maven-home>/lib/ext}}|{{.mvn/extensions.xml}}|(x) {{NoClassDefFoundError}}|
|{{.mvn/extensions.xml}}|{{.mvn/extensions.xml}}|(x) {{NoClassDefFoundError}}|
|{{-Dmaven.ext.class.path}}|{{.mvn/extensions.xml}}|(/)|
|{{<maven-home>/lib/ext}}|{{-Dmaven.ext.class.path}}|(x) {{NoClassDefFoundError}}|
|{{.mvn/extensions.xml}}|{{-Dmaven.ext.class.path}}|(x) {{NoClassDefFoundError}}|
|{{-Dmaven.ext.class.path}}|{{-Dmaven.ext.class.path}}|(/)|

With this workaround, I was able to get it working in all cases:

{code}
ClassRealm extensionRealm = (ClassRealm) this.getClass().getClassLoader();
if (!"maven.ext".equals(extensionRealm.getId())) {
  extensionRealm.getWorld().getRealms().stream()
    .filter(realm -> realm.getId().startsWith("coreExtension>com.gradle:gradle-enterprise-maven-extension:") || realm.getId().equals("maven.ext"))
    .max(comparing((ClassRealm realm) -> realm.getId().length()))
    .ifPresent(realm -> {
      try {
        extensionRealm.importFrom(realm.getId(), "com.gradle.maven.extension.api.scan");
      } catch (Exception e) {
        throw new RuntimeException("Could not import package from realm", e);
      }
    });
}
{code}

Please let me know if I have misunderstood the intention of declaring exported packages in {{extension.xml}}. I was unable to find documentation on it, unfortunately.



--
This message was sent by Atlassian Jira
(v8.3.4#803005)