Skip to content

Commit 4d7600e

Browse files
authored
Use official pug-lint to lint pug files (#65)
* WIP: Make sure to pass items to puglint * Group test cases instead of defining just valid/invalid cases * Make more complicated tests for pug-lint * Make sure pug-lint works well with indentation * Improve integration with Attribute Brackets rule * Make sure we work properly with one-line statements * Make sure we describe the rule for others * Make sure we mention pug-lint in main configuration
1 parent 5cd3cdd commit 4d7600e

File tree

10 files changed

+930
-355
lines changed

10 files changed

+930
-355
lines changed

README.md

+4
Original file line numberDiff line numberDiff line change
@@ -61,3 +61,7 @@ Then configure the rules you want to use under the rules section.
6161
* [`react-pug/quotes`](./docs/rules/quotes.md): Manage quotes in Pug
6262
* [`react-pug/uses-react`](./docs/rules/uses-react.md): Prevent React to be marked as unused
6363
* [`react-pug/uses-vars`](./docs/rules/uses-vars.md): Prevent variables used in Pug to be marked as unused
64+
65+
Experimental:
66+
67+
* [`react-pug/pug-lint`](./docs/rules/pug-lint.md): Inherit pug-lint to validate pug

docs/rules/pug-lint.md

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
# Inherit pug-lint to validate pug (experimental)
2+
3+
This rule applies pug-lint to pug templates in js.
4+
5+
[See pug-lint project](https://github.com/pugjs/pug-lint)
6+
7+
## Rule Details
8+
9+
The following patterns are considered warnings:
10+
11+
```jsx
12+
/*eslint react-pug/pug-lint: ["error", { "requireSpaceAfterCodeOperator": true }]*/
13+
pug`div=greeting`
14+
```
15+
16+
```jsx
17+
/*eslint react-pug/pug-lint: ["error", { "disallowTrailingSpaces": true }]*/
18+
pug`div: img() `
19+
```
20+
21+
The following patterns are **not** considered warnings:
22+
23+
```jsx
24+
/*eslint react-pug/pug-lint: ["error", { "requireSpaceAfterCodeOperator": true }]*/
25+
pug`div= greeting`
26+
```
27+
28+
```jsx
29+
/*eslint react-pug/pug-lint: ["error", { "disallowTrailingSpaces": true }]*/
30+
pug`div: img()`
31+
```

index.js

+2
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ const allRules = {
1010
'no-broken-template': require('./lib/rules/no-broken-template'),
1111
'no-interpolation': require('./lib/rules/no-interpolation'),
1212
'no-undef': require('./lib/rules/no-undef'),
13+
'pug-lint': require('./lib/rules/pug-lint'),
1314
quotes: require('./lib/rules/quotes'),
1415
'uses-react': require('./lib/rules/uses-react'),
1516
'uses-vars': require('./lib/rules/uses-vars'),
@@ -29,6 +30,7 @@ module.exports = {
2930
'react-pug/indent': 2,
3031
'react-pug/no-broken-template': 2,
3132
'react-pug/no-undef': 2,
33+
'react-pug/pug-lint': 2,
3234
'react-pug/no-interpolation': 2,
3335
'react-pug/quotes': 2,
3436
'react-pug/uses-react': 2,

lib/rules/pug-lint.js

+118
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,118 @@
1+
/**
2+
* @fileoverview Inherit pug-lint to validate pug
3+
* @author Eugene Zhlobo
4+
*/
5+
6+
const Linter = require('pug-lint')
7+
const common = require('common-prefix')
8+
9+
const { isReactPugReference, buildLocation, docsUrl } = require('../util/eslint')
10+
const getTemplate = require('../util/getTemplate')
11+
12+
//------------------------------------------------------------------------------
13+
// Rule Definition
14+
//------------------------------------------------------------------------------
15+
16+
const buildMessage = actual => (
17+
`Invalid indentation, found "${actual}" spaces`
18+
)
19+
20+
module.exports = {
21+
meta: {
22+
docs: {
23+
description: 'Inherit pug-lint to validate pug (experimental)',
24+
category: 'Stylistic Issues',
25+
recommended: false,
26+
url: docsUrl('pug-lint'),
27+
},
28+
schema: [
29+
{
30+
type: 'object',
31+
},
32+
],
33+
},
34+
35+
create: function (context) {
36+
return {
37+
TaggedTemplateExpression: function (node) {
38+
if (isReactPugReference(node)) {
39+
const template = getTemplate(node)
40+
const lines = template.split('\n')
41+
42+
const linter = new Linter()
43+
44+
linter.configure(context.options[0])
45+
46+
const firstTokenInLine = context
47+
.getSourceCode()
48+
.getTokensBefore(node, {
49+
filter: token => token.loc.end.line === node.loc.start.line,
50+
})[0]
51+
52+
const minimalIndent = firstTokenInLine
53+
? firstTokenInLine.loc.start.column
54+
: node.loc.start.column
55+
56+
const desiredIndent = lines.length > 1
57+
? minimalIndent + 2
58+
: 0
59+
60+
const amountOfUselessSpaces = common(lines.filter(item => item.trim() !== ''))
61+
.replace(/^(\s*).*/, '$1')
62+
.length
63+
64+
if (amountOfUselessSpaces > 0 && amountOfUselessSpaces < desiredIndent) {
65+
context.report({
66+
node,
67+
message: buildMessage(amountOfUselessSpaces),
68+
loc: buildLocation(
69+
[(node.loc.start.line + 1), 0],
70+
[(node.loc.start.line + 1), amountOfUselessSpaces],
71+
),
72+
})
73+
74+
return null
75+
}
76+
77+
// We need to pass the template without not valuable spaces in the
78+
// beginning of each line
79+
const preparedTemplate = lines
80+
.map(item => item.slice(desiredIndent))
81+
.join('\n')
82+
83+
const result = linter.checkString(preparedTemplate, 'testfile')
84+
85+
if (result.length) {
86+
result.forEach((error) => {
87+
const delta = error.line === 1
88+
// When template starts plus backtick
89+
? node.quasi.quasis[0].loc.start.column + 1
90+
: desiredIndent - 1
91+
92+
let columnStart = error.column + delta
93+
let columnEnd = error.column + delta
94+
let message = error.msg
95+
96+
if (error.msg === 'Invalid indentation') {
97+
columnStart = 0
98+
columnEnd = preparedTemplate.split('\n')[error.line - 1].replace(/^(\s*).*/, '$1').length + desiredIndent
99+
message = buildMessage(columnEnd)
100+
}
101+
102+
context.report({
103+
node,
104+
message,
105+
loc: buildLocation(
106+
[(node.loc.start.line + error.line) - 1, columnStart],
107+
[(node.loc.start.line + error.line) - 1, columnEnd],
108+
),
109+
})
110+
})
111+
}
112+
}
113+
114+
return null
115+
},
116+
}
117+
},
118+
}

lib/util/testBuildCases.js

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
const extractCase = type => item => ({
2+
options: item.options || [],
3+
...item[type],
4+
})
5+
6+
const extractCases = type => (items) => {
7+
if (items.some(item => item.only)) {
8+
return items.filter(item => item.only).map(extractCase(type))
9+
}
10+
11+
return items.map(extractCase(type))
12+
}
13+
14+
module.exports = function (cases) {
15+
return {
16+
valid: extractCases('valid')(cases),
17+
invalid: extractCases('invalid')(cases),
18+
}
19+
}

package.json

+2
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,9 @@
2828
"dependencies": {
2929
"@babel/parser": "^7.3.2",
3030
"@babel/traverse": "^7.2.3",
31+
"common-prefix": "^1.1.0",
3132
"pug-lexer": "^4.0.0",
33+
"pug-lint": "^2.5.0",
3234
"pug-uses-variables": "^3.0.0"
3335
},
3436
"devDependencies": {

tests/each-rule.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ describe('each rule', () => {
3434
describe('mentioned in README', () => {
3535
const listOfRulesInReadme = fs
3636
.readFileSync(path.resolve(__dirname, '..', 'README.md'), 'utf-8')
37-
.match(/## List of supported rules\n+((\*.+\n+)+)/m)[1]
37+
.match(/## List of supported rules\n+((\*.+\n+|^[^#]+)+)/m)[1]
3838
.split('\n')
3939
.filter(Boolean)
4040

tests/lib/rules/eslint.js

+2-17
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99

1010
const eslint = require('eslint')
1111
const buildError = require('../../../lib/util/testBuildError')
12+
const buildCases = require('../../../lib/util/testBuildCases')
1213

1314
const { RuleTester } = eslint
1415

@@ -216,20 +217,4 @@ const cases = [
216217
},
217218
]
218219

219-
const extractCase = type => item => ({
220-
options: item.options || [],
221-
...item[type],
222-
})
223-
224-
const extractCases = type => (items) => {
225-
if (items.some(item => item.only)) {
226-
return items.filter(item => item.only).map(extractCase(type))
227-
}
228-
229-
return items.map(extractCase(type))
230-
}
231-
232-
ruleTester.run('rule "eslint"', rule, {
233-
valid: extractCases('valid')(cases),
234-
invalid: extractCases('invalid')(cases),
235-
})
220+
ruleTester.run('rule "eslint"', rule, buildCases(cases))

0 commit comments

Comments
 (0)