mirror of
https://github.com/denoland/deno.git
synced 2025-01-21 13:00:36 -05:00
refactor(lint): renames and code flattening (#27386)
Working on loading plugin configuration for https://github.com/denoland/deno/pull/27203 I encountered a lot of complexity, so did some drive-by cleanups to make it easier to grok the code and have fewer duplicate names.
This commit is contained in:
parent
59f263409e
commit
f9add94e17
3 changed files with 263 additions and 230 deletions
|
@ -24,6 +24,7 @@ use deno_config::workspace::WorkspaceLintConfig;
|
||||||
use deno_config::workspace::WorkspaceResolver;
|
use deno_config::workspace::WorkspaceResolver;
|
||||||
use deno_core::resolve_url_or_path;
|
use deno_core::resolve_url_or_path;
|
||||||
use deno_graph::GraphKind;
|
use deno_graph::GraphKind;
|
||||||
|
use deno_lint::linter::LintConfig as DenoLintConfig;
|
||||||
use deno_npm::npm_rc::NpmRc;
|
use deno_npm::npm_rc::NpmRc;
|
||||||
use deno_npm::npm_rc::ResolvedNpmRc;
|
use deno_npm::npm_rc::ResolvedNpmRc;
|
||||||
use deno_npm::resolution::ValidSerializedNpmResolutionSnapshot;
|
use deno_npm::resolution::ValidSerializedNpmResolutionSnapshot;
|
||||||
|
@ -1351,9 +1352,7 @@ impl CliOptions {
|
||||||
Ok(result)
|
Ok(result)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn resolve_deno_lint_config(
|
pub fn resolve_deno_lint_config(&self) -> Result<DenoLintConfig, AnyError> {
|
||||||
&self,
|
|
||||||
) -> Result<deno_lint::linter::LintConfig, AnyError> {
|
|
||||||
let ts_config_result =
|
let ts_config_result =
|
||||||
self.resolve_ts_config_for_emit(TsConfigType::Emit)?;
|
self.resolve_ts_config_for_emit(TsConfigType::Emit)?;
|
||||||
|
|
||||||
|
@ -1362,7 +1361,7 @@ impl CliOptions {
|
||||||
ts_config_result.ts_config,
|
ts_config_result.ts_config,
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
Ok(deno_lint::linter::LintConfig {
|
Ok(DenoLintConfig {
|
||||||
default_jsx_factory: (!transpile_options.jsx_automatic)
|
default_jsx_factory: (!transpile_options.jsx_automatic)
|
||||||
.then(|| transpile_options.jsx_factory.clone()),
|
.then(|| transpile_options.jsx_factory.clone()),
|
||||||
default_jsx_fragment_factory: (!transpile_options.jsx_automatic)
|
default_jsx_fragment_factory: (!transpile_options.jsx_automatic)
|
||||||
|
|
|
@ -45,6 +45,7 @@ use deno_graph::source::ResolveError;
|
||||||
use deno_graph::Resolution;
|
use deno_graph::Resolution;
|
||||||
use deno_graph::ResolutionError;
|
use deno_graph::ResolutionError;
|
||||||
use deno_graph::SpecifierError;
|
use deno_graph::SpecifierError;
|
||||||
|
use deno_lint::linter::LintConfig as DenoLintConfig;
|
||||||
use deno_resolver::sloppy_imports::SloppyImportsResolution;
|
use deno_resolver::sloppy_imports::SloppyImportsResolution;
|
||||||
use deno_resolver::sloppy_imports::SloppyImportsResolutionKind;
|
use deno_resolver::sloppy_imports::SloppyImportsResolutionKind;
|
||||||
use deno_runtime::deno_fs;
|
use deno_runtime::deno_fs;
|
||||||
|
@ -834,7 +835,7 @@ fn generate_lint_diagnostics(
|
||||||
lint_rule_provider.resolve_lint_rules(Default::default(), None)
|
lint_rule_provider.resolve_lint_rules(Default::default(), None)
|
||||||
},
|
},
|
||||||
fix: false,
|
fix: false,
|
||||||
deno_lint_config: deno_lint::linter::LintConfig {
|
deno_lint_config: DenoLintConfig {
|
||||||
default_jsx_factory: None,
|
default_jsx_factory: None,
|
||||||
default_jsx_fragment_factory: None,
|
default_jsx_fragment_factory: None,
|
||||||
},
|
},
|
||||||
|
|
|
@ -20,7 +20,7 @@ use deno_core::unsync::future::LocalFutureExt;
|
||||||
use deno_core::unsync::future::SharedLocal;
|
use deno_core::unsync::future::SharedLocal;
|
||||||
use deno_graph::ModuleGraph;
|
use deno_graph::ModuleGraph;
|
||||||
use deno_lint::diagnostic::LintDiagnostic;
|
use deno_lint::diagnostic::LintDiagnostic;
|
||||||
use deno_lint::linter::LintConfig;
|
use deno_lint::linter::LintConfig as DenoLintConfig;
|
||||||
use log::debug;
|
use log::debug;
|
||||||
use reporters::create_reporter;
|
use reporters::create_reporter;
|
||||||
use reporters::LintReporter;
|
use reporters::LintReporter;
|
||||||
|
@ -29,7 +29,6 @@ use std::collections::HashSet;
|
||||||
use std::fs;
|
use std::fs;
|
||||||
use std::io::stdin;
|
use std::io::stdin;
|
||||||
use std::io::Read;
|
use std::io::Read;
|
||||||
use std::path::Path;
|
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
@ -47,6 +46,7 @@ use crate::graph_util::ModuleGraphCreator;
|
||||||
use crate::tools::fmt::run_parallelized;
|
use crate::tools::fmt::run_parallelized;
|
||||||
use crate::util::display;
|
use crate::util::display;
|
||||||
use crate::util::file_watcher;
|
use crate::util::file_watcher;
|
||||||
|
use crate::util::file_watcher::WatcherCommunicator;
|
||||||
use crate::util::fs::canonicalize_path;
|
use crate::util::fs::canonicalize_path;
|
||||||
use crate::util::path::is_script_ext;
|
use crate::util::path::is_script_ext;
|
||||||
use crate::util::sync::AtomicFlag;
|
use crate::util::sync::AtomicFlag;
|
||||||
|
@ -69,136 +69,139 @@ pub async fn lint(
|
||||||
flags: Arc<Flags>,
|
flags: Arc<Flags>,
|
||||||
lint_flags: LintFlags,
|
lint_flags: LintFlags,
|
||||||
) -> Result<(), AnyError> {
|
) -> Result<(), AnyError> {
|
||||||
if let Some(watch_flags) = &lint_flags.watch {
|
if lint_flags.watch.is_some() {
|
||||||
if lint_flags.is_stdin() {
|
if lint_flags.is_stdin() {
|
||||||
return Err(generic_error(
|
return Err(generic_error(
|
||||||
"Lint watch on standard input is not supported.",
|
"Lint watch on standard input is not supported.",
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
file_watcher::watch_func(
|
|
||||||
flags,
|
|
||||||
file_watcher::PrintConfig::new("Lint", !watch_flags.no_clear_screen),
|
|
||||||
move |flags, watcher_communicator, changed_paths| {
|
|
||||||
let lint_flags = lint_flags.clone();
|
|
||||||
watcher_communicator.show_path_changed(changed_paths.clone());
|
|
||||||
Ok(async move {
|
|
||||||
let factory = CliFactory::from_flags(flags);
|
|
||||||
let cli_options = factory.cli_options()?;
|
|
||||||
let lint_config = cli_options.resolve_deno_lint_config()?;
|
|
||||||
let mut paths_with_options_batches =
|
|
||||||
resolve_paths_with_options_batches(cli_options, &lint_flags)?;
|
|
||||||
for paths_with_options in &mut paths_with_options_batches {
|
|
||||||
_ = watcher_communicator
|
|
||||||
.watch_paths(paths_with_options.paths.clone());
|
|
||||||
|
|
||||||
let files = std::mem::take(&mut paths_with_options.paths);
|
return lint_with_watch(flags, lint_flags).await;
|
||||||
paths_with_options.paths = if let Some(paths) = &changed_paths {
|
}
|
||||||
// lint all files on any changed (https://github.com/denoland/deno/issues/12446)
|
|
||||||
files
|
|
||||||
.iter()
|
|
||||||
.any(|path| {
|
|
||||||
canonicalize_path(path)
|
|
||||||
.map(|p| paths.contains(&p))
|
|
||||||
.unwrap_or(false)
|
|
||||||
})
|
|
||||||
.then_some(files)
|
|
||||||
.unwrap_or_else(|| [].to_vec())
|
|
||||||
} else {
|
|
||||||
files
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut linter = WorkspaceLinter::new(
|
let factory = CliFactory::from_flags(flags);
|
||||||
factory.caches()?.clone(),
|
let cli_options = factory.cli_options()?;
|
||||||
factory.lint_rule_provider().await?,
|
let lint_rule_provider = factory.lint_rule_provider().await?;
|
||||||
factory.module_graph_creator().await?.clone(),
|
let is_stdin = lint_flags.is_stdin();
|
||||||
cli_options.start_dir.clone(),
|
let deno_lint_config = cli_options.resolve_deno_lint_config()?;
|
||||||
&cli_options.resolve_workspace_lint_options(&lint_flags)?,
|
let workspace_lint_options =
|
||||||
);
|
cli_options.resolve_workspace_lint_options(&lint_flags)?;
|
||||||
for paths_with_options in paths_with_options_batches {
|
let success = if is_stdin {
|
||||||
linter
|
lint_stdin(
|
||||||
.lint_files(
|
cli_options,
|
||||||
cli_options,
|
lint_rule_provider,
|
||||||
paths_with_options.options,
|
workspace_lint_options,
|
||||||
lint_config.clone(),
|
lint_flags,
|
||||||
paths_with_options.dir,
|
deno_lint_config,
|
||||||
paths_with_options.paths,
|
)?
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
}
|
|
||||||
|
|
||||||
linter.finish();
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
})
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
} else {
|
} else {
|
||||||
let factory = CliFactory::from_flags(flags);
|
let mut linter = WorkspaceLinter::new(
|
||||||
let cli_options = factory.cli_options()?;
|
factory.caches()?.clone(),
|
||||||
let is_stdin = lint_flags.is_stdin();
|
lint_rule_provider,
|
||||||
let deno_lint_config = cli_options.resolve_deno_lint_config()?;
|
factory.module_graph_creator().await?.clone(),
|
||||||
let workspace_lint_options =
|
cli_options.start_dir.clone(),
|
||||||
cli_options.resolve_workspace_lint_options(&lint_flags)?;
|
&workspace_lint_options,
|
||||||
let success = if is_stdin {
|
);
|
||||||
let start_dir = &cli_options.start_dir;
|
let paths_with_options_batches =
|
||||||
let reporter_lock = Arc::new(Mutex::new(create_reporter(
|
resolve_paths_with_options_batches(cli_options, &lint_flags)?;
|
||||||
workspace_lint_options.reporter_kind,
|
for paths_with_options in paths_with_options_batches {
|
||||||
)));
|
linter
|
||||||
let lint_config = start_dir
|
.lint_files(
|
||||||
.to_lint_config(FilePatterns::new_with_base(start_dir.dir_path()))?;
|
cli_options,
|
||||||
let lint_options = LintOptions::resolve(lint_config, &lint_flags);
|
paths_with_options.options,
|
||||||
let lint_rules = factory
|
deno_lint_config.clone(),
|
||||||
.lint_rule_provider()
|
paths_with_options.dir,
|
||||||
.await?
|
paths_with_options.paths,
|
||||||
.resolve_lint_rules_err_empty(
|
)
|
||||||
lint_options.rules,
|
.await?;
|
||||||
start_dir.maybe_deno_json().map(|c| c.as_ref()),
|
|
||||||
)?;
|
|
||||||
let mut file_path = cli_options.initial_cwd().join(STDIN_FILE_NAME);
|
|
||||||
if let Some(ext) = cli_options.ext_flag() {
|
|
||||||
file_path.set_extension(ext);
|
|
||||||
}
|
|
||||||
let r = lint_stdin(&file_path, lint_rules, deno_lint_config);
|
|
||||||
let success = handle_lint_result(
|
|
||||||
&file_path.to_string_lossy(),
|
|
||||||
r,
|
|
||||||
reporter_lock.clone(),
|
|
||||||
);
|
|
||||||
reporter_lock.lock().close(1);
|
|
||||||
success
|
|
||||||
} else {
|
|
||||||
let mut linter = WorkspaceLinter::new(
|
|
||||||
factory.caches()?.clone(),
|
|
||||||
factory.lint_rule_provider().await?,
|
|
||||||
factory.module_graph_creator().await?.clone(),
|
|
||||||
cli_options.start_dir.clone(),
|
|
||||||
&workspace_lint_options,
|
|
||||||
);
|
|
||||||
let paths_with_options_batches =
|
|
||||||
resolve_paths_with_options_batches(cli_options, &lint_flags)?;
|
|
||||||
for paths_with_options in paths_with_options_batches {
|
|
||||||
linter
|
|
||||||
.lint_files(
|
|
||||||
cli_options,
|
|
||||||
paths_with_options.options,
|
|
||||||
deno_lint_config.clone(),
|
|
||||||
paths_with_options.dir,
|
|
||||||
paths_with_options.paths,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
}
|
|
||||||
linter.finish()
|
|
||||||
};
|
|
||||||
if !success {
|
|
||||||
deno_runtime::exit(1);
|
|
||||||
}
|
}
|
||||||
|
linter.finish()
|
||||||
|
};
|
||||||
|
if !success {
|
||||||
|
deno_runtime::exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async fn lint_with_watch_inner(
|
||||||
|
flags: Arc<Flags>,
|
||||||
|
lint_flags: LintFlags,
|
||||||
|
watcher_communicator: Arc<WatcherCommunicator>,
|
||||||
|
changed_paths: Option<Vec<PathBuf>>,
|
||||||
|
) -> Result<(), AnyError> {
|
||||||
|
let factory = CliFactory::from_flags(flags);
|
||||||
|
let cli_options = factory.cli_options()?;
|
||||||
|
let lint_config = cli_options.resolve_deno_lint_config()?;
|
||||||
|
let mut paths_with_options_batches =
|
||||||
|
resolve_paths_with_options_batches(cli_options, &lint_flags)?;
|
||||||
|
for paths_with_options in &mut paths_with_options_batches {
|
||||||
|
_ = watcher_communicator.watch_paths(paths_with_options.paths.clone());
|
||||||
|
|
||||||
|
let files = std::mem::take(&mut paths_with_options.paths);
|
||||||
|
paths_with_options.paths = if let Some(paths) = &changed_paths {
|
||||||
|
// lint all files on any changed (https://github.com/denoland/deno/issues/12446)
|
||||||
|
files
|
||||||
|
.iter()
|
||||||
|
.any(|path| {
|
||||||
|
canonicalize_path(path)
|
||||||
|
.map(|p| paths.contains(&p))
|
||||||
|
.unwrap_or(false)
|
||||||
|
})
|
||||||
|
.then_some(files)
|
||||||
|
.unwrap_or_else(|| [].to_vec())
|
||||||
|
} else {
|
||||||
|
files
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut linter = WorkspaceLinter::new(
|
||||||
|
factory.caches()?.clone(),
|
||||||
|
factory.lint_rule_provider().await?,
|
||||||
|
factory.module_graph_creator().await?.clone(),
|
||||||
|
cli_options.start_dir.clone(),
|
||||||
|
&cli_options.resolve_workspace_lint_options(&lint_flags)?,
|
||||||
|
);
|
||||||
|
for paths_with_options in paths_with_options_batches {
|
||||||
|
linter
|
||||||
|
.lint_files(
|
||||||
|
cli_options,
|
||||||
|
paths_with_options.options,
|
||||||
|
lint_config.clone(),
|
||||||
|
paths_with_options.dir,
|
||||||
|
paths_with_options.paths,
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
}
|
||||||
|
|
||||||
|
linter.finish();
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn lint_with_watch(
|
||||||
|
flags: Arc<Flags>,
|
||||||
|
lint_flags: LintFlags,
|
||||||
|
) -> Result<(), AnyError> {
|
||||||
|
let watch_flags = lint_flags.watch.as_ref().unwrap();
|
||||||
|
|
||||||
|
file_watcher::watch_func(
|
||||||
|
flags,
|
||||||
|
file_watcher::PrintConfig::new("Lint", !watch_flags.no_clear_screen),
|
||||||
|
move |flags, watcher_communicator, changed_paths| {
|
||||||
|
let lint_flags = lint_flags.clone();
|
||||||
|
watcher_communicator.show_path_changed(changed_paths.clone());
|
||||||
|
Ok(lint_with_watch_inner(
|
||||||
|
flags,
|
||||||
|
lint_flags,
|
||||||
|
watcher_communicator,
|
||||||
|
changed_paths,
|
||||||
|
))
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
struct PathsWithOptions {
|
struct PathsWithOptions {
|
||||||
dir: WorkspaceDirectory,
|
dir: WorkspaceDirectory,
|
||||||
paths: Vec<PathBuf>,
|
paths: Vec<PathBuf>,
|
||||||
|
@ -269,7 +272,7 @@ impl WorkspaceLinter {
|
||||||
&mut self,
|
&mut self,
|
||||||
cli_options: &Arc<CliOptions>,
|
cli_options: &Arc<CliOptions>,
|
||||||
lint_options: LintOptions,
|
lint_options: LintOptions,
|
||||||
lint_config: LintConfig,
|
lint_config: DenoLintConfig,
|
||||||
member_dir: WorkspaceDirectory,
|
member_dir: WorkspaceDirectory,
|
||||||
paths: Vec<PathBuf>,
|
paths: Vec<PathBuf>,
|
||||||
) -> Result<(), AnyError> {
|
) -> Result<(), AnyError> {
|
||||||
|
@ -294,112 +297,63 @@ impl WorkspaceLinter {
|
||||||
deno_lint_config: lint_config,
|
deno_lint_config: lint_config,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
let has_error = self.has_error.clone();
|
||||||
|
let reporter_lock = self.reporter_lock.clone();
|
||||||
|
|
||||||
let mut futures = Vec::with_capacity(2);
|
let mut futures = Vec::with_capacity(2);
|
||||||
if linter.has_package_rules() {
|
if linter.has_package_rules() {
|
||||||
if self.workspace_module_graph.is_none() {
|
if let Some(fut) = self.run_package_rules(&linter, &member_dir, &paths) {
|
||||||
let module_graph_creator = self.module_graph_creator.clone();
|
futures.push(fut);
|
||||||
let packages = self.workspace_dir.jsr_packages_for_publish();
|
|
||||||
self.workspace_module_graph = Some(
|
|
||||||
async move {
|
|
||||||
module_graph_creator
|
|
||||||
.create_and_validate_publish_graph(&packages, true)
|
|
||||||
.await
|
|
||||||
.map(Rc::new)
|
|
||||||
.map_err(Rc::new)
|
|
||||||
}
|
|
||||||
.boxed_local()
|
|
||||||
.shared_local(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
let workspace_module_graph_future =
|
|
||||||
self.workspace_module_graph.as_ref().unwrap().clone();
|
|
||||||
let publish_config = member_dir.maybe_package_config();
|
|
||||||
if let Some(publish_config) = publish_config {
|
|
||||||
let has_error = self.has_error.clone();
|
|
||||||
let reporter_lock = self.reporter_lock.clone();
|
|
||||||
let linter = linter.clone();
|
|
||||||
let path_urls = paths
|
|
||||||
.iter()
|
|
||||||
.filter_map(|p| ModuleSpecifier::from_file_path(p).ok())
|
|
||||||
.collect::<HashSet<_>>();
|
|
||||||
futures.push(
|
|
||||||
async move {
|
|
||||||
let graph = workspace_module_graph_future
|
|
||||||
.await
|
|
||||||
.map_err(|err| anyhow!("{:#}", err))?;
|
|
||||||
let export_urls =
|
|
||||||
publish_config.config_file.resolve_export_value_urls()?;
|
|
||||||
if !export_urls.iter().any(|url| path_urls.contains(url)) {
|
|
||||||
return Ok(()); // entrypoint is not specified, so skip
|
|
||||||
}
|
|
||||||
let diagnostics = linter.lint_package(&graph, &export_urls);
|
|
||||||
if !diagnostics.is_empty() {
|
|
||||||
has_error.raise();
|
|
||||||
let mut reporter = reporter_lock.lock();
|
|
||||||
for diagnostic in &diagnostics {
|
|
||||||
reporter.visit_diagnostic(diagnostic);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
.boxed_local(),
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
futures.push({
|
let maybe_incremental_cache_ = maybe_incremental_cache.clone();
|
||||||
let has_error = self.has_error.clone();
|
let linter = linter.clone();
|
||||||
let reporter_lock = self.reporter_lock.clone();
|
let cli_options = cli_options.clone();
|
||||||
let maybe_incremental_cache = maybe_incremental_cache.clone();
|
let fut = async move {
|
||||||
let linter = linter.clone();
|
let operation = move |file_path: PathBuf| {
|
||||||
let cli_options = cli_options.clone();
|
let file_text = deno_ast::strip_bom(fs::read_to_string(&file_path)?);
|
||||||
async move {
|
|
||||||
run_parallelized(paths, {
|
|
||||||
move |file_path| {
|
|
||||||
let file_text =
|
|
||||||
deno_ast::strip_bom(fs::read_to_string(&file_path)?);
|
|
||||||
|
|
||||||
// don't bother rechecking this file if it didn't have any diagnostics before
|
// don't bother rechecking this file if it didn't have any diagnostics before
|
||||||
if let Some(incremental_cache) = &maybe_incremental_cache {
|
if let Some(incremental_cache) = &maybe_incremental_cache_ {
|
||||||
if incremental_cache.is_file_same(&file_path, &file_text) {
|
if incremental_cache.is_file_same(&file_path, &file_text) {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let r = linter.lint_file(
|
|
||||||
&file_path,
|
|
||||||
file_text,
|
|
||||||
cli_options.ext_flag().as_deref(),
|
|
||||||
);
|
|
||||||
if let Ok((file_source, file_diagnostics)) = &r {
|
|
||||||
if let Some(incremental_cache) = &maybe_incremental_cache {
|
|
||||||
if file_diagnostics.is_empty() {
|
|
||||||
// update the incremental cache if there were no diagnostics
|
|
||||||
incremental_cache.update_file(
|
|
||||||
&file_path,
|
|
||||||
// ensure the returned text is used here as it may have been modified via --fix
|
|
||||||
file_source.text(),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let success = handle_lint_result(
|
|
||||||
&file_path.to_string_lossy(),
|
|
||||||
r,
|
|
||||||
reporter_lock.clone(),
|
|
||||||
);
|
|
||||||
if !success {
|
|
||||||
has_error.raise();
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
})
|
}
|
||||||
.await
|
|
||||||
}
|
let r = linter.lint_file(
|
||||||
.boxed_local()
|
&file_path,
|
||||||
});
|
file_text,
|
||||||
|
cli_options.ext_flag().as_deref(),
|
||||||
|
);
|
||||||
|
if let Ok((file_source, file_diagnostics)) = &r {
|
||||||
|
if let Some(incremental_cache) = &maybe_incremental_cache_ {
|
||||||
|
if file_diagnostics.is_empty() {
|
||||||
|
// update the incremental cache if there were no diagnostics
|
||||||
|
incremental_cache.update_file(
|
||||||
|
&file_path,
|
||||||
|
// ensure the returned text is used here as it may have been modified via --fix
|
||||||
|
file_source.text(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let success = handle_lint_result(
|
||||||
|
&file_path.to_string_lossy(),
|
||||||
|
r,
|
||||||
|
reporter_lock.clone(),
|
||||||
|
);
|
||||||
|
if !success {
|
||||||
|
has_error.raise();
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
};
|
||||||
|
run_parallelized(paths, operation).await
|
||||||
|
}
|
||||||
|
.boxed_local();
|
||||||
|
futures.push(fut);
|
||||||
|
|
||||||
if lint_options.fix {
|
if lint_options.fix {
|
||||||
// run sequentially when using `--fix` to lower the chances of weird
|
// run sequentially when using `--fix` to lower the chances of weird
|
||||||
|
@ -419,6 +373,63 @@ impl WorkspaceLinter {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn run_package_rules(
|
||||||
|
&mut self,
|
||||||
|
linter: &Arc<CliLinter>,
|
||||||
|
member_dir: &WorkspaceDirectory,
|
||||||
|
paths: &[PathBuf],
|
||||||
|
) -> Option<LocalBoxFuture<Result<(), AnyError>>> {
|
||||||
|
if self.workspace_module_graph.is_none() {
|
||||||
|
let module_graph_creator = self.module_graph_creator.clone();
|
||||||
|
let packages = self.workspace_dir.jsr_packages_for_publish();
|
||||||
|
self.workspace_module_graph = Some(
|
||||||
|
async move {
|
||||||
|
module_graph_creator
|
||||||
|
.create_and_validate_publish_graph(&packages, true)
|
||||||
|
.await
|
||||||
|
.map(Rc::new)
|
||||||
|
.map_err(Rc::new)
|
||||||
|
}
|
||||||
|
.boxed_local()
|
||||||
|
.shared_local(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
let workspace_module_graph_future =
|
||||||
|
self.workspace_module_graph.as_ref().unwrap().clone();
|
||||||
|
let maybe_publish_config = member_dir.maybe_package_config();
|
||||||
|
let publish_config = maybe_publish_config?;
|
||||||
|
|
||||||
|
let has_error = self.has_error.clone();
|
||||||
|
let reporter_lock = self.reporter_lock.clone();
|
||||||
|
let linter = linter.clone();
|
||||||
|
let path_urls = paths
|
||||||
|
.iter()
|
||||||
|
.filter_map(|p| ModuleSpecifier::from_file_path(p).ok())
|
||||||
|
.collect::<HashSet<_>>();
|
||||||
|
let fut = async move {
|
||||||
|
let graph = workspace_module_graph_future
|
||||||
|
.await
|
||||||
|
.map_err(|err| anyhow!("{:#}", err))?;
|
||||||
|
let export_urls =
|
||||||
|
publish_config.config_file.resolve_export_value_urls()?;
|
||||||
|
if !export_urls.iter().any(|url| path_urls.contains(url)) {
|
||||||
|
return Ok(()); // entrypoint is not specified, so skip
|
||||||
|
}
|
||||||
|
let diagnostics = linter.lint_package(&graph, &export_urls);
|
||||||
|
if !diagnostics.is_empty() {
|
||||||
|
has_error.raise();
|
||||||
|
let mut reporter = reporter_lock.lock();
|
||||||
|
for diagnostic in &diagnostics {
|
||||||
|
reporter.visit_diagnostic(diagnostic);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
.boxed_local();
|
||||||
|
Some(fut)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn finish(self) -> bool {
|
pub fn finish(self) -> bool {
|
||||||
debug!("Found {} files", self.file_count);
|
debug!("Found {} files", self.file_count);
|
||||||
self.reporter_lock.lock().close(self.file_count);
|
self.reporter_lock.lock().close(self.file_count);
|
||||||
|
@ -494,10 +505,27 @@ pub fn print_rules_list(json: bool, maybe_rules_tags: Option<Vec<String>>) {
|
||||||
/// Treats input as TypeScript.
|
/// Treats input as TypeScript.
|
||||||
/// Compatible with `--json` flag.
|
/// Compatible with `--json` flag.
|
||||||
fn lint_stdin(
|
fn lint_stdin(
|
||||||
file_path: &Path,
|
cli_options: &Arc<CliOptions>,
|
||||||
configured_rules: ConfiguredRules,
|
lint_rule_provider: LintRuleProvider,
|
||||||
deno_lint_config: LintConfig,
|
workspace_lint_options: WorkspaceLintOptions,
|
||||||
) -> Result<(ParsedSource, Vec<LintDiagnostic>), AnyError> {
|
lint_flags: LintFlags,
|
||||||
|
deno_lint_config: DenoLintConfig,
|
||||||
|
) -> Result<bool, AnyError> {
|
||||||
|
let start_dir = &cli_options.start_dir;
|
||||||
|
let reporter_lock = Arc::new(Mutex::new(create_reporter(
|
||||||
|
workspace_lint_options.reporter_kind,
|
||||||
|
)));
|
||||||
|
let lint_config = start_dir
|
||||||
|
.to_lint_config(FilePatterns::new_with_base(start_dir.dir_path()))?;
|
||||||
|
let lint_options = LintOptions::resolve(lint_config, &lint_flags);
|
||||||
|
let configured_rules = lint_rule_provider.resolve_lint_rules_err_empty(
|
||||||
|
lint_options.rules,
|
||||||
|
start_dir.maybe_deno_json().map(|c| c.as_ref()),
|
||||||
|
)?;
|
||||||
|
let mut file_path = cli_options.initial_cwd().join(STDIN_FILE_NAME);
|
||||||
|
if let Some(ext) = cli_options.ext_flag() {
|
||||||
|
file_path.set_extension(ext);
|
||||||
|
}
|
||||||
let mut source_code = String::new();
|
let mut source_code = String::new();
|
||||||
if stdin().read_to_string(&mut source_code).is_err() {
|
if stdin().read_to_string(&mut source_code).is_err() {
|
||||||
return Err(generic_error("Failed to read from stdin"));
|
return Err(generic_error("Failed to read from stdin"));
|
||||||
|
@ -509,9 +537,14 @@ fn lint_stdin(
|
||||||
deno_lint_config,
|
deno_lint_config,
|
||||||
});
|
});
|
||||||
|
|
||||||
linter
|
let r = linter
|
||||||
.lint_file(file_path, deno_ast::strip_bom(source_code), None)
|
.lint_file(&file_path, deno_ast::strip_bom(source_code), None)
|
||||||
.map_err(AnyError::from)
|
.map_err(AnyError::from);
|
||||||
|
|
||||||
|
let success =
|
||||||
|
handle_lint_result(&file_path.to_string_lossy(), r, reporter_lock.clone());
|
||||||
|
reporter_lock.lock().close(1);
|
||||||
|
Ok(success)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn handle_lint_result(
|
fn handle_lint_result(
|
||||||
|
|
Loading…
Add table
Reference in a new issue