sass-graph.js 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. 'use strict';
  2. var fs = require('fs');
  3. var path = require('path');
  4. var _ = require('lodash');
  5. var glob = require('glob');
  6. var parseImports = require('./parse-imports');
  7. // resolve a sass module to a path
  8. function resolveSassPath(sassPath, loadPaths, extensions) {
  9. // trim sass file extensions
  10. var re = new RegExp('(\.('+extensions.join('|')+'))$', 'i');
  11. var sassPathName = sassPath.replace(re, '');
  12. // check all load paths
  13. var i, j, length = loadPaths.length, scssPath, partialPath;
  14. for (i = 0; i < length; i++) {
  15. for (j = 0; j < extensions.length; j++) {
  16. scssPath = path.normalize(loadPaths[i] + '/' + sassPathName + '.' + extensions[j]);
  17. try {
  18. if (fs.lstatSync(scssPath).isFile()) {
  19. return scssPath;
  20. }
  21. } catch (e) {}
  22. }
  23. // special case for _partials
  24. for (j = 0; j < extensions.length; j++) {
  25. scssPath = path.normalize(loadPaths[i] + '/' + sassPathName + '.' + extensions[j]);
  26. partialPath = path.join(path.dirname(scssPath), '_' + path.basename(scssPath));
  27. try {
  28. if (fs.lstatSync(partialPath).isFile()) {
  29. return partialPath;
  30. }
  31. } catch (e) {}
  32. }
  33. }
  34. // File to import not found or unreadable so we assume this is a custom import
  35. return false;
  36. }
  37. function Graph(options, dir) {
  38. this.dir = dir;
  39. this.extensions = options.extensions || [];
  40. this.exclude = options.exclude instanceof RegExp ? options.exclude : null;
  41. this.index = {};
  42. this.follow = options.follow || false;
  43. this.loadPaths = _(options.loadPaths).map(function(p) {
  44. return path.resolve(p);
  45. }).value();
  46. if (dir) {
  47. var graph = this;
  48. _.each(glob.sync(dir+'/**/*.@('+this.extensions.join('|')+')', { dot: true, nodir: true, follow: this.follow }), function(file) {
  49. try {
  50. graph.addFile(path.resolve(file));
  51. } catch (e) {}
  52. });
  53. }
  54. }
  55. // add a sass file to the graph
  56. Graph.prototype.addFile = function(filepath, parent) {
  57. if (this.exclude !== null && this.exclude.test(filepath)) return;
  58. var entry = this.index[filepath] = this.index[filepath] || {
  59. imports: [],
  60. importedBy: [],
  61. modified: fs.statSync(filepath).mtime
  62. };
  63. var resolvedParent;
  64. var isIndentedSyntax = path.extname(filepath) === '.sass';
  65. var imports = parseImports(fs.readFileSync(filepath, 'utf-8'), isIndentedSyntax);
  66. var cwd = path.dirname(filepath);
  67. var i, length = imports.length, loadPaths, resolved;
  68. for (i = 0; i < length; i++) {
  69. loadPaths = _([cwd, this.dir]).concat(this.loadPaths).filter().uniq().value();
  70. resolved = resolveSassPath(imports[i], loadPaths, this.extensions);
  71. if (!resolved) continue;
  72. // check exclcude regex
  73. if (this.exclude !== null && this.exclude.test(resolved)) continue;
  74. // recurse into dependencies if not already enumerated
  75. if (!_.includes(entry.imports, resolved)) {
  76. entry.imports.push(resolved);
  77. this.addFile(fs.realpathSync(resolved), filepath);
  78. }
  79. }
  80. // add link back to parent
  81. if (parent) {
  82. resolvedParent = _(parent).intersection(this.loadPaths).value();
  83. if (resolvedParent) {
  84. resolvedParent = parent.substr(parent.indexOf(resolvedParent));
  85. } else {
  86. resolvedParent = parent;
  87. }
  88. // check exclcude regex
  89. if (!(this.exclude !== null && this.exclude.test(resolvedParent))) {
  90. entry.importedBy.push(resolvedParent);
  91. }
  92. }
  93. };
  94. // visits all files that are ancestors of the provided file
  95. Graph.prototype.visitAncestors = function(filepath, callback) {
  96. this.visit(filepath, callback, function(err, node) {
  97. if (err || !node) return [];
  98. return node.importedBy;
  99. });
  100. };
  101. // visits all files that are descendents of the provided file
  102. Graph.prototype.visitDescendents = function(filepath, callback) {
  103. this.visit(filepath, callback, function(err, node) {
  104. if (err || !node) return [];
  105. return node.imports;
  106. });
  107. };
  108. // a generic visitor that uses an edgeCallback to find the edges to traverse for a node
  109. Graph.prototype.visit = function(filepath, callback, edgeCallback, visited) {
  110. filepath = fs.realpathSync(filepath);
  111. var visited = visited || [];
  112. if (!this.index.hasOwnProperty(filepath)) {
  113. edgeCallback('Graph doesn\'t contain ' + filepath, null);
  114. }
  115. var edges = edgeCallback(null, this.index[filepath]);
  116. var i, length = edges.length;
  117. for (i = 0; i < length; i++) {
  118. if (!_.includes(visited, edges[i])) {
  119. visited.push(edges[i]);
  120. callback(edges[i], this.index[edges[i]]);
  121. this.visit(edges[i], callback, edgeCallback, visited);
  122. }
  123. }
  124. };
  125. function processOptions(options) {
  126. return Object.assign({
  127. loadPaths: [process.cwd()],
  128. extensions: ['scss', 'sass'],
  129. }, options);
  130. }
  131. module.exports.parseFile = function(filepath, options) {
  132. if (fs.lstatSync(filepath).isFile()) {
  133. filepath = path.resolve(filepath);
  134. options = processOptions(options);
  135. var graph = new Graph(options);
  136. graph.addFile(filepath);
  137. return graph;
  138. }
  139. // throws
  140. };
  141. module.exports.parseDir = function(dirpath, options) {
  142. if (fs.lstatSync(dirpath).isDirectory()) {
  143. dirpath = path.resolve(dirpath);
  144. options = processOptions(options);
  145. var graph = new Graph(options, dirpath);
  146. return graph;
  147. }
  148. // throws
  149. };