|
|
|
/**
|
|
|
|
* @fileoverview Plugins manager
|
|
|
|
* @author Nicholas C. Zakas
|
|
|
|
*/
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// Requirements
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
let debug = require("debug"),
|
|
|
|
Environments = require("./environments"),
|
|
|
|
rules = require("../rules");
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// Private
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
debug = debug("eslint:plugins");
|
|
|
|
|
|
|
|
let plugins = Object.create(null);
|
|
|
|
|
|
|
|
let PLUGIN_NAME_PREFIX = "eslint-plugin-",
|
|
|
|
NAMESPACE_REGEX = /^@.*\//i;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes the prefix `eslint-plugin-` from a plugin name.
|
|
|
|
* @param {string} pluginName The name of the plugin which may have the prefix.
|
|
|
|
* @returns {string} The name of the plugin without prefix.
|
|
|
|
*/
|
|
|
|
function removePrefix(pluginName) {
|
|
|
|
return pluginName.indexOf(PLUGIN_NAME_PREFIX) === 0 ? pluginName.substring(PLUGIN_NAME_PREFIX.length) : pluginName;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the scope (namespace) of a plugin.
|
|
|
|
* @param {string} pluginName The name of the plugin which may have the prefix.
|
|
|
|
* @returns {string} The name of the plugins namepace if it has one.
|
|
|
|
*/
|
|
|
|
function getNamespace(pluginName) {
|
|
|
|
return pluginName.match(NAMESPACE_REGEX) ? pluginName.match(NAMESPACE_REGEX)[0] : "";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes the namespace from a plugin name.
|
|
|
|
* @param {string} pluginName The name of the plugin which may have the prefix.
|
|
|
|
* @returns {string} The name of the plugin without the namespace.
|
|
|
|
*/
|
|
|
|
function removeNamespace(pluginName) {
|
|
|
|
return pluginName.replace(NAMESPACE_REGEX, "");
|
|
|
|
}
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// Public Interface
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
|
|
|
|
removePrefix: removePrefix,
|
|
|
|
getNamespace: getNamespace,
|
|
|
|
removeNamespace: removeNamespace,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Defines a plugin with a given name rather than loading from disk.
|
|
|
|
* @param {string} pluginName The name of the plugin to load.
|
|
|
|
* @param {Object} plugin The plugin object.
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
define: function(pluginName, plugin) {
|
|
|
|
let pluginNameWithoutNamespace = removeNamespace(pluginName),
|
|
|
|
pluginNameWithoutPrefix = removePrefix(pluginNameWithoutNamespace);
|
|
|
|
|
|
|
|
plugins[pluginNameWithoutPrefix] = plugin;
|
|
|
|
|
|
|
|
// load up environments and rules
|
|
|
|
Environments.importPlugin(plugin, pluginNameWithoutPrefix);
|
|
|
|
|
|
|
|
if (plugin.rules) {
|
|
|
|
rules.import(plugin.rules, pluginNameWithoutPrefix);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a plugin with the given name.
|
|
|
|
* @param {string} pluginName The name of the plugin to retrieve.
|
|
|
|
* @returns {Object} The plugin or null if not loaded.
|
|
|
|
*/
|
|
|
|
get: function(pluginName) {
|
|
|
|
return plugins[pluginName] || null;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns all plugins that are loaded.
|
|
|
|
* @returns {Object} The plugins cache.
|
|
|
|
*/
|
|
|
|
getAll: function() {
|
|
|
|
return plugins;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Loads a plugin with the given name.
|
|
|
|
* @param {string} pluginName The name of the plugin to load.
|
|
|
|
* @returns {void}
|
|
|
|
* @throws {Error} If the plugin cannot be loaded.
|
|
|
|
*/
|
|
|
|
load: function(pluginName) {
|
|
|
|
let pluginNamespace = getNamespace(pluginName),
|
|
|
|
pluginNameWithoutNamespace = removeNamespace(pluginName),
|
|
|
|
pluginNameWithoutPrefix = removePrefix(pluginNameWithoutNamespace),
|
|
|
|
plugin = null;
|
|
|
|
|
|
|
|
if (!plugins[pluginNameWithoutPrefix]) {
|
|
|
|
try {
|
|
|
|
plugin = require(pluginNamespace + PLUGIN_NAME_PREFIX + pluginNameWithoutPrefix);
|
|
|
|
} catch (err) {
|
|
|
|
debug("Failed to load plugin eslint-plugin-" + pluginNameWithoutPrefix + ". Proceeding without it.");
|
|
|
|
err.message = "Failed to load plugin " + pluginName + ": " + err.message;
|
|
|
|
err.messageTemplate = "plugin-missing";
|
|
|
|
err.messageData = {
|
|
|
|
pluginName: pluginNameWithoutPrefix
|
|
|
|
};
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.define(pluginName, plugin);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Loads all plugins from an array.
|
|
|
|
* @param {string[]} pluginNames An array of plugins names.
|
|
|
|
* @returns {void}
|
|
|
|
* @throws {Error} If a plugin cannot be loaded.
|
|
|
|
*/
|
|
|
|
loadAll: function(pluginNames) {
|
|
|
|
pluginNames.forEach(this.load, this);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resets plugin information. Use for tests only.
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
testReset: function() {
|
|
|
|
plugins = Object.create(null);
|
|
|
|
}
|
|
|
|
};
|