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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: fc5c57133db8755341c7dea0030318f79296ab019eb026017ac993e62e2066f2
4
- data.tar.gz: 0fc112757a1f0023673778ef5581a5ad48c2bcccb25052cf403c31548cc21137
3
+ metadata.gz: afe08bb3ff3e8c0371bacff7e9b508e4efef53762832f1c654cf10304ca6f056
4
+ data.tar.gz: 55889ab55f094e8a5195c2bf99fd53e4e7a3a334a157ae0b9c8b52212f87a19d
5
5
  SHA512:
6
- metadata.gz: 9f2ad98bda1e6b473871427b1d8259ef8a6db0cb51bcd18ed02d194b268801ee4764301e797a954b21c786502eeaef1ba14416944be287c86970b42e97ab2fd0
7
- data.tar.gz: ca9859096a9954448f1bc618e3e068b696e98fe703e97bccc71c16590b87bd4e5b97504daeb8c8d8927d3accd35bce4a84fb6444ab69113a42f19390adb91759
6
+ metadata.gz: 14f1f84a35eba0140cebc8df823720be70ecfecb7be829dfe799bb087816a06c112883dddb9090c150b72a9b70c1db7db7bb7dfc0e5b77775e64fb9624a60c81
7
+ data.tar.gz: 8d18d9be48e914b21f4af508851efa1e3f31ab0ca6474e51c9fdb3b420655c7b25b29e672d8eb4380d010ed96d382ad684ce53df8343245b04d559b154fcec55
@@ -1,3 +1,3 @@
1
1
  module ImmosquareCleaner
2
- VERSION = "0.1.65".freeze
2
+ VERSION = "0.1.66".freeze
3
3
  end
@@ -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
- ## JS files
142
- ## 16/05/2024
143
- ## maj 06/01/2025
144
- ## ---------
145
- ## Depuis eslint V9 (acutellement en V9.17.0), il y a une erreur
146
- ## "warning File ignored because outside of base path"
147
- ## si le fichier à linté est dans un dossier supérieur à celui du fichier de config.
148
- ## ISSUE : https://github.com/eslint/eslint/issues/19118
149
- ## ---------
150
- ## Dans nos apps nous sommes tjs dans ce cas car le fichier de config est dans le dossier du gem.
151
- ## et les fichiers à linté sont dans les apps.
152
- ## ---------
153
- ## Pour éviter ce problème on met le fichier dans un dossier temporaire dans le dossier du gem
154
- ## et on le supprime par la suite.
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 = ["bun eslint --config #{gem_root}/linters/eslint.config.mjs #{temp_file_path} --fix"]
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
@@ -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
+ ![showcase-erb-lint-gif](https://github.com/Splines/eslint-plugin-erb/assets/37160523/623d6007-b4f5-41ce-be76-5bc0208ed636?raw=true)
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,3 @@
1
+ "use strict";
2
+
3
+ module.exports = require("./lib");
@@ -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
@@ -6,6 +6,7 @@
6
6
  "eslint": "^9.22.0",
7
7
  "eslint-plugin-align-assignments": "^1.1.2",
8
8
  "eslint-plugin-align-import": "^1.0.0",
9
+ "eslint-plugin-erb": "^2.1.1",
9
10
  "eslint-plugin-prefer-arrow": "^1.2.3",
10
11
  "eslint-plugin-sonarjs": "^3.0.2",
11
12
  "prettier": "^3.5.3"
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.65
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