From 32f905087099abbfec5cbd24bce84001ffe4997b Mon Sep 17 00:00:00 2001 From: Sysix Date: Fri, 20 Dec 2024 16:20:13 +0100 Subject: [PATCH 01/10] fix(linter): rule: `no-restricted-imports` support option `patterns` with `group` key --- .../src/rules/eslint/no_restricted_imports.rs | 141 +++++++++++++----- 1 file changed, 101 insertions(+), 40 deletions(-) diff --git a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs index 4b0c1ffe94b9b..49bd0a46dd5ae 100644 --- a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs +++ b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs @@ -26,13 +26,20 @@ fn no_restricted_imports_diagnostic( } #[derive(Debug, Default, Clone)] -pub struct NoRestrictedImports { - paths: Box, +pub struct NoRestrictedImports(Box); + +impl std::ops::Deref for NoRestrictedImports { + type Target = NoRestrictedImportsConfig; + + fn deref(&self) -> &Self::Target { + &self.0 + } } #[derive(Debug, Default, Clone)] -struct NoRestrictedImportsConfig { - paths: Box<[RestrictedPath]>, +pub struct NoRestrictedImportsConfig { + paths: Vec, + patterns: Vec, } #[derive(Debug, Clone, Deserialize)] @@ -44,6 +51,15 @@ struct RestrictedPath { message: Option, } +#[derive(Debug, Clone, Deserialize)] +#[serde(rename_all = "camelCase")] +struct RestrictedPattern { + group: Vec, + import_names: Option>, + allow_import_names: Option>, + message: Option, +} + declare_oxc_lint!( /// ### What it does /// This rule allows you to specify imports that you don’t want to use in your application. @@ -77,26 +93,17 @@ declare_oxc_lint!( nursery, ); -fn add_configuration_from_object( +fn add_configuration_path_from_object( paths: &mut Vec, - obj: &serde_json::Map, + paths_value: &serde_json::Value, ) { - let Some(paths_value) = obj.get("paths") else { - if let Ok(path) = - serde_json::from_value::(serde_json::Value::Object(obj.clone())) - { - paths.push(path); - } - return; - }; - let Some(paths_array) = paths_value.as_array() else { return; }; for path_value in paths_array { match path_value { - Value::String(module_name) => add_configuration_from_string(paths, module_name), + Value::String(module_name) => add_configuration_path_from_string(paths, module_name), Value::Object(_) => { if let Ok(path) = serde_json::from_value::(path_value.clone()) { paths.push(path); @@ -107,7 +114,7 @@ fn add_configuration_from_object( } } -fn add_configuration_from_string(paths: &mut Vec, module_name: &str) { +fn add_configuration_path_from_string(paths: &mut Vec, module_name: &str) { paths.push(RestrictedPath { name: CompactStr::new(module_name), import_names: None, @@ -116,39 +123,101 @@ fn add_configuration_from_string(paths: &mut Vec, module_name: & }); } +fn add_configuration_patterns_from_object( + patterns: &mut Vec, + patterns_value: &serde_json::Value, +) { + let Some(paths_array) = patterns_value.as_array() else { + return; + }; + + for path_value in paths_array { + match path_value { + Value::String(module_name) => { + add_configuration_patterns_from_string(patterns, module_name); + } + Value::Object(_) => { + if let Ok(path) = serde_json::from_value::(path_value.clone()) { + patterns.push(path); + } + } + _ => (), + } + } +} + +fn add_configuration_patterns_from_string(paths: &mut Vec, module_name: &str) { + paths.push(RestrictedPattern { + group: vec![CompactStr::new(module_name)], + import_names: None, + allow_import_names: None, + message: None, + }); +} + impl Rule for NoRestrictedImports { fn from_configuration(value: serde_json::Value) -> Self { - let mut paths = Vec::new(); + let mut paths: Vec = Vec::new(); + let mut patterns: Vec = Vec::new(); match &value { Value::Array(module_names) => { for module_name in module_names { match module_name { Value::String(module_string) => { - add_configuration_from_string(&mut paths, module_string); + add_configuration_path_from_string(&mut paths, module_string); + } + Value::Object(obj) => { + if let Some(paths_value) = obj.get("paths") { + add_configuration_path_from_object(&mut paths, paths_value); + } else if let Some(patterns_value) = obj.get("patterns") { + add_configuration_patterns_from_object( + &mut patterns, + patterns_value, + ); + } else if let Ok(path) = serde_json::from_value::( + serde_json::Value::Object(obj.clone()), + ) { + paths.push(path); + }; } - Value::Object(obj) => add_configuration_from_object(&mut paths, obj), _ => (), }; } } Value::String(module_name) => { - add_configuration_from_string(&mut paths, module_name); + add_configuration_path_from_string(&mut paths, module_name); } Value::Object(obj) => { - add_configuration_from_object(&mut paths, obj); + if let Some(paths_value) = obj.get("paths") { + add_configuration_path_from_object(&mut paths, paths_value); + } else if let Some(patterns_value) = obj.get("patterns") { + add_configuration_patterns_from_object(&mut patterns, patterns_value); + } else if let Ok(path) = + serde_json::from_value::(serde_json::Value::Object(obj.clone())) + { + paths.push(path); + } } _ => {} } - Self { paths: Box::new(NoRestrictedImportsConfig { paths: paths.into_boxed_slice() }) } + Self(Box::new(NoRestrictedImportsConfig { paths, patterns })) } fn run_once(&self, ctx: &LintContext<'_>) { let module_record = ctx.module_record(); let mut side_effect_import_map: FxHashMap<&CompactStr, Vec> = FxHashMap::default(); - for path in &self.paths.paths { + for (source, requests) in &module_record.requested_modules { + for request in requests { + if request.is_import && module_record.import_entries.is_empty() { + side_effect_import_map.entry(source).or_default().push(request.span); + } + } + } + + for path in &self.paths { for entry in &module_record.import_entries { let source = entry.module_request.name(); let span = entry.module_request.span(); @@ -164,14 +233,6 @@ impl Rule for NoRestrictedImports { no_restricted_imports_diagnostic(ctx, span, path.message.clone(), source); } - for (source, requests) in &module_record.requested_modules { - for request in requests { - if request.is_import && module_record.import_entries.is_empty() { - side_effect_import_map.entry(source).or_default().push(request.span); - } - } - } - for (source, spans) in &side_effect_import_map { if source.as_str() == path.name.as_str() && path.import_names.is_none() { if let Some(span) = spans.iter().next() { @@ -735,14 +796,14 @@ fn test() { r#"import withPaths from "foo/bar";"#, Some(serde_json::json!([{ "paths": ["foo/bar"] }])), ), - // ( - // r#"import withPatterns from "foo/bar";"#, - // Some(serde_json::json!([{ "patterns": ["foo"] }])), - // ), - // ( - // r#"import withPatterns from "foo/bar";"#, - // Some(serde_json::json!([{ "patterns": ["bar"] }])), - // ), + ( + r#"import withPatterns from "foo/bar";"#, + Some(serde_json::json!([{ "patterns": ["foo"] }])), + ), + ( + r#"import withPatterns from "foo/bar";"#, + Some(serde_json::json!([{ "patterns": ["bar"] }])), + ), // ( // r#"import withPatterns from "foo/baz";"#, // Some( From 6b6d508a9754a4176ee641faecd276a155373caf Mon Sep 17 00:00:00 2001 From: Sysix Date: Fri, 20 Dec 2024 17:13:17 +0100 Subject: [PATCH 02/10] fix(linter): rule: no-restricted-imports support option patterns with group key --- Cargo.lock | 1 + crates/oxc_linter/Cargo.toml | 1 + .../src/rules/eslint/no_restricted_imports.rs | 30 +++++++++++++++++++ 3 files changed, 32 insertions(+) diff --git a/Cargo.lock b/Cargo.lock index b0446737852f1..4db320efc1b57 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1691,6 +1691,7 @@ dependencies = [ "dashmap 6.1.0", "fast-glob", "globset", + "ignore", "insta", "itertools", "json-strip-comments", diff --git a/crates/oxc_linter/Cargo.toml b/crates/oxc_linter/Cargo.toml index de4dfe0fdddba..10105e422f152 100644 --- a/crates/oxc_linter/Cargo.toml +++ b/crates/oxc_linter/Cargo.toml @@ -42,6 +42,7 @@ cow-utils = { workspace = true } dashmap = { workspace = true } fast-glob = { workspace = true } globset = { workspace = true } +ignore = { workspace = true } itertools = { workspace = true } json-strip-comments = { workspace = true } language-tags = { workspace = true } diff --git a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs index 49bd0a46dd5ae..b1f97ccb8e58e 100644 --- a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs +++ b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs @@ -4,6 +4,7 @@ use oxc_span::{CompactStr, Span}; use rustc_hash::FxHashMap; use serde::Deserialize; use serde_json::Value; +use ignore::gitignore::GitignoreBuilder; use crate::{ context::LintContext, @@ -217,6 +218,35 @@ impl Rule for NoRestrictedImports { } } + for pattern in &self.patterns { + let mut builder = GitignoreBuilder::new("/"); + + for group in &pattern.group { + let _ = builder.add_line(None, group.as_str()); + } + + let Ok(gitignore) = builder.build() else { + continue; + }; + + for entry in &module_record.import_entries { + let source = entry.module_request.name(); + let span = entry.module_request.span(); + + let matched = gitignore.matched(source, true); + println!("{:?}", matched); + + // ToDo: better whitelist handling + if matched.is_whitelist() { + break; + } + + if !matched.is_none() { + no_restricted_imports_diagnostic(ctx, span, pattern.message.clone(), source); + } + } + } + for path in &self.paths { for entry in &module_record.import_entries { let source = entry.module_request.name(); From cb4d3326ca9965b51cd51a8c0fd1cb72fdc97ba4 Mon Sep 17 00:00:00 2001 From: "autofix-ci[bot]" <114827586+autofix-ci[bot]@users.noreply.github.com> Date: Fri, 20 Dec 2024 16:14:32 +0000 Subject: [PATCH 03/10] [autofix.ci] apply automated fixes --- crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs index b1f97ccb8e58e..7e73aabed3647 100644 --- a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs +++ b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs @@ -1,10 +1,10 @@ +use ignore::gitignore::GitignoreBuilder; use oxc_diagnostics::OxcDiagnostic; use oxc_macros::declare_oxc_lint; use oxc_span::{CompactStr, Span}; use rustc_hash::FxHashMap; use serde::Deserialize; use serde_json::Value; -use ignore::gitignore::GitignoreBuilder; use crate::{ context::LintContext, From f397cfd9e58bc7b80f31e0731efc7a0e2424fdeb Mon Sep 17 00:00:00 2001 From: Sysix Date: Fri, 20 Dec 2024 20:20:17 +0100 Subject: [PATCH 04/10] fix(linter): rule: no-restricted-imports support option patterns with group key --- .../src/rules/eslint/no_restricted_imports.rs | 266 ++++++++---------- .../eslint_no_restricted_imports.snap | 14 +- 2 files changed, 119 insertions(+), 161 deletions(-) diff --git a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs index 7e73aabed3647..5d2a80e12cd20 100644 --- a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs +++ b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs @@ -8,8 +8,8 @@ use serde_json::Value; use crate::{ context::LintContext, - module_record::{ExportImportName, ImportImportName}, - rule::Rule, + module_record::{ExportEntry, ExportImportName, ImportEntry, ImportImportName}, + rule::Rule, ModuleRecord, }; fn no_restricted_imports_diagnostic( @@ -208,61 +208,89 @@ impl Rule for NoRestrictedImports { fn run_once(&self, ctx: &LintContext<'_>) { let module_record = ctx.module_record(); - let mut side_effect_import_map: FxHashMap<&CompactStr, Vec> = FxHashMap::default(); - for (source, requests) in &module_record.requested_modules { - for request in requests { - if request.is_import && module_record.import_entries.is_empty() { - side_effect_import_map.entry(source).or_default().push(request.span); - } - } - } + // for pattern in &self.patterns { + // let mut builder = GitignoreBuilder::new("/"); + // + // for group in &pattern.group { + // let _ = builder.add_line(None, group.as_str()); + // } + // + // let Ok(gitignore) = builder.build() else { + // continue; + // }; + // + // for entry in &module_record.import_entries { + // let source = entry.module_request.name(); + // let span = entry.module_request.span(); + // + // let matched = gitignore.matched(source, true); + // println!("{:?}", matched); + // + // // ToDo: better whitelist handling + // if matched.is_whitelist() { + // break; + // } + // + // if !matched.is_none() { + // no_restricted_imports_diagnostic(ctx, span, pattern.message.clone(), source); + // } + // } + // } - for pattern in &self.patterns { - let mut builder = GitignoreBuilder::new("/"); + self.report_side_effects(ctx, module_record); - for group in &pattern.group { - let _ = builder.add_line(None, group.as_str()); - } + for entry in &module_record.import_entries { + self.report_import_name_allowed(ctx, entry); + } - let Ok(gitignore) = builder.build() else { - continue; - }; + for entry in &module_record.local_export_entries { + self.report_export_name_allowed(ctx, entry); + } - for entry in &module_record.import_entries { - let source = entry.module_request.name(); - let span = entry.module_request.span(); + for entry in &module_record.indirect_export_entries { + self.report_export_name_allowed(ctx, entry); + } - let matched = gitignore.matched(source, true); - println!("{:?}", matched); + for entry in &module_record.star_export_entries { + self.report_export_name_allowed(ctx, entry); + } + } +} - // ToDo: better whitelist handling - if matched.is_whitelist() { - break; - } +impl NoRestrictedImports { + fn is_name_span_allowed(&self, name: &CompactStr, path: &RestrictedPath) -> bool { + // fast check if this name is allowed + if path.allow_import_names.as_ref().is_some_and(|allowed| allowed.contains(&name)) { + return true; + } - if !matched.is_none() { - no_restricted_imports_diagnostic(ctx, span, pattern.message.clone(), source); - } - } + // when no importNames option is provided, no import in general is allowed + if path.import_names.as_ref().is_none() { + return false; } - for path in &self.paths { - for entry in &module_record.import_entries { - let source = entry.module_request.name(); - let span = entry.module_request.span(); + // the name is found is the importNames list + if path.import_names.as_ref().is_some_and(|disallowed| disallowed.contains(&name)) { + return false; + } - if source != path.name.as_str() { - continue; - } + // we allow it + true + } - if is_import_name_allowed(&entry.import_name, path) { - continue; - } + fn report_side_effects(&self, ctx: &LintContext<'_>, module_record: &ModuleRecord) { + let mut side_effect_import_map: FxHashMap<&CompactStr, Vec> = FxHashMap::default(); - no_restricted_imports_diagnostic(ctx, span, path.message.clone(), source); + for (source, requests) in &module_record.requested_modules { + for request in requests { + if request.is_import && module_record.import_entries.is_empty() { + side_effect_import_map.entry(source).or_default().push(request.span); + } } + } + for path in &self.paths { for (source, spans) in &side_effect_import_map { if source.as_str() == path.name.as_str() && path.import_names.is_none() { if let Some(span) = spans.iter().next() { @@ -270,129 +298,59 @@ impl Rule for NoRestrictedImports { } } } + } + } - for entry in &module_record.local_export_entries { - if let Some(module_request) = &entry.module_request { - let source = module_request.name(); - let span = entry.span; + fn report_import_name_allowed(&self, ctx: &LintContext<'_>, entry: &ImportEntry) { + let source = entry.module_request.name(); - if source == path.name.as_str() { - no_restricted_imports_diagnostic(ctx, span, path.message.clone(), source); - } - } + for path in &self.paths { + if source != path.name.as_str() { + continue; } - for entry in &module_record.indirect_export_entries { - if let Some(module_request) = &entry.module_request { - let source = module_request.name(); - let span = entry.span; - - if source != path.name.as_str() { - continue; - } - if is_export_name_allowed(&entry.import_name, path) { - continue; - } - - no_restricted_imports_diagnostic(ctx, span, path.message.clone(), source); + let skipable = match &entry.import_name { + ImportImportName::Name(import) => self.is_name_span_allowed(&import.name, path), + ImportImportName::Default(_) => { + self.is_name_span_allowed(&CompactStr::new("default"), path) } - } - - for entry in &module_record.star_export_entries { - if let Some(module_request) = &entry.module_request { - let source = module_request.name(); - let span = entry.span; + ImportImportName::NamespaceObject => false, + }; - if source != path.name.as_str() { - continue; - } + if skipable { + continue; + } - if is_export_name_allowed(&entry.import_name, path) { - continue; - } + let span = entry.module_request.span(); - no_restricted_imports_diagnostic(ctx, span, path.message.clone(), source); - } - } + no_restricted_imports_diagnostic(ctx, span, path.message.clone(), source); } } -} -fn is_import_name_allowed(name: &ImportImportName, path: &RestrictedPath) -> bool { - match &name { - ImportImportName::Name(import) => { - // fast check if this name is allowed - if path - .allow_import_names - .as_ref() - .is_some_and(|allowed| allowed.contains(&import.name)) - { - return true; - } + fn report_export_name_allowed(&self, ctx: &LintContext<'_>, entry: &ExportEntry) { + let Some(source) = entry.module_request.as_ref().map(|request| request.name()) else { + return; + }; - // when no importNames option is provided, no import in general is allowed - if path.import_names.as_ref().is_none() { - return false; - } - - // the name is found is the importNames list - if path - .import_names - .as_ref() - .is_some_and(|disallowed| disallowed.contains(&import.name)) - { - return false; - } - - // we allow it - true - } - ImportImportName::Default(_) => { - if path - .import_names - .as_ref() - .is_some_and(|disallowed| disallowed.contains(&CompactStr::new("default"))) - || path.import_names.as_ref().is_none() - { - return false; + for path in &self.paths { + if source != path.name.as_str() { + continue; } - true - } - ImportImportName::NamespaceObject => false, - } -} - -fn is_export_name_allowed(name: &ExportImportName, path: &RestrictedPath) -> bool { - match &name { - ExportImportName::Name(import_name) => { - // fast check if this name is allowed - if path - .allow_import_names - .as_ref() - .is_some_and(|allowed| allowed.contains(&import_name.name)) - { - return true; - } + let skipable = match &entry.import_name { + ExportImportName::Name(import) => self.is_name_span_allowed(&import.name, path), + ExportImportName::All | ExportImportName::AllButDefault => false, + ExportImportName::Null => true, + }; - // when no importNames option is provided, no import in general is allowed - if path.import_names.as_ref().is_none() { - return false; + if skipable { + continue; } - // the name is found is the importNames list - if path - .import_names - .as_ref() - .is_some_and(|disallowed| disallowed.contains(&import_name.name)) - { - return false; - } + let span = entry.span; - true + no_restricted_imports_diagnostic(ctx, span, path.message.clone(), source); } - ExportImportName::All | ExportImportName::AllButDefault => false, - ExportImportName::Null => true, } } @@ -826,14 +784,14 @@ fn test() { r#"import withPaths from "foo/bar";"#, Some(serde_json::json!([{ "paths": ["foo/bar"] }])), ), - ( - r#"import withPatterns from "foo/bar";"#, - Some(serde_json::json!([{ "patterns": ["foo"] }])), - ), - ( - r#"import withPatterns from "foo/bar";"#, - Some(serde_json::json!([{ "patterns": ["bar"] }])), - ), + // ( + // r#"import withPatterns from "foo/bar";"#, + // Some(serde_json::json!([{ "patterns": ["foo"] }])), + // ), + // ( + // r#"import withPatterns from "foo/bar";"#, + // Some(serde_json::json!([{ "patterns": ["bar"] }])), + // ), // ( // r#"import withPatterns from "foo/baz";"#, // Some( diff --git a/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap b/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap index 131a7e8bdd204..ce0cf44f21c72 100644 --- a/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap +++ b/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap @@ -275,21 +275,21 @@ snapshot_kind: text ╰──── help: Remove the import statement. - ⚠ eslint(no-restricted-imports): import Text from ui/_components instead + ⚠ eslint(no-restricted-imports): import Image from ui/_components instead ╭─[no_restricted_imports.tsx:1:41] 1 │ import { Image, Text, ScrollView } from 'react-native' · ────────────── ╰──── help: Remove the import statement. - ⚠ eslint(no-restricted-imports): import ScrollView from ui/_components instead + ⚠ eslint(no-restricted-imports): import Text from ui/_components instead ╭─[no_restricted_imports.tsx:1:41] 1 │ import { Image, Text, ScrollView } from 'react-native' · ────────────── ╰──── help: Remove the import statement. - ⚠ eslint(no-restricted-imports): import Image from ui/_components instead + ⚠ eslint(no-restricted-imports): import ScrollView from ui/_components instead ╭─[no_restricted_imports.tsx:1:41] 1 │ import { Image, Text, ScrollView } from 'react-native' · ────────────── @@ -310,14 +310,14 @@ snapshot_kind: text ╰──── help: Remove the import statement. - ⚠ eslint(no-restricted-imports): Use `barbaz` instead of `bar`. + ⚠ eslint(no-restricted-imports): Don"t use "foo" and `qux` from "mod". ╭─[no_restricted_imports.tsx:1:36] 1 │ import { foo, bar, baz, qux } from 'mod' · ───── ╰──── help: Remove the import statement. - ⚠ eslint(no-restricted-imports): Don"t use "foo" and `qux` from "mod". + ⚠ eslint(no-restricted-imports): Use `barbaz` instead of `bar`. ╭─[no_restricted_imports.tsx:1:36] 1 │ import { foo, bar, baz, qux } from 'mod' · ───── @@ -338,14 +338,14 @@ snapshot_kind: text ╰──── help: Remove the import statement. - ⚠ eslint(no-restricted-imports): Don"t use "foo" or "baz" from "mod". + ⚠ eslint(no-restricted-imports): Use "b" or `bar` from "quux/mod" instead. ╭─[no_restricted_imports.tsx:1:36] 1 │ import { foo, bar, baz, qux } from 'mod' · ───── ╰──── help: Remove the import statement. - ⚠ eslint(no-restricted-imports): Use "b" or `bar` from "quux/mod" instead. + ⚠ eslint(no-restricted-imports): Don"t use "foo" or "baz" from "mod". ╭─[no_restricted_imports.tsx:1:36] 1 │ import { foo, bar, baz, qux } from 'mod' · ───── From 29113095ff4efe98a146d37ab12f671a1ae18194 Mon Sep 17 00:00:00 2001 From: Sysix Date: Fri, 20 Dec 2024 21:44:12 +0100 Subject: [PATCH 05/10] fix(linter): rule: no-restricted-imports support option patterns with group key --- .../src/rules/eslint/no_restricted_imports.rs | 411 ++++++++++-------- 1 file changed, 230 insertions(+), 181 deletions(-) diff --git a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs index 5d2a80e12cd20..2ee679acf8fbb 100644 --- a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs +++ b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs @@ -9,7 +9,8 @@ use serde_json::Value; use crate::{ context::LintContext, module_record::{ExportEntry, ExportImportName, ImportEntry, ImportImportName}, - rule::Rule, ModuleRecord, + rule::Rule, + ModuleRecord, }; fn no_restricted_imports_diagnostic( @@ -209,35 +210,6 @@ impl Rule for NoRestrictedImports { fn run_once(&self, ctx: &LintContext<'_>) { let module_record = ctx.module_record(); - // for pattern in &self.patterns { - // let mut builder = GitignoreBuilder::new("/"); - // - // for group in &pattern.group { - // let _ = builder.add_line(None, group.as_str()); - // } - // - // let Ok(gitignore) = builder.build() else { - // continue; - // }; - // - // for entry in &module_record.import_entries { - // let source = entry.module_request.name(); - // let span = entry.module_request.span(); - // - // let matched = gitignore.matched(source, true); - // println!("{:?}", matched); - // - // // ToDo: better whitelist handling - // if matched.is_whitelist() { - // break; - // } - // - // if !matched.is_none() { - // no_restricted_imports_diagnostic(ctx, span, pattern.message.clone(), source); - // } - // } - // } - self.report_side_effects(ctx, module_record); for entry in &module_record.import_entries { @@ -259,9 +231,29 @@ impl Rule for NoRestrictedImports { } impl NoRestrictedImports { - fn is_name_span_allowed(&self, name: &CompactStr, path: &RestrictedPath) -> bool { + fn is_name_span_allowed_in_path(name: &CompactStr, path: &RestrictedPath) -> bool { + // fast check if this name is allowed + if path.allow_import_names.as_ref().is_some_and(|allowed| allowed.contains(name)) { + return true; + } + + // when no importNames option is provided, no import in general is allowed + if path.import_names.as_ref().is_none() { + return false; + } + + // the name is found is the importNames list + if path.import_names.as_ref().is_some_and(|disallowed| disallowed.contains(name)) { + return false; + } + + // we allow it + true + } + + fn is_name_span_allowed_in_pattern(name: &CompactStr, path: &RestrictedPattern) -> bool { // fast check if this name is allowed - if path.allow_import_names.as_ref().is_some_and(|allowed| allowed.contains(&name)) { + if path.allow_import_names.as_ref().is_some_and(|allowed| allowed.contains(name)) { return true; } @@ -271,7 +263,7 @@ impl NoRestrictedImports { } // the name is found is the importNames list - if path.import_names.as_ref().is_some_and(|disallowed| disallowed.contains(&name)) { + if path.import_names.as_ref().is_some_and(|disallowed| disallowed.contains(name)) { return false; } @@ -310,9 +302,11 @@ impl NoRestrictedImports { } let skipable = match &entry.import_name { - ImportImportName::Name(import) => self.is_name_span_allowed(&import.name, path), + ImportImportName::Name(import) => { + Self::is_name_span_allowed_in_path(&import.name, path) + } ImportImportName::Default(_) => { - self.is_name_span_allowed(&CompactStr::new("default"), path) + Self::is_name_span_allowed_in_path(&CompactStr::new("default"), path) } ImportImportName::NamespaceObject => false, }; @@ -325,10 +319,63 @@ impl NoRestrictedImports { no_restricted_imports_diagnostic(ctx, span, path.message.clone(), source); } + + // println!("{:?}", self.patterns); + + let mut whitelist_found = false; + let mut found_errors = vec![]; + + for pattern in &self.patterns { + let skipable = match &entry.import_name { + ImportImportName::Name(import) => { + Self::is_name_span_allowed_in_pattern(&import.name, pattern) + } + ImportImportName::Default(_) => { + Self::is_name_span_allowed_in_pattern(&CompactStr::new("default"), pattern) + } + ImportImportName::NamespaceObject => false, + }; + + if skipable { + continue; + } + + let mut builder = GitignoreBuilder::new("/"); + + for group in &pattern.group { + let _ = builder.add_line(None, group.as_str()); + } + + let Ok(gitignore) = builder.build() else { + continue; + }; + + let source = entry.module_request.name(); + let span = entry.module_request.span(); + + let matched = gitignore.matched(source, true); + // println!("{:?}", matched); + + if matched.is_whitelist() { + whitelist_found = true; + break; + } + + if !matched.is_none() { + found_errors.push((span, pattern)); + } + } + + if !whitelist_found && !found_errors.is_empty() { + for (span, pattern) in found_errors { + no_restricted_imports_diagnostic(ctx, span, pattern.message.clone(), source); + } + } } fn report_export_name_allowed(&self, ctx: &LintContext<'_>, entry: &ExportEntry) { - let Some(source) = entry.module_request.as_ref().map(|request| request.name()) else { + let Some(source) = entry.module_request.as_ref().map(crate::module_record::NameSpan::name) + else { return; }; @@ -338,7 +385,9 @@ impl NoRestrictedImports { } let skipable = match &entry.import_name { - ExportImportName::Name(import) => self.is_name_span_allowed(&import.name, path), + ExportImportName::Name(import) => { + Self::is_name_span_allowed_in_path(&import.name, path) + } ExportImportName::All | ExportImportName::AllButDefault => false, ExportImportName::Null => true, }; @@ -592,99 +641,99 @@ fn test() { }] }])), ), - ( - "import Foo from 'foo';", - Some(serde_json::json!([{ - "patterns": [{ - "group": ["foo"], - "importNamePattern": "^Foo" - }] - }])), - ), - ( - "import Foo from 'foo';", - Some(serde_json::json!([{ - "patterns": [{ - "importNames": ["Foo"], - "group": ["foo"], - "importNamePattern": "^Foo" - }] - }])), - ), - ( - "import Foo from '../../my/relative-module';", - Some(serde_json::json!([{ - "patterns": [{ - "group": ["**/my/relative-module"], - "importNamePattern": "^Foo" - }] - }])), - ), - ( - "import { Bar } from '../../my/relative-module';", - Some(serde_json::json!([{ - "patterns": [{ - "group": ["**/my/relative-module"], - "importNamePattern": "^Foo" - }] - }])), - ), - ( - "import { Bar as Foo } from '../../my/relative-module';", - Some(serde_json::json!([{ - "patterns": [{ - "group": ["**/my/relative-module"], - "importNamePattern": "^Foo" - }] - }])), - ), - ( - "import { Bar as Foo } from '../../my/relative-module';", - Some(serde_json::json!([{ - "patterns": [{ - "importNames": ["Foo"], - "group": ["**/my/relative-module"], - "importNamePattern": "^Foo" - }] - }])), - ), - ( - "import Foo, { Baz as Bar } from '../../my/relative-module';", - Some(serde_json::json!([{ - "patterns": [{ - "group": ["**/my/relative-module"], - "importNamePattern": "^(Foo|Bar)" - }] - }])), - ), - ( - "import Foo, { Baz as Bar } from '../../my/relative-module';", - Some(serde_json::json!([{ - "patterns": [{ - "importNames": ["Foo"], - "group": ["**/my/relative-module"], - "importNamePattern": "^Bar" - }] - }])), - ), - ( - "export { Bar } from 'foo';", - Some(serde_json::json!([{ - "patterns": [{ - "group": ["foo"], - "importNamePattern": "^Foo" - }] - }])), - ), - ( - "export { Bar as Foo } from 'foo';", - Some(serde_json::json!([{ - "patterns": [{ - "group": ["foo"], - "importNamePattern": "^Foo" - }] - }])), - ), + // ( + // "import Foo from 'foo';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "group": ["foo"], + // "importNamePattern": "^Foo" + // }] + // }])), + // ), + // ( + // "import Foo from 'foo';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "importNames": ["Foo"], + // "group": ["foo"], + // "importNamePattern": "^Foo" + // }] + // }])), + // ), + // ( + // "import Foo from '../../my/relative-module';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "group": ["**/my/relative-module"], + // "importNamePattern": "^Foo" + // }] + // }])), + // ), + // ( + // "import { Bar } from '../../my/relative-module';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "group": ["**/my/relative-module"], + // "importNamePattern": "^Foo" + // }] + // }])), + // ), + // ( + // "import { Bar as Foo } from '../../my/relative-module';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "group": ["**/my/relative-module"], + // "importNamePattern": "^Foo" + // }] + // }])), + // ), + // ( + // "import { Bar as Foo } from '../../my/relative-module';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "importNames": ["Foo"], + // "group": ["**/my/relative-module"], + // "importNamePattern": "^Foo" + // }] + // }])), + // ), + // ( + // "import Foo, { Baz as Bar } from '../../my/relative-module';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "group": ["**/my/relative-module"], + // "importNamePattern": "^(Foo|Bar)" + // }] + // }])), + // ), + // ( + // "import Foo, { Baz as Bar } from '../../my/relative-module';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "importNames": ["Foo"], + // "group": ["**/my/relative-module"], + // "importNamePattern": "^Bar" + // }] + // }])), + // ), + // ( + // "export { Bar } from 'foo';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "group": ["foo"], + // "importNamePattern": "^Foo" + // }] + // }])), + // ), + // ( + // "export { Bar as Foo } from 'foo';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "group": ["foo"], + // "importNamePattern": "^Foo" + // }] + // }])), + // ), ( r#"import { AllowedObject } from "foo";"#, Some(serde_json::json!([{ @@ -731,49 +780,49 @@ fn test() { }] }])), ), - ( - "import { Foo } from 'foo';", - Some(serde_json::json!([{ - "patterns": [{ - "group": ["foo"], - "allowImportNamePattern": "^Foo" - }] - }])), - ), - ( - r#"import withPatterns from "foo/bar";"#, - Some( - serde_json::json!([{ "patterns": [{ "regex": "foo/(?!bar)", "message": "foo is forbidden, use bar instead" }] }]), - ), - ), - ( - "import withPatternsCaseSensitive from 'foo';", - Some(serde_json::json!([{ - "patterns": [{ - "regex": "FOO", - "message": "foo is forbidden, use bar instead", - "caseSensitive": true - }] - }])), - ), - ( - "import Foo from '../../my/relative-module';", - Some(serde_json::json!([{ - "patterns": [{ - "regex": "my/relative-module", - "importNamePattern": "^Foo" - }] - }])), - ), - ( - "import { Bar } from '../../my/relative-module';", - Some(serde_json::json!([{ - "patterns": [{ - "regex": "my/relative-module", - "importNamePattern": "^Foo" - }] - }])), - ), + // ( + // "import { Foo } from 'foo';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "group": ["foo"], + // "allowImportNamePattern": "^Foo" + // }] + // }])), + // ), + // ( + // r#"import withPatterns from "foo/bar";"#, + // Some( + // serde_json::json!([{ "patterns": [{ "regex": "foo/(?!bar)", "message": "foo is forbidden, use bar instead" }] }]), + // ), + // ), + // ( + // "import withPatternsCaseSensitive from 'foo';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "regex": "FOO", + // "message": "foo is forbidden, use bar instead", + // "caseSensitive": true + // }] + // }])), + // ), + // ( + // "import Foo from '../../my/relative-module';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "regex": "my/relative-module", + // "importNamePattern": "^Foo" + // }] + // }])), + // ), + // ( + // "import { Bar } from '../../my/relative-module';", + // Some(serde_json::json!([{ + // "patterns": [{ + // "regex": "my/relative-module", + // "importNamePattern": "^Foo" + // }] + // }])), + // ), ]; let fail = vec![ @@ -784,14 +833,14 @@ fn test() { r#"import withPaths from "foo/bar";"#, Some(serde_json::json!([{ "paths": ["foo/bar"] }])), ), - // ( - // r#"import withPatterns from "foo/bar";"#, - // Some(serde_json::json!([{ "patterns": ["foo"] }])), - // ), - // ( - // r#"import withPatterns from "foo/bar";"#, - // Some(serde_json::json!([{ "patterns": ["bar"] }])), - // ), + ( + r#"import withPatterns from "foo/bar";"#, + Some(serde_json::json!([{ "patterns": ["foo"] }])), + ), + ( + r#"import withPatterns from "foo/bar";"#, + Some(serde_json::json!([{ "patterns": ["bar"] }])), + ), // ( // r#"import withPatterns from "foo/baz";"#, // Some( From f829c6344bc1504fc790b87e083d8192c8d53543 Mon Sep 17 00:00:00 2001 From: Sysix Date: Fri, 20 Dec 2024 22:12:54 +0100 Subject: [PATCH 06/10] fix(linter): rule: no-restricted-imports support option patterns with group key --- .../src/rules/eslint/no_restricted_imports.rs | 174 +++++++++--------- .../eslint_no_restricted_imports.snap | 98 ++++++++++ 2 files changed, 185 insertions(+), 87 deletions(-) diff --git a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs index 2ee679acf8fbb..aa34748e4aefc 100644 --- a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs +++ b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs @@ -833,38 +833,38 @@ fn test() { r#"import withPaths from "foo/bar";"#, Some(serde_json::json!([{ "paths": ["foo/bar"] }])), ), - ( - r#"import withPatterns from "foo/bar";"#, - Some(serde_json::json!([{ "patterns": ["foo"] }])), - ), - ( - r#"import withPatterns from "foo/bar";"#, - Some(serde_json::json!([{ "patterns": ["bar"] }])), - ), - // ( - // r#"import withPatterns from "foo/baz";"#, - // Some( - // serde_json::json!([{ "patterns": [{ "group": ["foo/*", "!foo/bar"], "message": "foo is forbidden, use foo/bar instead" }] }]), - // ), - // ), // ( - // r#"import withPatterns from "foo/baz";"#, - // Some( - // serde_json::json!([{ "patterns": [{ "group": ["foo/bar", "foo/baz"], "message": "some foo subimports are restricted" }] }]), - // ), + // r#"import withPatterns from "foo/bar";"#, + // Some(serde_json::json!([{ "patterns": ["foo"] }])), // ), // ( // r#"import withPatterns from "foo/bar";"#, - // Some(serde_json::json!([{ "patterns": [{ "group": ["foo/bar"] }] }])), + // Some(serde_json::json!([{ "patterns": ["bar"] }])), // ), + ( + r#"import withPatterns from "foo/baz";"#, + Some( + serde_json::json!([{ "patterns": [{ "group": ["foo/*", "!foo/bar"], "message": "foo is forbidden, use foo/bar instead" }] }]), + ), + ), + ( + r#"import withPatterns from "foo/baz";"#, + Some( + serde_json::json!([{ "patterns": [{ "group": ["foo/bar", "foo/baz"], "message": "some foo subimports are restricted" }] }]), + ), + ), + ( + r#"import withPatterns from "foo/bar";"#, + Some(serde_json::json!([{ "patterns": [{ "group": ["foo/bar"] }] }])), + ), // ( // "import withPatternsCaseInsensitive from 'foo';", // Some(serde_json::json!([{ "patterns": [{ "group": ["FOO"] }] }])), // ), - // ( - // r#"import withGitignores from "foo/bar";"#, - // Some(serde_json::json!([{ "patterns": ["foo/*", "!foo/baz"] }])), - // ), + ( + r#"import withGitignores from "foo/bar";"#, + Some(serde_json::json!([{ "patterns": ["foo/*", "!foo/baz"] }])), + ), (r#"export * from "fs";"#, Some(serde_json::json!(["fs"]))), (r#"export * as ns from "fs";"#, Some(serde_json::json!(["fs"]))), (r#"export {a} from "fs";"#, Some(serde_json::json!(["fs"]))), @@ -1336,76 +1336,76 @@ fn test() { "import relativeWithPaths from '../foo';", Some(serde_json::json!([{ "paths": ["../foo"] }])), ), - // ( - // "import relativeWithPatterns from '../foo';", - // Some(serde_json::json!([{ "patterns": ["../foo"] }])), - // ), + ( + "import relativeWithPatterns from '../foo';", + Some(serde_json::json!([{ "patterns": ["../foo"] }])), + ), ("import absolute from '/foo';", Some(serde_json::json!(["/foo"]))), ("import absoluteWithPaths from '/foo';", Some(serde_json::json!([{ "paths": ["/foo"] }]))), - // ( - // "import absoluteWithPatterns from '/foo';", - // Some(serde_json::json!([{ "patterns": ["foo"] }])), - // ), + ( + "import absoluteWithPatterns from '/foo';", + Some(serde_json::json!([{ "patterns": ["foo"] }])), + ), // ( // "import absoluteWithPatterns from '#foo/bar';", // Some(serde_json::json!([{ "patterns": ["\\#foo"] }])), // ), - // ( - // "import { Foo } from '../../my/relative-module';", - // Some(serde_json::json!([{ - // "patterns": [{ - // "group": ["**/my/relative-module"], - // "importNames": ["Foo"] - // }] - // }])), - // ), - // ( - // "import { Foo, Bar } from '../../my/relative-module';", - // Some(serde_json::json!([{ - // "patterns": [{ - // "group": ["**/my/relative-module"], - // "importNames": ["Foo", "Bar"], - // "message": "Import from @/utils instead." - // }] - // }])), - // ), - // ( - // "import * as All from '../../my/relative-module';", - // Some(serde_json::json!([{ - // "patterns": [{ - // "group": ["**/my/relative-module"], - // "importNames": ["Foo"] - // }] - // }])), - // ), - // ( - // "import * as AllWithCustomMessage from '../../my/relative-module';", - // Some(serde_json::json!([{ - // "patterns": [{ - // "group": ["**/my/relative-module"], - // "importNames": ["Foo"], - // "message": "Import from @/utils instead." - // }] - // }])), - // ), - // ( - // "import def, * as ns from 'mod';", - // Some(serde_json::json!([{ - // "patterns": [{ - // "group": ["mod"], - // "importNames": ["default"] - // }] - // }])), - // ), - // ( - // "import Foo from 'mod';", - // Some(serde_json::json!([{ - // "patterns": [{ - // "group": ["mod"], - // "importNames": ["default"] - // }] - // }])), - // ), + ( + "import { Foo } from '../../my/relative-module';", + Some(serde_json::json!([{ + "patterns": [{ + "group": ["**/my/relative-module"], + "importNames": ["Foo"] + }] + }])), + ), + ( + "import { Foo, Bar } from '../../my/relative-module';", + Some(serde_json::json!([{ + "patterns": [{ + "group": ["**/my/relative-module"], + "importNames": ["Foo", "Bar"], + "message": "Import from @/utils instead." + }] + }])), + ), + ( + "import * as All from '../../my/relative-module';", + Some(serde_json::json!([{ + "patterns": [{ + "group": ["**/my/relative-module"], + "importNames": ["Foo"] + }] + }])), + ), + ( + "import * as AllWithCustomMessage from '../../my/relative-module';", + Some(serde_json::json!([{ + "patterns": [{ + "group": ["**/my/relative-module"], + "importNames": ["Foo"], + "message": "Import from @/utils instead." + }] + }])), + ), + ( + "import def, * as ns from 'mod';", + Some(serde_json::json!([{ + "patterns": [{ + "group": ["mod"], + "importNames": ["default"] + }] + }])), + ), + ( + "import Foo from 'mod';", + Some(serde_json::json!([{ + "patterns": [{ + "group": ["mod"], + "importNames": ["default"] + }] + }])), + ), // ( // "import { Foo } from 'foo';", // Some(serde_json::json!([{ diff --git a/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap b/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap index ce0cf44f21c72..50cd1c37e49b0 100644 --- a/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap +++ b/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap @@ -30,6 +30,34 @@ snapshot_kind: text ╰──── help: Remove the import statement. + ⚠ eslint(no-restricted-imports): foo is forbidden, use foo/bar instead + ╭─[no_restricted_imports.tsx:1:26] + 1 │ import withPatterns from "foo/baz"; + · ───────── + ╰──── + help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): some foo subimports are restricted + ╭─[no_restricted_imports.tsx:1:26] + 1 │ import withPatterns from "foo/baz"; + · ───────── + ╰──── + help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): 'foo/bar' import is restricted from being used. + ╭─[no_restricted_imports.tsx:1:26] + 1 │ import withPatterns from "foo/bar"; + · ───────── + ╰──── + help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): 'foo/bar' import is restricted from being used. + ╭─[no_restricted_imports.tsx:1:28] + 1 │ import withGitignores from "foo/bar"; + · ───────── + ╰──── + help: Remove the import statement. + ⚠ eslint(no-restricted-imports): 'fs' import is restricted from being used. ╭─[no_restricted_imports.tsx:1:1] 1 │ export * from "fs"; @@ -471,6 +499,13 @@ snapshot_kind: text ╰──── help: Remove the import statement. + ⚠ eslint(no-restricted-imports): '../foo' import is restricted from being used. + ╭─[no_restricted_imports.tsx:1:34] + 1 │ import relativeWithPatterns from '../foo'; + · ──────── + ╰──── + help: Remove the import statement. + ⚠ eslint(no-restricted-imports): '/foo' import is restricted from being used. ╭─[no_restricted_imports.tsx:1:22] 1 │ import absolute from '/foo'; @@ -485,6 +520,69 @@ snapshot_kind: text ╰──── help: Remove the import statement. + ⚠ eslint(no-restricted-imports): '/foo' import is restricted from being used. + ╭─[no_restricted_imports.tsx:1:34] + 1 │ import absoluteWithPatterns from '/foo'; + · ────── + ╰──── + help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): '../../my/relative-module' import is restricted from being used. + ╭─[no_restricted_imports.tsx:1:21] + 1 │ import { Foo } from '../../my/relative-module'; + · ────────────────────────── + ╰──── + help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): Import from @/utils instead. + ╭─[no_restricted_imports.tsx:1:26] + 1 │ import { Foo, Bar } from '../../my/relative-module'; + · ────────────────────────── + ╰──── + help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): Import from @/utils instead. + ╭─[no_restricted_imports.tsx:1:26] + 1 │ import { Foo, Bar } from '../../my/relative-module'; + · ────────────────────────── + ╰──── + help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): '../../my/relative-module' import is restricted from being used. + ╭─[no_restricted_imports.tsx:1:22] + 1 │ import * as All from '../../my/relative-module'; + · ────────────────────────── + ╰──── + help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): Import from @/utils instead. + ╭─[no_restricted_imports.tsx:1:39] + 1 │ import * as AllWithCustomMessage from '../../my/relative-module'; + · ────────────────────────── + ╰──── + help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): 'mod' import is restricted from being used. + ╭─[no_restricted_imports.tsx:1:26] + 1 │ import def, * as ns from 'mod'; + · ───── + ╰──── + help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): 'mod' import is restricted from being used. + ╭─[no_restricted_imports.tsx:1:26] + 1 │ import def, * as ns from 'mod'; + · ───── + ╰──── + help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): 'mod' import is restricted from being used. + ╭─[no_restricted_imports.tsx:1:17] + 1 │ import Foo from 'mod'; + · ───── + ╰──── + help: Remove the import statement. + ⚠ eslint(no-restricted-imports): 'foo' import is restricted from being used. ╭─[no_restricted_imports.tsx:1:49] 1 │ import { AllowedObject, DisallowedObject } from "foo"; From 9aa76d39a8f321f2aa6ec74da30d7a0c8a920f9d Mon Sep 17 00:00:00 2001 From: Sysix Date: Fri, 20 Dec 2024 22:21:07 +0100 Subject: [PATCH 07/10] fix(linter): rule: no-restricted-imports support option patterns with group key --- .../src/rules/eslint/no_restricted_imports.rs | 144 +++++++++--------- .../eslint_no_restricted_imports.snap | 28 ++++ 2 files changed, 100 insertions(+), 72 deletions(-) diff --git a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs index aa34748e4aefc..14e830c2382d6 100644 --- a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs +++ b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs @@ -789,40 +789,40 @@ fn test() { // }] // }])), // ), - // ( - // r#"import withPatterns from "foo/bar";"#, - // Some( - // serde_json::json!([{ "patterns": [{ "regex": "foo/(?!bar)", "message": "foo is forbidden, use bar instead" }] }]), - // ), - // ), - // ( - // "import withPatternsCaseSensitive from 'foo';", - // Some(serde_json::json!([{ - // "patterns": [{ - // "regex": "FOO", - // "message": "foo is forbidden, use bar instead", - // "caseSensitive": true - // }] - // }])), - // ), - // ( - // "import Foo from '../../my/relative-module';", - // Some(serde_json::json!([{ - // "patterns": [{ - // "regex": "my/relative-module", - // "importNamePattern": "^Foo" - // }] - // }])), - // ), - // ( - // "import { Bar } from '../../my/relative-module';", - // Some(serde_json::json!([{ - // "patterns": [{ - // "regex": "my/relative-module", - // "importNamePattern": "^Foo" - // }] - // }])), - // ), + ( + r#"import withPatterns from "foo/bar";"#, + Some( + serde_json::json!([{ "patterns": [{ "regex": "foo/(?!bar)", "message": "foo is forbidden, use bar instead" }] }]), + ), + ), + ( + "import withPatternsCaseSensitive from 'foo';", + Some(serde_json::json!([{ + "patterns": [{ + "regex": "FOO", + "message": "foo is forbidden, use bar instead", + "caseSensitive": true + }] + }])), + ), + ( + "import Foo from '../../my/relative-module';", + Some(serde_json::json!([{ + "patterns": [{ + "regex": "my/relative-module", + "importNamePattern": "^Foo" + }] + }])), + ), + ( + "import { Bar } from '../../my/relative-module';", + Some(serde_json::json!([{ + "patterns": [{ + "regex": "my/relative-module", + "importNamePattern": "^Foo" + }] + }])), + ), ]; let fail = vec![ @@ -1633,25 +1633,25 @@ fn test() { }] }])), ), - // ( - // r#"import { AllowedObject, DisallowedObject } from "foo";"#, - // Some(serde_json::json!([{ - // "patterns": [{ - // "group": ["foo"], - // "allowImportNames": ["AllowedObject"] - // }] - // }])), - // ), - // ( - // r#"import { AllowedObject, DisallowedObject } from "foo";"#, - // Some(serde_json::json!([{ - // "patterns": [{ - // "group": ["foo"], - // "allowImportNames": ["AllowedObject"], - // "message": r#"Only "AllowedObject" is allowed to be imported from "foo"."# - // }] - // }])), - // ), + ( + r#"import { AllowedObject, DisallowedObject } from "foo";"#, + Some(serde_json::json!([{ + "patterns": [{ + "group": ["foo"], + "allowImportNames": ["AllowedObject"] + }] + }])), + ), + ( + r#"import { AllowedObject, DisallowedObject } from "foo";"#, + Some(serde_json::json!([{ + "patterns": [{ + "group": ["foo"], + "allowImportNames": ["AllowedObject"], + "message": r#"Only "AllowedObject" is allowed to be imported from "foo"."# + }] + }])), + ), ( r#"import * as AllowedObject from "foo";"#, Some(serde_json::json!([{ @@ -1671,25 +1671,25 @@ fn test() { }] }])), ), - // ( - // r#"import * as AllowedObject from "foo/bar";"#, - // Some(serde_json::json!([{ - // "patterns": [{ - // "group": ["foo/*"], - // "allowImportNames": ["AllowedObject"] - // }] - // }])), - // ), - // ( - // r#"import * as AllowedObject from "foo/bar";"#, - // Some(serde_json::json!([{ - // "patterns": [{ - // "group": ["foo/*"], - // "allowImportNames": ["AllowedObject"], - // "message": r#"Only "AllowedObject" is allowed to be imported from "foo"."# - // }] - // }])), - // ), + ( + r#"import * as AllowedObject from "foo/bar";"#, + Some(serde_json::json!([{ + "patterns": [{ + "group": ["foo/*"], + "allowImportNames": ["AllowedObject"] + }] + }])), + ), + ( + r#"import * as AllowedObject from "foo/bar";"#, + Some(serde_json::json!([{ + "patterns": [{ + "group": ["foo/*"], + "allowImportNames": ["AllowedObject"], + "message": r#"Only "AllowedObject" is allowed to be imported from "foo"."# + }] + }])), + ), // ( // r#"import * as AllowedObject from "foo/bar";"#, // Some(serde_json::json!([{ diff --git a/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap b/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap index 50cd1c37e49b0..972b0993fbdef 100644 --- a/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap +++ b/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap @@ -597,6 +597,20 @@ snapshot_kind: text ╰──── help: Remove the import statement. + ⚠ eslint(no-restricted-imports): 'foo' import is restricted from being used. + ╭─[no_restricted_imports.tsx:1:49] + 1 │ import { AllowedObject, DisallowedObject } from "foo"; + · ───── + ╰──── + help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): Only "AllowedObject" is allowed to be imported from "foo". + ╭─[no_restricted_imports.tsx:1:49] + 1 │ import { AllowedObject, DisallowedObject } from "foo"; + · ───── + ╰──── + help: Remove the import statement. + ⚠ eslint(no-restricted-imports): 'foo' import is restricted from being used. ╭─[no_restricted_imports.tsx:1:32] 1 │ import * as AllowedObject from "foo"; @@ -610,3 +624,17 @@ snapshot_kind: text · ───── ╰──── help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): 'foo/bar' import is restricted from being used. + ╭─[no_restricted_imports.tsx:1:32] + 1 │ import * as AllowedObject from "foo/bar"; + · ───────── + ╰──── + help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): Only "AllowedObject" is allowed to be imported from "foo". + ╭─[no_restricted_imports.tsx:1:32] + 1 │ import * as AllowedObject from "foo/bar"; + · ───────── + ╰──── + help: Remove the import statement. From 5f8d01092579c9190b68f86f17844b64a3928842 Mon Sep 17 00:00:00 2001 From: Sysix Date: Fri, 20 Dec 2024 22:47:34 +0100 Subject: [PATCH 08/10] fix(linter): rule: no-restricted-imports support option patterns with group key --- .../src/rules/eslint/no_restricted_imports.rs | 31 ++++++++++--------- .../eslint_no_restricted_imports.snap | 14 +++++++++ 2 files changed, 31 insertions(+), 14 deletions(-) diff --git a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs index 14e830c2382d6..19d8e0990917d 100644 --- a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs +++ b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs @@ -59,6 +59,7 @@ struct RestrictedPattern { group: Vec, import_names: Option>, allow_import_names: Option>, + case_sensitive: Option, message: Option, } @@ -153,6 +154,7 @@ fn add_configuration_patterns_from_string(paths: &mut Vec, mo group: vec![CompactStr::new(module_name)], import_names: None, allow_import_names: None, + case_sensitive: None, message: None, }); } @@ -341,6 +343,7 @@ impl NoRestrictedImports { } let mut builder = GitignoreBuilder::new("/"); + let _ = builder.case_insensitive(!pattern.case_sensitive.unwrap_or(false)); for group in &pattern.group { let _ = builder.add_line(None, group.as_str()); @@ -857,10 +860,10 @@ fn test() { r#"import withPatterns from "foo/bar";"#, Some(serde_json::json!([{ "patterns": [{ "group": ["foo/bar"] }] }])), ), - // ( - // "import withPatternsCaseInsensitive from 'foo';", - // Some(serde_json::json!([{ "patterns": [{ "group": ["FOO"] }] }])), - // ), + ( + "import withPatternsCaseInsensitive from 'foo';", + Some(serde_json::json!([{ "patterns": [{ "group": ["FOO"] }] }])), + ), ( r#"import withGitignores from "foo/bar";"#, Some(serde_json::json!([{ "patterns": ["foo/*", "!foo/baz"] }])), @@ -1734,16 +1737,16 @@ fn test() { // }] // }])), // ), - // ( - // "import withPatternsCaseSensitive from 'foo';", - // Some(serde_json::json!([{ - // "patterns": [{ - // "group": ["FOO"], - // "message": "foo is forbidden, use bar instead", - // "caseSensitive": false - // }] - // }])), - // ), + ( + "import withPatternsCaseSensitive from 'foo';", + Some(serde_json::json!([{ + "patterns": [{ + "group": ["FOO"], + "message": "foo is forbidden, use bar instead", + "caseSensitive": false + }] + }])), + ), // ( // " // // error diff --git a/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap b/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap index 972b0993fbdef..9d213f568e089 100644 --- a/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap +++ b/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap @@ -51,6 +51,13 @@ snapshot_kind: text ╰──── help: Remove the import statement. + ⚠ eslint(no-restricted-imports): 'foo' import is restricted from being used. + ╭─[no_restricted_imports.tsx:1:41] + 1 │ import withPatternsCaseInsensitive from 'foo'; + · ───── + ╰──── + help: Remove the import statement. + ⚠ eslint(no-restricted-imports): 'foo/bar' import is restricted from being used. ╭─[no_restricted_imports.tsx:1:28] 1 │ import withGitignores from "foo/bar"; @@ -638,3 +645,10 @@ snapshot_kind: text · ───────── ╰──── help: Remove the import statement. + + ⚠ eslint(no-restricted-imports): foo is forbidden, use bar instead + ╭─[no_restricted_imports.tsx:1:39] + 1 │ import withPatternsCaseSensitive from 'foo'; + · ───── + ╰──── + help: Remove the import statement. From 5a0036c1fbd223064d7cf3fba2b39ad93c5cc7b3 Mon Sep 17 00:00:00 2001 From: Sysix Date: Fri, 20 Dec 2024 23:15:08 +0100 Subject: [PATCH 09/10] fix(linter): rule: no-restricted-imports support option patterns with group key --- .../src/rules/eslint/no_restricted_imports.rs | 12 ++++++------ .../src/snapshots/eslint_no_restricted_imports.snap | 7 +++++++ 2 files changed, 13 insertions(+), 6 deletions(-) diff --git a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs index 19d8e0990917d..051241e8efa07 100644 --- a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs +++ b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs @@ -342,7 +342,7 @@ impl NoRestrictedImports { continue; } - let mut builder = GitignoreBuilder::new("/"); + let mut builder = GitignoreBuilder::new(""); let _ = builder.case_insensitive(!pattern.case_sensitive.unwrap_or(false)); for group in &pattern.group { @@ -356,7 +356,7 @@ impl NoRestrictedImports { let source = entry.module_request.name(); let span = entry.module_request.span(); - let matched = gitignore.matched(source, true); + let matched = gitignore.matched(source, false); // println!("{:?}", matched); if matched.is_whitelist() { @@ -840,10 +840,10 @@ fn test() { // r#"import withPatterns from "foo/bar";"#, // Some(serde_json::json!([{ "patterns": ["foo"] }])), // ), - // ( - // r#"import withPatterns from "foo/bar";"#, - // Some(serde_json::json!([{ "patterns": ["bar"] }])), - // ), + ( + r#"import withPatterns from "foo/bar";"#, + Some(serde_json::json!([{ "patterns": ["bar"] }])), + ), ( r#"import withPatterns from "foo/baz";"#, Some( diff --git a/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap b/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap index 9d213f568e089..fc8dde990dd5e 100644 --- a/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap +++ b/crates/oxc_linter/src/snapshots/eslint_no_restricted_imports.snap @@ -30,6 +30,13 @@ snapshot_kind: text ╰──── help: Remove the import statement. + ⚠ eslint(no-restricted-imports): 'foo/bar' import is restricted from being used. + ╭─[no_restricted_imports.tsx:1:26] + 1 │ import withPatterns from "foo/bar"; + · ───────── + ╰──── + help: Remove the import statement. + ⚠ eslint(no-restricted-imports): foo is forbidden, use foo/bar instead ╭─[no_restricted_imports.tsx:1:26] 1 │ import withPatterns from "foo/baz"; From a85cc9658dc914be828bd16b467dfac6ebae87e5 Mon Sep 17 00:00:00 2001 From: Sysix Date: Sun, 22 Dec 2024 17:40:28 +0100 Subject: [PATCH 10/10] fix(linter): rule: no-restricted-imports support option patterns with group key --- .../src/rules/eslint/no_restricted_imports.rs | 259 +++++++++++------- 1 file changed, 162 insertions(+), 97 deletions(-) diff --git a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs index 051241e8efa07..6b6976c9409d2 100644 --- a/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs +++ b/crates/oxc_linter/src/rules/eslint/no_restricted_imports.rs @@ -8,7 +8,7 @@ use serde_json::Value; use crate::{ context::LintContext, - module_record::{ExportEntry, ExportImportName, ImportEntry, ImportImportName}, + module_record::{ExportEntry, ExportImportName, ImportEntry, ImportImportName, NameSpan}, rule::Rule, ModuleRecord, }; @@ -63,6 +63,13 @@ struct RestrictedPattern { message: Option, } +#[derive(Debug)] +enum GlobResult { + Found, + Whitelist, + None, +} + declare_oxc_lint!( /// ### What it does /// This rule allows you to specify imports that you don’t want to use in your application. @@ -159,6 +166,115 @@ fn add_configuration_patterns_from_string(paths: &mut Vec, mo }); } +fn is_name_span_allowed_in_path(name: &CompactStr, path: &RestrictedPath) -> bool { + // fast check if this name is allowed + if path.allow_import_names.as_ref().is_some_and(|allowed| allowed.contains(name)) { + return true; + } + + // when no importNames option is provided, no import in general is allowed + if path.import_names.as_ref().is_none() { + return false; + } + + // the name is found is the importNames list + if path.import_names.as_ref().is_some_and(|disallowed| disallowed.contains(name)) { + return false; + } + + // we allow it + true +} + +fn is_name_span_allowed_in_pattern(name: &CompactStr, pattern: &RestrictedPattern) -> bool { + // fast check if this name is allowed + if pattern.allow_import_names.as_ref().is_some_and(|allowed| allowed.contains(name)) { + return true; + } + + // when no importNames option is provided, no import in general is allowed + if pattern.import_names.as_ref().is_none() { + return false; + } + + // the name is found is the importNames list + if pattern.import_names.as_ref().is_some_and(|disallowed| disallowed.contains(name)) { + return false; + } + + // we allow it + true +} + +impl RestrictedPath { + fn is_skip_able_import(&self, name: &ImportImportName) -> bool { + match &name { + ImportImportName::Name(import) => is_name_span_allowed_in_path(&import.name, self), + ImportImportName::Default(_) => { + is_name_span_allowed_in_path(&CompactStr::new("default"), self) + } + ImportImportName::NamespaceObject => false, + } + } + + fn is_skip_able_export(&self, name: &ExportImportName) -> bool { + match &name { + ExportImportName::Name(import) => is_name_span_allowed_in_path(&import.name, self), + ExportImportName::All | ExportImportName::AllButDefault => false, + ExportImportName::Null => true, + } + } +} + +impl RestrictedPattern { + fn is_skip_able_import(&self, name: &ImportImportName) -> bool { + match &name { + ImportImportName::Name(import) => is_name_span_allowed_in_pattern(&import.name, self), + ImportImportName::Default(_) => { + is_name_span_allowed_in_pattern(&CompactStr::new("default"), self) + } + ImportImportName::NamespaceObject => false, + } + } + + fn is_skip_able_export(&self, name: &ExportImportName) -> bool { + match &name { + ExportImportName::Name(import) => is_name_span_allowed_in_pattern(&import.name, self), + ExportImportName::All | ExportImportName::AllButDefault => false, + ExportImportName::Null => true, + } + } + + fn get_gitignore_glob_result(&self, name: &NameSpan) -> GlobResult { + let mut builder = GitignoreBuilder::new(""); + // returns always OK, will be fixed in the next version + let _ = builder.case_insensitive(!self.case_sensitive.unwrap_or(false)); + + for group in &self.group { + // returns always OK + let _ = builder.add_line(None, group.as_str()); + } + + let Ok(gitignore) = builder.build() else { + return GlobResult::None; + }; + + let source = name.name(); + + let matched = gitignore.matched(source, false); + + if matched.is_whitelist() { + return GlobResult::Whitelist; + } + + if matched.is_none() { + return GlobResult::None; + } + + GlobResult::Found + } +} + impl Rule for NoRestrictedImports { fn from_configuration(value: serde_json::Value) -> Self { let mut paths: Vec = Vec::new(); @@ -233,46 +349,6 @@ impl Rule for NoRestrictedImports { } impl NoRestrictedImports { - fn is_name_span_allowed_in_path(name: &CompactStr, path: &RestrictedPath) -> bool { - // fast check if this name is allowed - if path.allow_import_names.as_ref().is_some_and(|allowed| allowed.contains(name)) { - return true; - } - - // when no importNames option is provided, no import in general is allowed - if path.import_names.as_ref().is_none() { - return false; - } - - // the name is found is the importNames list - if path.import_names.as_ref().is_some_and(|disallowed| disallowed.contains(name)) { - return false; - } - - // we allow it - true - } - - fn is_name_span_allowed_in_pattern(name: &CompactStr, path: &RestrictedPattern) -> bool { - // fast check if this name is allowed - if path.allow_import_names.as_ref().is_some_and(|allowed| allowed.contains(name)) { - return true; - } - - // when no importNames option is provided, no import in general is allowed - if path.import_names.as_ref().is_none() { - return false; - } - - // the name is found is the importNames list - if path.import_names.as_ref().is_some_and(|disallowed| disallowed.contains(name)) { - return false; - } - - // we allow it - true - } - fn report_side_effects(&self, ctx: &LintContext<'_>, module_record: &ModuleRecord) { let mut side_effect_import_map: FxHashMap<&CompactStr, Vec> = FxHashMap::default(); @@ -303,17 +379,7 @@ impl NoRestrictedImports { continue; } - let skipable = match &entry.import_name { - ImportImportName::Name(import) => { - Self::is_name_span_allowed_in_path(&import.name, path) - } - ImportImportName::Default(_) => { - Self::is_name_span_allowed_in_path(&CompactStr::new("default"), path) - } - ImportImportName::NamespaceObject => false, - }; - - if skipable { + if path.is_skip_able_import(&entry.import_name) { continue; } @@ -322,51 +388,26 @@ impl NoRestrictedImports { no_restricted_imports_diagnostic(ctx, span, path.message.clone(), source); } - // println!("{:?}", self.patterns); - let mut whitelist_found = false; let mut found_errors = vec![]; for pattern in &self.patterns { - let skipable = match &entry.import_name { - ImportImportName::Name(import) => { - Self::is_name_span_allowed_in_pattern(&import.name, pattern) - } - ImportImportName::Default(_) => { - Self::is_name_span_allowed_in_pattern(&CompactStr::new("default"), pattern) - } - ImportImportName::NamespaceObject => false, - }; - - if skipable { + if pattern.is_skip_able_import(&entry.import_name) { continue; } - let mut builder = GitignoreBuilder::new(""); - let _ = builder.case_insensitive(!pattern.case_sensitive.unwrap_or(false)); - - for group in &pattern.group { - let _ = builder.add_line(None, group.as_str()); - } + match pattern.get_gitignore_glob_result(&entry.module_request) { + GlobResult::Whitelist => { + whitelist_found = true; + break; + } + GlobResult::Found => { + let span = entry.module_request.span(); - let Ok(gitignore) = builder.build() else { - continue; + found_errors.push((span, pattern)); + } + GlobResult::None => (), }; - - let source = entry.module_request.name(); - let span = entry.module_request.span(); - - let matched = gitignore.matched(source, false); - // println!("{:?}", matched); - - if matched.is_whitelist() { - whitelist_found = true; - break; - } - - if !matched.is_none() { - found_errors.push((span, pattern)); - } } if !whitelist_found && !found_errors.is_empty() { @@ -387,15 +428,7 @@ impl NoRestrictedImports { continue; } - let skipable = match &entry.import_name { - ExportImportName::Name(import) => { - Self::is_name_span_allowed_in_path(&import.name, path) - } - ExportImportName::All | ExportImportName::AllButDefault => false, - ExportImportName::Null => true, - }; - - if skipable { + if path.is_skip_able_export(&entry.import_name) { continue; } @@ -403,6 +436,38 @@ impl NoRestrictedImports { no_restricted_imports_diagnostic(ctx, span, path.message.clone(), source); } + + let mut whitelist_found = false; + let mut found_errors = vec![]; + + for pattern in &self.patterns { + if pattern.is_skip_able_export(&entry.import_name) { + continue; + } + + let Some(module_request) = &entry.module_request else { + continue; + }; + + match pattern.get_gitignore_glob_result(module_request) { + GlobResult::Whitelist => { + whitelist_found = true; + break; + } + GlobResult::Found => { + let span = module_request.span(); + + found_errors.push((span, pattern)); + } + GlobResult::None => (), + }; + } + + if !whitelist_found && !found_errors.is_empty() { + for (span, pattern) in found_errors { + no_restricted_imports_diagnostic(ctx, span, pattern.message.clone(), source); + } + } } }