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

org.neo4j.kernel.extension.AbstractExtensions Maven / Gradle / Ivy

Go to download

Neo4j kernel is a lightweight, embedded Java database designed to store data structured as graphs rather than tables. For more information, see http://neo4j.org.

There is a newer version: 5.25.1
Show newest version
/*
 * Copyright (c) "Neo4j"
 * Neo4j Sweden AB [http://neo4j.com]
 *
 * This file is part of Neo4j.
 *
 * Neo4j is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */
package org.neo4j.kernel.extension;

import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.List;
import java.util.Objects;

import org.neo4j.collection.Dependencies;
import org.neo4j.common.DependencyResolver;
import org.neo4j.exceptions.UnsatisfiedDependencyException;
import org.neo4j.kernel.extension.context.ExtensionContext;
import org.neo4j.kernel.impl.util.DependenciesProxy;
import org.neo4j.kernel.lifecycle.LifeSupport;
import org.neo4j.kernel.lifecycle.Lifecycle;

import static java.util.stream.Collectors.toList;
import static org.neo4j.internal.helpers.collection.Iterables.stream;

public abstract class AbstractExtensions extends DependencyResolver.Adapter implements Lifecycle
{
    private final ExtensionContext extensionContext;
    private final List> extensionFactories;
    private final Dependencies dependencies;
    private final LifeSupport life = new LifeSupport();
    private final ExtensionFailureStrategy extensionFailureStrategy;

    AbstractExtensions( ExtensionContext extensionContext, Iterable> extensionFactories, Dependencies dependencies,
            ExtensionFailureStrategy extensionFailureStrategy, ExtensionType extensionType )
    {
        this.extensionContext = extensionContext;
        this.extensionFailureStrategy = extensionFailureStrategy;
        this.extensionFactories = stream( extensionFactories ).filter( e -> e.getExtensionType() == extensionType ).collect( toList() );
        this.dependencies = dependencies;
    }

    @Override
    public void init()
    {
        for ( ExtensionFactory extensionFactory : extensionFactories )
        {
            try
            {
                Object extensionDependencies = getExtensionDependencies( extensionFactory );
                Lifecycle dependency = newInstance( extensionContext, extensionFactory, extensionDependencies );
                Objects.requireNonNull( dependency, extensionFactory + " returned a null extension." );
                life.add( dependencies.satisfyDependency( dependency ) );
            }
            catch ( UnsatisfiedDependencyException exception )
            {
                extensionFailureStrategy.handle( extensionFactory, exception );
            }
            catch ( Throwable throwable )
            {
                extensionFailureStrategy.handle( extensionFactory, throwable );
            }
        }

        life.init();
    }

    @Override
    public void start()
    {
        life.start();
    }

    @Override
    public void stop()
    {
        life.stop();
    }

    @Override
    public void shutdown()
    {
        life.shutdown();
    }

    @Override
    public  T resolveDependency( Class type, SelectionStrategy selector )
    {
        Iterable typeDependencies = resolveTypeDependencies( type );
        return selector.select( type, typeDependencies );
    }

    @Override
    public  Iterable resolveTypeDependencies( Class type )
    {
        return life.getLifecycleInstances().stream().filter( type::isInstance ).map( type::cast ).collect( toList() );
    }

    @Override
    public boolean containsDependency( Class type )
    {
        return life.getLifecycleInstances().stream().anyMatch( type::isInstance );
    }

    private Object getExtensionDependencies( ExtensionFactory factory )
    {
        // superclass is either ExtensionFactory or it's subclass that can be generic too
        Class factoryType = factory.getClass().getSuperclass();
        Type genericSuperclass = factory.getClass().getGenericSuperclass();
        while ( factoryType.getGenericSuperclass() instanceof ParameterizedType )
        {
            genericSuperclass = factoryType.getGenericSuperclass();
            factoryType = factoryType.getSuperclass();
        }
        Class configurationClass = (Class) ((ParameterizedType) genericSuperclass).getActualTypeArguments()[0];
        return DependenciesProxy.dependencies( dependencies, configurationClass );
    }

    @SuppressWarnings( "unchecked" )
    private static  Lifecycle newInstance( ExtensionContext extensionContext, ExtensionFactory factory, Object dependencies )
    {
        return factory.newInstance( extensionContext, (T)dependencies );
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy