All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.infinispan.graalvm.NativeMetadataProvider Maven / Gradle / Ivy

There is a newer version: 15.1.3.Final
Show newest version
package org.infinispan.graalvm;

import static org.infinispan.commons.configuration.io.xml.XmlPullParser.END_DOCUMENT;
import static org.infinispan.commons.configuration.io.xml.XmlPullParser.END_TAG;
import static org.infinispan.commons.configuration.io.xml.XmlPullParser.FEATURE_PROCESS_NAMESPACES;
import static org.infinispan.commons.configuration.io.xml.XmlPullParser.START_TAG;

import java.io.IOException;
import java.io.InputStream;
import java.util.Collection;
import java.util.HashSet;
import java.util.Set;
import java.util.function.Function;
import java.util.stream.Stream;

import org.graalvm.nativeimage.hosted.Feature;
import org.infinispan.commons.configuration.io.xml.MXParser;
import org.infinispan.commons.configuration.io.xml.XmlPullParser;
import org.infinispan.commons.configuration.io.xml.XmlPullParserException;
import org.infinispan.commons.graalvm.Bundle;
import org.infinispan.commons.graalvm.ClassLoaderFeatureAccess;
import org.infinispan.commons.graalvm.Jandex;
import org.infinispan.commons.graalvm.ReflectionProcessor;
import org.infinispan.commons.graalvm.ReflectiveClass;
import org.infinispan.commons.graalvm.Resource;
import org.infinispan.commons.marshall.SerializeWith;
import org.infinispan.configuration.serializing.SerializedWith;
import org.infinispan.marshall.exts.CollectionExternalizer;
import org.infinispan.marshall.exts.EnumExternalizer;
import org.infinispan.marshall.exts.EnumSetExternalizer;
import org.infinispan.marshall.exts.MapExternalizer;
import org.infinispan.notifications.Listener;
import org.jboss.jandex.IndexView;
import org.jgroups.Version;
import org.jgroups.conf.ClassConfigurator;

public class NativeMetadataProvider implements org.infinispan.commons.graalvm.NativeMetadataProvider {

   static final Collection resourceFiles = Resource.of(
         "META-INF/services/org\\.infinispan\\.configuration\\.parsing\\.ConfigurationParser",
         "META-INF/services/org\\.infinispan\\.factories\\.impl\\.ModuleMetadataBuilder",
         "META-INF/infinispan-version\\.properties",
         "org/infinispan/protostream/message-wrapping\\.proto",
         "protostream/common-java-types\\.proto",
         "protostream/common-java-container-types\\.proto",
         "proto/generated/user\\.commons\\.proto",
         "proto/generated/persistence\\.commons\\.proto",
         "proto/generated/persistence\\.core\\.proto",
         "proto/generated/global\\.commons\\.proto",
         "default-configs/default-jgroups-udp\\.xml",
         "default-configs/default-jgroups-tcp\\.xml",
         "default-configs/default-jgroups-kubernetes\\.xml",
         "default-configs/default-jgroups-ec2\\.xml",
         "default-configs/default-jgroups-google\\.xml",
         "default-configs/default-jgroups-azure\\.xml",
         ClassConfigurator.MAGIC_NUMBER_FILE,
         ClassConfigurator.PROTOCOL_ID_FILE,
         Version.VERSION_FILE
   );

   static final Collection resourceRegexps = Resource.of(
         "protostream/google/protobuf/.*.proto"
   );

   static final Collection bundles = Bundle.of(
         "jg-messages",
         "java.base:sun.security.util.Resources",
         "com.sun.org.apache.xerces.internal.impl.msg.XMLMessages"
   );

   final Feature.FeatureAccess featureAccess;
   final ReflectionProcessor reflection;

   public NativeMetadataProvider() {
      this(new ClassLoaderFeatureAccess(NativeMetadataProvider.class.getClassLoader()));
   }

   public NativeMetadataProvider(Feature.FeatureAccess featureAccess) {
      this.featureAccess = featureAccess;
      this.reflection = reflectionProcessor();
   }

