NodeStuffPlugin.js 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. /*
  2. MIT License http://www.opensource.org/licenses/mit-license.php
  3. Author Tobias Koppers @sokra
  4. */
  5. "use strict";
  6. const NodeStuffInWebError = require("./NodeStuffInWebError");
  7. const RuntimeGlobals = require("./RuntimeGlobals");
  8. const CachedConstDependency = require("./dependencies/CachedConstDependency");
  9. const ConstDependency = require("./dependencies/ConstDependency");
  10. const {
  11. evaluateToString,
  12. expressionIsUnsupported
  13. } = require("./javascript/JavascriptParserHelpers");
  14. const { relative } = require("./util/fs");
  15. const { parseResource } = require("./util/identifier");
  16. /** @typedef {import("webpack-sources").ReplaceSource} ReplaceSource */
  17. /** @typedef {import("./Compiler")} Compiler */
  18. /** @typedef {import("./Dependency")} Dependency */
  19. /** @typedef {import("./DependencyTemplates")} DependencyTemplates */
  20. /** @typedef {import("./RuntimeTemplate")} RuntimeTemplate */
  21. class NodeStuffPlugin {
  22. constructor(options) {
  23. this.options = options;
  24. }
  25. /**
  26. * Apply the plugin
  27. * @param {Compiler} compiler the compiler instance
  28. * @returns {void}
  29. */
  30. apply(compiler) {
  31. const options = this.options;
  32. compiler.hooks.compilation.tap(
  33. "NodeStuffPlugin",
  34. (compilation, { normalModuleFactory }) => {
  35. const handler = (parser, parserOptions) => {
  36. if (parserOptions.node === false) return;
  37. let localOptions = options;
  38. if (parserOptions.node) {
  39. localOptions = { ...localOptions, ...parserOptions.node };
  40. }
  41. if (localOptions.global !== false) {
  42. const withWarning = localOptions.global === "warn";
  43. parser.hooks.expression
  44. .for("global")
  45. .tap("NodeStuffPlugin", expr => {
  46. const dep = new ConstDependency(
  47. RuntimeGlobals.global,
  48. expr.range,
  49. [RuntimeGlobals.global]
  50. );
  51. dep.loc = expr.loc;
  52. parser.state.module.addPresentationalDependency(dep);
  53. // TODO webpack 6 remove
  54. if (withWarning) {
  55. parser.state.module.addWarning(
  56. new NodeStuffInWebError(
  57. dep.loc,
  58. "global",
  59. "The global namespace object is Node.js feature and doesn't present in browser."
  60. )
  61. );
  62. }
  63. });
  64. }
  65. const setModuleConstant = (expressionName, fn, warning) => {
  66. parser.hooks.expression
  67. .for(expressionName)
  68. .tap("NodeStuffPlugin", expr => {
  69. const dep = new CachedConstDependency(
  70. JSON.stringify(fn(parser.state.module)),
  71. expr.range,
  72. expressionName
  73. );
  74. dep.loc = expr.loc;
  75. parser.state.module.addPresentationalDependency(dep);
  76. // TODO webpack 6 remove
  77. if (warning) {
  78. parser.state.module.addWarning(
  79. new NodeStuffInWebError(dep.loc, expressionName, warning)
  80. );
  81. }
  82. return true;
  83. });
  84. };
  85. const setConstant = (expressionName, value, warning) =>
  86. setModuleConstant(expressionName, () => value, warning);
  87. const context = compiler.context;
  88. if (localOptions.__filename) {
  89. switch (localOptions.__filename) {
  90. case "mock":
  91. setConstant("__filename", "/index.js");
  92. break;
  93. case "warn-mock":
  94. setConstant(
  95. "__filename",
  96. "/index.js",
  97. "The __filename is Node.js feature and doesn't present in browser."
  98. );
  99. break;
  100. case true:
  101. setModuleConstant("__filename", module =>
  102. relative(compiler.inputFileSystem, context, module.resource)
  103. );
  104. break;
  105. }
  106. parser.hooks.evaluateIdentifier
  107. .for("__filename")
  108. .tap("NodeStuffPlugin", expr => {
  109. if (!parser.state.module) return;
  110. const resource = parseResource(parser.state.module.resource);
  111. return evaluateToString(resource.path)(expr);
  112. });
  113. }
  114. if (localOptions.__dirname) {
  115. switch (localOptions.__dirname) {
  116. case "mock":
  117. setConstant("__dirname", "/");
  118. break;
  119. case "warn-mock":
  120. setConstant(
  121. "__dirname",
  122. "/",
  123. "The __dirname is Node.js feature and doesn't present in browser."
  124. );
  125. break;
  126. case true:
  127. setModuleConstant("__dirname", module =>
  128. relative(compiler.inputFileSystem, context, module.context)
  129. );
  130. break;
  131. }
  132. parser.hooks.evaluateIdentifier
  133. .for("__dirname")
  134. .tap("NodeStuffPlugin", expr => {
  135. if (!parser.state.module) return;
  136. return evaluateToString(parser.state.module.context)(expr);
  137. });
  138. }
  139. parser.hooks.expression
  140. .for("require.extensions")
  141. .tap(
  142. "NodeStuffPlugin",
  143. expressionIsUnsupported(
  144. parser,
  145. "require.extensions is not supported by webpack. Use a loader instead."
  146. )
  147. );
  148. };
  149. normalModuleFactory.hooks.parser
  150. .for("javascript/auto")
  151. .tap("NodeStuffPlugin", handler);
  152. normalModuleFactory.hooks.parser
  153. .for("javascript/dynamic")
  154. .tap("NodeStuffPlugin", handler);
  155. }
  156. );
  157. }
  158. }
  159. module.exports = NodeStuffPlugin;