index.js 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. "use strict";
  2. Object.defineProperty(exports, "__esModule", {
  3. value: true
  4. });
  5. exports.default = loader;
  6. var _path = _interopRequireDefault(require("path"));
  7. var _semver = require("semver");
  8. var _package = _interopRequireDefault(require("postcss/package.json"));
  9. var _Warning = _interopRequireDefault(require("./Warning"));
  10. var _Error = _interopRequireDefault(require("./Error"));
  11. var _options = _interopRequireDefault(require("./options.json"));
  12. var _utils = require("./utils");
  13. function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
  14. let hasExplicitDependencyOnPostCSS = false;
  15. /**
  16. * **PostCSS Loader**
  17. *
  18. * Loads && processes CSS with [PostCSS](https://github.com/postcss/postcss)
  19. *
  20. * @method loader
  21. *
  22. * @param {String} content Source
  23. * @param {Object} sourceMap Source Map
  24. * @param {Object} meta Meta
  25. *
  26. * @return {callback} callback Result
  27. */
  28. async function loader(content, sourceMap, meta) {
  29. const options = this.getOptions(_options.default);
  30. const callback = this.async();
  31. const configOption = typeof options.postcssOptions === "undefined" || typeof options.postcssOptions.config === "undefined" ? true : options.postcssOptions.config;
  32. const postcssFactory = (0, _utils.getPostcssImplementation)(this, options.implementation);
  33. if (!postcssFactory) {
  34. callback(new Error(`The Postcss implementation "${options.implementation}" not found`));
  35. return;
  36. }
  37. let loadedConfig;
  38. if (configOption) {
  39. try {
  40. loadedConfig = await (0, _utils.loadConfig)(this, configOption, options.postcssOptions);
  41. } catch (error) {
  42. callback(error);
  43. return;
  44. }
  45. }
  46. const useSourceMap = typeof options.sourceMap !== "undefined" ? options.sourceMap : this.sourceMap;
  47. const {
  48. plugins,
  49. processOptions
  50. } = await (0, _utils.getPostcssOptions)(this, loadedConfig, options.postcssOptions);
  51. if (useSourceMap) {
  52. processOptions.map = {
  53. inline: false,
  54. annotation: false,
  55. ...processOptions.map
  56. };
  57. }
  58. if (sourceMap && processOptions.map) {
  59. processOptions.map.prev = (0, _utils.normalizeSourceMap)(sourceMap, this.context);
  60. }
  61. let root; // Reuse PostCSS AST from other loaders
  62. if (meta && meta.ast && meta.ast.type === "postcss" && (0, _semver.satisfies)(meta.ast.version, `^${_package.default.version}`)) {
  63. ({
  64. root
  65. } = meta.ast);
  66. }
  67. if (!root && options.execute) {
  68. // eslint-disable-next-line no-param-reassign
  69. content = (0, _utils.exec)(content, this);
  70. }
  71. let result;
  72. let processor;
  73. try {
  74. processor = postcssFactory(plugins);
  75. result = await processor.process(root || content, processOptions);
  76. } catch (error) {
  77. // Check postcss versions to avoid using PostCSS 7.
  78. // For caching reasons, we use the readFileSync and existsSync functions from the context,
  79. // not the functions from the `fs` module.
  80. if (!hasExplicitDependencyOnPostCSS && processor && processor.version && processor.version.startsWith("7.")) {
  81. // The `findPackageJsonDir` function returns `string` or `null`.
  82. // This is used to do for caching, that is, an explicit comparison with `undefined`
  83. // is used to make the condition body run once.
  84. const packageJSONDir = (0, _utils.findPackageJSONDir)(process.cwd(), this.fs.statSync);
  85. if (packageJSONDir) {
  86. let bufferOfPackageJSON;
  87. try {
  88. bufferOfPackageJSON = this.fs.readFileSync(_path.default.resolve(packageJSONDir, "package.json"), "utf8");
  89. } catch (_error) {// Nothing
  90. }
  91. if (bufferOfPackageJSON) {
  92. let pkg;
  93. try {
  94. pkg = JSON.parse(bufferOfPackageJSON);
  95. } catch (_error) {// Nothing
  96. }
  97. if (pkg) {
  98. if (!pkg.dependencies.postcss && !pkg.devDependencies.postcss) {
  99. this.emitWarning(new Error("Add postcss as project dependency. postcss is not a peer dependency for postcss-loader. " + "Use `npm install postcss` or `yarn add postcss`"));
  100. } else {
  101. hasExplicitDependencyOnPostCSS = true;
  102. }
  103. }
  104. }
  105. }
  106. }
  107. if (error.file) {
  108. this.addDependency(error.file);
  109. }
  110. if (error.name === "CssSyntaxError") {
  111. callback(new _Error.default(error));
  112. } else {
  113. callback(error);
  114. }
  115. return;
  116. }
  117. for (const warning of result.warnings()) {
  118. this.emitWarning(new _Warning.default(warning));
  119. }
  120. for (const message of result.messages) {
  121. // eslint-disable-next-line default-case
  122. switch (message.type) {
  123. case "dependency":
  124. this.addDependency(message.file);
  125. break;
  126. case "build-dependency":
  127. this.addBuildDependency(message.file);
  128. break;
  129. case "missing-dependency":
  130. this.addMissingDependency(message.file);
  131. break;
  132. case "context-dependency":
  133. this.addContextDependency(message.file);
  134. break;
  135. case "dir-dependency":
  136. this.addContextDependency(message.dir);
  137. break;
  138. case "asset":
  139. if (message.content && message.file) {
  140. this.emitFile(message.file, message.content, message.sourceMap, message.info);
  141. }
  142. }
  143. } // eslint-disable-next-line no-undefined
  144. let map = result.map ? result.map.toJSON() : undefined;
  145. if (map && useSourceMap) {
  146. map = (0, _utils.normalizeSourceMapAfterPostcss)(map, this.context);
  147. }
  148. const ast = {
  149. type: "postcss",
  150. version: result.processor.version,
  151. root: result.root
  152. };
  153. callback(null, result.css, map, {
  154. ast
  155. });
  156. }