-
-
Notifications
You must be signed in to change notification settings - Fork 1.6k
/
Copy pathno-useless-path-segments.js
119 lines (101 loc) · 2.9 KB
/
no-useless-path-segments.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
/**
* @fileOverview Ensures that there are no useless path segments
* @author Thomas Grainger
*/
import path from 'path'
import sumBy from 'lodash/sumBy'
import resolve from 'eslint-module-utils/resolve'
import moduleVisitor from 'eslint-module-utils/moduleVisitor'
import docsUrl from '../docsUrl'
/**
* convert a potentially relative path from node utils into a true
* relative path.
*
* ../ -> ..
* ./ -> .
* .foo/bar -> ./.foo/bar
* ..foo/bar -> ./..foo/bar
* foo/bar -> ./foo/bar
*
* @param rel {string} relative posix path potentially missing leading './'
* @returns {string} relative posix path that always starts with a ./
**/
function toRel(rel) {
const stripped = rel.replace(/\/$/g, '')
return /^((\.\.)|(\.))($|\/)/.test(stripped) ? stripped : `./${stripped}`
}
function normalize(fn) {
return toRel(path.posix.normalize(fn))
}
const countRelParent = x => sumBy(x, v => v === '..')
module.exports = {
meta: {
docs: {
url: docsUrl('no-useless-path-segments'),
},
schema: [
{
type: 'object',
properties: {
commonjs: { type: 'boolean' },
requireResolve: {
oneOf: [
{ type: 'boolean' },
{
type: 'object',
properties: {
commonjs: { type: 'boolean' },
},
},
],
},
},
additionalProperties: false,
},
],
fixable: 'code',
},
create: function (context) {
const currentDir = path.dirname(context.getFilename())
function checkSourceValue(source) {
const { value } = source
function report(proposed) {
context.report({
node: source,
message: `Useless path segments for "${value}", should be "${proposed}"`,
fix: fixer => fixer.replaceText(source, JSON.stringify(proposed)),
})
}
if (!value.startsWith('.')) {
return
}
const resolvedPath = resolve(value, context)
const normed = normalize(value)
if (normed !== value && resolvedPath === resolve(normed, context)) {
return report(normed)
}
if (value.startsWith('./')) {
return
}
if (resolvedPath === undefined) {
return
}
const expected = path.relative(currentDir, resolvedPath)
const expectedSplit = expected.split(path.sep)
const valueSplit = value.replace(/^\.\//, '').split('/')
const valueNRelParents = countRelParent(valueSplit)
const expectedNRelParents = countRelParent(expectedSplit)
const diff = valueNRelParents - expectedNRelParents
if (diff <= 0) {
return
}
return report(
toRel(valueSplit
.slice(0, expectedNRelParents)
.concat(valueSplit.slice(valueNRelParents + diff))
.join('/'))
)
}
return moduleVisitor(checkSourceValue, context.options[0])
},
}