Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(linter): implement class sorting rule (first pass) #1362

Merged
merged 86 commits into from
Jan 24, 2024
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
86 commits
Select commit Hold shift + click to select a range
53e104f
wip
DaniGuardiola Dec 29, 2023
0494556
fixes
DaniGuardiola Dec 29, 2023
da3c54c
add layers and pre-process options
DaniGuardiola Dec 31, 2023
beec56b
class parser!
DaniGuardiola Dec 31, 2023
d50aeab
split into modules
DaniGuardiola Dec 31, 2023
67ccb2c
updated obsolete PoC code and cleaned up
DaniGuardiola Dec 31, 2023
e2672fe
use declare_node_union
DaniGuardiola Dec 31, 2023
737005a
cleanup, docs, unit test TODOs
DaniGuardiola Jan 1, 2024
838edba
Merge remote-tracking branch 'origin/main' into feat/class-sort-rule
ematipico Jan 11, 2024
06cbfb3
resolve merge
ematipico Jan 11, 2024
e58fdb7
chore some code
ematipico Jan 11, 2024
9ddba19
shut down clippy
ematipico Jan 11, 2024
e990013
public docs
DaniGuardiola Jan 11, 2024
9e7d666
Merge remote-tracking branch 'origin/main' into feat/class-sort-rule
ematipico Jan 12, 2024
cff3479
fix: conflict
ematipico Jan 12, 2024
508a93f
address review comments
DaniGuardiola Jan 12, 2024
debd8e6
fix query bug and lexer bug
DaniGuardiola Jan 15, 2024
57d6cbf
fix: test and clippy
ematipico Jan 16, 2024
57043b2
chore: make fix unsafe, and make classes static
ematipico Jan 16, 2024
89295d1
wip integration tests
DaniGuardiola Jan 21, 2024
3f7904d
fix options and tests
DaniGuardiola Jan 22, 2024
71039f5
class info unit tests
DaniGuardiola Jan 22, 2024
9144a65
TODOs
DaniGuardiola Jan 22, 2024
56c4478
improve preset structure
DaniGuardiola Jan 22, 2024
2195b0b
one more class info unit test
DaniGuardiola Jan 22, 2024
627484d
fixes
DaniGuardiola Jan 22, 2024
0afcef5
lexer unit tests
DaniGuardiola Jan 23, 2024
84720be
remove TODO (already covered by integration tests)
DaniGuardiola Jan 23, 2024
0d0b4ed
remove panics
DaniGuardiola Jan 23, 2024
72ee1f1
improve docs
DaniGuardiola Jan 23, 2024
e078cbf
tweak docs
ematipico Jan 24, 2024
6174cee
Merge branch 'feat/class-sort-rule' of github.com:DaniGuardiola/biome…
DaniGuardiola Jan 24, 2024
da104cd
docs(pt-br): translated git hooks page (#1498)
hknsh Jan 12, 2024
841f8cb
feat: HTML grammar (#1544)
ematipico Jan 12, 2024
e78e927
Refactor CSS Parser to rename CssRule to CssQualifiedRule (#1546)
denbezrukov Jan 12, 2024
b20f0a4
fix(js_formatter): fix #1511 (#1547)
Conaclos Jan 12, 2024
34c919a
feat: foudations for migration from prettier (#1545)
ematipico Jan 13, 2024
de9b3e1
fix(config): don't ignore `include` when `ignore` is set (#1548)
Conaclos Jan 13, 2024
801c9ba
refactor(cli): do not emit warnings for protected ignored files (#1552)
Conaclos Jan 13, 2024
e79fca1
chore: move `nhedger` to core contributors (#1557)
nhedger Jan 13, 2024
73d57fc
fix(lint/noUselessFragments): trim trivia to avoid suggesting invalid…
togami2864 Jan 14, 2024
c8d0981
fix(config): apply global include/ignore before tools' include/ignore…
Conaclos Jan 14, 2024
e1119f7
release: 1.5.2 (#1562)
Conaclos Jan 15, 2024
f55f2a7
refactor(parser): Update lexer methods to handle only ASCII identifie…
denbezrukov Jan 15, 2024
4807e98
chore(CHANGELOG): add Unreleased section and fix typo (#1566)
Conaclos Jan 15, 2024
31dc7ac
feat(cli): file system prettier migration (#1567)
ematipico Jan 16, 2024
259a839
feat(linter): show dependency variable name by useExhaustiveDependenc…
mehm8128 Jan 16, 2024
1b5f261
feat(css_formatter): Formatting for `border` property (#1453)
faultyserver Jan 16, 2024
f6b12c4
fix(exhaustiveDeps): perform nested capture check in a correct way (#…
XiNiHa Jan 16, 2024
76fbdb8
fix(js_formatter): fix invalid formatting of nested multiline comment…
ah-yu Jan 17, 2024
a15397c
feat(project): `Deserializable` derive macro (#1564)
arendjr Jan 17, 2024
e39e1b1
fix(lint/noArrayIndexKey): false negative in template literals (#1586)
vasucp1207 Jan 17, 2024
61ce44a
docs(website): fix lefthook pre-push recipe (#1587)
lmauromb Jan 17, 2024
63dbae0
refactor(css_parser): Renamed 'CssSimpleFunction' to 'CssFunction' (#…
denbezrukov Jan 18, 2024
bf0004b
fix(formatter): apply line ending option (#1591)
ematipico Jan 18, 2024
472e702
fix(lsp): register formatter dynamically if possible (#1590)
nhedger Jan 18, 2024
8f79c55
fix(linter): correctly handle multibyte chars in regexes (#1592)
Conaclos Jan 18, 2024
0c03701
docs(website): tables from source rules to Biome rules (#1583)
ematipico Jan 18, 2024
fa3bd5b
fix(lint/useExhaustiveDependencies): do not panic on TS import equal …
Conaclos Jan 18, 2024
300d246
chore: fix links in rules sources page
ematipico Jan 18, 2024
9b25002
chore: fix links in rules sources page /pt2
ematipico Jan 18, 2024
ae46172
chore: update documentation (#1599)
ematipico Jan 19, 2024
f9c92e1
feat(docs): add exclusive rules and fix headings (#1602)
ematipico Jan 19, 2024
3e04fd8
docs(formatter): fix rustdoc example (#1604)
spanishpear Jan 19, 2024
c063e28
fix(formatter): correctly handle comments at end of export/import lis…
spanishpear Jan 20, 2024
e03edce
docs: fix pre-commit sample version v0.1.0 (#1612)
9renpoto Jan 20, 2024
d25694f
fix(linter): correctly override linter presets (#1606)
Conaclos Jan 20, 2024
0b6464b
fix(website): git hook shell scripts (#1615)
Conaclos Jan 20, 2024
f8aea9a
fix(config): don't include ignored files known to be jsonc files when…
Conaclos Jan 20, 2024
543248e
fix(website): fix lint-staged glob pattern (#1616)
Conaclos Jan 20, 2024
4bfcfaf
Update no_nodejs_modules.rs (#1621)
huseeiin Jan 21, 2024
c4500f7
fix(js_parser): Allow `const` in TsMethodSignatureTypeMember and TsCa…
magic-akari Jan 21, 2024
8b3ed5f
feat(website): generate open graph images (#1622)
ematipico Jan 21, 2024
c4d72ee
feat(css_parser): CSS parser to improve selector error handling (#1619)
denbezrukov Jan 21, 2024
3214be6
fix(website): rules sources page link (#1632)
Sec-ant Jan 22, 2024
0c12315
release: 1.5.3 (#1629)
Conaclos Jan 22, 2024
74865f0
fix: remove unneeded semicolon in the autofix suggestion (#1638)
togami2864 Jan 22, 2024
20d73ea
Update language-support.mdx (#1639)
millette Jan 22, 2024
db5c6d8
fix(js_formatter): incorrect call chain break (#1646)
kalleep Jan 23, 2024
4176bde
feat(linter): add rule `noSkippedTests` (#1635)
ematipico Jan 23, 2024
997df23
fix(js_parser): correctly parse type arguments in expression (#1645)
ah-yu Jan 24, 2024
e0c1d37
Revert "tweak docs"
ematipico Jan 24, 2024
88357f4
update codegen
ematipico Jan 24, 2024
e23182d
Merge remote-tracking branch 'origin/main' into feat/class-sort-rule
ematipico Jan 24, 2024
f74aec0
codegen
ematipico Jan 24, 2024
a1b156d
test/docs fixes and improvements
DaniGuardiola Jan 24, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions crates/biome_diagnostics_categories/src/categories.rs
Original file line number Diff line number Diff line change
Expand Up @@ -128,6 +128,7 @@ define_categories! {
"lint/nursery/useNodejsImportProtocol": "https://biomejs.dev/linter/rules/use-nodejs-import-protocol",
"lint/nursery/useNumberNamespace": "https://biomejs.dev/linter/rules/use-number-namespace",
"lint/nursery/useShorthandFunctionType": "https://biomejs.dev/linter/rules/use-shorthand-function-type",
"lint/nursery/useSortedClasses": "https://biomejs.dev/linter/rules/use-sorted-classes",
"lint/performance/noAccumulatingSpread": "https://biomejs.dev/linter/rules/no-accumulating-spread",
"lint/performance/noDelete": "https://biomejs.dev/linter/rules/no-delete",
"lint/security/noDangerouslySetInnerHtml": "https://biomejs.dev/linter/rules/no-dangerously-set-inner-html",
Expand Down
2 changes: 2 additions & 0 deletions crates/biome_js_analyze/src/semantic_analyzers/nursery.rs

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Original file line number Diff line number Diff line change
@@ -0,0 +1,122 @@
mod any_class_string_like;
mod class_info;
mod class_lexer;
mod options;
mod presets;
mod sort;
mod sort_config;

use biome_analyze::{
context::RuleContext, declare_rule, ActionCategory, Ast, FixKind, Rule, RuleDiagnostic,
};
use biome_console::markup;
use biome_diagnostics::Applicability;
use biome_js_factory::make::{js_string_literal, js_string_literal_expression, jsx_string};
use biome_rowan::{AstNode, BatchMutationExt};

use crate::JsRuleAction;

pub use self::options::UseSortedClassesOptions;
use self::{
any_class_string_like::AnyClassStringLike,
presets::{get_utilities_preset, UseSortedClassesPreset},
sort::sort_class_name,
sort_config::SortConfig,
};

declare_rule! {
/// Enforce the sorting of CSS classes.
///
/// TODO: description
///
/// ## Examples
///
/// ### Invalid
///
/// ```jsx,expect_diagnostic
ematipico marked this conversation as resolved.
Show resolved Hide resolved
/// <div class="px-2 foo px-4 bar" />;
/// ```
///
/// ## Valid
///
/// ```js
/// // TODO: examples
/// ```
///
pub(crate) UseSortedClasses {
version: "next",
name: "useSortedClasses",
ematipico marked this conversation as resolved.
Show resolved Hide resolved
recommended: false,
fix_kind: FixKind::Safe,
}
}

impl Rule for UseSortedClasses {
type Query = Ast<AnyClassStringLike>;
type State = String;
type Signals = Option<Self::State>;
type Options = UseSortedClassesOptions;

fn run(ctx: &RuleContext<Self>) -> Option<Self::State> {
// TODO: unsure if options are needed here. The sort config should ideally be created once
// from the options and then reused for all queries.
// let options = &ctx.options();
// TODO: the sort config should already exist at this point, and be generated from the options,
// including the preset and extended options as well.
let sort_config = SortConfig::new(
get_utilities_preset(&UseSortedClassesPreset::default()),
Vec::new(),
);

let value = ctx.query().value()?;
let sorted_value = sort_class_name(value.as_str(), &sort_config);
DaniGuardiola marked this conversation as resolved.
Show resolved Hide resolved
if value != sorted_value {
Some(sorted_value)
} else {
None
}
}

fn diagnostic(ctx: &RuleContext<Self>, _: &Self::State) -> Option<RuleDiagnostic> {
Some(
RuleDiagnostic::new(rule_category!(), ctx.query().range(), "TODO: title").note(
markup! {
"TODO: description."
},
),
)
}

fn action(ctx: &RuleContext<Self>, state: &Self::State) -> Option<JsRuleAction> {
let mut mutation = ctx.root().begin();
match ctx.query() {
AnyClassStringLike::JsStringLiteralExpression(string_literal) => {
let replacement = js_string_literal_expression(js_string_literal(state));
mutation.replace_node(string_literal.clone(), replacement);
Some(JsRuleAction {
category: ActionCategory::QuickFix,
applicability: Applicability::Always,
message: markup! {
"TODO: message."
}
.to_owned(),
mutation,
})
}
AnyClassStringLike::JsxString(jsx_string_node) => {
let replacement = jsx_string(js_string_literal(state));
mutation.replace_node(jsx_string_node.clone(), replacement);
Some(JsRuleAction {
category: ActionCategory::QuickFix,
applicability: Applicability::Always,
message: markup! {
"TODO: message."
}
.to_owned(),
mutation,
})
}
_ => None,
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,226 @@
use biome_analyze::{
DaniGuardiola marked this conversation as resolved.
Show resolved Hide resolved
AddVisitor, AnalyzerOptions, Phases, QueryMatch, Queryable, RuleKey, ServiceBag, Visitor,
VisitorContext,
};
use biome_js_syntax::{
JsCallArguments, JsCallExpression, JsLanguage, JsStringLiteralExpression,
JsTemplateChunkElement, JsxAttribute, JsxString,
};
use biome_rowan::{declare_node_union, AstNode, Language, SyntaxNode, TextRange, WalkEvent};

use super::UseSortedClassesOptions;

// utils
// -----

fn get_options_from_analyzer(analyzer_options: &AnalyzerOptions) -> UseSortedClassesOptions {
match analyzer_options
.configuration
.rules
.get_rule_options::<UseSortedClassesOptions>(&RuleKey::new("nursery", "useSortedClasses"))
{
Some(options) => options.clone(),
None => UseSortedClassesOptions::default(),
}
DaniGuardiola marked this conversation as resolved.
Show resolved Hide resolved
}

fn get_callee_name(call_expression: &JsCallExpression) -> Option<String> {
Some(
call_expression
.callee()
.ok()?
.as_js_identifier_expression()?
.name()
.ok()?
.name()
.ok()?
.to_string(),
)
}

fn is_call_expression_of_target_function(
call_expression: &JsCallExpression,
target_functions: &[String],
) -> bool {
match get_callee_name(call_expression) {
Some(name) => target_functions.contains(&name.to_string()),
None => false,
}
DaniGuardiola marked this conversation as resolved.
Show resolved Hide resolved
}

fn get_attribute_name(attribute: &JsxAttribute) -> Option<String> {
Some(attribute.name().ok()?.as_jsx_name()?.to_string())
}
DaniGuardiola marked this conversation as resolved.
Show resolved Hide resolved

fn is_target_attribute(attribute: &JsxAttribute, target_attributes: &[String]) -> bool {
match get_attribute_name(attribute) {
Some(name) => target_attributes.contains(&name.to_string()),
None => false,
}
}

// attributes visitor
// ------------------

#[derive(Default)]
struct StringLiteralInAttributeVisitor {
in_target_attribute: bool,
}

// Finds class-like strings in JSX attributes, including class, className, and others defined in the options.
impl Visitor for StringLiteralInAttributeVisitor {
type Language = JsLanguage;
fn visit(
&mut self,
event: &WalkEvent<SyntaxNode<Self::Language>>,
mut ctx: VisitorContext<Self::Language>,
) {
let options = get_options_from_analyzer(ctx.options);
ematipico marked this conversation as resolved.
Show resolved Hide resolved
match event {
WalkEvent::Enter(node) => {
// When entering an attribute node, track if we are in a target attribute.
if let Some(attribute) = JsxAttribute::cast_ref(node) {
self.in_target_attribute = is_target_attribute(&attribute, &options.attributes);
}

// When entering a JSX string node, and we are in a target attribute, emit.
if let Some(jsx_string) = JsxString::cast_ref(node) {
if self.in_target_attribute {
ctx.match_query(AnyClassStringLike::JsxString(jsx_string));
}
}

// When entering a string literal node, and we are in a target attribute, emit.
if let Some(string_literal) = JsStringLiteralExpression::cast_ref(node) {
if self.in_target_attribute {
ctx.match_query(AnyClassStringLike::JsStringLiteralExpression(
string_literal,
));
}
}
}
WalkEvent::Leave(node) => {
// When leaving an attribute node, reset in_target_attribute.
if JsxAttribute::cast_ref(node).is_some() {
self.in_target_attribute = false;
}
}
}
}
}

// functions (call expression) visitor
// -----------------------------------

#[derive(Default)]
struct StringLiteralInCallExpressionVisitor {
in_target_function: bool,
in_arguments: bool,
}

// Finds class-like strings inside function calls defined in the options, e.g. clsx(classes).
impl Visitor for StringLiteralInCallExpressionVisitor {
type Language = JsLanguage;

fn visit(
&mut self,
event: &WalkEvent<SyntaxNode<Self::Language>>,
mut ctx: VisitorContext<Self::Language>,
) {
let options = get_options_from_analyzer(ctx.options);
if options.functions.is_empty() {
return;
}
match event {
WalkEvent::Enter(node) => {
// When entering a call expression node, track if we are in a target function and reset
// in_arguments.
if let Some(call_expression) = JsCallExpression::cast_ref(node) {
self.in_target_function =
is_call_expression_of_target_function(&call_expression, &options.functions);
self.in_arguments = false;
}

// When entering a call arguments node, set in_arguments.
if JsCallArguments::cast_ref(node).is_some() {
self.in_arguments = true;
}

// When entering a string literal node, and we are in a target function and in arguments, emit.
if let Some(string_literal) = JsStringLiteralExpression::cast_ref(node) {
if self.in_target_function && self.in_arguments {
ctx.match_query(AnyClassStringLike::JsStringLiteralExpression(
string_literal,
));
}
}
}
WalkEvent::Leave(node) => {
// When leaving a call arguments node, reset in_arguments.
if JsCallArguments::cast_ref(node).is_some() {
self.in_arguments = false;
}
}
}
}
}

// functions (template chunk) visitor
// ----------------------------------

// Finds class-like template chunks in tagged template calls defined in the options, e.g. tw`classes`.
// TODO: template chunk visitor

// query
// -----

declare_node_union! {
/// A string literal, JSX string, or template chunk representing a CSS class string.
pub AnyClassStringLike = JsStringLiteralExpression | JsxString | JsTemplateChunkElement
}

impl AnyClassStringLike {
/// Returns the value of the string literal, JSX string, or template chunk.
pub fn value(&self) -> Option<String> {
match self {
AnyClassStringLike::JsStringLiteralExpression(string_literal) => {
Some(string_literal.inner_string_text().ok()?.to_string())
}
AnyClassStringLike::JsxString(jsx_string) => {
Some(jsx_string.inner_string_text().ok()?.to_string())
}
AnyClassStringLike::JsTemplateChunkElement(template_chunk) => {
Some(template_chunk.to_string())
}
}
}
DaniGuardiola marked this conversation as resolved.
Show resolved Hide resolved
}

impl QueryMatch for AnyClassStringLike {
fn text_range(&self) -> TextRange {
self.range()
}
}

impl Queryable for AnyClassStringLike {
type Input = Self;
type Language = JsLanguage;
type Output = AnyClassStringLike;
type Services = ();

fn build_visitor(
analyzer: &mut impl AddVisitor<Self::Language>,
_: &<Self::Language as Language>::Root,
) {
analyzer.add_visitor(Phases::Syntax, || {
StringLiteralInAttributeVisitor::default()
});
analyzer.add_visitor(Phases::Syntax, || {
StringLiteralInCallExpressionVisitor::default()
});
}

fn unwrap_match(_: &ServiceBag, query: &Self::Input) -> Self::Output {
query.clone()
}
}
Loading
Loading