immosquare-cleaner 0.1.65 → 0.1.66
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/lib/immosquare-cleaner/version.rb +1 -1
- data/lib/immosquare-cleaner.rb +21 -20
- data/linters/eslint.config.mjs +14 -0
- data/node_modules/eslint-plugin-erb/LICENSE +21 -0
- data/node_modules/eslint-plugin-erb/README.md +257 -0
- data/node_modules/eslint-plugin-erb/index.js +3 -0
- data/node_modules/eslint-plugin-erb/lib/autofix.js +44 -0
- data/node_modules/eslint-plugin-erb/lib/cache.js +21 -0
- data/node_modules/eslint-plugin-erb/lib/dummy_replacements_map.js +30 -0
- data/node_modules/eslint-plugin-erb/lib/file_coordinates.js +41 -0
- data/node_modules/eslint-plugin-erb/lib/index.js +50 -0
- data/node_modules/eslint-plugin-erb/lib/offset_calculation.js +50 -0
- data/node_modules/eslint-plugin-erb/lib/offset_map.js +111 -0
- data/node_modules/eslint-plugin-erb/lib/postprocess.js +72 -0
- data/node_modules/eslint-plugin-erb/lib/preprocess.js +113 -0
- data/node_modules/eslint-plugin-erb/package.json +50 -0
- data/package.json +1 -0
- metadata +14 -1
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: afe08bb3ff3e8c0371bacff7e9b508e4efef53762832f1c654cf10304ca6f056
|
4
|
+
data.tar.gz: 55889ab55f094e8a5195c2bf99fd53e4e7a3a334a157ae0b9c8b52212f87a19d
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 14f1f84a35eba0140cebc8df823720be70ecfecb7be829dfe799bb087816a06c112883dddb9090c150b72a9b70c1db7db7bb7dfc0e5b77775e64fb9624a60c81
|
7
|
+
data.tar.gz: 8d18d9be48e914b21f4af508851efa1e3f31ab0ca6474e51c9fdb3b420655c7b25b29e672d8eb4380d010ed96d382ad684ce53df8343245b04d559b154fcec55
|
data/lib/immosquare-cleaner.rb
CHANGED
@@ -136,34 +136,35 @@ module ImmosquareCleaner
|
|
136
136
|
elsif file_path =~ %r{locales/.*\.yml$}
|
137
137
|
ImmosquareYaml.clean(file_path)
|
138
138
|
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
elsif file_path.end_with?(".js") || file_path.end_with?(".mjs")
|
139
|
+
##============================================================##
|
140
|
+
## JS files
|
141
|
+
## 16/05/2024
|
142
|
+
## maj 06/01/2025
|
143
|
+
## ---------
|
144
|
+
## Depuis eslint V9 (acutellement en V9.17.0), il y a une erreur
|
145
|
+
## "warning File ignored because outside of base path"
|
146
|
+
## si le fichier à linté est dans un dossier supérieur à celui du fichier de config.
|
147
|
+
## ISSUE : https://github.com/eslint/eslint/issues/19118
|
148
|
+
## ---------
|
149
|
+
## Dans nos apps nous sommes tjs dans ce cas car le fichier de config est dans le dossier du gem.
|
150
|
+
## et les fichiers à linté sont dans les apps.
|
151
|
+
## ---------
|
152
|
+
## Pour éviter ce problème on met le fichier dans un dossier temporaire dans le dossier du gem
|
153
|
+
## et on le supprime par la suite.
|
154
|
+
##============================================================##
|
155
|
+
elsif file_path.end_with?(".js", ".mjs", "js.erb")
|
157
156
|
begin
|
158
157
|
temp_folder_path = "#{gem_root}/tmp"
|
159
158
|
temp_file_path = "#{temp_folder_path}/#{File.basename(file_path)}"
|
160
159
|
FileUtils.mkdir_p(temp_folder_path)
|
161
160
|
File.write(temp_file_path, File.read(file_path))
|
162
|
-
cmds = [
|
161
|
+
cmds = []
|
162
|
+
cmds << "bundle exec erb_lint --config #{erblint_config_with_version_path} #{file_path} #{ImmosquareCleaner.configuration.erblint_options || "--autocorrect"}" if file_path.end_with?("js.erb")
|
163
|
+
cmds << "bun eslint --config #{gem_root}/linters/eslint.config.mjs #{temp_file_path} --fix"
|
164
|
+
|
163
165
|
launch_cmds(cmds)
|
164
166
|
File.normalize_last_line(temp_file_path)
|
165
167
|
File.write(file_path, File.read(temp_file_path))
|
166
|
-
rescue StandardError => e
|
167
168
|
ensure
|
168
169
|
FileUtils.rm_f(temp_file_path)
|
169
170
|
end
|
data/linters/eslint.config.mjs
CHANGED
@@ -3,6 +3,7 @@ import preferArrow from "eslint-plugin-prefer-arrow"
|
|
3
3
|
import sonarjs from "eslint-plugin-sonarjs"
|
4
4
|
import alignAssignments from "eslint-plugin-align-assignments"
|
5
5
|
import alignImport from "eslint-plugin-align-import"
|
6
|
+
import erb from "eslint-plugin-erb"
|
6
7
|
|
7
8
|
export default [
|
8
9
|
js.configs.recommended,
|
@@ -112,5 +113,18 @@ export default [
|
|
112
113
|
// ==============================================================================================##
|
113
114
|
"no-magic-numbers": ["off", { ignore: [], ignoreArrayIndexes: true, enforceConst: true, detectObjects: false }]
|
114
115
|
}
|
116
|
+
},
|
117
|
+
erb.configs.recommended,
|
118
|
+
{
|
119
|
+
linterOptions: {
|
120
|
+
// The "unused disable directive" is set to "warn" by default.
|
121
|
+
// For the ERB plugin to work correctly, you must disable
|
122
|
+
// this directive to avoid issues described here
|
123
|
+
// https://github.com/eslint/eslint/discussions/18114
|
124
|
+
// If you're using the CLI, you might also use the following flag:
|
125
|
+
// --report-unused-disable-directives-severity=off
|
126
|
+
reportUnusedDisableDirectives: "off"
|
127
|
+
}
|
128
|
+
// your other configuration options
|
115
129
|
}
|
116
130
|
]
|
@@ -0,0 +1,21 @@
|
|
1
|
+
MIT License
|
2
|
+
|
3
|
+
Copyright (c) 2023-2025 Splines
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
7
|
+
in the Software without restriction, including without limitation the rights
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
10
|
+
furnished to do so, subject to the following conditions:
|
11
|
+
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
13
|
+
copies or substantial portions of the Software.
|
14
|
+
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
21
|
+
SOFTWARE.
|
@@ -0,0 +1,257 @@
|
|
1
|
+
# eslint-plugin-erb
|
2
|
+
|
3
|
+
**Lint your JavaScript code inside ERB files (`.js.erb`).**
|
4
|
+
A zero-dependency plugin for [ESLint](https://eslint.org/).
|
5
|
+
<br>Also lints your **HTML code** in `.html.erb` if you want to.
|
6
|
+
|
7
|
+

|
8
|
+
|
9
|
+
## Usage
|
10
|
+
|
11
|
+
### Install
|
12
|
+
|
13
|
+
Install the plugin alongside [ESLint](https://eslint.org/docs/latest/use/getting-started):
|
14
|
+
|
15
|
+
```sh
|
16
|
+
npm install --save-dev eslint eslint-plugin-erb
|
17
|
+
```
|
18
|
+
|
19
|
+
### Configure
|
20
|
+
|
21
|
+
Starting of v9 ESLint provides a [new flat config format](https://eslint.org/docs/latest/use/configure/configuration-files-new) (`eslint.config.js`). Also see the [configuration migration guide](https://eslint.org/docs/latest/use/configure/migration-guide). Use it as follows and it will automatically lint all your **JavaScript code** in `.js.erb` files:
|
22
|
+
|
23
|
+
```js
|
24
|
+
// eslint.config.js
|
25
|
+
import erb from "eslint-plugin-erb";
|
26
|
+
|
27
|
+
export default [
|
28
|
+
erb.configs.recommended,
|
29
|
+
{
|
30
|
+
linterOptions: {
|
31
|
+
// The "unused disable directive" is set to "warn" by default.
|
32
|
+
// For the ERB plugin to work correctly, you must disable
|
33
|
+
// this directive to avoid issues described here
|
34
|
+
// https://github.com/eslint/eslint/discussions/18114
|
35
|
+
// If you're using the CLI, you might also use the following flag:
|
36
|
+
// --report-unused-disable-directives-severity=off
|
37
|
+
reportUnusedDisableDirectives: "off",
|
38
|
+
},
|
39
|
+
// your other configuration options
|
40
|
+
}
|
41
|
+
];
|
42
|
+
```
|
43
|
+
|
44
|
+
<details>
|
45
|
+
<summary>See more complete example</summary>
|
46
|
+
|
47
|
+
```js
|
48
|
+
// eslint.config.js
|
49
|
+
import js from "@eslint/js";
|
50
|
+
import stylistic from "@stylistic/eslint-plugin";
|
51
|
+
import globals from "globals";
|
52
|
+
import erb from "eslint-plugin-erb";
|
53
|
+
|
54
|
+
const customizedStylistic = stylistic.configs.customize({
|
55
|
+
"indent": 2,
|
56
|
+
"jsx": false,
|
57
|
+
"quote-props": "always",
|
58
|
+
"semi": "always",
|
59
|
+
"brace-style": "1tbs",
|
60
|
+
});
|
61
|
+
|
62
|
+
const customGlobals = {
|
63
|
+
MyGlobalVariableOrFunctionOrClassOrWhatever: "readable",
|
64
|
+
};
|
65
|
+
|
66
|
+
// [1] https://eslint.org/docs/latest/use/configure/configuration-files-new#globally-ignoring-files-with-ignores
|
67
|
+
|
68
|
+
export default [
|
69
|
+
js.configs.recommended,
|
70
|
+
erb.configs.recommended,
|
71
|
+
// Globally ignoring the following files
|
72
|
+
// "Note that only global ignores patterns can match directories.
|
73
|
+
// 'ignores' patterns that are specific to a configuration will
|
74
|
+
// only match file names." ~ see [1]
|
75
|
+
{
|
76
|
+
ignores: [
|
77
|
+
"node_modules/",
|
78
|
+
"tests/fixtures/",
|
79
|
+
"tmp/",
|
80
|
+
],
|
81
|
+
},
|
82
|
+
{
|
83
|
+
plugins: {
|
84
|
+
"@stylistic": stylistic,
|
85
|
+
},
|
86
|
+
rules: {
|
87
|
+
...customizedStylistic.rules,
|
88
|
+
"no-unused-vars": ["warn", { argsIgnorePattern: "^_" }],
|
89
|
+
"@stylistic/quotes": ["error", "double", { avoidEscape: true }],
|
90
|
+
},
|
91
|
+
languageOptions: {
|
92
|
+
ecmaVersion: 2022,
|
93
|
+
sourceType: "module",
|
94
|
+
globals: {
|
95
|
+
...customGlobals,
|
96
|
+
...globals.browser,
|
97
|
+
...globals.node,
|
98
|
+
},
|
99
|
+
},
|
100
|
+
linterOptions: {
|
101
|
+
// The "unused disable directive" is set to "warn" by default.
|
102
|
+
// For the ERB plugin to work correctly, you must disable
|
103
|
+
// this directive to avoid issues described here
|
104
|
+
// https://github.com/eslint/eslint/discussions/18114
|
105
|
+
// If you're using the CLI, you might also use the following flag:
|
106
|
+
// --report-unused-disable-directives-severity=off
|
107
|
+
reportUnusedDisableDirectives: "off",
|
108
|
+
},
|
109
|
+
},
|
110
|
+
];
|
111
|
+
```
|
112
|
+
|
113
|
+
</details>
|
114
|
+
|
115
|
+
<details>
|
116
|
+
|
117
|
+
<summary>Alternative way to configure the processor</summary>
|
118
|
+
|
119
|
+
With this variant you have a bit more control over what is going on, e.g. you could name your files `.js.special-erb` and still lint them (if they contain JS and ERB syntax).
|
120
|
+
|
121
|
+
```js
|
122
|
+
// eslint.config.js
|
123
|
+
import erb from "eslint-plugin-erb";
|
124
|
+
|
125
|
+
export default [
|
126
|
+
{
|
127
|
+
files: ["**/*.js.erb"],
|
128
|
+
processor: erb.processors.processorJs,
|
129
|
+
},
|
130
|
+
{
|
131
|
+
linterOptions: {
|
132
|
+
// The "unused disable directive" is set to "warn" by default.
|
133
|
+
// For the ERB plugin to work correctly, you must disable
|
134
|
+
// this directive to avoid issues described here
|
135
|
+
// https://github.com/eslint/eslint/discussions/18114
|
136
|
+
// If you're using the CLI, you might also use the following flag:
|
137
|
+
// --report-unused-disable-directives-severity=off
|
138
|
+
reportUnusedDisableDirectives: "off",
|
139
|
+
},
|
140
|
+
// your other configuration options
|
141
|
+
}
|
142
|
+
];
|
143
|
+
```
|
144
|
+
|
145
|
+
</details>
|
146
|
+
|
147
|
+
<details>
|
148
|
+
<summary>Legacy: you can still use the old `.eslintrc.js` format</summary>
|
149
|
+
|
150
|
+
You can extend the `plugin:erb/recommended-legacy` config that will enable the ERB processor on all `.js.erb` files.
|
151
|
+
|
152
|
+
```js
|
153
|
+
// .eslintrc.js
|
154
|
+
module.exports = {
|
155
|
+
extends: "plugin:erb/recommended-legacy"
|
156
|
+
};
|
157
|
+
```
|
158
|
+
|
159
|
+
Or you can configure the processor manually:
|
160
|
+
|
161
|
+
```js
|
162
|
+
// .eslintrc.js
|
163
|
+
module.exports = {
|
164
|
+
plugins: ["erb"],
|
165
|
+
overrides: [
|
166
|
+
{
|
167
|
+
files: ["**/*.js.erb"],
|
168
|
+
processor: "erb/processorJs"
|
169
|
+
}
|
170
|
+
]
|
171
|
+
};
|
172
|
+
```
|
173
|
+
|
174
|
+
</details>
|
175
|
+
|
176
|
+
<br>
|
177
|
+
|
178
|
+
If you also want to lint **HTML code** in `.html.erb` files, you can use our preprocessor in conjunction with the amazing [`html-eslint`](https://html-eslint.org/) plugin. Install `html-eslint`, then add the following to your ESLint config file (flat config format):
|
179
|
+
|
180
|
+
```js
|
181
|
+
// eslint.config.js
|
182
|
+
import erb from "eslint-plugin-erb";
|
183
|
+
|
184
|
+
export default [
|
185
|
+
// your other configurations...
|
186
|
+
{
|
187
|
+
processor: erb.processors["processorHtml"],
|
188
|
+
...html.configs["flat/recommended"],
|
189
|
+
files: ["**/*.html", "**/*.html.erb"],
|
190
|
+
rules: {
|
191
|
+
...html.configs["flat/recommended"].rules,
|
192
|
+
"@html-eslint/indent": ["error", 2],
|
193
|
+
// other rules...
|
194
|
+
},
|
195
|
+
}
|
196
|
+
];
|
197
|
+
```
|
198
|
+
|
199
|
+
Additionally, you might want to add the following option to the other objects (`{}`) in `export default []` (at the same level like the `files` key above), since other rules might be incompatible with HTML files:
|
200
|
+
|
201
|
+
```js
|
202
|
+
ignores: ["**/*.html**"],
|
203
|
+
```
|
204
|
+
|
205
|
+
## Editor Integrations
|
206
|
+
|
207
|
+
The [ESLint extension](https://marketplace.visualstudio.com/items?itemName=dbaeumer.vscode-eslint) for VSCode has built-in support for the ERB processor once you've configured it in your `.eslintrc.js` file as shown above.
|
208
|
+
|
209
|
+
If you're using VSCode, you may find this `settings.json` options useful:
|
210
|
+
|
211
|
+
```jsonc
|
212
|
+
{
|
213
|
+
"editor.formatOnSave": false, // it still autosaves with the options below
|
214
|
+
//////////////////////////////////////
|
215
|
+
// JS (ESLint)
|
216
|
+
//////////////////////////////////////
|
217
|
+
// https://eslint.style/guide/faq#how-to-auto-format-on-save
|
218
|
+
// https://github.com/microsoft/vscode-eslint#settings-options
|
219
|
+
"eslint.format.enable": true,
|
220
|
+
"[javascript]": {
|
221
|
+
"editor.formatOnSave": false, // to avoid formatting twice (ESLint + VSCode)
|
222
|
+
"editor.defaultFormatter": "dbaeumer.vscode-eslint" // use ESLint plugin
|
223
|
+
},
|
224
|
+
"editor.codeActionsOnSave": {
|
225
|
+
"source.fixAll.eslint": "explicit" // Auto-fix ESLint errors on save
|
226
|
+
},
|
227
|
+
// this disables VSCode built-int formatter (instead we want to use ESLint)
|
228
|
+
"javascript.validate.enable": false,
|
229
|
+
//////////////////////////////////////
|
230
|
+
// Files
|
231
|
+
//////////////////////////////////////
|
232
|
+
"files.associations": {
|
233
|
+
"*.js.erb": "javascript"
|
234
|
+
},
|
235
|
+
}
|
236
|
+
```
|
237
|
+
|
238
|
+
## Limitations
|
239
|
+
|
240
|
+
- Does not account for code indentation inside `if/else` ERB statements, e.g.
|
241
|
+
this snippet
|
242
|
+
|
243
|
+
```js
|
244
|
+
<% if you_feel_lucky %>
|
245
|
+
console.log("You are lucky 🍀");
|
246
|
+
<% end %>
|
247
|
+
```
|
248
|
+
|
249
|
+
will be autofixed to
|
250
|
+
|
251
|
+
```js
|
252
|
+
<% if you_feel_lucky %>
|
253
|
+
console.log("You are lucky 🍀");
|
254
|
+
<% end %>
|
255
|
+
```
|
256
|
+
|
257
|
+
- No support for ESLint _suggestions_ (but full support for _autofixes_ as shown in the GIF above)
|
@@ -0,0 +1,44 @@
|
|
1
|
+
const { coordinatesToIndex, indexToColumn } = require("./file_coordinates.js");
|
2
|
+
const { calculateOffset } = require("./offset_calculation.js");
|
3
|
+
|
4
|
+
function transformFix(message, originalText, lintableText, offsetMap, dummyReplacementsMap) {
|
5
|
+
const newRange = calculateNewRange(message, originalText, lintableText, offsetMap);
|
6
|
+
const newFixText = transformFixText(message, newRange, originalText,
|
7
|
+
offsetMap, dummyReplacementsMap);
|
8
|
+
|
9
|
+
return {
|
10
|
+
range: newRange,
|
11
|
+
text: newFixText,
|
12
|
+
};
|
13
|
+
}
|
14
|
+
|
15
|
+
function calculateNewRange(message, originalText, lintableText, offsetMap) {
|
16
|
+
let lastIndex = -1;
|
17
|
+
let lastIndexMapped = -1;
|
18
|
+
|
19
|
+
return message.fix.range.map((index) => {
|
20
|
+
if (index == lastIndex) {
|
21
|
+
return lastIndexMapped;
|
22
|
+
}
|
23
|
+
|
24
|
+
const pos = indexToColumn(lintableText, index);
|
25
|
+
const offset = calculateOffset(index, pos.line, offsetMap);
|
26
|
+
const posOriginal = {
|
27
|
+
line: pos.line + offset.lineOffset,
|
28
|
+
column: pos.column + offset.columnOffset,
|
29
|
+
};
|
30
|
+
const indexMapped = coordinatesToIndex(originalText, posOriginal.line, posOriginal.column);
|
31
|
+
|
32
|
+
lastIndex = index;
|
33
|
+
lastIndexMapped = indexMapped;
|
34
|
+
|
35
|
+
return indexMapped;
|
36
|
+
});
|
37
|
+
}
|
38
|
+
|
39
|
+
function transformFixText(message, newRange, originalText, offsetMap, dummyReplacementsMap) {
|
40
|
+
const fixText = message.fix.text;
|
41
|
+
return dummyReplacementsMap.apply(fixText);
|
42
|
+
}
|
43
|
+
|
44
|
+
module.exports = { transformFix };
|
@@ -0,0 +1,21 @@
|
|
1
|
+
class Cache {
|
2
|
+
constructor() {
|
3
|
+
// filename -> { originalText, lintableText, offsetMap }
|
4
|
+
this.cache = new Map();
|
5
|
+
}
|
6
|
+
|
7
|
+
add(filename, originalText, lintableText, offsetMap, dummyReplacementsMap) {
|
8
|
+
this.cache.set(filename, { originalText, lintableText, offsetMap, dummyReplacementsMap });
|
9
|
+
}
|
10
|
+
|
11
|
+
get(filename) {
|
12
|
+
return this.cache.get(filename);
|
13
|
+
}
|
14
|
+
|
15
|
+
delete(filename) {
|
16
|
+
this.cache.delete(filename);
|
17
|
+
}
|
18
|
+
}
|
19
|
+
|
20
|
+
const cache = new Cache();
|
21
|
+
module.exports = cache;
|
@@ -0,0 +1,30 @@
|
|
1
|
+
/**
|
2
|
+
* Array that tracks the replacements that have been made in the processed text,
|
3
|
+
* so that we can reverse them later.
|
4
|
+
*/
|
5
|
+
class DummyReplacementsMap {
|
6
|
+
constructor() {
|
7
|
+
this.replacements = [];
|
8
|
+
}
|
9
|
+
|
10
|
+
addMapping(originalText, dummyText) {
|
11
|
+
this.replacements.push([originalText, dummyText]);
|
12
|
+
}
|
13
|
+
|
14
|
+
/**
|
15
|
+
* Tries to apply all stored replacements to the given text.
|
16
|
+
* @param {string} text - The text to apply the replacements to.
|
17
|
+
* @returns {string} - The text with the replacements applied.
|
18
|
+
*/
|
19
|
+
apply(text) {
|
20
|
+
let lintableText = text;
|
21
|
+
for (const [originalText, dummyText] of this.replacements) {
|
22
|
+
lintableText = lintableText.replace(dummyText, originalText);
|
23
|
+
}
|
24
|
+
return lintableText;
|
25
|
+
}
|
26
|
+
}
|
27
|
+
|
28
|
+
module.exports = {
|
29
|
+
DummyReplacementsMap,
|
30
|
+
};
|
@@ -0,0 +1,41 @@
|
|
1
|
+
/**
|
2
|
+
*
|
3
|
+
* @param {string} text Text
|
4
|
+
* @param {number} line 1-based line number
|
5
|
+
* @param {number} column 1-based column number
|
6
|
+
* @returns {number} 0-based index of the character in the text
|
7
|
+
* at the given line and column
|
8
|
+
*/
|
9
|
+
function coordinatesToIndex(text, line, column) {
|
10
|
+
const lines = text.split("\n");
|
11
|
+
|
12
|
+
// Lines
|
13
|
+
let index = 0;
|
14
|
+
for (let i = 0; i < line - 1; i++) { // lines are 1-based (!)
|
15
|
+
index += lines[i].length + 1; // +1 for newline character
|
16
|
+
}
|
17
|
+
|
18
|
+
// Column
|
19
|
+
index += (column - 1); // column is 1-based (while index is 0-based)
|
20
|
+
|
21
|
+
return index;
|
22
|
+
}
|
23
|
+
|
24
|
+
/**
|
25
|
+
*
|
26
|
+
* @param {*string} text Text
|
27
|
+
* @param {*number} index 0-based index in text
|
28
|
+
* @returns {*number} 1-based column number
|
29
|
+
*/
|
30
|
+
function indexToColumn(text, index) {
|
31
|
+
const linesUpToIndex = text.slice(0, index).split("\n");
|
32
|
+
const column = linesUpToIndex[linesUpToIndex.length - 1].length;
|
33
|
+
const lineNumber = linesUpToIndex.length;
|
34
|
+
|
35
|
+
return {
|
36
|
+
line: lineNumber,
|
37
|
+
column: column + 1, // column is 1-based
|
38
|
+
};
|
39
|
+
}
|
40
|
+
|
41
|
+
module.exports = { coordinatesToIndex, indexToColumn };
|
@@ -0,0 +1,50 @@
|
|
1
|
+
const { preprocessJs, preprocessHtml } = require("./preprocess.js");
|
2
|
+
const postprocess = require("./postprocess.js");
|
3
|
+
|
4
|
+
// Load processors
|
5
|
+
const processorJs = {
|
6
|
+
meta: {
|
7
|
+
name: "processJs",
|
8
|
+
},
|
9
|
+
preprocess: preprocessJs,
|
10
|
+
postprocess,
|
11
|
+
supportsAutofix: true,
|
12
|
+
};
|
13
|
+
const processorHtml = {
|
14
|
+
meta: {
|
15
|
+
name: "processHtml",
|
16
|
+
},
|
17
|
+
preprocess: preprocessHtml,
|
18
|
+
postprocess,
|
19
|
+
supportsAutofix: true,
|
20
|
+
};
|
21
|
+
|
22
|
+
// Define & export plugin
|
23
|
+
const plugin = {
|
24
|
+
meta: {
|
25
|
+
name: "eslint-plugin-erb",
|
26
|
+
version: "2.1.1",
|
27
|
+
},
|
28
|
+
configs: {
|
29
|
+
"recommended": {
|
30
|
+
files: ["**/*.js.erb"],
|
31
|
+
processor: processorJs,
|
32
|
+
},
|
33
|
+
// for the old non-flat config ESLint API
|
34
|
+
"recommended-legacy": {
|
35
|
+
plugins: ["erb"],
|
36
|
+
overrides: [
|
37
|
+
{
|
38
|
+
files: ["**/*.js.erb"],
|
39
|
+
processor: "erb/processorJs",
|
40
|
+
},
|
41
|
+
],
|
42
|
+
},
|
43
|
+
},
|
44
|
+
processors: {
|
45
|
+
processorJs: processorJs,
|
46
|
+
processorHtml: processorHtml,
|
47
|
+
},
|
48
|
+
};
|
49
|
+
|
50
|
+
module.exports = plugin;
|
@@ -0,0 +1,50 @@
|
|
1
|
+
////////////////////////////////////////////////////
|
2
|
+
// Offset calculation
|
3
|
+
////////////////////////////////////////////////////
|
4
|
+
|
5
|
+
function calculateOffset(index, line, offsetMap) {
|
6
|
+
const lineOffsets = offsetMap.lookupLine(line);
|
7
|
+
if (lineOffsets) {
|
8
|
+
return calculateOffsetLineStrategy(index, line, lineOffsets, offsetMap);
|
9
|
+
}
|
10
|
+
else {
|
11
|
+
return calculateOffsetNormalIndexStrategy(index, offsetMap);
|
12
|
+
}
|
13
|
+
}
|
14
|
+
|
15
|
+
function calculateOffsetLineStrategy(index, line, lineOffsets, offsetMap) {
|
16
|
+
const indexKey = offsetMap.findKeyJustSmallerThan(index, line);
|
17
|
+
|
18
|
+
if (indexKey !== undefined) {
|
19
|
+
const offsets = lineOffsets.get(indexKey);
|
20
|
+
return {
|
21
|
+
lineOffset: offsets.lineOffset,
|
22
|
+
columnOffset: offsets.columnOffset,
|
23
|
+
};
|
24
|
+
}
|
25
|
+
|
26
|
+
// Situation: our index is on the left side of [..., ..., ...]
|
27
|
+
// We need to find the next smaller line number that is present in the map
|
28
|
+
// Therefore: Fallback to normal index strategy
|
29
|
+
return calculateOffsetNormalIndexStrategy(index, offsetMap);
|
30
|
+
}
|
31
|
+
|
32
|
+
function calculateOffsetNormalIndexStrategy(index, offsetMap) {
|
33
|
+
const indexKey = offsetMap.findKeyJustSmallerThan(index, null);
|
34
|
+
|
35
|
+
if (indexKey !== undefined) {
|
36
|
+
const offsets = offsetMap.lookupIndex(indexKey);
|
37
|
+
return {
|
38
|
+
lineOffset: offsets.lineOffset,
|
39
|
+
columnOffset: 0,
|
40
|
+
};
|
41
|
+
}
|
42
|
+
|
43
|
+
// Can only happen at the beginning of the file before first ERB tag
|
44
|
+
return {
|
45
|
+
lineOffset: 0,
|
46
|
+
columnOffset: 0,
|
47
|
+
};
|
48
|
+
}
|
49
|
+
|
50
|
+
module.exports = { calculateOffset };
|
@@ -0,0 +1,111 @@
|
|
1
|
+
/* Sample view of the relevant data structures:
|
2
|
+
|
3
|
+
this.lineOffsetLookup = {
|
4
|
+
1 => Map(1) { 43 => [ 0, -23 ] },
|
5
|
+
2 => Map(1) { 92 => [ 0, -17 ] },
|
6
|
+
6 => Map(1) { 224 => [ 3, 5 ] },
|
7
|
+
7 => Map(1) { 303 => [ 6, 6 ] },
|
8
|
+
12 => Map(1) { 605 => [ 6, -3 ] },
|
9
|
+
18 => Map(2) { 813 => [ 6, -26 ], 856 => [ 6, -43 ] },
|
10
|
+
21 => Map(1) { 942 => [ 6, 95 ] }
|
11
|
+
}
|
12
|
+
|
13
|
+
For brevity, we emit the respective property names and show the values in an
|
14
|
+
array instead. First number is line offset, second number is column offset.
|
15
|
+
|
16
|
+
this.offsetLookup = {
|
17
|
+
43 => [ 0, -23 ],
|
18
|
+
92 => [ 0, -17 ],
|
19
|
+
224 => [ 3, 5 ],
|
20
|
+
303 => [ 6, 6 ],
|
21
|
+
605 => [ 6, -3 ],
|
22
|
+
813 => [ 6, -26 ],
|
23
|
+
856 => [ 6, -43 ],
|
24
|
+
942 => [ 6, 95 ]
|
25
|
+
}
|
26
|
+
|
27
|
+
We store the end index of the respective ERB tag and use that as key for our map.
|
28
|
+
Note that this end index refers to the processed text, i. e. the text after
|
29
|
+
replacing ERB tags with dummy comments.
|
30
|
+
The index is an index to the whole text.
|
31
|
+
Note that line offsets accumulate.
|
32
|
+
|
33
|
+
And to add to the good old off-by-one confusion:
|
34
|
+
Line offsets: 1-based
|
35
|
+
Column offsets: 0-based
|
36
|
+
(also see file coordinates)
|
37
|
+
|
38
|
+
*/
|
39
|
+
|
40
|
+
class OffsetMap {
|
41
|
+
constructor() {
|
42
|
+
// (line number in processed text) -> (this.offsetLookup map with keys for that line)
|
43
|
+
this.lineOffsetLookup = new Map();
|
44
|
+
|
45
|
+
// (index in processed text) -> { lineOffset, columnOffset }
|
46
|
+
this.offsetLookup = new Map();
|
47
|
+
}
|
48
|
+
|
49
|
+
/**
|
50
|
+
*
|
51
|
+
* @param {Number} index end index in processed text
|
52
|
+
* @param {Number} line line number in processed text
|
53
|
+
* @param {Number} lineOffset number of lines that have to be added to a warning
|
54
|
+
* in the processed text
|
55
|
+
* @param {Number} columnOffset number of columns that have to be added to a
|
56
|
+
* warning in the processed text
|
57
|
+
*/
|
58
|
+
addMapping(index, line, lineOffset, columnOffset) {
|
59
|
+
// Multiple ERB tags can be opened and closed on the same line
|
60
|
+
let sameLineAdditionalColumnOffset = 0;
|
61
|
+
let existingLineMap = new Map();
|
62
|
+
|
63
|
+
if (this.lineOffsetLookup.has(line)) {
|
64
|
+
existingLineMap = this.lineOffsetLookup.get(line);
|
65
|
+
const highestIndex = Math.max(...Array.from(existingLineMap.keys()));
|
66
|
+
sameLineAdditionalColumnOffset = existingLineMap.get(highestIndex).columnOffset;
|
67
|
+
}
|
68
|
+
|
69
|
+
// Construct new entry for data structures
|
70
|
+
const newPartialMap = new Map();
|
71
|
+
newPartialMap.set(index, {
|
72
|
+
lineOffset: lineOffset,
|
73
|
+
columnOffset: columnOffset + sameLineAdditionalColumnOffset,
|
74
|
+
});
|
75
|
+
|
76
|
+
// Update data structures (don't care if its inefficient, it works fast enough)
|
77
|
+
this.lineOffsetLookup.set(line, new Map([...existingLineMap, ...newPartialMap]));
|
78
|
+
this.offsetLookup = new Map([...this.offsetLookup, ...newPartialMap]);
|
79
|
+
}
|
80
|
+
|
81
|
+
lookupIndex(index) {
|
82
|
+
return this.offsetLookup.get(index);
|
83
|
+
}
|
84
|
+
|
85
|
+
lookupLine(line) {
|
86
|
+
return this.lineOffsetLookup.get(line);
|
87
|
+
}
|
88
|
+
|
89
|
+
// we assume that line has a valid entry in the map, caller needs to check (!)
|
90
|
+
findKeyJustSmallerThan(index, line) {
|
91
|
+
// Determine which map to use
|
92
|
+
let offsetLookup = line ? this.lineOffsetLookup.get(line) : this.offsetLookup;
|
93
|
+
|
94
|
+
const mapIndizes = Array.from(offsetLookup.keys());
|
95
|
+
mapIndizes.sort((a, b) => b - a); // from highest to lowest
|
96
|
+
|
97
|
+
let keyJustSmallerThanIndex;
|
98
|
+
for (let key of mapIndizes) {
|
99
|
+
if (key <= index) {
|
100
|
+
keyJustSmallerThanIndex = key;
|
101
|
+
return keyJustSmallerThanIndex;
|
102
|
+
}
|
103
|
+
}
|
104
|
+
|
105
|
+
return undefined;
|
106
|
+
}
|
107
|
+
}
|
108
|
+
|
109
|
+
module.exports = {
|
110
|
+
OffsetMap,
|
111
|
+
};
|
@@ -0,0 +1,72 @@
|
|
1
|
+
const cache = require("./cache.js");
|
2
|
+
const { coordinatesToIndex } = require("./file_coordinates.js");
|
3
|
+
const { calculateOffset } = require("./offset_calculation.js");
|
4
|
+
const { transformFix } = require("./autofix.js");
|
5
|
+
|
6
|
+
/**
|
7
|
+
* Transforms the messages form ESLint such that they point to the correct
|
8
|
+
* location in the source code.
|
9
|
+
* @param {Array<Message[]>} messages An array containing one array of messages
|
10
|
+
* for each code block returned from `preprocess`. As we only deal with one
|
11
|
+
* code block, this array contains only *one* array of messages.
|
12
|
+
* @param {string} filename filename of the file.
|
13
|
+
* @returns {Message[]} A flattened array of messages with mapped locations.
|
14
|
+
*/
|
15
|
+
function postprocess(messages, filename) {
|
16
|
+
const { originalText, lintableText, offsetMap, dummyReplacementsMap } = cache.get(filename);
|
17
|
+
|
18
|
+
const transformedMessages = messages[0].map((message) => {
|
19
|
+
return adjustMessage(message, originalText, lintableText, offsetMap, dummyReplacementsMap);
|
20
|
+
});
|
21
|
+
cache.delete(filename);
|
22
|
+
|
23
|
+
return transformedMessages;
|
24
|
+
}
|
25
|
+
|
26
|
+
/**
|
27
|
+
* Adjusts ESLint messages to point to the correct location in the source code.
|
28
|
+
* @param {Message} message message form ESLint
|
29
|
+
* @returns {Message} same message, but adjusted to the correct location
|
30
|
+
*/
|
31
|
+
function adjustMessage(message, originalText, lintableText, offsetMap, dummyReplacementsMap) {
|
32
|
+
if (!Number.isInteger(message.line)) {
|
33
|
+
return {
|
34
|
+
...message,
|
35
|
+
line: 0,
|
36
|
+
column: 0,
|
37
|
+
};
|
38
|
+
}
|
39
|
+
|
40
|
+
// From now on: assume some kind of positivity, e.g. message.line and
|
41
|
+
// message.column exists.
|
42
|
+
|
43
|
+
// Start line/column
|
44
|
+
const index = coordinatesToIndex(lintableText, message.line, message.column);
|
45
|
+
const offset = calculateOffset(index, message.line, offsetMap);
|
46
|
+
let newCoordinates = {
|
47
|
+
line: message.line + offset.lineOffset,
|
48
|
+
column: message.column + offset.columnOffset,
|
49
|
+
};
|
50
|
+
|
51
|
+
// End line/column
|
52
|
+
if (Number.isInteger(message.endLine)) {
|
53
|
+
const indexEnd = coordinatesToIndex(lintableText, message.endLine, message.endColumn);
|
54
|
+
const offsetEnd = calculateOffset(indexEnd, message.endLine, offsetMap);
|
55
|
+
|
56
|
+
newCoordinates = {
|
57
|
+
...newCoordinates,
|
58
|
+
endLine: message.endLine + offsetEnd.lineOffset,
|
59
|
+
endColumn: message.endColumn + offsetEnd.columnOffset };
|
60
|
+
}
|
61
|
+
|
62
|
+
// Autofixes
|
63
|
+
const adjustedFix = {};
|
64
|
+
if (message.fix) {
|
65
|
+
adjustedFix.fix = transformFix(message, originalText, lintableText,
|
66
|
+
offsetMap, dummyReplacementsMap);
|
67
|
+
}
|
68
|
+
|
69
|
+
return { ...message, ...newCoordinates, ...adjustedFix };
|
70
|
+
}
|
71
|
+
|
72
|
+
module.exports = postprocess;
|
@@ -0,0 +1,113 @@
|
|
1
|
+
// The words "after" and "before" refer to "after" and "before" processing,
|
2
|
+
// i.e. replacing of respective ERB tags with dummy comments.
|
3
|
+
|
4
|
+
const cache = require("./cache.js");
|
5
|
+
|
6
|
+
const { indexToColumn } = require("./file_coordinates.js");
|
7
|
+
|
8
|
+
// how annoying is that kind of import in JS ?!
|
9
|
+
var OffsetMap = require("./offset_map.js").OffsetMap;
|
10
|
+
var DummyReplacementsMap = require("./dummy_replacements_map.js").DummyReplacementsMap;
|
11
|
+
|
12
|
+
const ERB_REGEX = /<%[\s\S]*?%>/g;
|
13
|
+
const HASH = "566513c5d83ac26e15414f2754"; // to avoid collisions with user code
|
14
|
+
|
15
|
+
/**
|
16
|
+
* Transforms the given text into lintable text. We do this by stripping out
|
17
|
+
* ERB tags and replacing them with dummy comments. Additionally, we keep track
|
18
|
+
* of the offset introduced by this replacement, so that we can adjust the
|
19
|
+
* location of messages in the postprocess step later.
|
20
|
+
* @param {string} text text of the file
|
21
|
+
* @param {string} filename filename of the file
|
22
|
+
* @param {(id) => string} toDummyString function that takes an id and the
|
23
|
+
* matched text and returns a dummy string to replace the matched text with.
|
24
|
+
* This dummy string must be unique.
|
25
|
+
* @returns {Array<{ filename: string, text: string }>} source code blocks to lint
|
26
|
+
*/
|
27
|
+
function preprocess(text, filename, toDummyString) {
|
28
|
+
let lintableTextArr = text.split("");
|
29
|
+
|
30
|
+
let match;
|
31
|
+
let numAddLines = 0;
|
32
|
+
let numDiffChars = 0;
|
33
|
+
const offsetMap = new OffsetMap();
|
34
|
+
const dummyReplacementsMap = new DummyReplacementsMap();
|
35
|
+
|
36
|
+
let matchedId = 0;
|
37
|
+
while ((match = ERB_REGEX.exec(text)) !== null) {
|
38
|
+
// Match information
|
39
|
+
const startIndex = match.index;
|
40
|
+
const matchText = match[0];
|
41
|
+
const matchLength = matchText.length;
|
42
|
+
const endIndex = startIndex + matchLength;
|
43
|
+
const matchLines = matchText.split("\n");
|
44
|
+
|
45
|
+
// Lines
|
46
|
+
numAddLines += matchLines.length - 1;
|
47
|
+
|
48
|
+
// Columns
|
49
|
+
const dummy = toDummyString(matchedId, matchText);
|
50
|
+
const coordStartIndex = indexToColumn(text, startIndex);
|
51
|
+
const endColumnAfter = coordStartIndex.column + dummy.length;
|
52
|
+
const coordEndIndex = indexToColumn(text, endIndex);
|
53
|
+
const endColumnBefore = coordEndIndex.column;
|
54
|
+
const numAddColumns = endColumnBefore - endColumnAfter;
|
55
|
+
replaceTextWithDummy(lintableTextArr, startIndex, matchLength - 1, dummy);
|
56
|
+
|
57
|
+
const textWithErbSyntax = text.slice(startIndex, endIndex);
|
58
|
+
dummyReplacementsMap.addMapping(textWithErbSyntax, dummy);
|
59
|
+
|
60
|
+
// Store in map
|
61
|
+
const lineAfter = coordEndIndex.line - numAddLines;
|
62
|
+
numDiffChars += dummy.length - matchLength;
|
63
|
+
const endIndexAfter = endIndex + numDiffChars;
|
64
|
+
offsetMap.addMapping(endIndexAfter, lineAfter, numAddLines, numAddColumns);
|
65
|
+
|
66
|
+
matchedId += 1;
|
67
|
+
}
|
68
|
+
|
69
|
+
const lintableText = lintableTextArr.join("");
|
70
|
+
cache.add(filename, text, lintableText, offsetMap, dummyReplacementsMap);
|
71
|
+
return [lintableText];
|
72
|
+
}
|
73
|
+
|
74
|
+
/**
|
75
|
+
* In-place replaces the text (as array) at the given index, for a given length,
|
76
|
+
* with a dummy string.
|
77
|
+
*
|
78
|
+
* Note that the dummy string is inserted at the given index as one big string.
|
79
|
+
* For the length of the match, subsequent characters are replaced with empty
|
80
|
+
* strings in the array.
|
81
|
+
*/
|
82
|
+
function replaceTextWithDummy(lintableTextArr, startIndex, length, dummy) {
|
83
|
+
lintableTextArr[startIndex] = dummy;
|
84
|
+
const replaceArgs = Array(length).join(".").split(".");
|
85
|
+
// -> results in ['', '', '', '', ...]
|
86
|
+
lintableTextArr.splice(startIndex + 1, length, ...replaceArgs);
|
87
|
+
}
|
88
|
+
|
89
|
+
function preprocessJs(text, filename) {
|
90
|
+
function wrapInEslintDisable(text) {
|
91
|
+
return `/* eslint-disable */${text}/* eslint-enable */`;
|
92
|
+
}
|
93
|
+
|
94
|
+
function toDummyString(id, matchText) {
|
95
|
+
if (matchText.startsWith("<%=")) {
|
96
|
+
return wrapInEslintDisable(`{}/* eslint-plugin-erb ${HASH} ${id} */`);
|
97
|
+
}
|
98
|
+
return wrapInEslintDisable(`/* eslint-plugin-erb ${HASH} ${id} */`);
|
99
|
+
}
|
100
|
+
return preprocess(text, filename, toDummyString);
|
101
|
+
}
|
102
|
+
|
103
|
+
function preprocessHtml(text, filename) {
|
104
|
+
function toDummyString(id, _matchText) {
|
105
|
+
return `<!-- ${HASH} ${id} -->`;
|
106
|
+
}
|
107
|
+
return preprocess(text, filename, toDummyString);
|
108
|
+
}
|
109
|
+
|
110
|
+
module.exports = {
|
111
|
+
preprocessJs,
|
112
|
+
preprocessHtml,
|
113
|
+
};
|
@@ -0,0 +1,50 @@
|
|
1
|
+
{
|
2
|
+
"name": "eslint-plugin-erb",
|
3
|
+
"version": "2.1.1",
|
4
|
+
"description": "An ESLint plugin to lint JavaScript in ERB files (.js.erb)",
|
5
|
+
"license": "MIT",
|
6
|
+
"author": {
|
7
|
+
"name": "Splines",
|
8
|
+
"url": "https://github.com/splines"
|
9
|
+
},
|
10
|
+
"repository": {
|
11
|
+
"type": "git",
|
12
|
+
"url": "https://github.com/Splines/eslint-plugin-erb.git"
|
13
|
+
},
|
14
|
+
"bugs": {
|
15
|
+
"url": "https://github.com/Splines/eslint-plugin-erb/issues"
|
16
|
+
},
|
17
|
+
"keywords": [
|
18
|
+
"eslint",
|
19
|
+
"eslintplugin",
|
20
|
+
"eslint-processor",
|
21
|
+
"erb",
|
22
|
+
"ruby",
|
23
|
+
"embedded ruby",
|
24
|
+
"javascript",
|
25
|
+
"lint",
|
26
|
+
"linter"
|
27
|
+
],
|
28
|
+
"main": "index.js",
|
29
|
+
"files": [
|
30
|
+
"index.js",
|
31
|
+
"lib"
|
32
|
+
],
|
33
|
+
"devDependencies": {
|
34
|
+
"@html-eslint/eslint-plugin": "^0.31.1",
|
35
|
+
"@html-eslint/parser": "^0.31.0",
|
36
|
+
"@stylistic/eslint-plugin": "^1.3.3",
|
37
|
+
"chai": "^4.3.10",
|
38
|
+
"eslint": "^9.17.0",
|
39
|
+
"globals": "^13.24.0",
|
40
|
+
"mocha": "^10.2.0"
|
41
|
+
},
|
42
|
+
"scripts": {
|
43
|
+
"lint": "eslint .",
|
44
|
+
"lint-fix": "eslint --fix .",
|
45
|
+
"test": "mocha",
|
46
|
+
"whats-included": "npm pack --dry-run",
|
47
|
+
"bump-version": "npm run test && npm version --no-git-tag-version",
|
48
|
+
"publish-final": "npm run test && npm publish"
|
49
|
+
}
|
50
|
+
}
|
data/package.json
CHANGED
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: immosquare-cleaner
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.1.
|
4
|
+
version: 0.1.66
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- immosquare
|
@@ -5638,6 +5638,19 @@ files:
|
|
5638
5638
|
- node_modules/eslint-plugin-align-import/package.json
|
5639
5639
|
- node_modules/eslint-plugin-align-import/tests/lib/rules/align-import.js
|
5640
5640
|
- node_modules/eslint-plugin-align-import/tests/lib/rules/trim-import.js
|
5641
|
+
- node_modules/eslint-plugin-erb/LICENSE
|
5642
|
+
- node_modules/eslint-plugin-erb/README.md
|
5643
|
+
- node_modules/eslint-plugin-erb/index.js
|
5644
|
+
- node_modules/eslint-plugin-erb/lib/autofix.js
|
5645
|
+
- node_modules/eslint-plugin-erb/lib/cache.js
|
5646
|
+
- node_modules/eslint-plugin-erb/lib/dummy_replacements_map.js
|
5647
|
+
- node_modules/eslint-plugin-erb/lib/file_coordinates.js
|
5648
|
+
- node_modules/eslint-plugin-erb/lib/index.js
|
5649
|
+
- node_modules/eslint-plugin-erb/lib/offset_calculation.js
|
5650
|
+
- node_modules/eslint-plugin-erb/lib/offset_map.js
|
5651
|
+
- node_modules/eslint-plugin-erb/lib/postprocess.js
|
5652
|
+
- node_modules/eslint-plugin-erb/lib/preprocess.js
|
5653
|
+
- node_modules/eslint-plugin-erb/package.json
|
5641
5654
|
- node_modules/eslint-plugin-prefer-arrow/LICENSE
|
5642
5655
|
- node_modules/eslint-plugin-prefer-arrow/README.md
|
5643
5656
|
- node_modules/eslint-plugin-prefer-arrow/index.js
|