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

io.helidon.microprofile.graphql.server.JandexUtils Maven / Gradle / Ivy

/*
 * Copyright (c) 2020, 2023 Oracle and/or its affiliates.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package io.helidon.microprofile.graphql.server;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.lang.System.Logger.Level;
import java.lang.reflect.Modifier;
import java.net.URL;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Enumeration;
import java.util.HashSet;
import java.util.List;
import java.util.Set;

import org.jboss.jandex.ClassInfo;
import org.jboss.jandex.DotName;
import org.jboss.jandex.Index;
import org.jboss.jandex.IndexReader;

/**
 * Utilities for working with Jandex indexes.
 */
class JandexUtils {

    private static final System.Logger LOGGER = System.getLogger(JandexUtils.class.getName());

    /**
     * Default Jandex index file.
     */
    protected static final String DEFAULT_INDEX_FILE = "META-INF/jandex.idx";

    /**
     * Property to override the default index file. (Normally used for functional tests)
     */
    public static final String PROP_INDEX_FILE = "io.helidon.microprofile.graphql.indexfile";

    /**
     * The {@link Set} of loaded indexes.
     */
    private Set setIndexes = new HashSet<>();

    /**
     * The file used to load the index.
     */
    private String indexFile;

    /**
     * Construct an instance of the utilities class..
     */
    private JandexUtils() {
        indexFile = System.getProperty(PROP_INDEX_FILE, DEFAULT_INDEX_FILE);
    }

    /**
     * Create a new {@link JandexUtils}.
     * @return a new {@link JandexUtils}
     */
    public static JandexUtils create() {
         return new JandexUtils();
    }

    /**
     * Load all the index files of the given name.
     */
    public void loadIndexes() {
        try {
            List listUrls = findIndexFiles(indexFile);

            // loop through each URL and load the index
            for (URL url : listUrls) {
                try (InputStream input = url.openStream()) {
                    setIndexes.add(new IndexReader(input).read());
                } catch (Exception e) {
                    LOGGER.log(Level.WARNING, "Unable to load default Jandex index file: " + url
                                           + " : " + e.getMessage());
                }
            }
        } catch (IOException ignore) {
            // any Exception coming from getResources() or toURL() is ignored and
            // the Map of indexes remain empty
        }
    }

    /**
     * Return all the Jandex index files with the given name. If the name is absolute then
     * return the single file.
     *
     * @param indexFileName  index file name
     * @return a {@link List} of the index file names
     *
     * @throws IOException if any error
     */
    private List findIndexFiles(String indexFileName) throws IOException {
        List result = new ArrayList<>();
        ClassLoader contextClassLoader = Thread.currentThread().getContextClassLoader();
        File file = new File(indexFile);
        if (file.isAbsolute()) {
            result.add(file.toPath().toUri().toURL());
            return result;
        }

        Enumeration urls = contextClassLoader.getResources(indexFileName);
        while (urls.hasMoreElements()) {
            result.add(urls.nextElement());
        }

        return result;
    }


    /**
     * Return a {@link Collection} of {@link Class}es which are implementors of a given class/interface.
     *
     * @param clazz           {@link Class} to check for implementors
     * @param includeAbstract indicates if abstract classes should be included
     * @return a {@link Collection} of {@link Class}es
     */
    public Collection> getKnownImplementors(String clazz, boolean includeAbstract) {
        Set> setResults = new HashSet<>();
        if (!hasIndex()) {
            return null;
        }

        for (Index index : setIndexes) {
            Set allKnownImplementors = index.getAllKnownImplementors(DotName.createSimple(clazz));
            for (ClassInfo classInfo : allKnownImplementors) {
                Class clazzName = null;
                try {
                    clazzName = Class.forName(classInfo.toString());
                } catch (ClassNotFoundException e) {
                    // ignore as class should exist
                }
                if (includeAbstract || !Modifier.isAbstract(clazzName.getModifiers())) {
                    setResults.add(clazzName);
                }
            }
        }

        return setResults;
    }

    /**
     * Return a {@link Collection} of {@link Class}es which are implementors of a given class/interface and are not abstract.
     *
     * @param clazz {@link Class} to check for implementors
     * @return a {@link Collection} of {@link Class}es
     */
    public Collection> getKnownImplementors(String clazz) {
        return getKnownImplementors(clazz, false);
    }

    /**
     * Indicates if an index was found.
     *
     * @return true if an index was found
     */
    public boolean hasIndex() {
        return setIndexes != null && setIndexes.size() > 0;
    }

    /**
     * Return the generated {@link Set} of {@link Index}es.
     *
     * @return the generated {@link Set} of {@link Index}es
     */
    public Set getIndexes() {
        return setIndexes;
    }

    /**
     * The index file used to load the index. (may not exist).
     *
     * @return index file used to load the index
     */
    public String getIndexFile() {
        return indexFile;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy