diff --git a/crates/ruff/resources/test/fixtures/pydocstyle/D.py b/crates/ruff/resources/test/fixtures/pydocstyle/D.py index 94abc1c906f2ec..23a74c381f42ef 100644 --- a/crates/ruff/resources/test/fixtures/pydocstyle/D.py +++ b/crates/ruff/resources/test/fixtures/pydocstyle/D.py @@ -634,3 +634,8 @@ def starts_with_this(): @expect('D404: First word of the docstring should not be "This"') def starts_with_space_then_this(): """ This is a docstring that starts with a space.""" # noqa: D210 + + +class SameLine: """This is a docstring on the same line""" + +def same_line(): """This is a docstring on the same line""" diff --git a/crates/ruff/src/rules/pydocstyle/rules/blank_before_after_class.rs b/crates/ruff/src/rules/pydocstyle/rules/blank_before_after_class.rs index 77f55b07441ae6..556f1624bcbda1 100644 --- a/crates/ruff/src/rules/pydocstyle/rules/blank_before_after_class.rs +++ b/crates/ruff/src/rules/pydocstyle/rules/blank_before_after_class.rs @@ -43,9 +43,7 @@ use crate::registry::{AsRule, Rule}; /// /// [D211]: https://beta.ruff.rs/docs/rules/blank-line-before-class #[violation] -pub struct OneBlankLineBeforeClass { - lines: usize, -} +pub struct OneBlankLineBeforeClass; impl AlwaysAutofixableViolation for OneBlankLineBeforeClass { #[derive_message_formats] @@ -97,9 +95,7 @@ impl AlwaysAutofixableViolation for OneBlankLineBeforeClass { /// /// [PEP 257]: https://peps.python.org/pep-0257/ #[violation] -pub struct OneBlankLineAfterClass { - lines: usize, -} +pub struct OneBlankLineAfterClass; impl AlwaysAutofixableViolation for OneBlankLineAfterClass { #[derive_message_formats] @@ -144,9 +140,7 @@ impl AlwaysAutofixableViolation for OneBlankLineAfterClass { /// /// [D203]: https://beta.ruff.rs/docs/rules/one-blank-line-before-class #[violation] -pub struct BlankLineBeforeClass { - lines: usize, -} +pub struct BlankLineBeforeClass; impl AlwaysAutofixableViolation for BlankLineBeforeClass { #[derive_message_formats] @@ -170,14 +164,24 @@ pub(crate) fn blank_before_after_class(checker: &mut Checker, docstring: &Docstr return; }; + // Special-case: the docstring is on the same line as the class. For example: + // ```python + // class PhotoMetadata: """Metadata about a photo.""" + // ``` + let between_range = TextRange::new(stmt.start(), docstring.start()); + if !checker.locator().contains_line_break(between_range) { + return; + } + if checker.enabled(Rule::OneBlankLineBeforeClass) || checker.enabled(Rule::BlankLineBeforeClass) { - let before = checker - .locator() - .slice(TextRange::new(stmt.start(), docstring.start())); + let mut lines = UniversalNewlineIterator::with_offset( + checker.locator().slice(between_range), + between_range.start(), + ) + .rev(); let mut blank_lines_before = 0usize; - let mut lines = UniversalNewlineIterator::with_offset(before, stmt.start()).rev(); let mut blank_lines_start = lines.next().map(|line| line.start()).unwrap_or_default(); for line in lines { @@ -191,12 +195,7 @@ pub(crate) fn blank_before_after_class(checker: &mut Checker, docstring: &Docstr if checker.enabled(Rule::BlankLineBeforeClass) { if blank_lines_before != 0 { - let mut diagnostic = Diagnostic::new( - BlankLineBeforeClass { - lines: blank_lines_before, - }, - docstring.range(), - ); + let mut diagnostic = Diagnostic::new(BlankLineBeforeClass, docstring.range()); if checker.patch(diagnostic.kind.rule()) { // Delete the blank line before the class. diagnostic.set_fix(Fix::automatic(Edit::deletion( @@ -209,12 +208,7 @@ pub(crate) fn blank_before_after_class(checker: &mut Checker, docstring: &Docstr } if checker.enabled(Rule::OneBlankLineBeforeClass) { if blank_lines_before != 1 { - let mut diagnostic = Diagnostic::new( - OneBlankLineBeforeClass { - lines: blank_lines_before, - }, - docstring.range(), - ); + let mut diagnostic = Diagnostic::new(OneBlankLineBeforeClass, docstring.range()); if checker.patch(diagnostic.kind.rule()) { // Insert one blank line before the class. diagnostic.set_fix(Fix::automatic(Edit::replacement( @@ -229,9 +223,9 @@ pub(crate) fn blank_before_after_class(checker: &mut Checker, docstring: &Docstr } if checker.enabled(Rule::OneBlankLineAfterClass) { - let after = checker - .locator() - .slice(TextRange::new(docstring.end(), stmt.end())); + let after_range = TextRange::new(docstring.end(), stmt.end()); + + let after = checker.locator().slice(after_range); let all_blank_after = after.universal_newlines().skip(1).all(|line| { line.trim_whitespace().is_empty() || line.trim_whitespace_start().starts_with('#') @@ -240,9 +234,10 @@ pub(crate) fn blank_before_after_class(checker: &mut Checker, docstring: &Docstr return; } - let mut blank_lines_after = 0usize; - let mut lines = UniversalNewlineIterator::with_offset(after, docstring.end()); + let mut lines = UniversalNewlineIterator::with_offset(after, after_range.start()); + let first_line_start = lines.next().map(|l| l.start()).unwrap_or_default(); + let mut blank_lines_after = 0usize; let mut blank_lines_end = docstring.end(); for line in lines { @@ -255,12 +250,7 @@ pub(crate) fn blank_before_after_class(checker: &mut Checker, docstring: &Docstr } if blank_lines_after != 1 { - let mut diagnostic = Diagnostic::new( - OneBlankLineAfterClass { - lines: blank_lines_after, - }, - docstring.range(), - ); + let mut diagnostic = Diagnostic::new(OneBlankLineAfterClass, docstring.range()); if checker.patch(diagnostic.kind.rule()) { // Insert a blank line before the class (replacing any existing lines). diagnostic.set_fix(Fix::automatic(Edit::replacement( diff --git a/crates/ruff/src/rules/pydocstyle/snapshots/ruff__rules__pydocstyle__tests__D400_D.py.snap b/crates/ruff/src/rules/pydocstyle/snapshots/ruff__rules__pydocstyle__tests__D400_D.py.snap index 144c7cb86d5dd2..3b92b97fd1edd3 100644 --- a/crates/ruff/src/rules/pydocstyle/snapshots/ruff__rules__pydocstyle__tests__D400_D.py.snap +++ b/crates/ruff/src/rules/pydocstyle/snapshots/ruff__rules__pydocstyle__tests__D400_D.py.snap @@ -272,4 +272,38 @@ D.py:615:5: D400 [*] First line should end with a period 617 617 | """ 618 618 | +D.py:639:17: D400 [*] First line should end with a period + | +639 | class SameLine: """This is a docstring on the same line""" + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ D400 +640 | +641 | def same_line(): """This is a docstring on the same line""" + | + = help: Add period + +ℹ Suggested fix +636 636 | """ This is a docstring that starts with a space.""" # noqa: D210 +637 637 | +638 638 | +639 |-class SameLine: """This is a docstring on the same line""" + 639 |+class SameLine: """This is a docstring on the same line.""" +640 640 | +641 641 | def same_line(): """This is a docstring on the same line""" + +D.py:641:18: D400 [*] First line should end with a period + | +639 | class SameLine: """This is a docstring on the same line""" +640 | +641 | def same_line(): """This is a docstring on the same line""" + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ D400 + | + = help: Add period + +ℹ Suggested fix +638 638 | +639 639 | class SameLine: """This is a docstring on the same line""" +640 640 | +641 |-def same_line(): """This is a docstring on the same line""" + 641 |+def same_line(): """This is a docstring on the same line.""" + diff --git a/crates/ruff/src/rules/pydocstyle/snapshots/ruff__rules__pydocstyle__tests__D404_D.py.snap b/crates/ruff/src/rules/pydocstyle/snapshots/ruff__rules__pydocstyle__tests__D404_D.py.snap index 009f74f62a2cbe..d372af4445f1e8 100644 --- a/crates/ruff/src/rules/pydocstyle/snapshots/ruff__rules__pydocstyle__tests__D404_D.py.snap +++ b/crates/ruff/src/rules/pydocstyle/snapshots/ruff__rules__pydocstyle__tests__D404_D.py.snap @@ -17,4 +17,20 @@ D.py:636:5: D404 First word of the docstring should not be "This" | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ D404 | +D.py:639:17: D404 First word of the docstring should not be "This" + | +639 | class SameLine: """This is a docstring on the same line""" + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ D404 +640 | +641 | def same_line(): """This is a docstring on the same line""" + | + +D.py:641:18: D404 First word of the docstring should not be "This" + | +639 | class SameLine: """This is a docstring on the same line""" +640 | +641 | def same_line(): """This is a docstring on the same line""" + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ D404 + | + diff --git a/crates/ruff/src/rules/pydocstyle/snapshots/ruff__rules__pydocstyle__tests__D415_D.py.snap b/crates/ruff/src/rules/pydocstyle/snapshots/ruff__rules__pydocstyle__tests__D415_D.py.snap index 2b457967e8b282..73725b09cf1812 100644 --- a/crates/ruff/src/rules/pydocstyle/snapshots/ruff__rules__pydocstyle__tests__D415_D.py.snap +++ b/crates/ruff/src/rules/pydocstyle/snapshots/ruff__rules__pydocstyle__tests__D415_D.py.snap @@ -254,4 +254,38 @@ D.py:615:5: D415 [*] First line should end with a period, question mark, or excl 617 617 | """ 618 618 | +D.py:639:17: D415 [*] First line should end with a period, question mark, or exclamation point + | +639 | class SameLine: """This is a docstring on the same line""" + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ D415 +640 | +641 | def same_line(): """This is a docstring on the same line""" + | + = help: Add closing punctuation + +ℹ Suggested fix +636 636 | """ This is a docstring that starts with a space.""" # noqa: D210 +637 637 | +638 638 | +639 |-class SameLine: """This is a docstring on the same line""" + 639 |+class SameLine: """This is a docstring on the same line.""" +640 640 | +641 641 | def same_line(): """This is a docstring on the same line""" + +D.py:641:18: D415 [*] First line should end with a period, question mark, or exclamation point + | +639 | class SameLine: """This is a docstring on the same line""" +640 | +641 | def same_line(): """This is a docstring on the same line""" + | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ D415 + | + = help: Add closing punctuation + +ℹ Suggested fix +638 638 | +639 639 | class SameLine: """This is a docstring on the same line""" +640 640 | +641 |-def same_line(): """This is a docstring on the same line""" + 641 |+def same_line(): """This is a docstring on the same line.""" +