mirror of
https://github.com/astral-sh/ruff.git
synced 2025-09-26 11:59:35 +00:00
Use BestFits for non-fluent attribute chains (#6817)
This commit is contained in:
parent
d376cb4c2a
commit
1cd7790a8a
6 changed files with 95 additions and 21 deletions
|
@ -52,3 +52,16 @@ for converter in connection.ops.get_db_converters(
|
||||||
expression
|
expression
|
||||||
) + expression.get_db_converters(connection):
|
) + expression.get_db_converters(connection):
|
||||||
...
|
...
|
||||||
|
|
||||||
|
|
||||||
|
aaa = (
|
||||||
|
bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb # awkward comment
|
||||||
|
)
|
||||||
|
|
||||||
|
def test():
|
||||||
|
m2m_also_quite_long_zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz = models.ManyToManyField(Person, blank=True)
|
||||||
|
|
||||||
|
m2m_also_quite_long_zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz = models.ManyToManyFieldAttributeChainField
|
||||||
|
|
||||||
|
m2m_also_quite_long_zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz = models.ManyToManyField(Person, blank=True)
|
||||||
|
m2m_also_quite_long_zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz = models.ManyToManyFieldAttributeChainFieeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeld
|
||||||
|
|
|
@ -66,6 +66,13 @@ def foo():
|
||||||
1
|
1
|
||||||
)
|
)
|
||||||
|
|
||||||
|
yield (
|
||||||
|
"# * Make sure each ForeignKey and OneToOneField has `on_delete` set "
|
||||||
|
"to the desired behavior"
|
||||||
|
)
|
||||||
|
|
||||||
|
yield aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa + bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb + ccccccccccccccccccccccccccccccccccccccccccccccccccccccc
|
||||||
|
|
||||||
yield ("Cache key will cause errors if used with memcached: %r " "(longer than %s)" % (
|
yield ("Cache key will cause errors if used with memcached: %r " "(longer than %s)" % (
|
||||||
key,
|
key,
|
||||||
MEMCACHE_MAX_KEY_LENGTH,
|
MEMCACHE_MAX_KEY_LENGTH,
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
use crate::expression::CallChainLayout;
|
use crate::expression::CallChainLayout;
|
||||||
use ruff_formatter::FormatRuleWithOptions;
|
|
||||||
|
use ruff_formatter::{format_args, write, FormatRuleWithOptions};
|
||||||
use ruff_python_ast::node::AnyNodeRef;
|
use ruff_python_ast::node::AnyNodeRef;
|
||||||
use ruff_python_ast::{Expr, ExprCall};
|
use ruff_python_ast::{Expr, ExprCall};
|
||||||
|
|
||||||
|
@ -31,15 +32,11 @@ impl FormatNodeRule<ExprCall> for FormatExprCall {
|
||||||
|
|
||||||
let call_chain_layout = self.call_chain_layout.apply_in_node(item, f);
|
let call_chain_layout = self.call_chain_layout.apply_in_node(item, f);
|
||||||
|
|
||||||
let fmt_inner = format_with(|f| {
|
let fmt_func = format_with(|f| match func.as_ref() {
|
||||||
match func.as_ref() {
|
Expr::Attribute(expr) => expr.format().with_options(call_chain_layout).fmt(f),
|
||||||
Expr::Attribute(expr) => expr.format().with_options(call_chain_layout).fmt(f)?,
|
Expr::Call(expr) => expr.format().with_options(call_chain_layout).fmt(f),
|
||||||
Expr::Call(expr) => expr.format().with_options(call_chain_layout).fmt(f)?,
|
Expr::Subscript(expr) => expr.format().with_options(call_chain_layout).fmt(f),
|
||||||
Expr::Subscript(expr) => expr.format().with_options(call_chain_layout).fmt(f)?,
|
_ => func.format().fmt(f),
|
||||||
_ => func.format().fmt(f)?,
|
|
||||||
}
|
|
||||||
|
|
||||||
arguments.format().fmt(f)
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// Allow to indent the parentheses while
|
// Allow to indent the parentheses while
|
||||||
|
@ -51,9 +48,9 @@ impl FormatNodeRule<ExprCall> for FormatExprCall {
|
||||||
if call_chain_layout == CallChainLayout::Fluent
|
if call_chain_layout == CallChainLayout::Fluent
|
||||||
&& self.call_chain_layout == CallChainLayout::Default
|
&& self.call_chain_layout == CallChainLayout::Default
|
||||||
{
|
{
|
||||||
group(&fmt_inner).fmt(f)
|
group(&format_args![fmt_func, arguments.format()]).fmt(f)
|
||||||
} else {
|
} else {
|
||||||
fmt_inner.fmt(f)
|
write!(f, [fmt_func, arguments.format()])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -69,10 +66,7 @@ impl NeedsParentheses for ExprCall {
|
||||||
{
|
{
|
||||||
OptionalParentheses::Multiline
|
OptionalParentheses::Multiline
|
||||||
} else {
|
} else {
|
||||||
match self.func.needs_parentheses(self.into(), context) {
|
self.func.needs_parentheses(self.into(), context)
|
||||||
OptionalParentheses::BestFit => OptionalParentheses::Never,
|
|
||||||
parentheses => parentheses,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -56,6 +56,18 @@ assert xxxxxxxxx.xxxxxxxxx.xxxxxxxxx(
|
||||||
|
|
||||||
|
|
||||||
# Example from https://github.com/psf/black/issues/3229
|
# Example from https://github.com/psf/black/issues/3229
|
||||||
|
@@ -43,8 +41,6 @@
|
||||||
|
)
|
||||||
|
|
||||||
|
# Regression test for https://github.com/psf/black/issues/3414.
|
||||||
|
-assert xxxxxxxxx.xxxxxxxxx.xxxxxxxxx(
|
||||||
|
- xxxxxxxxx
|
||||||
|
-).xxxxxxxxxxxxxxxxxx(), (
|
||||||
|
- "xxx {xxxxxxxxx} xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"
|
||||||
|
-)
|
||||||
|
+assert (
|
||||||
|
+ xxxxxxxxx.xxxxxxxxx.xxxxxxxxx(xxxxxxxxx).xxxxxxxxxxxxxxxxxx()
|
||||||
|
+), "xxx {xxxxxxxxx} xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"
|
||||||
```
|
```
|
||||||
|
|
||||||
## Ruff Output
|
## Ruff Output
|
||||||
|
@ -104,11 +116,9 @@ assert (
|
||||||
)
|
)
|
||||||
|
|
||||||
# Regression test for https://github.com/psf/black/issues/3414.
|
# Regression test for https://github.com/psf/black/issues/3414.
|
||||||
assert xxxxxxxxx.xxxxxxxxx.xxxxxxxxx(
|
assert (
|
||||||
xxxxxxxxx
|
xxxxxxxxx.xxxxxxxxx.xxxxxxxxx(xxxxxxxxx).xxxxxxxxxxxxxxxxxx()
|
||||||
).xxxxxxxxxxxxxxxxxx(), (
|
), "xxx {xxxxxxxxx} xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"
|
||||||
"xxx {xxxxxxxxx} xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"
|
|
||||||
)
|
|
||||||
```
|
```
|
||||||
|
|
||||||
## Black Output
|
## Black Output
|
||||||
|
|
|
@ -58,6 +58,19 @@ for converter in connection.ops.get_db_converters(
|
||||||
expression
|
expression
|
||||||
) + expression.get_db_converters(connection):
|
) + expression.get_db_converters(connection):
|
||||||
...
|
...
|
||||||
|
|
||||||
|
|
||||||
|
aaa = (
|
||||||
|
bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb # awkward comment
|
||||||
|
)
|
||||||
|
|
||||||
|
def test():
|
||||||
|
m2m_also_quite_long_zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz = models.ManyToManyField(Person, blank=True)
|
||||||
|
|
||||||
|
m2m_also_quite_long_zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz = models.ManyToManyFieldAttributeChainField
|
||||||
|
|
||||||
|
m2m_also_quite_long_zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz = models.ManyToManyField(Person, blank=True)
|
||||||
|
m2m_also_quite_long_zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz = models.ManyToManyFieldAttributeChainFieeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeld
|
||||||
```
|
```
|
||||||
|
|
||||||
## Output
|
## Output
|
||||||
|
@ -120,6 +133,25 @@ for converter in connection.ops.get_db_converters(
|
||||||
expression
|
expression
|
||||||
) + expression.get_db_converters(connection):
|
) + expression.get_db_converters(connection):
|
||||||
...
|
...
|
||||||
|
|
||||||
|
|
||||||
|
aaa = bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb # awkward comment
|
||||||
|
|
||||||
|
|
||||||
|
def test():
|
||||||
|
m2m_also_quite_long_zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz = (
|
||||||
|
models.ManyToManyField(Person, blank=True)
|
||||||
|
)
|
||||||
|
|
||||||
|
m2m_also_quite_long_zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz = (
|
||||||
|
models.ManyToManyFieldAttributeChainField
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
m2m_also_quite_long_zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz = (
|
||||||
|
models.ManyToManyField(Person, blank=True)
|
||||||
|
)
|
||||||
|
m2m_also_quite_long_zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz = models.ManyToManyFieldAttributeChainFieeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeld
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -72,6 +72,13 @@ def foo():
|
||||||
1
|
1
|
||||||
)
|
)
|
||||||
|
|
||||||
|
yield (
|
||||||
|
"# * Make sure each ForeignKey and OneToOneField has `on_delete` set "
|
||||||
|
"to the desired behavior"
|
||||||
|
)
|
||||||
|
|
||||||
|
yield aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa + bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb + ccccccccccccccccccccccccccccccccccccccccccccccccccccccc
|
||||||
|
|
||||||
yield ("Cache key will cause errors if used with memcached: %r " "(longer than %s)" % (
|
yield ("Cache key will cause errors if used with memcached: %r " "(longer than %s)" % (
|
||||||
key,
|
key,
|
||||||
MEMCACHE_MAX_KEY_LENGTH,
|
MEMCACHE_MAX_KEY_LENGTH,
|
||||||
|
@ -168,6 +175,17 @@ def foo():
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
yield (
|
||||||
|
"# * Make sure each ForeignKey and OneToOneField has `on_delete` set "
|
||||||
|
"to the desired behavior"
|
||||||
|
)
|
||||||
|
|
||||||
|
yield (
|
||||||
|
aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa
|
||||||
|
+ bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb
|
||||||
|
+ ccccccccccccccccccccccccccccccccccccccccccccccccccccccc
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
yield (
|
yield (
|
||||||
"Cache key will cause errors if used with memcached: %r "
|
"Cache key will cause errors if used with memcached: %r "
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue