Skip to content

Commit

Permalink
Small cleanups
Browse files Browse the repository at this point in the history
  • Loading branch information
AlexWaygood committed Jan 22, 2024
1 parent 0ab0682 commit d5606fa
Show file tree
Hide file tree
Showing 2 changed files with 50 additions and 37 deletions.
30 changes: 13 additions & 17 deletions crates/ruff_linter/src/rules/ruff/rules/sequence_sorting.rs
Original file line number Diff line number Diff line change
Expand Up @@ -89,12 +89,6 @@ impl<'a> From<&'a str> for IsortSortKey<'a> {
}
}

impl<'a> From<&'a StringSequenceItem> for IsortSortKey<'a> {
fn from(item: &'a StringSequenceItem) -> Self {
Self::from(item.value.as_str())
}
}

/// Classification for the casing of an element in a
/// sequence of literal strings.
///
Expand Down Expand Up @@ -135,9 +129,9 @@ impl InferredMemberType {
/// since in terms of the AST structure it's almost identical
/// to tuples/lists.)
///
/// Whereas lists, dicts and sets are always parenthesized
/// Whereas list, dict and set literals are always parenthesized
/// (e.g. lists always start with `[` and end with `]`),
/// single-line tuples *can* be unparenthesized.
/// single-line tuple literals *can* be unparenthesized.
/// We keep the original AST node around for the
/// Tuple variant so that this can be queried later.
#[derive(Debug)]
Expand All @@ -148,6 +142,9 @@ pub(super) enum SequenceKind<'a> {
}

impl SequenceKind<'_> {
// N.B. We only need the source code for the Tuple variant here,
// but if you already have a `Locator` instance handy,
// getting the source code is very cheap.
fn surrounding_brackets(&self, source: &str) -> (&'static str, &'static str) {
match self {
Self::List => ("[", "]"),
Expand Down Expand Up @@ -179,15 +176,14 @@ impl SequenceKind<'_> {
}
}

/// An enumeration of the various kinds of
/// [display literals](https://docs.python.org/3/reference/expressions.html#displays-for-lists-sets-and-dictionaries)
/// Python provides for builtin containers.
#[derive(Debug, is_macro::Is)]
pub(super) enum DisplayKind<'a> {
Sequence(SequenceKind<'a>),
Dict { values: &'a [ast::Expr] },
}

/// A newtype that zips together the string values of a display literal's elts,
/// together with the original AST nodes for that display literal's elts.
///
/// The main purpose of separating this out into a separate struct
/// is to enforce the invariants that:
///
/// 1. The two iterables that are zipped together have the same length; and,
/// 2. The length of both iterables is >= 2
struct SequenceElements<'a>(Vec<(&'a &'a str, &'a ast::Expr)>);

impl<'a> SequenceElements<'a> {
Expand Down
57 changes: 37 additions & 20 deletions crates/ruff_linter/src/rules/ruff/rules/sort_dunder_slots.rs
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ use ruff_text_size::{Ranged, TextRange};

use crate::checkers::ast::Checker;
use crate::rules::ruff::rules::sequence_sorting::{
sort_single_line_elements_sequence, DisplayKind, MultilineStringSequenceValue, SequenceKind,
sort_single_line_elements_sequence, MultilineStringSequenceValue, SequenceKind,
SortClassification, SortingStyle,
};

Expand Down Expand Up @@ -245,30 +245,27 @@ impl<'a> StringLiteralDisplay<'a> {
fn generate_fix(&self, items: &[&str], checker: &Checker) -> Option<Fix> {
let locator = checker.locator();
let is_multiline = locator.contains_line_break(self.range());
let sorted_source_code = {
if is_multiline {
// Sorting multiline dicts is unsupported
let display_kind = self.display_kind.as_sequence()?;
let sorted_source_code = match (&self.display_kind, is_multiline) {
(DisplayKind::Sequence(sequence_kind), true) => {
let analyzed_sequence = MultilineStringSequenceValue::from_source_range(
self.range(),
display_kind,
sequence_kind,
locator,
)?;
assert_eq!(analyzed_sequence.len(), self.elts.len());
analyzed_sequence.into_sorted_source_code(SORTING_STYLE, locator, checker.stylist())
} else {
match &self.display_kind {
DisplayKind::Dict { values } => {
sort_single_line_elements_dict(&self.elts, items, values, locator)
}
DisplayKind::Sequence(sequence_kind) => sort_single_line_elements_sequence(
sequence_kind,
&self.elts,
items,
locator,
SORTING_STYLE,
),
}
}
// Sorting multiline dicts is unsupported
(DisplayKind::Dict { .. }, true) => return None,
(DisplayKind::Sequence(sequence_kind), false) => sort_single_line_elements_sequence(
sequence_kind,
&self.elts,
items,
locator,
SORTING_STYLE,
),
(DisplayKind::Dict { values }, false) => {
sort_single_line_elements_dict(&self.elts, items, values, locator)
}
};
Some(Fix::safe_edit(Edit::range_replacement(
Expand All @@ -278,6 +275,26 @@ impl<'a> StringLiteralDisplay<'a> {
}
}

/// An enumeration of the various kinds of
/// [display literals](https://docs.python.org/3/reference/expressions.html#displays-for-lists-sets-and-dictionaries)
/// Python provides for builtin containers.
#[derive(Debug)]
enum DisplayKind<'a> {
Sequence(SequenceKind<'a>),
Dict { values: &'a [ast::Expr] },
}

/// A newtype that zips together three iterables:
///
/// 1. The string values of a dict literal's keys;
/// 2. The original AST nodes for the dict literal's keys; and,
/// 3. The original AST nodes for the dict literal's values
///
/// The main purpose of separating this out into a separate struct
/// is to enforce the invariants that:
///
/// 1. The three iterables that are zipped together have the same length; and,
/// 2. The length of all three iterables is >= 2
struct DictElements<'a>(Vec<(&'a &'a str, &'a ast::Expr, &'a ast::Expr)>);

impl<'a> DictElements<'a> {
Expand Down Expand Up @@ -313,7 +330,7 @@ impl<'a> DictElements<'a> {
/// `sequence_sorting.rs` if any other modules need it,
/// but stays here for now, since this is currently the
/// only module that needs it
pub(super) fn sort_single_line_elements_dict(
fn sort_single_line_elements_dict(
key_elts: &[ast::Expr],
elements: &[&str],
value_elts: &[ast::Expr],
Expand Down

0 comments on commit d5606fa

Please sign in to comment.