   @Override
   public Stream reflectiveClasses() {
      return reflection.classes();
   }

   @Override
   public Stream includedResources() {
      return Stream.of(
            resourceFiles.stream(),
            resourceRegexps.stream()
      ).flatMap(Function.identity());
   }

   @Override
   public Stream bundles() {
      return bundles.stream();
   }

   private ReflectionProcessor reflectionProcessor() {
      IndexView index = Jandex.createIndex(
            com.github.benmanes.caffeine.cache.Cache.class, // Caffeine
            org.jgroups.stack.Protocol.class, // JGroups
            org.infinispan.protostream.GeneratedSchema.class, // ProtoStream
            org.infinispan.commons.CacheException.class, // Commons
            org.infinispan.AdvancedCache.class // Core
      );
      ReflectionProcessor reflection = new ReflectionProcessor(featureAccess, index);
      caffeine(reflection);
      jgroups(reflection);
      infinispan(reflection);
      return reflection;
   }

   private void caffeine(ReflectionProcessor processor) {
      processor.addClasses(
            "com.github.benmanes.caffeine.cache.PDMS",
            "com.github.benmanes.caffeine.cache.PSA",
            "com.github.benmanes.caffeine.cache.PSMS",
            "com.github.benmanes.caffeine.cache.PSW",
            "com.github.benmanes.caffeine.cache.PSMW",
            "com.github.benmanes.caffeine.cache.PSAMW",
            "com.github.benmanes.caffeine.cache.PSAWMW",
            "com.github.benmanes.caffeine.cache.PSWMS",
            "com.github.benmanes.caffeine.cache.PSWMW",
            "com.github.benmanes.caffeine.cache.SILMS",
            "com.github.benmanes.caffeine.cache.SSA",
            "com.github.benmanes.caffeine.cache.SSLA",
            "com.github.benmanes.caffeine.cache.SSLMS",
            "com.github.benmanes.caffeine.cache.SSLMW",
            "com.github.benmanes.caffeine.cache.SSMS",
            "com.github.benmanes.caffeine.cache.SSMSA",
            "com.github.benmanes.caffeine.cache.SSMSAW",
            "com.github.benmanes.caffeine.cache.SSMSW",
            "com.github.benmanes.caffeine.cache.SSW"
      );
   }

   private void jgroups(ReflectionProcessor processor) {
      processor
            .addClasses(
                  org.jgroups.blocks.RequestCorrelator.class,
                  org.jgroups.protocols.MsgStats.class,
                  org.jgroups.protocols.raft.AppendEntriesResponse.class,
                  org.jgroups.tests.perf.MPerf.class,
                  org.jgroups.util.Util.AddressScope.class,
                  org.jgroups.Version.class
            )
            .addClasses(true, true,
                  org.jgroups.protocols.MsgStats.class,
                  org.jgroups.protocols.raft.AppendEntriesRequest.class,
                  org.jgroups.stack.DiagnosticsHandler.class,
                  org.jgroups.util.ThreadPool.class
            ).addImplementations(false, false,
                  org.jgroups.conf.PropertyConverter.class,
                  org.jgroups.protocols.LocalTransport.class,
                  org.jgroups.stack.MessageProcessingPolicy.class
            ).addImplementations(true, false,
                  org.jgroups.protocols.Bundler.class,
                  org.jgroups.stack.Protocol.class
            );

      processor.addClasses(jgroupsClasses().toArray(new String[0]));
   }

   private Collection jgroupsClasses() {
      Set classes = new HashSet<>();
      try (InputStream source = org.jgroups.Message.class.getClassLoader().getResourceAsStream(ClassConfigurator.MAGIC_NUMBER_FILE)) {
         MXParser reader = new MXParser();
         reader.setFeature(FEATURE_PROCESS_NAMESPACES, false);
         reader.setInput(source, null);
         int eventType;
         while ((eventType = reader.next()) != END_DOCUMENT) {
            if (eventType == START_TAG) {
               if (reader.getName().equals("magic-number-class-mapping")) {
                  parseJGroupsMagicNumbers(reader, classes);
               }
            }
         }
      } catch (IOException e) {
         throw new RuntimeException(e);
      }
      return classes;
   }

