| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160 | 'use strict';const webpack = require('webpack');const createDomain = require('./createDomain');/** * A Entry, it can be of type string or string[] or Object<string | string[],string> * @typedef {(string[] | string | Object<string | string[],string>)} Entry *//** * Add entries Method * @param {?Object} config - Webpack config * @param {?Object} options - Dev-Server options * @param {?Object} server * @returns {void} */function addEntries(config, options, server) {  if (options.inline !== false) {    // we're stubbing the app in this method as it's static and doesn't require    // a server to be supplied. createDomain requires an app with the    // address() signature.    const app = server || {      address() {        return { port: options.port };      },    };    /** @type {string} */    const domain = createDomain(options, app);    /** @type {string} */    const sockHost = options.sockHost ? `&sockHost=${options.sockHost}` : '';    /** @type {string} */    const sockPath = options.sockPath ? `&sockPath=${options.sockPath}` : '';    /** @type {string} */    const sockPort = options.sockPort ? `&sockPort=${options.sockPort}` : '';    /** @type {string} */    const clientEntry = `${require.resolve(      '../../client/'    )}?${domain}${sockHost}${sockPath}${sockPort}`;    /** @type {(string[] | string)} */    let hotEntry;    if (options.hotOnly) {      hotEntry = require.resolve('webpack/hot/only-dev-server');    } else if (options.hot) {      hotEntry = require.resolve('webpack/hot/dev-server');    }    /**     * prependEntry Method     * @param {Entry} originalEntry     * @param {Entry} additionalEntries     * @returns {Entry}     */    const prependEntry = (originalEntry, additionalEntries) => {      if (typeof originalEntry === 'function') {        return () =>          Promise.resolve(originalEntry()).then((entry) =>            prependEntry(entry, additionalEntries)          );      }      if (typeof originalEntry === 'object' && !Array.isArray(originalEntry)) {        /** @type {Object<string,string>} */        const clone = {};        Object.keys(originalEntry).forEach((key) => {          // entry[key] should be a string here          const entryDescription = originalEntry[key];          if (typeof entryDescription === 'object' && entryDescription.import) {            clone[key] = Object.assign({}, entryDescription, {              import: prependEntry(entryDescription.import, additionalEntries),            });          } else {            clone[key] = prependEntry(entryDescription, additionalEntries);          }        });        return clone;      }      // in this case, entry is a string or an array.      // make sure that we do not add duplicates.      /** @type {Entry} */      const entriesClone = additionalEntries.slice(0);      [].concat(originalEntry).forEach((newEntry) => {        if (!entriesClone.includes(newEntry)) {          entriesClone.push(newEntry);        }      });      return entriesClone;    };    /**     *     * Description of the option for checkInject method     * @typedef {Function} checkInjectOptionsParam     * @param {Object} _config - compilerConfig     * @return {Boolean}     */    /**     *     * @param {Boolean | checkInjectOptionsParam} option - inject(Hot|Client) it is Boolean | fn => Boolean     * @param {Object} _config     * @param {Boolean} defaultValue     * @return {Boolean}     */    // eslint-disable-next-line no-shadow    const checkInject = (option, _config, defaultValue) => {      if (typeof option === 'boolean') return option;      if (typeof option === 'function') return option(_config);      return defaultValue;    };    // eslint-disable-next-line no-shadow    [].concat(config).forEach((config) => {      /** @type {Boolean} */      const webTarget = [        'web',        'webworker',        'electron-renderer',        'node-webkit',        undefined, // eslint-disable-line        null,      ].includes(config.target);      /** @type {Entry} */      const additionalEntries = checkInject(        options.injectClient,        config,        webTarget      )        ? [clientEntry]        : [];      if (hotEntry && checkInject(options.injectHot, config, true)) {        additionalEntries.push(hotEntry);      }      config.entry = prependEntry(config.entry || './src', additionalEntries);      if (options.hot || options.hotOnly) {        config.plugins = config.plugins || [];        if (          !config.plugins.find(            // Check for the name rather than the constructor reference in case            // there are multiple copies of webpack installed            (plugin) => plugin.constructor.name === 'HotModuleReplacementPlugin'          )        ) {          config.plugins.push(new webpack.HotModuleReplacementPlugin());        }      }    });  }}module.exports = addEntries;
 |