UNPKG

verdaccio

Version:

A lightweight private npm proxy registry

154 lines (114 loc) 15.5 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.default = loadPlugin; var _path = _interopRequireDefault(require("path")); var _lodash = _interopRequireDefault(require("lodash")); var _logger = require("./logger"); var _constants = require("./constants"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } /** * Requires a module. * @param {*} path the module's path * @return {Object} */ function tryLoad(path) { try { return require(path); } catch (err) { if (err.code === _constants.MODULE_NOT_FOUND) { return null; } throw err; } } function mergeConfig(appConfig, pluginConfig) { return _lodash.default.merge(appConfig, pluginConfig); } function isValid(plugin) { return _lodash.default.isFunction(plugin) || _lodash.default.isFunction(plugin.default); } function isES6(plugin) { return Object.keys(plugin).includes('default'); } // export type PluginGeneric<R, T extends IPlugin<R> = ; /** * Load a plugin following the rules * - First try to load from the internal directory plugins (which will disappear soon or later). * - A second attempt from the external plugin directory * - A third attempt from node_modules, in case to have multiple match as for instance verdaccio-ldap * and sinopia-ldap. All verdaccio prefix will have preferences. * @param {*} config a reference of the configuration settings * @param {*} pluginConfigs * @param {*} params a set of params to initialize the plugin * @param {*} sanityCheck callback that check the shape that should fulfill the plugin * @return {Array} list of plugins */ function loadPlugin(config, pluginConfigs = {}, params, sanityCheck, prefix = 'verdaccio') { return Object.keys(pluginConfigs).map(pluginId => { let plugin; const localPlugin = _path.default.resolve(__dirname + '/../plugins', pluginId); // try local plugins first plugin = tryLoad(localPlugin); // try the external plugin directory if (plugin === null && config.plugins) { const pluginDir = config.plugins; const externalFilePlugin = _path.default.resolve(pluginDir, pluginId); plugin = tryLoad(externalFilePlugin); // npm package if (plugin === null && pluginId.match(/^[^\.\/]/)) { plugin = tryLoad(_path.default.resolve(pluginDir, `${prefix}-${pluginId}`)); // compatibility for old sinopia plugins if (!plugin) { plugin = tryLoad(_path.default.resolve(pluginDir, `sinopia-${pluginId}`)); } } } // npm package if (plugin === null && pluginId.match(/^[^\.\/]/)) { plugin = tryLoad(`${prefix}-${pluginId}`); // compatibility for old sinopia plugins if (!plugin) { plugin = tryLoad(`sinopia-${pluginId}`); } } if (plugin === null) { plugin = tryLoad(pluginId); } // relative to config path if (plugin === null && pluginId.match(/^\.\.?($|\/)/)) { plugin = tryLoad(_path.default.resolve(_path.default.dirname(config.self_path), pluginId)); } if (plugin === null) { _logger.logger.error({ content: pluginId, prefix }, 'plugin not found. try npm install @{prefix}-@{content}'); throw Error(` ${prefix}-${pluginId} plugin not found. try "npm install ${prefix}-${pluginId}"`); } if (!isValid(plugin)) { _logger.logger.error({ content: pluginId }, '@{prefix}-@{content} plugin does not have the right code structure'); throw Error(`"${pluginId}" plugin does not have the right code structure`); } /* eslint new-cap:off */ try { plugin = isES6(plugin) ? new plugin.default(mergeConfig(config, pluginConfigs[pluginId]), params) : plugin(pluginConfigs[pluginId], params); } catch (error) { plugin = null; _logger.logger.error({ error, pluginId }, 'error loading a plugin @{pluginId}: @{error}'); } /* eslint new-cap:off */ if (plugin === null || !sanityCheck(plugin)) { _logger.logger.error({ content: pluginId, prefix }, "@{prefix}-@{content} doesn't look like a valid plugin"); throw Error(`sanity check has failed, "${pluginId}" is not a valid plugin`); } _logger.logger.warn({ content: pluginId, prefix }, 'Plugin successfully loaded: @{prefix}-@{content}'); return plugin; }); } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/lib/plugin-loader.ts"],"names":["tryLoad","path","require","err","code","MODULE_NOT_FOUND","mergeConfig","appConfig","pluginConfig","_","merge","isValid","plugin","isFunction","default","isES6","Object","keys","includes","loadPlugin","config","pluginConfigs","params","sanityCheck","prefix","map","pluginId","localPlugin","Path","resolve","__dirname","plugins","pluginDir","externalFilePlugin","match","dirname","self_path","logger","error","content","Error","warn"],"mappings":"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","sourcesContent":["import Path from 'path';\nimport _ from 'lodash';\nimport { Config, IPlugin } from '@verdaccio/types';\nimport { logger } from './logger';\nimport { MODULE_NOT_FOUND } from './constants';\n\n/**\n * Requires a module.\n * @param {*} path the module's path\n * @return {Object}\n */\nfunction tryLoad(path: string): any {\n  try {\n    return require(path);\n  } catch (err) {\n    if (err.code === MODULE_NOT_FOUND) {\n      return null;\n    }\n    throw err;\n  }\n}\n\nfunction mergeConfig(appConfig, pluginConfig): Config {\n  return _.merge(appConfig, pluginConfig);\n}\n\nfunction isValid(plugin): boolean {\n  return _.isFunction(plugin) || _.isFunction(plugin.default);\n}\n\nfunction isES6(plugin): boolean {\n  return Object.keys(plugin).includes('default');\n}\n\n// export type PluginGeneric<R, T extends IPlugin<R> = ;\n\n/**\n * Load a plugin following the rules\n * - First try to load from the internal directory plugins (which will disappear soon or later).\n * - A second attempt from the external plugin directory\n * - A third attempt from node_modules, in case to have multiple match as for instance verdaccio-ldap\n * and sinopia-ldap. All verdaccio prefix will have preferences.\n * @param {*} config a reference of the configuration settings\n * @param {*} pluginConfigs\n * @param {*} params a set of params to initialize the plugin\n * @param {*} sanityCheck callback that check the shape that should fulfill the plugin\n * @return {Array} list of plugins\n */\nexport default function loadPlugin<T extends IPlugin<T>>(\n  config: Config,\n  pluginConfigs: any = {},\n  params: any,\n  sanityCheck: any,\n  prefix: string = 'verdaccio'\n): any[] {\n  return Object.keys(pluginConfigs).map(\n    (pluginId: string): IPlugin<T> => {\n      let plugin;\n\n      const localPlugin = Path.resolve(__dirname + '/../plugins', pluginId);\n      // try local plugins first\n      plugin = tryLoad(localPlugin);\n\n      // try the external plugin directory\n      if (plugin === null && config.plugins) {\n        const pluginDir = config.plugins;\n        const externalFilePlugin = Path.resolve(pluginDir, pluginId);\n        plugin = tryLoad(externalFilePlugin);\n\n        // npm package\n        if (plugin === null && pluginId.match(/^[^\\.\\/]/)) {\n          plugin = tryLoad(Path.resolve(pluginDir, `${prefix}-${pluginId}`));\n          // compatibility for old sinopia plugins\n          if (!plugin) {\n            plugin = tryLoad(Path.resolve(pluginDir, `sinopia-${pluginId}`));\n          }\n        }\n      }\n\n      // npm package\n      if (plugin === null && pluginId.match(/^[^\\.\\/]/)) {\n        plugin = tryLoad(`${prefix}-${pluginId}`);\n        // compatibility for old sinopia plugins\n        if (!plugin) {\n          plugin = tryLoad(`sinopia-${pluginId}`);\n        }\n      }\n\n      if (plugin === null) {\n        plugin = tryLoad(pluginId);\n      }\n\n      // relative to config path\n      if (plugin === null && pluginId.match(/^\\.\\.?($|\\/)/)) {\n        plugin = tryLoad(Path.resolve(Path.dirname(config.self_path), pluginId));\n      }\n\n      if (plugin === null) {\n        logger.error(\n          { content: pluginId, prefix },\n          'plugin not found. try npm install @{prefix}-@{content}'\n        );\n        throw Error(`\n        ${prefix}-${pluginId} plugin not found. try \"npm install ${prefix}-${pluginId}\"`);\n      }\n\n      if (!isValid(plugin)) {\n        logger.error(\n          { content: pluginId },\n          '@{prefix}-@{content} plugin does not have the right code structure'\n        );\n        throw Error(`\"${pluginId}\" plugin does not have the right code structure`);\n      }\n\n      /* eslint new-cap:off */\n      try {\n        plugin = isES6(plugin)\n          ? new plugin.default(mergeConfig(config, pluginConfigs[pluginId]), params)\n          : plugin(pluginConfigs[pluginId], params);\n      } catch (error) {\n        plugin = null;\n        logger.error({ error, pluginId }, 'error loading a plugin @{pluginId}: @{error}');\n      }\n      /* eslint new-cap:off */\n\n      if (plugin === null || !sanityCheck(plugin)) {\n        logger.error(\n          { content: pluginId, prefix },\n          \"@{prefix}-@{content} doesn't look like a valid plugin\"\n        );\n        throw Error(`sanity check has failed, \"${pluginId}\" is not a valid plugin`);\n      }\n\n      logger.warn(\n        { content: pluginId, prefix },\n        'Plugin successfully loaded: @{prefix}-@{content}'\n      );\n      return plugin;\n    }\n  );\n}\n"]}