123 lines
16 KiB
JavaScript
123 lines
16 KiB
JavaScript
|
'use strict';
|
||
|
|
||
|
var _docsUrl = require('../docsUrl');
|
||
|
|
||
|
var _docsUrl2 = _interopRequireDefault(_docsUrl);
|
||
|
|
||
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
||
|
|
||
|
module.exports = {
|
||
|
meta: {
|
||
|
docs: {
|
||
|
url: (0, _docsUrl2.default)('first')
|
||
|
},
|
||
|
fixable: 'code'
|
||
|
},
|
||
|
|
||
|
create: function (context) {
|
||
|
function isPossibleDirective(node) {
|
||
|
return node.type === 'ExpressionStatement' && node.expression.type === 'Literal' && typeof node.expression.value === 'string';
|
||
|
}
|
||
|
|
||
|
return {
|
||
|
'Program': function (n) {
|
||
|
const body = n.body,
|
||
|
absoluteFirst = context.options[0] === 'absolute-first',
|
||
|
message = 'Import in body of module; reorder to top.',
|
||
|
sourceCode = context.getSourceCode(),
|
||
|
originSourceCode = sourceCode.getText();
|
||
|
let nonImportCount = 0,
|
||
|
anyExpressions = false,
|
||
|
anyRelative = false,
|
||
|
lastLegalImp = null,
|
||
|
errorInfos = [],
|
||
|
shouldSort = true,
|
||
|
lastSortNodesIndex = 0;
|
||
|
body.forEach(function (node, index) {
|
||
|
if (!anyExpressions && isPossibleDirective(node)) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
anyExpressions = true;
|
||
|
|
||
|
if (node.type === 'ImportDeclaration') {
|
||
|
if (absoluteFirst) {
|
||
|
if (/^\./.test(node.source.value)) {
|
||
|
anyRelative = true;
|
||
|
} else if (anyRelative) {
|
||
|
context.report({
|
||
|
node: node.source,
|
||
|
message: 'Absolute imports should come before relative imports.'
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
if (nonImportCount > 0) {
|
||
|
for (let variable of context.getDeclaredVariables(node)) {
|
||
|
if (!shouldSort) break;
|
||
|
const references = variable.references;
|
||
|
if (references.length) {
|
||
|
for (let reference of references) {
|
||
|
if (reference.identifier.range[0] < node.range[1]) {
|
||
|
shouldSort = false;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
shouldSort && (lastSortNodesIndex = errorInfos.length);
|
||
|
errorInfos.push({
|
||
|
node,
|
||
|
range: [body[index - 1].range[1], node.range[1]]
|
||
|
});
|
||
|
} else {
|
||
|
lastLegalImp = node;
|
||
|
}
|
||
|
} else {
|
||
|
nonImportCount++;
|
||
|
}
|
||
|
});
|
||
|
if (!errorInfos.length) return;
|
||
|
errorInfos.forEach(function (errorInfo, index) {
|
||
|
const node = errorInfo.node,
|
||
|
infos = {
|
||
|
node,
|
||
|
message
|
||
|
};
|
||
|
if (index < lastSortNodesIndex) {
|
||
|
infos.fix = function (fixer) {
|
||
|
return fixer.insertTextAfter(node, '');
|
||
|
};
|
||
|
} else if (index === lastSortNodesIndex) {
|
||
|
const sortNodes = errorInfos.slice(0, lastSortNodesIndex + 1);
|
||
|
infos.fix = function (fixer) {
|
||
|
const removeFixers = sortNodes.map(function (_errorInfo) {
|
||
|
return fixer.removeRange(_errorInfo.range);
|
||
|
}),
|
||
|
range = [0, removeFixers[removeFixers.length - 1].range[1]];
|
||
|
let insertSourceCode = sortNodes.map(function (_errorInfo) {
|
||
|
const nodeSourceCode = String.prototype.slice.apply(originSourceCode, _errorInfo.range);
|
||
|
if (/\S/.test(nodeSourceCode[0])) {
|
||
|
return '\n' + nodeSourceCode;
|
||
|
}
|
||
|
return nodeSourceCode;
|
||
|
}).join(''),
|
||
|
insertFixer = null,
|
||
|
replaceSourceCode = '';
|
||
|
if (!lastLegalImp) {
|
||
|
insertSourceCode = insertSourceCode.trim() + insertSourceCode.match(/^(\s+)/)[0];
|
||
|
}
|
||
|
insertFixer = lastLegalImp ? fixer.insertTextAfter(lastLegalImp, insertSourceCode) : fixer.insertTextBefore(body[0], insertSourceCode);
|
||
|
const fixers = [insertFixer].concat(removeFixers);
|
||
|
fixers.forEach(function (computedFixer, i) {
|
||
|
replaceSourceCode += originSourceCode.slice(fixers[i - 1] ? fixers[i - 1].range[1] : 0, computedFixer.range[0]) + computedFixer.text;
|
||
|
});
|
||
|
return fixer.replaceTextRange(range, replaceSourceCode);
|
||
|
};
|
||
|
}
|
||
|
context.report(infos);
|
||
|
});
|
||
|
}
|
||
|
};
|
||
|
}
|
||
|
};
|
||
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbInJ1bGVzL2ZpcnN0LmpzIl0sIm5hbWVzIjpbIm1vZHVsZSIsImV4cG9ydHMiLCJtZXRhIiwiZG9jcyIsInVybCIsImZpeGFibGUiLCJjcmVhdGUiLCJjb250ZXh0IiwiaXNQb3NzaWJsZURpcmVjdGl2ZSIsIm5vZGUiLCJ0eXBlIiwiZXhwcmVzc2lvbiIsInZhbHVlIiwibiIsImJvZHkiLCJhYnNvbHV0ZUZpcnN0Iiwib3B0aW9ucyIsIm1lc3NhZ2UiLCJzb3VyY2VDb2RlIiwiZ2V0U291cmNlQ29kZSIsIm9yaWdpblNvdXJjZUNvZGUiLCJnZXRUZXh0Iiwibm9uSW1wb3J0Q291bnQiLCJhbnlFeHByZXNzaW9ucyIsImFueVJlbGF0aXZlIiwibGFzdExlZ2FsSW1wIiwiZXJyb3JJbmZvcyIsInNob3VsZFNvcnQiLCJsYXN0U29ydE5vZGVzSW5kZXgiLCJmb3JFYWNoIiwiaW5kZXgiLCJ0ZXN0Iiwic291cmNlIiwicmVwb3J0IiwidmFyaWFibGUiLCJnZXREZWNsYXJlZFZhcmlhYmxlcyIsInJlZmVyZW5jZXMiLCJsZW5ndGgiLCJyZWZlcmVuY2UiLCJpZGVudGlmaWVyIiwicmFuZ2UiLCJwdXNoIiwiZXJyb3JJbmZvIiwiaW5mb3MiLCJmaXgiLCJmaXhlciIsImluc2VydFRleHRBZnRlciIsInNvcnROb2RlcyIsInNsaWNlIiwicmVtb3ZlRml4ZXJzIiwibWFwIiwiX2Vycm9ySW5mbyIsInJlbW92ZVJhbmdlIiwiaW5zZXJ0U291cmNlQ29kZSIsIm5vZGVTb3VyY2VDb2RlIiwiU3RyaW5nIiwicHJvdG90eXBlIiwiYXBwbHkiLCJqb2luIiwiaW5zZXJ0Rml4ZXIiLCJyZXBsYWNlU291cmNlQ29kZSIsInRyaW0iLCJtYXRjaCIsImluc2VydFRleHRCZWZvcmUiLCJmaXhlcnMiLCJjb25jYXQiLCJjb21wdXRlZEZpeGVyIiwiaSIsInRleHQiLCJyZXBsYWNlVGV4dFJhbmdlIl0sIm1hcHBpbmdzIjoiOztBQUFBOzs7Ozs7QUFFQUEsT0FBT0MsT0FBUCxHQUFp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
|