d6a86f
From 0fd7b0d19bd96456292585e883e3ba2ebbaf579b Mon Sep 17 00:00:00 2001
d6a86f
From: Mikolaj Izdebski <mizdebsk@redhat.com>
d6a86f
Date: Wed, 21 Jun 2017 10:21:10 +0200
d6a86f
Subject: [PATCH] Fix installer plugin loading
d6a86f
d6a86f
---
d6a86f
 .../install/impl/ArtifactInstallerFactory.java     |  97 ++++++--
d6a86f
 .../tools/install/impl/IsolatedClassRealm.java     | 245 +++++++++++++++++++++
d6a86f
 2 files changed, 320 insertions(+), 22 deletions(-)
d6a86f
 create mode 100644 xmvn-tools/xmvn-install/src/main/java/org/fedoraproject/xmvn/tools/install/impl/IsolatedClassRealm.java
d6a86f
d6a86f
diff --git a/xmvn-tools/xmvn-install/src/main/java/org/fedoraproject/xmvn/tools/install/impl/ArtifactInstallerFactory.java b/xmvn-tools/xmvn-install/src/main/java/org/fedoraproject/xmvn/tools/install/impl/ArtifactInstallerFactory.java
d6a86f
index 7a80571..e6a9a2d 100644
d6a86f
--- a/xmvn-tools/xmvn-install/src/main/java/org/fedoraproject/xmvn/tools/install/impl/ArtifactInstallerFactory.java
d6a86f
+++ b/xmvn-tools/xmvn-install/src/main/java/org/fedoraproject/xmvn/tools/install/impl/ArtifactInstallerFactory.java
d6a86f
@@ -15,8 +15,17 @@
d6a86f
  */
d6a86f
 package org.fedoraproject.xmvn.tools.install.impl;
d6a86f
 
d6a86f
+import java.io.BufferedReader;
d6a86f
+import java.io.IOException;
d6a86f
+import java.io.InputStream;
d6a86f
+import java.io.InputStreamReader;
d6a86f
+import java.nio.file.Files;
d6a86f
+import java.nio.file.Path;
d6a86f
+import java.nio.file.Paths;
d6a86f
 import java.util.Arrays;
d6a86f
-import java.util.Collection;
d6a86f
+import java.util.LinkedHashMap;
d6a86f
+import java.util.List;
d6a86f
+import java.util.Map;
d6a86f
 import java.util.Properties;
d6a86f
 
d6a86f
 import org.slf4j.Logger;
d6a86f
@@ -35,49 +44,93 @@ class ArtifactInstallerFactory
d6a86f
 
d6a86f
     private final ArtifactInstaller defaultArtifactInstaller;
d6a86f
 
d6a86f
-    private final ArtifactInstaller eclipseArtifactInstaller;
d6a86f
+    private final IsolatedClassRealm pluginRealm;
d6a86f
 
