mirror of
https://github.com/astral-sh/ruff.git
synced 2025-08-02 09:52:32 +00:00

## Summary This PR removes the following dependencies from the `ruff_python_parser` crate: * `anyhow` (moved to dev dependencies) * `is-macro` * `itertools` The main motivation is that they aren't used much. Additionally, it updates the return type of `parse_type_annotation` to use a more specific `ParseError` instead of the generic `anyhow::Error`. ## Test Plan `cargo insta test`
81 lines
2.9 KiB
Rust
81 lines
2.9 KiB
Rust
//! This module takes care of parsing a type annotation.
|
|
|
|
use ruff_python_ast::relocate::relocate_expr;
|
|
use ruff_python_ast::str::raw_contents;
|
|
use ruff_python_ast::{Expr, ExprStringLiteral, StringFlags, StringLiteral};
|
|
use ruff_text_size::Ranged;
|
|
|
|
use crate::{parse_expression, parse_expression_range, ParseError};
|
|
|
|
#[derive(Copy, Clone, Debug)]
|
|
pub enum AnnotationKind {
|
|
/// The annotation is defined as part a simple string literal,
|
|
/// e.g. `x: "List[int]" = []`. Annotations within simple literals
|
|
/// can be accurately located. For example, we can underline specific
|
|
/// expressions within the annotation and apply automatic fixes, which is
|
|
/// not possible for complex string literals.
|
|
Simple,
|
|
|
|
/// The annotation is defined as part of a complex string literal, such as
|
|
/// a literal containing an implicit concatenation or escaped characters,
|
|
/// e.g. `x: "List" "[int]" = []`. These are comparatively rare, but valid.
|
|
Complex,
|
|
}
|
|
|
|
impl AnnotationKind {
|
|
/// Returns `true` if the annotation kind is simple.
|
|
pub const fn is_simple(self) -> bool {
|
|
matches!(self, AnnotationKind::Simple)
|
|
}
|
|
}
|
|
|
|
/// Parses the given string expression node as a type annotation. The given `source` is the entire
|
|
/// source code.
|
|
pub fn parse_type_annotation(
|
|
string_expr: &ExprStringLiteral,
|
|
source: &str,
|
|
) -> Result<(Expr, AnnotationKind), ParseError> {
|
|
let expr_text = &source[string_expr.range()];
|
|
|
|
if let [string_literal] = string_expr.value.as_slice() {
|
|
// Compare the raw contents (without quotes) of the expression with the parsed contents
|
|
// contained in the string literal.
|
|
if raw_contents(expr_text)
|
|
.is_some_and(|raw_contents| raw_contents == string_literal.as_str())
|
|
{
|
|
parse_simple_type_annotation(string_literal, source)
|
|
} else {
|
|
// The raw contents of the string doesn't match the parsed content. This could be the
|
|
// case for annotations that contain escaped quotes.
|
|
parse_complex_type_annotation(string_expr)
|
|
}
|
|
} else {
|
|
// String is implicitly concatenated.
|
|
parse_complex_type_annotation(string_expr)
|
|
}
|
|
}
|
|
|
|
fn parse_simple_type_annotation(
|
|
string_literal: &StringLiteral,
|
|
source: &str,
|
|
) -> Result<(Expr, AnnotationKind), ParseError> {
|
|
Ok((
|
|
parse_expression_range(
|
|
source,
|
|
string_literal
|
|
.range()
|
|
.add_start(string_literal.flags.opener_len())
|
|
.sub_end(string_literal.flags.closer_len()),
|
|
)?
|
|
.into_expr(),
|
|
AnnotationKind::Simple,
|
|
))
|
|
}
|
|
|
|
fn parse_complex_type_annotation(
|
|
string_expr: &ExprStringLiteral,
|
|
) -> Result<(Expr, AnnotationKind), ParseError> {
|
|
let mut parsed = parse_expression(string_expr.value.to_str())?.into_expr();
|
|
relocate_expr(&mut parsed, string_expr.range());
|
|
Ok((parsed, AnnotationKind::Complex))
|
|
}
|