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

org.jmeterplugins.repository.PluginManager Maven / Gradle / Ivy

There is a newer version: 1.10
Show newest version
package org.jmeterplugins.repository;


import net.sf.json.JSON;
import net.sf.json.JSONArray;
import net.sf.json.JSONObject;
import net.sf.json.JSONSerializer;
import org.apache.jmeter.assertions.Assertion;
import org.apache.jmeter.config.ConfigElement;
import org.apache.jmeter.control.Controller;
import org.apache.jmeter.engine.JMeterEngine;
import org.apache.jmeter.gui.JMeterGUIComponent;
import org.apache.jmeter.processor.PostProcessor;
import org.apache.jmeter.processor.PreProcessor;
import org.apache.jmeter.samplers.SampleListener;
import org.apache.jmeter.samplers.Sampler;
import org.apache.jmeter.testbeans.TestBean;
import org.apache.jmeter.timers.Timer;
import org.apache.jmeter.util.JMeterUtils;
import org.apache.jorphan.reflect.ClassFinder;
import org.jmeterplugins.repository.exception.DownloadException;
import org.jmeterplugins.repository.http.StatsReporter;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.File;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.net.URLDecoder;
import java.nio.file.AccessDeniedException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.TreeSet;

public class PluginManager {
    private static final Logger log = LoggerFactory.getLogger(PluginManager.class);
    private static PluginManager staticManager = new PluginManager();
    private final JARSource jarSource;
    private boolean isSendRepoStats = true;
    protected Map allPlugins = new HashMap<>();

    public PluginManager() {
        String sysProp = System.getProperty("jpgc.repo.address", "https://jmeter-plugins.org/repo/");
        String jmProp = JMeterUtils.getPropDefault("jpgc.repo.address", sysProp);
        File jsonFile = new File(jmProp);
        if (jsonFile.isFile()) {
            jarSource = new JARSourceFilesystem(jsonFile);
        } else {
            jarSource = new JARSourceHTTP(jmProp);
        }
    }

    public boolean hasPlugins() {
        return allPlugins.size() > 0;
    }

    public synchronized void load() throws Throwable {
        detectJARConflicts();

        if (hasPlugins()) {
            return;
        }

        JSON json = jarSource.getRepo();

        if (!(json instanceof JSONArray)) {
            throw new RuntimeException("Result is not array");
        }

        for (Object elm : (JSONArray) json) {
            if (elm instanceof JSONObject) {
                Plugin plugin = Plugin.fromJSON((JSONObject) elm);
                if (plugin.getName().isEmpty()) {
                    log.debug("Skip empty name: " + plugin);
                    continue;
                }

                if (!plugin.isVirtual()) {
                    plugin.detectInstalled(getInstalledPlugins());
                }
                allPlugins.put(plugin, plugin.isInstalled());
            } else {
                log.warn("Invalid array element: " + elm);
            }
        }

        // after all usual plugins detected, detect virtual sets
        for (Plugin plugin : allPlugins.keySet()) {
            if (plugin.isVirtual()) {
                plugin.detectInstalled(getInstalledPlugins());
                allPlugins.put(plugin, plugin.isInstalled());
            }
        }

        if (isSendRepoStats && JMeterUtils.getPropDefault("jpgc.repo.sendstats", "true").equals("true")) {
            try {
                StatsReporter reporter = new StatsReporter(jarSource, getUsageStats());
                log.debug("Start sending repo stats");
                reporter.start();
            } catch (Exception e) {
                log.debug("Failed to report usage stats", e);
            }
        }

        log.info("Plugins Status: " + getAllPluginsStatusString());
    }

    private void checkRW() throws UnsupportedEncodingException, AccessDeniedException {
        String jarPath = Plugin.getJARPath(JMeterEngine.class.getCanonicalName());
        if (jarPath != null) {
            File libext = new File(URLDecoder.decode(jarPath, "UTF-8")).getParentFile();
            if (!isWritable(libext)) {
                String msg = "Have no write access for JMeter directories, not possible to use Plugins Manager: ";
                throw new AccessDeniedException(msg + libext);
            }
        }
    }

    private boolean isWritable(File path) {
        File sample = new File(path, "empty.txt");
        try {
            sample.createNewFile();
            sample.delete();
            return true;
        } catch (IOException e) {
            log.debug("Write check failed for " + path, e);
            return false;
        }
    }

    public void startModifications(Set delPlugins, Set installPlugins, Set installLibs,
                                   Set libDeletions, boolean doRestart, LinkedList additionalJMeterOptions) throws IOException {
        ChangesMaker maker = new ChangesMaker(allPlugins);
        File moveFile = maker.getMovementsFile(delPlugins, installPlugins, installLibs, libDeletions);
        File installFile = maker.getInstallFile(installPlugins, installLibs);
        File restartFile;
        if (doRestart) {
            restartFile = maker.getRestartFile(additionalJMeterOptions);
        } else {
            restartFile = null;
        }
        final ProcessBuilder builder = maker.getProcessBuilder(moveFile, installFile, restartFile);
        log.info("JAR Modifications log will be saved into: " + builder.redirectOutput().file().getPath());
        builder.start();
    }

    public void applyChanges(GenericCallback statusChanged, boolean doRestart, LinkedList additionalJMeterOptions) {
        try {
            checkRW();
        } catch (Throwable e) {
            throw new RuntimeException("Cannot apply changes: " + e.getMessage(), e);
        }

        DependencyResolver resolver = new DependencyResolver(allPlugins);
        Set additions = resolver.getAdditions();
        Set libInstalls = new HashSet<>();

        for (Map.Entry entry : resolver.getLibAdditions().entrySet()) {
            try {
                JARSource.DownloadResult dwn = jarSource.getJAR(entry.getKey(), entry.getValue(), statusChanged);
                libInstalls.add(new Library.InstallationInfo(entry.getKey(), dwn.getTmpFile(), dwn.getFilename()));
            } catch (Throwable e) {
                String msg = "Failed to download " + entry.getKey();
                log.error(msg, e);
                statusChanged.notify(msg);
                throw new DownloadException("Failed to download library " + entry.getKey(), e);
            }
        }

        for (Plugin plugin : additions) {
            try {
                plugin.download(jarSource, statusChanged);
            } catch (IOException e) {
                String msg = "Failed to download " + plugin;
                log.error(msg, e);
                statusChanged.notify(msg);
                throw new DownloadException("Failed to download plugin " + plugin, e);
            }
        }

        if (doRestart) {
            log.info("Restarting JMeter...");
            statusChanged.notify("Restarting JMeter...");
        }

        Set libDeletions = new HashSet<>();
        for (String lib : resolver.getLibDeletions()) {
            libDeletions.add(Plugin.getLibInstallPath(lib));
        }

        modifierHook(resolver.getDeletions(), additions, libInstalls, libDeletions, doRestart, additionalJMeterOptions);
    }

    private void modifierHook(final Set deletions, final Set additions, final Set libInstalls,
                              final Set libDeletions, final boolean doRestart, final LinkedList additionalJMeterOptions) {
        if (deletions.isEmpty() && additions.isEmpty() && libInstalls.isEmpty() && libDeletions.isEmpty()) {
            log.info("Finishing without changes");
        } else {
            log.info("Plugins manager will apply some modifications");
            Runtime.getRuntime().addShutdownHook(new Thread() {
                @Override
                public void run() {
                    try {
                        log.info("Starting JMeter Plugins modifications");
                        startModifications(deletions, additions, libInstalls, libDeletions, doRestart, additionalJMeterOptions);
                    } catch (Exception e) {
                        log.warn("Failed to run plugin cleaner job", e);
                    }
                }
            });
        }
    }

    protected String[] getUsageStats() {
        ArrayList data = new ArrayList<>();
        data.add(JMeterUtils.getJMeterVersion());

        for (Plugin p : getInstalledPlugins()) {
            data.add(p.getID() + "=" + p.getInstalledVersion());
        }
        log.debug("Usage stats: " + data);
        return data.toArray(new String[0]);
    }

    public String getChangesAsText() {
        DependencyResolver resolver = new DependencyResolver(allPlugins);

        StringBuilder text = new StringBuilder();

        for (Plugin pl : resolver.getDeletions()) {
            text.append("Uninstall plugin: ").append(pl).append(" ").append(pl.getInstalledVersion()).append("\n");
        }

        for (String pl : resolver.getLibDeletions()) {
            text.append("Uninstall library: ").append(pl).append("\n");
        }

        for (String pl : resolver.getLibAdditions().keySet()) {
            text.append("Install library: ").append(pl).append("\n");
        }

        for (Plugin pl : resolver.getAdditions()) {
            text.append("Install plugin: ").append(pl).append(" ").append(pl.getCandidateVersion()).append("\n");
        }

        return text.toString();
    }

    public Set getInstalledPlugins() {
        Set result = new TreeSet<>(new PluginComparator());
        for (Plugin plugin : allPlugins.keySet()) {
            if (plugin.isInstalled()) {
                result.add(plugin);
            }
        }
        return result;
    }

    public static Set getInstalledPlugins(Map allPlugins) {
        Set result = new HashSet<>();
        for (Plugin plugin : allPlugins.keySet()) {
            if (plugin.isInstalled()) {
                result.add(plugin);
            }
        }
        return result;
    }

    public Set getAvailablePlugins() {
        Set result = new TreeSet<>(new PluginComparator());
        for (Plugin plugin : allPlugins.keySet()) {
            if (!plugin.isInstalled()) {
                result.add(plugin);
            }
        }
        return result;
    }

    public Set getUpgradablePlugins() {
        Set result = new TreeSet<>(new PluginComparator());
        for (Plugin plugin : allPlugins.keySet()) {
            if (plugin.isUpgradable()) {
                result.add(plugin);
            }
        }
        return result;
    }

    public void togglePlugins(Set pluginsToInstall, boolean isInstall) {
        for (Plugin plugin : pluginsToInstall) {
            toggleInstalled(plugin, isInstall);
        }
    }

    public void toggleInstalled(Plugin plugin, boolean cbState) {
        if (!cbState && !plugin.canUninstall()) {
            log.warn("Cannot uninstall plugin: " + plugin);
            cbState = true;
        }
        allPlugins.put(plugin, cbState);
    }

    public boolean hasAnyUpdates() {
        for (Plugin p : allPlugins.keySet()) {
            if (p.isUpgradable()) {
                return true;
            }
        }
        return false;
    }

    public Plugin getPluginByID(String key) {
        for (Plugin p : allPlugins.keySet()) {
            if (p.getID().equals(key)) {
                return p;
            }
        }
        throw new IllegalArgumentException("Plugin not found in repo: " + key);
    }

    private class PluginComparator implements java.util.Comparator {
        @Override
        public int compare(Plugin o1, Plugin o2) {
            return o1.getName().compareTo(o2.getName());
        }
    }

    public void setTimeout(int timeout) {
        jarSource.setTimeout(timeout);
    }

    /**
     * @return Static instance of manager, used to spare resources on repo loading
     */
    public static PluginManager getStaticManager() {
        try {
            staticManager.load();
        } catch (Throwable e) {
            throw new RuntimeException("Failed to get plugin repositories", e);
        }
        return staticManager;
    }

    /**
     * @param id ID of the plugin to check
     * @return Version name for the plugin if it is installed, null otherwise
     */
    public static String getPluginStatus(String id) {
        PluginManager manager = getStaticManager();

        for (Plugin plugin : manager.allPlugins.keySet()) {
            if (plugin.id.equals(id)) {
                return plugin.getInstalledVersion();
            }
        }
        return null;
    }