d6a86f
-    private static ArtifactInstaller loadPlugin( String className )
d6a86f
+    private final Map<String, ArtifactInstaller> cachedPluginsByType = new LinkedHashMap<>();
d6a86f
+
d6a86f
+    private final Map<String, ArtifactInstaller> cachedPluginsByImplClass = new LinkedHashMap<>();
d6a86f
+
d6a86f
+    private ArtifactInstaller tryLoadPlugin( String type )
d6a86f
     {
d6a86f
+        if ( cachedPluginsByType.containsKey( type ) )
d6a86f
+            return cachedPluginsByType.get( type );
d6a86f
+
d6a86f
         try
d6a86f
         {
d6a86f
-            return (ArtifactInstaller) ArtifactInstallerFactory.class.getClassLoader().loadClass( className ).newInstance();
d6a86f
+            String resourceName = ArtifactInstaller.class.getCanonicalName() + "/" + type;
d6a86f
+            InputStream resourceStream = pluginRealm != null ? pluginRealm.getResourceAsStream( resourceName ) : null;
d6a86f
+            if ( resourceStream == null )
d6a86f
+            {
d6a86f
+                logger.debug( "No XMvn Installer plugin found for packaging type {}", type );
d6a86f
+                cachedPluginsByType.put( type, null );
d6a86f
+                return null;
d6a86f
+            }
d6a86f
+
d6a86f
+            String pluginImplClass;
d6a86f
+            try ( BufferedReader resourceReader = new BufferedReader( new InputStreamReader( resourceStream ) ) )
d6a86f
+            {
d6a86f
+                pluginImplClass = resourceReader.readLine();
d6a86f
+            }
d6a86f
+
d6a86f
+            ArtifactInstaller pluggedInInstaller = cachedPluginsByImplClass.get( pluginImplClass );
d6a86f
+            if ( pluggedInInstaller == null )
d6a86f
+            {
d6a86f
+                pluggedInInstaller = (ArtifactInstaller) pluginRealm.loadClass( pluginImplClass ).newInstance();
d6a86f
+                cachedPluginsByImplClass.put( pluginImplClass, pluggedInInstaller );
d6a86f
+            }
d6a86f
+
d6a86f
+            cachedPluginsByType.put( type, pluggedInInstaller );
d6a86f
+            return pluggedInInstaller;
d6a86f
         }
d6a86f
-        catch ( ReflectiveOperationException e )
d6a86f
+        catch ( IOException | ReflectiveOperationException e )
d6a86f
         {
d6a86f
-            return null;
d6a86f
+            throw new RuntimeException( "Unable to load XMvn Installer plugin for packaging type " + type, e );
d6a86f
         }
d6a86f
     }
d6a86f
 
d6a86f
     public ArtifactInstallerFactory( Configurator configurator )
d6a86f
     {
d6a86f
         defaultArtifactInstaller = new DefaultArtifactInstaller( configurator );
d6a86f
-        // FIXME Don't hardcode plugin class name
d6a86f
-        eclipseArtifactInstaller = loadPlugin( "org.fedoraproject.p2.xmvn.EclipseArtifactInstaller" );
d6a86f
+
d6a86f
+        Path pluginDir = Paths.get( "/usr/share/xmvn/lib/installer" );
d6a86f
+        if ( Files.isDirectory( pluginDir ) )
d6a86f
+        {
d6a86f
+            ClassLoader parentClassLoader = ArtifactInstallerFactory.class.getClassLoader();
d6a86f
+            pluginRealm = new IsolatedClassRealm( parentClassLoader );
d6a86f
+            pluginRealm.addJarDirectory( pluginDir );
d6a86f
+            PLUGIN_IMPORTS.forEach( pluginRealm::importPackage );
d6a86f
+        }
d6a86f
+        else
d6a86f
+        {
d6a86f
+            pluginRealm = null;
d6a86f
+        }
d6a86f
     }
d6a86f
 
d6a86f
     /**
d6a86f
-     * List of Tycho pacgkaging types.
d6a86f
+     * List of packages imported from XMvn Installer class loader to plug-in realms.
d6a86f
      */
d6a86f
-    private static final Collection<String> ECLIPSE_PACKAGING_TYPES = Arrays.asList( "eclipse-plugin", //
d6a86f
-                                                                                     "eclipse-test-plugin", //
d6a86f
-                                                                                     "eclipse-feature", //
d6a86f
-                                                                                     "eclipse-repository", //
d6a86f
-                                                                                     "eclipse-application", //
d6a86f
-                                                                                     "eclipse-update-site", //
d6a86f
-                                                                                     "eclipse-target-definition" );
d6a86f
+    private static final List<String> PLUGIN_IMPORTS = Arrays.asList( // XMvn API
d6a86f
+                                                                      "org.fedoraproject.xmvn.artifact", //
d6a86f
+                                                                      "org.fedoraproject.xmvn.config", //
d6a86f
+                                                                      "org.fedoraproject.xmvn.deployer", //
d6a86f
+                                                                      "org.fedoraproject.xmvn.locator", //
d6a86f
+                                                                      "org.fedoraproject.xmvn.metadata", //
d6a86f
+                                                                      "org.fedoraproject.xmvn.resolver", //
d6a86f
+                                                                      // XMvn Installer SPI
d6a86f
+                                                                      "org.fedoraproject.xmvn.tools.install", //
d6a86f
+                                                                      // SLF4J API
d6a86f
+                                                                      "org.slf4j" //
d6a86f
+    );
d6a86f
 
