ModuleScopePlugin.js 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. /**
  2. * Copyright (c) 2015-present, Facebook, Inc.
  3. *
  4. * This source code is licensed under the MIT license found in the
  5. * LICENSE file in the root directory of this source tree.
  6. */
  7. 'use strict';
  8. const chalk = require('chalk');
  9. const path = require('path');
  10. const os = require('os');
  11. class ModuleScopePlugin {
  12. constructor(appSrc, allowedFiles = []) {
  13. this.appSrcs = Array.isArray(appSrc) ? appSrc : [appSrc];
  14. this.allowedFiles = new Set(allowedFiles);
  15. this.allowedPaths = [...allowedFiles].map(path.dirname).filter(p => path.relative(p, process.cwd()) !== '');
  16. }
  17. apply(resolver) {
  18. const { appSrcs } = this;
  19. resolver.hooks.file.tapAsync(
  20. 'ModuleScopePlugin',
  21. (request, contextResolver, callback) => {
  22. // Unknown issuer, probably webpack internals
  23. if (!request.context.issuer) {
  24. return callback();
  25. }
  26. if (
  27. // If this resolves to a node_module, we don't care what happens next
  28. request.descriptionFileRoot.indexOf('/node_modules/') !== -1 ||
  29. request.descriptionFileRoot.indexOf('\\node_modules\\') !== -1 ||
  30. // Make sure this request was manual
  31. !request.__innerRequest_request
  32. ) {
  33. return callback();
  34. }
  35. // Resolve the issuer from our appSrc and make sure it's one of our files
  36. // Maybe an indexOf === 0 would be better?
  37. if (
  38. appSrcs.every(appSrc => {
  39. const relative = path.relative(appSrc, request.context.issuer);
  40. // If it's not in one of our app src or a subdirectory, not our request!
  41. return relative.startsWith('../') || relative.startsWith('..\\');
  42. })
  43. ) {
  44. return callback();
  45. }
  46. const requestFullPath = path.resolve(
  47. path.dirname(request.context.issuer),
  48. request.__innerRequest_request
  49. );
  50. if (this.allowedFiles.has(requestFullPath)) {
  51. return callback();
  52. }
  53. if (this.allowedPaths.some((allowedFile) => {
  54. return requestFullPath.startsWith(allowedFile);
  55. })) {
  56. return callback();
  57. }
  58. // Find path from src to the requested file
  59. // Error if in a parent directory of all given appSrcs
  60. if (
  61. appSrcs.every(appSrc => {
  62. const requestRelative = path.relative(appSrc, requestFullPath);
  63. return (
  64. requestRelative.startsWith('../') ||
  65. requestRelative.startsWith('..\\')
  66. );
  67. })
  68. ) {
  69. const scopeError = new Error(
  70. `You attempted to import ${chalk.cyan(
  71. request.__innerRequest_request
  72. )} which falls outside of the project ${chalk.cyan(
  73. 'src/'
  74. )} directory. ` +
  75. `Relative imports outside of ${chalk.cyan(
  76. 'src/'
  77. )} are not supported.` +
  78. os.EOL +
  79. `You can either move it inside ${chalk.cyan(
  80. 'src/'
  81. )}, or add a symlink to it from project's ${chalk.cyan(
  82. 'node_modules/'
  83. )}.`
  84. );
  85. Object.defineProperty(scopeError, '__module_scope_plugin', {
  86. value: true,
  87. writable: false,
  88. enumerable: false,
  89. });
  90. callback(scopeError, request);
  91. } else {
  92. callback();
  93. }
  94. }
  95. );
  96. }
  97. }
  98. module.exports = ModuleScopePlugin;