Use BestFits for non-fluent attribute chains (#6817)

This commit is contained in:
Micha Reiser 2023-08-24 14:09:25 +02:00 committed by GitHub
parent d376cb4c2a
commit 1cd7790a8a
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 95 additions and 21 deletions

View file

@ -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

View file

@ -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,

View file

@ -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,
}
} }
} }
} }

View file

@ -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

View file

@ -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
``` ```

View file

@ -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 "