d6a86f
     @SuppressWarnings( "unused" )
d6a86f
     public ArtifactInstaller getInstallerFor( Artifact artifact, Properties properties )
d6a86f
     {
d6a86f
         String type = properties.getProperty( "type" );
d6a86f
-        if ( type != null && ECLIPSE_PACKAGING_TYPES.contains( type ) )
d6a86f
+        if ( type != null )
d6a86f
         {
d6a86f
-            if ( eclipseArtifactInstaller != null )
d6a86f
-                return eclipseArtifactInstaller;
d6a86f
-
d6a86f
-            logger.error( "Unable to load XMvn P2 plugin, Eclipse artifact installation will be impossible" );
d6a86f
-            throw new RuntimeException( "Unable to load XMvn P2 plugin" );
d6a86f
+            ArtifactInstaller pluggedInInstaller = tryLoadPlugin( type );
d6a86f
+            if ( pluggedInInstaller != null )
d6a86f
+                return pluggedInInstaller;
d6a86f
         }
d6a86f
 
d6a86f
         return defaultArtifactInstaller;
d6a86f
diff --git a/xmvn-tools/xmvn-install/src/main/java/org/fedoraproject/xmvn/tools/install/impl/IsolatedClassRealm.java b/xmvn-tools/xmvn-install/src/main/java/org/fedoraproject/xmvn/tools/install/impl/IsolatedClassRealm.java
d6a86f
new file mode 100644
d6a86f
index 0000000..3324604
d6a86f
--- /dev/null
d6a86f
+++ b/xmvn-tools/xmvn-install/src/main/java/org/fedoraproject/xmvn/tools/install/impl/IsolatedClassRealm.java
d6a86f
@@ -0,0 +1,245 @@
d6a86f
+/*-
d6a86f
+ * Copyright (c) 2014-2017 Red Hat, Inc.
d6a86f
+ *
d6a86f
+ * Licensed under the Apache License, Version 2.0 (the "License");
d6a86f
+ * you may not use this file except in compliance with the License.
d6a86f
+ * You may obtain a copy of the License at
d6a86f
+ *
d6a86f
+ *     http://www.apache.org/licenses/LICENSE-2.0
d6a86f
+ *
d6a86f
+ * Unless required by applicable law or agreed to in writing, software
d6a86f
+ * distributed under the License is distributed on an "AS IS" BASIS,
d6a86f
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
d6a86f
+ * See the License for the specific language governing permissions and
d6a86f
+ * limitations under the License.
d6a86f
+ */
d6a86f
+package org.fedoraproject.xmvn.tools.install.impl;
d6a86f
+
d6a86f
+import java.io.IOException;
d6a86f
+import java.net.MalformedURLException;
d6a86f
+import java.net.URL;
d6a86f
+import java.net.URLClassLoader;
d6a86f
+import java.nio.file.DirectoryStream;
d6a86f
+import java.nio.file.Files;
d6a86f
+import java.nio.file.Path;
d6a86f
+import java.util.Collection;
d6a86f
+import java.util.Collections;
d6a86f
+import java.util.Enumeration;
d6a86f
+import java.util.HashSet;
d6a86f
+import java.util.LinkedHashSet;
d6a86f
+import java.util.Set;
d6a86f
+
d6a86f
+/**
d6a86f
+ * A generic, isolated class loader.
d6a86f
+ * 

d6a86f
+ * This class loader has its own classpath, separate from the primary Java classpath. It has a parent class loader, to
d6a86f
+ * which it delegates loading a set of imported classes. All other classes are loaded from its own classpath.
d6a86f
+ * 
d6a86f
+ * @author Mikolaj Izdebski
d6a86f
+ */
d6a86f
+class IsolatedClassRealm
d6a86f
+    extends URLClassLoader
d6a86f
+{
d6a86f
+    static
d6a86f
+    {
d6a86f
+        registerAsParallelCapable();
d6a86f
+    }
d6a86f
+
d6a86f
+    private final ClassLoader parent;
d6a86f
+
d6a86f
+    private final Set<String> imports = new HashSet<>();
d6a86f
+
d6a86f
+    private final Set<String> importsAll = new HashSet<>();
d6a86f
+
d6a86f
+    public IsolatedClassRealm( ClassLoader parent )
d6a86f
+    {
d6a86f
+        super( new URL[0], null );
d6a86f
+        this.parent = parent;
d6a86f
+    }
d6a86f
+
d6a86f
+    public void addJar( Path jar )
d6a86f
+    {
d6a86f
+        try
d6a86f
+        {
d6a86f
+            addURL( jar.toUri().toURL() );
d6a86f
+        }
d6a86f
+        catch ( MalformedURLException e )
d6a86f
+        {
d6a86f
+            throw new RuntimeException( e );
d6a86f
+        }
d6a86f
+    }
d6a86f
+
d6a86f
+    public void addJarDirectory( Path dir )
d6a86f
+    {
d6a86f
+        try ( DirectoryStream<Path> stream = Files.newDirectoryStream( dir, "*.jar" ) )
d6a86f
+        {
d6a86f
+            for ( Path path : stream )
d6a86f
+            {
d6a86f
+                addJar( path );
d6a86f
+            }
d6a86f
+        }
d6a86f
+        catch ( IOException e )
d6a86f
+        {
d6a86f
+            throw new RuntimeException( e );
d6a86f
+        }
d6a86f
+    }
d6a86f
+
d6a86f
+    public void importPackage( String packageName )
d6a86f
+    {
d6a86f
+        imports.add( packageName );
d6a86f
+    }
d6a86f
+
d6a86f
+    public void importAllPackages( String packageName )
d6a86f
+    {
d6a86f
+        importsAll.add( packageName );
d6a86f
+    }
d6a86f
+
d6a86f
+    boolean isImported( String name )
d6a86f
+    {
d6a86f
+        int index = name.lastIndexOf( '/' );
d6a86f
+
d6a86f
+        if ( index >= 0 )
d6a86f
+        {
d6a86f
+            name = name.replace( '/', '.' );
d6a86f
+        }
d6a86f
+        else
d6a86f
+        {
d6a86f
+            index = Math.max( name.lastIndexOf( '.' ), 0 );
d6a86f
+        }
d6a86f
+
d6a86f
+        String namespace = name.substring( 0, index );
d6a86f
+
d6a86f
+        if ( imports.contains( namespace ) )
d6a86f
+            return true;
d6a86f
+
d6a86f
+        while ( !namespace.isEmpty() )
d6a86f
+        {
d6a86f
+            if ( importsAll.contains( namespace ) )
d6a86f
+                return true;
d6a86f
+
d6a86f
+            namespace = namespace.substring( 0, Math.max( namespace.lastIndexOf( '.' ), 0 ) );
d6a86f
+        }
d6a86f
+
d6a86f
+        return false;
d6a86f
+    }
d6a86f
+
d6a86f
+    @Override
d6a86f
+    public Class loadClass( String name )
d6a86f
+        throws ClassNotFoundException
d6a86f
+    {
d6a86f
+        return loadClass( name, false );
d6a86f
+    }
d6a86f
+
d6a86f
+    @Override
d6a86f
+    protected Class loadClass( String name, boolean resolve )
d6a86f
+        throws ClassNotFoundException
d6a86f
+    {
d6a86f
+        if ( isImported( name ) )
d6a86f
+        {
d6a86f
+            try
d6a86f
+            {
d6a86f
+                return parent.loadClass( name );
d6a86f
+            }
d6a86f
+            catch ( ClassNotFoundException e )
d6a86f
+            {
d6a86f
+            }
d6a86f
+        }
d6a86f
+
d6a86f
+        try
d6a86f
+        {
d6a86f
+            return super.loadClass( name, resolve );
d6a86f
+        }
d6a86f
+        catch ( ClassNotFoundException e )
d6a86f
+        {
d6a86f
+        }
d6a86f
+
d6a86f
+        synchronized ( getClassLoadingLock( name ) )
d6a86f
+        {
d6a86f
+            Class clazz = findLoadedClass( name );
d6a86f
+            if ( clazz != null )
d6a86f
+            {
d6a86f
+                return clazz;
d6a86f
+            }
d6a86f
+
d6a86f
+            try
d6a86f
+            {
d6a86f
+                return super.findClass( name );
d6a86f
+            }
d6a86f
+            catch ( ClassNotFoundException e )
d6a86f
+            {
d6a86f
+            }
d6a86f
+        }
d6a86f
+
d6a86f
+        throw new ClassNotFoundException( name );
d6a86f
+    }
d6a86f
+
d6a86f
+    @Override
d6a86f
+    protected Class findClass( String name )
d6a86f
+        throws ClassNotFoundException
d6a86f
+    {
d6a86f
+        throw new ClassNotFoundException( name );
d6a86f
+    }
d6a86f
+
d6a86f
+    @Override
d6a86f
+    public URL getResource( String name )
d6a86f
+    {
d6a86f
+        if ( isImported( name ) )
d6a86f
+        {
d6a86f
+            URL resource = parent.getResource( name );
d6a86f
+            if ( resource != null )
d6a86f
+            {
d6a86f
+                return resource;
d6a86f
+            }
d6a86f
+        }
d6a86f
+
d6a86f
+        URL resource = super.getResource( name );
d6a86f
+        if ( resource != null )
d6a86f
+        {
d6a86f
+            return resource;
d6a86f
+        }
d6a86f
+
d6a86f
+        resource = super.findResource( name );
d6a86f
+        if ( resource != null )
d6a86f
+        {
d6a86f
+            return resource;
d6a86f
+        }
d6a86f
+
d6a86f
+        return null;
d6a86f
+    }
d6a86f
+
d6a86f
+    @Override
d6a86f
+    public Enumeration<URL> getResources( String name )
d6a86f
+        throws IOException
d6a86f
+    {
d6a86f
+        Collection<URL> resources = new LinkedHashSet<>();
d6a86f
+
d6a86f
+        if ( isImported( name ) )
d6a86f
+        {
d6a86f
+            try
d6a86f
+            {
d6a86f
+                resources.addAll( Collections.list( parent.getResources( name ) ) );
d6a86f
+            }
d6a86f
+            catch ( IOException e )
d6a86f
+            {
d6a86f
+            }
d6a86f
+        }
d6a86f
+
d6a86f
+        try
d6a86f
+        {
d6a86f
+            resources.addAll( Collections.list( super.getResources( name ) ) );
d6a86f
+        }
d6a86f
+        catch ( IOException e )
d6a86f
+        {
d6a86f
+        }
d6a86f
+
d6a86f
+        try
d6a86f
+        {
d6a86f
+            resources.addAll( Collections.list( super.findResources( name ) ) );
d6a86f
+        }
d6a86f
+        catch ( IOException e )
d6a86f
+        {
d6a86f
+        }
d6a86f
+
d6a86f
+        return Collections.enumeration( resources );
d6a86f
+    }
d6a86f
+}
d6a86f
-- 
d6a86f
2.9.3
d6a86f