org.neo4j.server.plugins.PluginManager.java Source code

Java tutorial

Introduction

Here is the source code for org.neo4j.server.plugins.PluginManager.java

Source

/**
 * Copyright (c) 2002-2011 "Neo Technology,"
 * Network Engine for Objects in Lund AB [http://neotechnology.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 <http://www.gnu.org/licenses/>.
 */
package org.neo4j.server.plugins;

import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.apache.commons.configuration.Configuration;
import org.neo4j.helpers.Pair;
import org.neo4j.kernel.AbstractGraphDatabase;
import org.neo4j.kernel.impl.util.StringLogger;
import org.neo4j.server.logging.Logger;
import org.neo4j.server.rest.repr.BadInputException;
import org.neo4j.server.rest.repr.ExtensionInjector;
import org.neo4j.server.rest.repr.ExtensionPointRepresentation;
import org.neo4j.server.rest.repr.Representation;

public final class PluginManager implements ExtensionInjector, PluginInvocator {
    private static final Logger log = Logger.getLogger(PluginManager.class);
    private final Map<String/*name*/, ServerExtender> extensions = new HashMap<String, ServerExtender>();

    public PluginManager(Configuration serverConfig, StringLogger logger) {
        this(serverConfig, ServerPlugin.load(), logger);
    }

    PluginManager(Configuration serverConfig, Iterable<ServerPlugin> plugins, StringLogger logger) {
        Map<String, Pair<ServerPlugin, ServerExtender>> extensions = new HashMap<String, Pair<ServerPlugin, ServerExtender>>();
        for (ServerPlugin plugin : plugins) {
            PluginPointFactory factory = new PluginPointFactoryImpl();
            final ServerExtender extender = new ServerExtender(factory);
            try {
                plugin.loadServerExtender(extender);
            } catch (Exception ex) {
                log.warn("Failed to load plugin [%s]: %s", plugin.toString(), ex.getMessage());
                if (logger != null)
                    logger.logMessage("Failed to load plugin: " + plugin, ex);
                continue;
            } catch (LinkageError err) {
                log.warn("Failed to load plugin [%s]: %s", plugin.toString(), err.getMessage());
                if (logger != null)
                    logger.logMessage("Failed to load plugin: " + plugin, err);
                continue;
            }
            Pair<ServerPlugin, ServerExtender> old = extensions.put(plugin.name, Pair.of(plugin, extender));
            if (old != null) {
                log.warn(String.format("Extension naming conflict \"%s\" between \"%s\" and \"%s\"", plugin.name,
                        old.first().getClass(), plugin.getClass()));
                if (logger != null)
                    logger.logMessage(String.format("Extension naming conflict \"%s\" between \"%s\" and \"%s\"",
                            plugin.name, old.first().getClass(), plugin.getClass()));
            }
        }
        for (Pair<ServerPlugin, ServerExtender> extension : extensions.values()) {
            log.info(String.format("Loaded server plugin \"%s\"", extension.first().name));
            if (logger != null) {
                logger.logMessage(String.format("Loaded server plugin \"%s\" (%s)", extension.first().name,
                        extension.first().getClass().getName()));
                for (PluginPoint point : extension.other().all()) {
                    logger.logMessage(String.format("  %s.%s: %s", point.forType().getSimpleName(), point.name(),
                            point.getDescription()));
                }
            }
            this.extensions.put(extension.first().name, extension.other());
        }
    }

    @Override
    public Map<String, List<String>> getExensionsFor(Class<?> type) {
        Map<String, List<String>> result = new HashMap<String, List<String>>();
        for (Map.Entry<String, ServerExtender> extension : extensions.entrySet()) {
            List<String> methods = new ArrayList<String>();
            for (PluginPoint method : extension.getValue().getExtensionsFor(type)) {
                methods.add(method.name());
            }
            if (!methods.isEmpty()) {
                result.put(extension.getKey(), methods);
            }
        }
        return result;
    }

    private PluginPoint extension(String name, Class<?> type, String method) throws PluginLookupException {
        ServerExtender extender = extensions.get(name);
        if (extender == null) {
            throw new PluginLookupException("No such ServerPlugin: \"" + name + "\"");
        }
        return extender.getExtensionPoint(type, method);
    }

    @Override
    public ExtensionPointRepresentation describe(String name, Class<?> type, String method)
            throws PluginLookupException {
        return describe(extension(name, type, method));
    }

    private ExtensionPointRepresentation describe(PluginPoint extension) {
        ExtensionPointRepresentation representation = new ExtensionPointRepresentation(extension.name(),
                extension.forType(), extension.getDescription());
        extension.describeParameters(representation);
        return representation;
    }

    @Override
    public List<ExtensionPointRepresentation> describeAll(String name) throws PluginLookupException {
        ServerExtender extender = extensions.get(name);
        if (extender == null) {
            throw new PluginLookupException("No such ServerPlugin: \"" + name + "\"");
        }
        List<ExtensionPointRepresentation> result = new ArrayList<ExtensionPointRepresentation>();
        for (PluginPoint plugin : extender.all()) {
            result.add(describe(plugin));
        }
        return result;
    }

    @Override
    public <T> Representation invoke(AbstractGraphDatabase graphDb, String name, Class<T> type, String method,
            T context, ParameterList params) throws PluginLookupException, BadInputException,
            PluginInvocationFailureException, BadPluginInvocationException {
        PluginPoint plugin = extension(name, type, method);
        try {
            return plugin.invoke(graphDb, context, params);
        } catch (BadInputException e) {
            throw e;
        } catch (BadPluginInvocationException e) {
            throw e;
        } catch (PluginInvocationFailureException e) {
            throw e;
        } catch (Exception e) {
            throw new PluginInvocationFailureException(e);
        }
    }

    @Override
    public Set<String> extensionNames() {
        return Collections.unmodifiableSet(extensions.keySet());
    }
}