   private void parseJGroupsMagicNumbers(XmlPullParser reader, Set classes) throws XmlPullParserException, IOException {
      int eventType;
      while ((eventType = reader.nextTag()) != END_DOCUMENT) {
         switch (eventType) {
            case END_TAG: {
               return;
            }
            case START_TAG: {
               if (reader.getName().equals("class")) {
                  String clazz = reader.getAttributeValue(null, "name");
                  classes.add(clazz);
                  reader.next();
                  continue;
               }
               throw new IOException("Unexpected content");
            }
            default: {
               throw new IOException("Unexpected content");
            }
         }
      }
      throw new IOException("Unexpected end of document");
   }

   private void infinispan(ReflectionProcessor processor) {
      processor
            .addClasses(
                  org.infinispan.CoreModuleImpl.class,
                  org.infinispan.distribution.ch.impl.HashFunctionPartitioner.class,
                  org.infinispan.distribution.ch.impl.RESPHashFunctionPartitioner.class,
                  org.infinispan.notifications.cachelistener.annotation.CacheEntryCreated.class,
                  org.infinispan.notifications.cachelistener.annotation.CacheEntryExpired.class,
                  org.infinispan.notifications.cachelistener.annotation.CacheEntryModified.class,
                  org.infinispan.notifications.cachelistener.annotation.CacheEntryRemoved.class,
                  org.infinispan.remoting.transport.jgroups.JGroupsTransport.class,
                  org.infinispan.transaction.lookup.EmbeddedTransactionManagerLookup.class
            )
            .addImplementations(false, false,
                  org.jboss.logging.BasicLogger.class,

                  org.infinispan.commons.marshall.AbstractExternalizer.class,
                  org.infinispan.commons.marshall.AdvancedExternalizer.class,

                  org.infinispan.configuration.parsing.ConfigurationParser.class,
                  org.infinispan.configuration.cache.AbstractModuleConfigurationBuilder.class,
                  org.infinispan.configuration.cache.StoreConfigurationBuilder.class,
                  org.infinispan.configuration.serializing.ConfigurationSerializer.class,
                  org.infinispan.distribution.ch.ConsistentHashFactory.class,
                  org.infinispan.factories.impl.ModuleMetadataBuilder.class,
                  org.infinispan.persistence.spi.NonBlockingStore.class,
                  org.infinispan.persistence.spi.CacheLoader.class,
                  org.infinispan.persistence.spi.CacheWriter.class
            )
            .addImplementations(true, false,
                  org.infinispan.util.logging.events.Messages.class,
                  org.infinispan.interceptors.AsyncInterceptor.class
            )
            .addImplementation(false, true,
                  org.infinispan.configuration.cache.StoreConfiguration.class
            )
            .addClassesWithAnnotation(false, true, Listener.class)
            .addClassFromAnnotationValue(false, false, SerializeWith.class)
            .addClassFromAnnotationValue(false, false, SerializedWith.class)
            .addClasses("org.infinispan.remoting.transport.jgroups.JGroupsTransport$ChannelCallbacks")
            .addClasses(
                  // XML parsing
                  "com.sun.org.apache.xerces.internal.jaxp.DocumentBuilderFactoryImpl",
                  "com.sun.org.apache.xerces.internal.jaxp.datatype.DatatypeFactoryImpl",
                  "com.sun.org.apache.xalan.internal.xsltc.trax.TransformerFactoryImpl",
                  "com.sun.org.apache.xerces.internal.jaxp.SAXParserFactoryImpl"
            );

      CollectionExternalizer.getSupportedPrivateClasses().forEach(processor::addClasses);
      MapExternalizer.getSupportedPrivateClasses().forEach(processor::addClasses);
      EnumExternalizer.INSTANCE.getTypeClasses().forEach(processor::addClasses);
      new EnumSetExternalizer().getTypeClasses().forEach(processor::addClasses);
   }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy