123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674 |
- "use strict";
- const astUtils = require("./utils/ast-utils");
- function containsLineTerminator(str) {
- return astUtils.LINEBREAK_MATCHER.test(str);
- }
- function last(arr) {
- return arr[arr.length - 1];
- }
- function isSingleLine(node) {
- return (node.loc.end.line === node.loc.start.line);
- }
- function isSingleLineProperties(properties) {
- const [firstProp] = properties,
- lastProp = last(properties);
- return firstProp.loc.start.line === lastProp.loc.end.line;
- }
- function initOptionProperty(toOptions, fromOptions) {
- toOptions.mode = fromOptions.mode || "strict";
-
- if (typeof fromOptions.beforeColon !== "undefined") {
- toOptions.beforeColon = +fromOptions.beforeColon;
- } else {
- toOptions.beforeColon = 0;
- }
-
- if (typeof fromOptions.afterColon !== "undefined") {
- toOptions.afterColon = +fromOptions.afterColon;
- } else {
- toOptions.afterColon = 1;
- }
-
- if (typeof fromOptions.align !== "undefined") {
- if (typeof fromOptions.align === "object") {
- toOptions.align = fromOptions.align;
- } else {
- toOptions.align = {
- on: fromOptions.align,
- mode: toOptions.mode,
- beforeColon: toOptions.beforeColon,
- afterColon: toOptions.afterColon
- };
- }
- }
- return toOptions;
- }
- function initOptions(toOptions, fromOptions) {
- if (typeof fromOptions.align === "object") {
-
- toOptions.align = initOptionProperty({}, fromOptions.align);
- toOptions.align.on = fromOptions.align.on || "colon";
- toOptions.align.mode = fromOptions.align.mode || "strict";
- toOptions.multiLine = initOptionProperty({}, (fromOptions.multiLine || fromOptions));
- toOptions.singleLine = initOptionProperty({}, (fromOptions.singleLine || fromOptions));
- } else {
- toOptions.multiLine = initOptionProperty({}, (fromOptions.multiLine || fromOptions));
- toOptions.singleLine = initOptionProperty({}, (fromOptions.singleLine || fromOptions));
-
- if (toOptions.multiLine.align) {
- toOptions.align = {
- on: toOptions.multiLine.align.on,
- mode: toOptions.multiLine.align.mode || toOptions.multiLine.mode,
- beforeColon: toOptions.multiLine.align.beforeColon,
- afterColon: toOptions.multiLine.align.afterColon
- };
- }
- }
- return toOptions;
- }
- module.exports = {
- meta: {
- type: "layout",
- docs: {
- description: "enforce consistent spacing between keys and values in object literal properties",
- category: "Stylistic Issues",
- recommended: false,
- url: "https://eslint.org/docs/rules/key-spacing"
- },
- fixable: "whitespace",
- schema: [{
- anyOf: [
- {
- type: "object",
- properties: {
- align: {
- anyOf: [
- {
- enum: ["colon", "value"]
- },
- {
- type: "object",
- properties: {
- mode: {
- enum: ["strict", "minimum"]
- },
- on: {
- enum: ["colon", "value"]
- },
- beforeColon: {
- type: "boolean"
- },
- afterColon: {
- type: "boolean"
- }
- },
- additionalProperties: false
- }
- ]
- },
- mode: {
- enum: ["strict", "minimum"]
- },
- beforeColon: {
- type: "boolean"
- },
- afterColon: {
- type: "boolean"
- }
- },
- additionalProperties: false
- },
- {
- type: "object",
- properties: {
- singleLine: {
- type: "object",
- properties: {
- mode: {
- enum: ["strict", "minimum"]
- },
- beforeColon: {
- type: "boolean"
- },
- afterColon: {
- type: "boolean"
- }
- },
- additionalProperties: false
- },
- multiLine: {
- type: "object",
- properties: {
- align: {
- anyOf: [
- {
- enum: ["colon", "value"]
- },
- {
- type: "object",
- properties: {
- mode: {
- enum: ["strict", "minimum"]
- },
- on: {
- enum: ["colon", "value"]
- },
- beforeColon: {
- type: "boolean"
- },
- afterColon: {
- type: "boolean"
- }
- },
- additionalProperties: false
- }
- ]
- },
- mode: {
- enum: ["strict", "minimum"]
- },
- beforeColon: {
- type: "boolean"
- },
- afterColon: {
- type: "boolean"
- }
- },
- additionalProperties: false
- }
- },
- additionalProperties: false
- },
- {
- type: "object",
- properties: {
- singleLine: {
- type: "object",
- properties: {
- mode: {
- enum: ["strict", "minimum"]
- },
- beforeColon: {
- type: "boolean"
- },
- afterColon: {
- type: "boolean"
- }
- },
- additionalProperties: false
- },
- multiLine: {
- type: "object",
- properties: {
- mode: {
- enum: ["strict", "minimum"]
- },
- beforeColon: {
- type: "boolean"
- },
- afterColon: {
- type: "boolean"
- }
- },
- additionalProperties: false
- },
- align: {
- type: "object",
- properties: {
- mode: {
- enum: ["strict", "minimum"]
- },
- on: {
- enum: ["colon", "value"]
- },
- beforeColon: {
- type: "boolean"
- },
- afterColon: {
- type: "boolean"
- }
- },
- additionalProperties: false
- }
- },
- additionalProperties: false
- }
- ]
- }],
- messages: {
- extraKey: "Extra space after {{computed}}key '{{key}}'.",
- extraValue: "Extra space before value for {{computed}}key '{{key}}'.",
- missingKey: "Missing space after {{computed}}key '{{key}}'.",
- missingValue: "Missing space before value for {{computed}}key '{{key}}'."
- }
- },
- create(context) {
-
- const options = context.options[0] || {},
- ruleOptions = initOptions({}, options),
- multiLineOptions = ruleOptions.multiLine,
- singleLineOptions = ruleOptions.singleLine,
- alignmentOptions = ruleOptions.align || null;
- const sourceCode = context.getSourceCode();
-
- function continuesPropertyGroup(lastMember, candidate) {
- const groupEndLine = lastMember.loc.start.line,
- candidateStartLine = candidate.loc.start.line;
- if (candidateStartLine - groupEndLine <= 1) {
- return true;
- }
-
- const leadingComments = sourceCode.getCommentsBefore(candidate);
- if (
- leadingComments.length &&
- leadingComments[0].loc.start.line - groupEndLine <= 1 &&
- candidateStartLine - last(leadingComments).loc.end.line <= 1
- ) {
- for (let i = 1; i < leadingComments.length; i++) {
- if (leadingComments[i].loc.start.line - leadingComments[i - 1].loc.end.line > 1) {
- return false;
- }
- }
- return true;
- }
- return false;
- }
-
- function isKeyValueProperty(property) {
- return !(
- (property.method ||
- property.shorthand ||
- property.kind !== "init" || property.type !== "Property")
- );
- }
-
- function getLastTokenBeforeColon(node) {
- const colonToken = sourceCode.getTokenAfter(node, astUtils.isColonToken);
- return sourceCode.getTokenBefore(colonToken);
- }
-
- function getNextColon(node) {
- return sourceCode.getTokenAfter(node, astUtils.isColonToken);
- }
-
- function getKey(property) {
- const key = property.key;
- if (property.computed) {
- return sourceCode.getText().slice(key.range[0], key.range[1]);
- }
- return astUtils.getStaticPropertyName(property);
- }
-
- function report(property, side, whitespace, expected, mode) {
- const diff = whitespace.length - expected,
- nextColon = getNextColon(property.key),
- tokenBeforeColon = sourceCode.getTokenBefore(nextColon, { includeComments: true }),
- tokenAfterColon = sourceCode.getTokenAfter(nextColon, { includeComments: true }),
- isKeySide = side === "key",
- isExtra = diff > 0,
- diffAbs = Math.abs(diff),
- spaces = Array(diffAbs + 1).join(" ");
- const locStart = isKeySide ? tokenBeforeColon.loc.end : nextColon.loc.start;
- const locEnd = isKeySide ? nextColon.loc.start : tokenAfterColon.loc.start;
- const missingLoc = isKeySide ? tokenBeforeColon.loc : tokenAfterColon.loc;
- const loc = isExtra ? { start: locStart, end: locEnd } : missingLoc;
- if ((
- diff && mode === "strict" ||
- diff < 0 && mode === "minimum" ||
- diff > 0 && !expected && mode === "minimum") &&
- !(expected && containsLineTerminator(whitespace))
- ) {
- let fix;
- if (isExtra) {
- let range;
-
- if (isKeySide) {
- range = [tokenBeforeColon.range[1], tokenBeforeColon.range[1] + diffAbs];
- } else {
- range = [tokenAfterColon.range[0] - diffAbs, tokenAfterColon.range[0]];
- }
- fix = function(fixer) {
- return fixer.removeRange(range);
- };
- } else {
-
- if (isKeySide) {
- fix = function(fixer) {
- return fixer.insertTextAfter(tokenBeforeColon, spaces);
- };
- } else {
- fix = function(fixer) {
- return fixer.insertTextBefore(tokenAfterColon, spaces);
- };
- }
- }
- let messageId = "";
- if (isExtra) {
- messageId = side === "key" ? "extraKey" : "extraValue";
- } else {
- messageId = side === "key" ? "missingKey" : "missingValue";
- }
- context.report({
- node: property[side],
- loc,
- messageId,
- data: {
- computed: property.computed ? "computed " : "",
- key: getKey(property)
- },
- fix
- });
- }
- }
-
- function getKeyWidth(property) {
- const startToken = sourceCode.getFirstToken(property);
- const endToken = getLastTokenBeforeColon(property.key);
- return endToken.range[1] - startToken.range[0];
- }
-
- function getPropertyWhitespace(property) {
- const whitespace = /(\s*):(\s*)/u.exec(sourceCode.getText().slice(
- property.key.range[1], property.value.range[0]
- ));
- if (whitespace) {
- return {
- beforeColon: whitespace[1],
- afterColon: whitespace[2]
- };
- }
- return null;
- }
-
- function createGroups(node) {
- if (node.properties.length === 1) {
- return [node.properties];
- }
- return node.properties.reduce((groups, property) => {
- const currentGroup = last(groups),
- prev = last(currentGroup);
- if (!prev || continuesPropertyGroup(prev, property)) {
- currentGroup.push(property);
- } else {
- groups.push([property]);
- }
- return groups;
- }, [
- []
- ]);
- }
-
- function verifyGroupAlignment(properties) {
- const length = properties.length,
- widths = properties.map(getKeyWidth),
- align = alignmentOptions.on;
- let targetWidth = Math.max(...widths),
- beforeColon, afterColon, mode;
- if (alignmentOptions && length > 1) {
- beforeColon = alignmentOptions.beforeColon;
- afterColon = alignmentOptions.afterColon;
- mode = alignmentOptions.mode;
- } else {
- beforeColon = multiLineOptions.beforeColon;
- afterColon = multiLineOptions.afterColon;
- mode = alignmentOptions.mode;
- }
-
- targetWidth += (align === "colon" ? beforeColon : afterColon);
- for (let i = 0; i < length; i++) {
- const property = properties[i];
- const whitespace = getPropertyWhitespace(property);
- if (whitespace) {
- const width = widths[i];
- if (align === "value") {
- report(property, "key", whitespace.beforeColon, beforeColon, mode);
- report(property, "value", whitespace.afterColon, targetWidth - width, mode);
- } else {
- report(property, "key", whitespace.beforeColon, targetWidth - width, mode);
- report(property, "value", whitespace.afterColon, afterColon, mode);
- }
- }
- }
- }
-
- function verifySpacing(node, lineOptions) {
- const actual = getPropertyWhitespace(node);
- if (actual) {
- report(node, "key", actual.beforeColon, lineOptions.beforeColon, lineOptions.mode);
- report(node, "value", actual.afterColon, lineOptions.afterColon, lineOptions.mode);
- }
- }
-
- function verifyListSpacing(properties, lineOptions) {
- const length = properties.length;
- for (let i = 0; i < length; i++) {
- verifySpacing(properties[i], lineOptions);
- }
- }
-
- function verifyAlignment(node) {
- createGroups(node).forEach(group => {
- const properties = group.filter(isKeyValueProperty);
- if (properties.length > 0 && isSingleLineProperties(properties)) {
- verifyListSpacing(properties, multiLineOptions);
- } else {
- verifyGroupAlignment(properties);
- }
- });
- }
-
-
-
- if (alignmentOptions) {
- return {
- ObjectExpression(node) {
- if (isSingleLine(node)) {
- verifyListSpacing(node.properties.filter(isKeyValueProperty), singleLineOptions);
- } else {
- verifyAlignment(node);
- }
- }
- };
- }
-
- return {
- Property(node) {
- verifySpacing(node, isSingleLine(node.parent) ? singleLineOptions : multiLineOptions);
- }
- };
- }
- };
|