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)

Reply via email to