    /**
     * @return Status for all plugins
     */
    public static String getAllPluginsStatus() {
        PluginManager manager = getStaticManager();
        return manager.getAllPluginsStatusString();
    }

    private String getAllPluginsStatusString() {
        ArrayList res = new ArrayList<>();
        for (Plugin plugin : getInstalledPlugins()) {
            res.add(plugin.getID() + "=" + plugin.getInstalledVersion());
        }
        return Arrays.toString(res.toArray());
    }

    /**
     * @return Available plugins
     */
    public static String getAvailablePluginsAsString() {
        PluginManager manager = getStaticManager();
        return manager.getAvailablePluginsString();
    }

    private String getAvailablePluginsString() {
        ArrayList res = new ArrayList<>();
        for (Plugin plugin : getAvailablePlugins()) {
            List versions = new ArrayList<>(plugin.getVersions());
            Collections.reverse(versions);
            res.add(plugin.getID() + "=" + Arrays.toString(versions.toArray()));
        }
        return Arrays.toString(res.toArray());
    }

    /**
     * @return Upgradable plugins
     */
    public static String getUpgradablePluginsAsString() {
        PluginManager manager = getStaticManager();
        return manager.getUpgradablePluginsString();
    }

    private String getUpgradablePluginsString() {
        ArrayList res = new ArrayList<>();
        for (Plugin plugin : getUpgradablePlugins()) {
            res.add(plugin.getID() + "=" + plugin.getMaxVersion());
        }
        return (res.size() != 0) ?
                Arrays.toString(res.toArray()) :
                "There is nothing to update.";
    }


    public static void detectJARConflicts() {
        String[] paths = System.getProperty("java.class.path").split(File.pathSeparator);
        final Map jarNames = new HashMap<>();

        for (String path : paths) {
            String name = path;
            int start = path.lastIndexOf(File.separator);
            if (start > 0) {
                name = name.substring(start + 1);
            }
            if (path.endsWith(".jar")) {
                name = name.substring(0, name.length() - 4);
            }
            name = removeJARVersion(name);
            if (jarNames.containsKey(name)) {
                log.warn("Found JAR conflict: " + path + " and " + jarNames.get(name));
            }
            jarNames.put(name, path);
        }
    }

    protected static String removeJARVersion(String path) {
        StringBuilder result = new StringBuilder();
        String data[] = path.split("-");
        for (int i = 0; i < data.length; i++) {
            String ch = data[i];
            if (!ch.isEmpty() && (!Character.isDigit(ch.charAt(0)) || (i < data.length - 1))) {
                result.append(ch);
            }
        }
        return result.toString();
    }

    public void logPluginComponents() {
        StringBuilder report = new StringBuilder("Plugin Components:\n");
        for (Plugin plugin : getInstalledPlugins()) {
            try {
                Class[] superClasses = {
                        Sampler.class,
                        Controller.class,
                        Timer.class,
                        ConfigElement.class,
                        PreProcessor.class,
                        PostProcessor.class,
                        Assertion.class,
                        SampleListener.class,
                        JMeterGUIComponent.class,
                        TestBean.class
                };
                String[] searchPaths = {plugin.installedPath};
                List list = ClassFinder.findClassesThatExtend(searchPaths, superClasses);
                report.append(plugin.id).append("\n").append("\"componentClasses\":").append(JSONSerializer.toJSON(list.toArray()).toString()).append(",\n");
            } catch (Throwable e) {
                log.error("Failed to get classes", e);
            }
        }
        log.info(report.toString());
    }

    public boolean isSendRepoStats() {
        return isSendRepoStats;
    }

    public void setSendRepoStats(boolean sendRepoStats) {
        isSendRepoStats = sendRepoStats;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy