mirror of
https://github.com/astral-sh/ruff.git
synced 2025-09-27 12:29:28 +00:00

## Summary This PR removes the `debug_assertion` in the `Indexer` to allow unterminated f-strings. This is mainly a fix in the development build which now matches the release build. The fix is simple: remove the `debug_assertion` which means that the there could be `FStringStart` and possibly `FStringMiddle` tokens without a corresponding f-string range in the `Indexer`. This means that the code requesting for the f-string index need to account for the `None` case, making the code safer. This also updates the code which queries the `FStringRanges` to account for the `None` case. This will happen when the `FStringStart` / `FStringMiddle` tokens are present but the `FStringEnd` token isn't which means that the `Indexer` won't contain the range for that f-string. ## Test Plan `cargo test` Taking the following code as an example: ```python f"{123} ``` This only emits a `FStringStart` token, but no `FStringMiddle` or `FStringEnd` tokens. And, ```python f"\.png${ ``` This emits a `FStringStart` and `FStringMiddle` token, but no `FStringEnd` token. fixes: #8065
97 lines
3.3 KiB
Rust
97 lines
3.3 KiB
Rust
use std::collections::BTreeMap;
|
|
|
|
use ruff_python_parser::Tok;
|
|
use ruff_text_size::{TextRange, TextSize};
|
|
|
|
/// Stores the ranges of all f-strings in a file sorted by [`TextRange::start`].
|
|
/// There can be multiple overlapping ranges for nested f-strings.
|
|
///
|
|
/// Note that the ranges for all unterminated f-strings are not stored.
|
|
#[derive(Debug)]
|
|
pub struct FStringRanges {
|
|
// Mapping from the f-string start location to its range.
|
|
raw: BTreeMap<TextSize, TextRange>,
|
|
}
|
|
|
|
impl FStringRanges {
|
|
/// Return the [`TextRange`] of the innermost f-string at the given offset.
|
|
pub fn innermost(&self, offset: TextSize) -> Option<TextRange> {
|
|
self.raw
|
|
.range(..=offset)
|
|
.rev()
|
|
.find(|(_, range)| range.contains(offset))
|
|
.map(|(_, range)| *range)
|
|
}
|
|
|
|
/// Return the [`TextRange`] of the outermost f-string at the given offset.
|
|
pub fn outermost(&self, offset: TextSize) -> Option<TextRange> {
|
|
// Explanation of the algorithm:
|
|
//
|
|
// ```python
|
|
// # v
|
|
// f"normal" f"another" f"first {f"second {f"third"} second"} first"
|
|
// # ^^(1)^^^
|
|
// # ^^^^^^^^^^^^(2)^^^^^^^^^^^^
|
|
// # ^^^^^^^^^^^^^^^^^^^^^(3)^^^^^^^^^^^^^^^^^^^^
|
|
// # ^^^(4)^^^^
|
|
// # ^^^(5)^^^
|
|
// ```
|
|
//
|
|
// The offset is marked with a `v` and the ranges are numbered in the order
|
|
// they are yielded by the iterator in the reverse order. The algorithm
|
|
// works as follows:
|
|
// 1. Skip all ranges that don't contain the offset (1).
|
|
// 2. Take all ranges that contain the offset (2, 3).
|
|
// 3. Stop taking ranges when the offset is no longer contained.
|
|
// 4. Take the last range that contained the offset (3, the outermost).
|
|
self.raw
|
|
.range(..=offset)
|
|
.rev()
|
|
.skip_while(|(_, range)| !range.contains(offset))
|
|
.take_while(|(_, range)| range.contains(offset))
|
|
.last()
|
|
.map(|(_, range)| *range)
|
|
}
|
|
|
|
/// Returns an iterator over all f-string [`TextRange`] sorted by their
|
|
/// start location.
|
|
///
|
|
/// For nested f-strings, the outermost f-string is yielded first, moving
|
|
/// inwards with each iteration.
|
|
#[inline]
|
|
pub fn values(&self) -> impl Iterator<Item = &TextRange> + '_ {
|
|
self.raw.values()
|
|
}
|
|
|
|
/// Returns the number of f-string ranges stored.
|
|
#[inline]
|
|
pub fn len(&self) -> usize {
|
|
self.raw.len()
|
|
}
|
|
}
|
|
|
|
#[derive(Default)]
|
|
pub(crate) struct FStringRangesBuilder {
|
|
start_locations: Vec<TextSize>,
|
|
raw: BTreeMap<TextSize, TextRange>,
|
|
}
|
|
|
|
impl FStringRangesBuilder {
|
|
pub(crate) fn visit_token(&mut self, token: &Tok, range: TextRange) {
|
|
match token {
|
|
Tok::FStringStart => {
|
|
self.start_locations.push(range.start());
|
|
}
|
|
Tok::FStringEnd => {
|
|
if let Some(start) = self.start_locations.pop() {
|
|
self.raw.insert(start, TextRange::new(start, range.end()));
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
pub(crate) fn finish(self) -> FStringRanges {
|
|
FStringRanges { raw: self.raw }
|
|
}
|
|
}
|