forked from ember-cli/eslint-plugin-ember
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge branch 'master' into eslint-doc-generator
* master: Support `<template>` (no-undef, etc) (ember-cli#1395)
- Loading branch information
Showing
6 changed files
with
1,812 additions
and
1,353 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,143 @@ | ||
'use strict'; | ||
|
||
const { getTemplateLocals } = require('@glimmer/syntax'); | ||
const { | ||
preprocessEmbeddedTemplates, | ||
} = require('ember-template-imports/lib/preprocess-embedded-templates'); | ||
const util = require('ember-template-imports/src/util'); | ||
|
||
const TRANSFORM_CACHE = new Map(); | ||
const TEXT_CACHE = new Map(); | ||
|
||
/** | ||
* This function is responsible for running the embedded templates transform | ||
* from ember-template-imports. | ||
* | ||
* It takes a gjs or gts file and converts <template> tags to their | ||
* intermediary (private) format. | ||
* | ||
* This is needed so that eslint knows what variables accessed within the template | ||
* are missing or accessed within the JS scope, because | ||
* part of the output includes a snippet that looks like: | ||
* | ||
* `, { strictMode: true, scope: () => ({on,noop}) })] | ||
* | ||
* So if on and noop are not present in the JS, we will have an error to work with | ||
*/ | ||
function gjs(text, filename) { | ||
const isGlimmerFile = filename.endsWith('.gjs') || filename.endsWith('.gts'); | ||
|
||
if (!isGlimmerFile) { | ||
return [ | ||
{ | ||
filename, | ||
text, | ||
}, | ||
]; | ||
} | ||
|
||
/** | ||
* If the file is already processed, we don't need to process it again. | ||
* | ||
* a file will not have a TEMPLATE_TAG_PLACEHOLDER unless it has been processed | ||
*/ | ||
if (text.includes(util.TEMPLATE_TAG_PLACEHOLDER)) { | ||
return [ | ||
{ | ||
filename, | ||
text, | ||
}, | ||
]; | ||
} | ||
|
||
const preprocessed = preprocessEmbeddedTemplates(text, { | ||
getTemplateLocals, | ||
relativePath: filename, | ||
|
||
templateTag: util.TEMPLATE_TAG_NAME, | ||
templateTagReplacement: util.TEMPLATE_TAG_PLACEHOLDER, | ||
|
||
includeSourceMaps: false, | ||
includeTemplateTokens: true, | ||
}); | ||
|
||
// Because the output looks like | ||
// `, { strictMode: true, scope: () => ({on,noop}) })] | ||
// we could accidentally have this line formatted with | ||
// prettier, or trailing commas, or any variety of formatting rules. | ||
// | ||
// We need to disable all of eslint around this "scope" return | ||
// value and allow-list the rules that check for undefined identifiers | ||
const transformed = preprocessed.output; | ||
|
||
TRANSFORM_CACHE.set(filename, transformed); | ||
TEXT_CACHE.set(filename, text); | ||
|
||
return [ | ||
{ | ||
filename, | ||
text: transformed, | ||
}, | ||
]; | ||
} | ||
|
||
/** | ||
* This function is for mapping back from the intermediary (private) format | ||
* back to gjs/gts / <template> so that we can appropriately display errors | ||
* or warnings in the correct location in the original source file in the user's | ||
* editor. | ||
*/ | ||
function mapRange(messages, filename) { | ||
const transformed = TRANSFORM_CACHE.get(filename); | ||
const original = TEXT_CACHE.get(filename); | ||
const flattened = messages.flat(); | ||
|
||
if (!transformed) { | ||
return flattened; | ||
} | ||
|
||
if (!transformed.includes(util.TEMPLATE_TAG_PLACEHOLDER)) { | ||
return flattened; | ||
} | ||
|
||
const lines = transformed.split('\n'); | ||
const originalLines = original.split('\n'); | ||
|
||
return flattened.map((message) => { | ||
const line = lines[message.line - 1]; | ||
const token = line.slice(message.column - 1, message.endColumn - 1); | ||
|
||
// Now that we have the token, we need to find the location | ||
// in the original text | ||
// | ||
// TODO: Long term, we should use glimmer syntax parsing to find | ||
// the correct node -- otherwise it's too easy to | ||
// partially match on similar text | ||
let originalLineNumber = 0; | ||
let originalColumnNumber = 0; | ||
|
||
for (const [index, line] of originalLines.entries()) { | ||
const column = line.search(new RegExp(`\\b${token}\\b`)); | ||
if (column > -1) { | ||
originalLineNumber = index + 1; | ||
originalColumnNumber = column + 1; | ||
break; | ||
} | ||
} | ||
|
||
const modified = { | ||
...message, | ||
line: originalLineNumber, | ||
column: originalColumnNumber, | ||
endLine: originalLineNumber, | ||
endColumn: originalColumnNumber + token.length, | ||
}; | ||
|
||
return modified; | ||
}); | ||
} | ||
|
||
module.exports = { | ||
preprocess: gjs, | ||
postprocess: mapRange, | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.