Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

Fallback to requires-python in certain cases when target-version is not found #16319

Merged
merged 36 commits into from
Mar 13, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
7bbda72
test fixtures
dylwil3 Feb 22, 2025
186235a
ruff.toml fallback to requires-python in same directory
dylwil3 Feb 22, 2025
92bc35a
no config fallback to ancestor requires-python
dylwil3 Feb 22, 2025
7c2c9b1
clippy
dylwil3 Feb 22, 2025
812fb20
update snapshot
dylwil3 Feb 22, 2025
759c0ba
try to appease windows
dylwil3 Feb 22, 2025
cc7d0bb
add TargetVersionStrategy to toggle fallback to requires_python
dylwil3 Feb 23, 2025
c8c10cd
add ConfigurationProvenance to switch resolution behavior
dylwil3 Feb 23, 2025
3c3c812
clippy
dylwil3 Feb 23, 2025
9fbe52a
debug log when parsing fails during fallback discovery
dylwil3 Feb 23, 2025
4a2d9e1
change name to ConfigurationOrigin
dylwil3 Feb 28, 2025
cd37ed8
make ConfigurationOrigin Copy
dylwil3 Feb 28, 2025
7f4403f
use Unknown ConfigurationOrigin instead of None
dylwil3 Feb 28, 2025
3613121
TargetVersionStrategy Standard --> UseDefault
dylwil3 Feb 28, 2025
1b52717
derive Relativity from ConfigurationOrigin
dylwil3 Feb 28, 2025
e2b30e9
nit: move test attr to above fn
dylwil3 Feb 28, 2025
9039594
nit: deriving --> derived
dylwil3 Feb 28, 2025
7e1f777
nit: use From instead of Into
dylwil3 Feb 28, 2025
0d99dba
nit: spacing
dylwil3 Feb 28, 2025
081518a
nit: shadow with as_ref
dylwil3 Feb 28, 2025
3c45426
clarify logic for default settings fallback
dylwil3 Feb 28, 2025
2a20b69
add test for shared config
dylwil3 Feb 28, 2025
123006f
update snapshots
dylwil3 Feb 28, 2025
513b1c8
display derived version in log
dylwil3 Mar 4, 2025
5f3ec8c
update snapshots
dylwil3 Mar 4, 2025
2d22b76
test --show-settings and add test with tool table
dylwil3 Mar 4, 2025
511129e
windows strikes again
dylwil3 Mar 4, 2025
ac55239
test check on directory instead of single file
dylwil3 Mar 12, 2025
54acafa
config origin knwon to be Ancestor in resolver - pass test
dylwil3 Mar 12, 2025
a1a7548
update snapshots from rebase
dylwil3 Mar 12, 2025
b098255
one more missed snapshot
dylwil3 Mar 12, 2025
229e905
test cli override
dylwil3 Mar 12, 2025
6093d74
for server use editor config with fallback version as fallback
dylwil3 Mar 12, 2025
f5d0203
Collect logs from the `log` crate in the server logs
MichaReiser Mar 13, 2025
49a593e
Use the `requires-python` fallback with a user-level configuration
MichaReiser Mar 13, 2025
dc26d44
Align server indexing with CLI behavior
MichaReiser Mar 13, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions Cargo.lock

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

1 change: 1 addition & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -154,6 +154,7 @@ toml = { version = "0.8.11" }
tracing = { version = "0.1.40" }
tracing-flame = { version = "0.2.0" }
tracing-indicatif = { version = "0.3.6" }
tracing-log = { version = "0.2.0" }
tracing-subscriber = { version = "0.3.18", default-features = false, features = [
"env-filter",
"fmt",
Expand Down
6 changes: 5 additions & 1 deletion crates/ruff/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -153,7 +153,11 @@ pub fn run(
}));
}

set_up_logging(global_options.log_level())?;
// Don't set up logging for the server command, as it has its own logging setup
// and setting the global logger can only be done once.
if !matches!(command, Command::Server { .. }) {
set_up_logging(global_options.log_level())?;
}

match command {
Command::Version { output_format } => {
Expand Down
62 changes: 55 additions & 7 deletions crates/ruff/src/resolve.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,12 +5,14 @@ use log::debug;
use path_absolutize::path_dedot;

use ruff_workspace::configuration::Configuration;
use ruff_workspace::pyproject;
use ruff_workspace::pyproject::{self, find_fallback_target_version};
use ruff_workspace::resolver::{
resolve_root_settings, ConfigurationTransformer, PyprojectConfig, PyprojectDiscoveryStrategy,
Relativity,
resolve_root_settings, ConfigurationOrigin, ConfigurationTransformer, PyprojectConfig,
PyprojectDiscoveryStrategy,
};

use ruff_python_ast as ast;

use crate::args::ConfigArguments;

/// Resolve the relevant settings strategy and defaults for the current
Expand All @@ -35,7 +37,11 @@ pub fn resolve(
// `pyproject.toml` for _all_ configuration, and resolve paths relative to the
// current working directory. (This matches ESLint's behavior.)
if let Some(pyproject) = config_arguments.config_file() {
let settings = resolve_root_settings(pyproject, Relativity::Cwd, config_arguments)?;
let settings = resolve_root_settings(
pyproject,
config_arguments,
ConfigurationOrigin::UserSpecified,
)?;
debug!(
"Using user-specified configuration file at: {}",
pyproject.display()
Expand All @@ -61,7 +67,8 @@ pub fn resolve(
"Using configuration file (via parent) at: {}",
pyproject.display()
);
let settings = resolve_root_settings(&pyproject, Relativity::Parent, config_arguments)?;
let settings =
resolve_root_settings(&pyproject, config_arguments, ConfigurationOrigin::Ancestor)?;
return Ok(PyprojectConfig::new(
PyprojectDiscoveryStrategy::Hierarchical,
settings,
Expand All @@ -74,11 +81,35 @@ pub fn resolve(
// end up the "closest" `pyproject.toml` file for every Python file later on, so
// these act as the "default" settings.)
if let Some(pyproject) = pyproject::find_user_settings_toml() {
struct FallbackTransformer<'a> {
arguments: &'a ConfigArguments,
}

impl ConfigurationTransformer for FallbackTransformer<'_> {
fn transform(&self, mut configuration: Configuration) -> Configuration {
// The `requires-python` constraint from the `pyproject.toml` takes precedence
// over the `target-version` from the user configuration.
let fallback = find_fallback_target_version(&*path_dedot::CWD);
if let Some(fallback) = fallback {
debug!("Derived `target-version` from found `requires-python`: {fallback:?}");
configuration.target_version = Some(fallback.into());
}

self.arguments.transform(configuration)
}
}

debug!(
"Using configuration file (via cwd) at: {}",
pyproject.display()
);
let settings = resolve_root_settings(&pyproject, Relativity::Cwd, config_arguments)?;
let settings = resolve_root_settings(
&pyproject,
&FallbackTransformer {
arguments: config_arguments,
},
ConfigurationOrigin::UserSettings,
)?;
return Ok(PyprojectConfig::new(
PyprojectDiscoveryStrategy::Hierarchical,
settings,
Expand All @@ -91,7 +122,24 @@ pub fn resolve(
// "closest" `pyproject.toml` file for every Python file later on, so these act
// as the "default" settings.)
debug!("Using Ruff default settings");
let config = config_arguments.transform(Configuration::default());
let mut config = config_arguments.transform(Configuration::default());
if config.target_version.is_none() {
// If we have arrived here we know that there was no `pyproject.toml`
// containing a `[tool.ruff]` section found in an ancestral directory.
// (This is an implicit requirement in the function
// `pyproject::find_settings_toml`.)
// However, there may be a `pyproject.toml` with a `requires-python`
// specified, and that is what we look for in this step.
let fallback = find_fallback_target_version(
stdin_filename
.as_ref()
.unwrap_or(&path_dedot::CWD.as_path()),
);
if let Some(version) = fallback {
debug!("Derived `target-version` from found `requires-python`: {version:?}");
}
config.target_version = fallback.map(ast::PythonVersion::from);
}
let settings = config.into_settings(&path_dedot::CWD)?;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we extract the path into a variable and use it both in find_fallback_target_version and when calling into_settings. It required me few cycles to understand whether path_dedot::CWD is correct but I then noticed that it's already what into_setting uses

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure, it depends on what behavior we want.

The way I have it currently set up I'm trying to make the behavior change as little as possible, so I've kept the settings resolution path the same no matter what, but attempted to find a fallback version using the same logic as we did when searching for a user configuration file. That logic would start from the stdin filename directory if it was passed in.

In other words, I don't think I can extract a path variable here since I may be passing a different value to find_fallback_target_version than I am to into_settings.

Ok(PyprojectConfig::new(
PyprojectDiscoveryStrategy::Hierarchical,
Expand Down
Loading
Loading