mirror of
https://github.com/astral-sh/ruff.git
synced 2025-08-24 04:14:44 +00:00
809 lines
27 KiB
Rust
809 lines
27 KiB
Rust
//! Token kinds for Python source code created by the lexer and consumed by the `ruff_python_parser`.
|
|
//!
|
|
//! This module defines the tokens that the lexer recognizes. The tokens are
|
|
//! loosely based on the token definitions found in the [CPython source].
|
|
//!
|
|
//! [CPython source]: https://github.com/python/cpython/blob/dfc2e065a2e71011017077e549cd2f9bf4944c54/Grammar/Tokens
|
|
|
|
use std::fmt;
|
|
|
|
use bitflags::bitflags;
|
|
|
|
use ruff_python_ast::name::Name;
|
|
use ruff_python_ast::str::Quote;
|
|
use ruff_python_ast::str_prefix::{
|
|
AnyStringPrefix, ByteStringPrefix, FStringPrefix, StringLiteralPrefix,
|
|
};
|
|
use ruff_python_ast::{AnyStringFlags, BoolOp, Int, IpyEscapeKind, Operator, StringFlags, UnaryOp};
|
|
use ruff_text_size::{Ranged, TextRange};
|
|
|
|
#[derive(Clone, Copy, PartialEq, Eq)]
|
|
pub struct Token {
|
|
/// The kind of the token.
|
|
kind: TokenKind,
|
|
/// The range of the token.
|
|
range: TextRange,
|
|
/// The set of flags describing this token.
|
|
flags: TokenFlags,
|
|
}
|
|
|
|
impl Token {
|
|
pub(crate) fn new(kind: TokenKind, range: TextRange, flags: TokenFlags) -> Token {
|
|
Self { kind, range, flags }
|
|
}
|
|
|
|
/// Returns the token kind.
|
|
#[inline]
|
|
pub const fn kind(&self) -> TokenKind {
|
|
self.kind
|
|
}
|
|
|
|
/// Returns the token as a tuple of (kind, range).
|
|
#[inline]
|
|
pub const fn as_tuple(&self) -> (TokenKind, TextRange) {
|
|
(self.kind, self.range)
|
|
}
|
|
|
|
/// Returns `true` if the current token is a triple-quoted string of any kind.
|
|
///
|
|
/// # Panics
|
|
///
|
|
/// If it isn't a string or any f-string tokens.
|
|
pub fn is_triple_quoted_string(self) -> bool {
|
|
assert!(self.is_any_string());
|
|
self.flags.is_triple_quoted()
|
|
}
|
|
|
|
/// Returns the [`Quote`] style for the current string token of any kind.
|
|
///
|
|
/// # Panics
|
|
///
|
|
/// If it isn't a string or any f-string tokens.
|
|
pub fn string_quote_style(self) -> Quote {
|
|
assert!(self.is_any_string());
|
|
self.flags.quote_style()
|
|
}
|
|
|
|
/// Returns `true` if this is any kind of string token.
|
|
const fn is_any_string(self) -> bool {
|
|
matches!(
|
|
self.kind,
|
|
TokenKind::String
|
|
| TokenKind::FStringStart
|
|
| TokenKind::FStringMiddle
|
|
| TokenKind::FStringEnd
|
|
)
|
|
}
|
|
}
|
|
|
|
impl Ranged for Token {
|
|
fn range(&self) -> TextRange {
|
|
self.range
|
|
}
|
|
}
|
|
|
|
impl fmt::Debug for Token {
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
write!(f, "{:?} {:?}", self.kind, self.range)?;
|
|
if !self.flags.is_empty() {
|
|
f.write_str(" (flags = ")?;
|
|
let mut first = true;
|
|
for (name, _) in self.flags.iter_names() {
|
|
if first {
|
|
first = false;
|
|
} else {
|
|
f.write_str(" | ")?;
|
|
}
|
|
f.write_str(name)?;
|
|
}
|
|
f.write_str(")")?;
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
/// A kind of a token.
|
|
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug, PartialOrd, Ord)]
|
|
pub enum TokenKind {
|
|
/// Token kind for a name, commonly known as an identifier.
|
|
Name,
|
|
/// Token kind for an integer.
|
|
Int,
|
|
/// Token kind for a floating point number.
|
|
Float,
|
|
/// Token kind for a complex number.
|
|
Complex,
|
|
/// Token kind for a string.
|
|
String,
|
|
/// Token kind for the start of an f-string. This includes the `f`/`F`/`fr` prefix
|
|
/// and the opening quote(s).
|
|
FStringStart,
|
|
/// Token kind that includes the portion of text inside the f-string that's not
|
|
/// part of the expression part and isn't an opening or closing brace.
|
|
FStringMiddle,
|
|
/// Token kind for the end of an f-string. This includes the closing quote.
|
|
FStringEnd,
|
|
/// Token kind for a IPython escape command.
|
|
IpyEscapeCommand,
|
|
/// Token kind for a comment. These are filtered out of the token stream prior to parsing.
|
|
Comment,
|
|
/// Token kind for a newline.
|
|
Newline,
|
|
/// Token kind for a newline that is not a logical line break. These are filtered out of
|
|
/// the token stream prior to parsing.
|
|
NonLogicalNewline,
|
|
/// Token kind for an indent.
|
|
Indent,
|
|
/// Token kind for a dedent.
|
|
Dedent,
|
|
EndOfFile,
|
|
/// Token kind for a question mark `?`.
|
|
Question,
|
|
/// Token kind for an exclamation mark `!`.
|
|
Exclamation,
|
|
/// Token kind for a left parenthesis `(`.
|
|
Lpar,
|
|
/// Token kind for a right parenthesis `)`.
|
|
Rpar,
|
|
/// Token kind for a left square bracket `[`.
|
|
Lsqb,
|
|
/// Token kind for a right square bracket `]`.
|
|
Rsqb,
|
|
/// Token kind for a colon `:`.
|
|
Colon,
|
|
/// Token kind for a comma `,`.
|
|
Comma,
|
|
/// Token kind for a semicolon `;`.
|
|
Semi,
|
|
/// Token kind for plus `+`.
|
|
Plus,
|
|
/// Token kind for minus `-`.
|
|
Minus,
|
|
/// Token kind for star `*`.
|
|
Star,
|
|
/// Token kind for slash `/`.
|
|
Slash,
|
|
/// Token kind for vertical bar `|`.
|
|
Vbar,
|
|
/// Token kind for ampersand `&`.
|
|
Amper,
|
|
/// Token kind for less than `<`.
|
|
Less,
|
|
/// Token kind for greater than `>`.
|
|
Greater,
|
|
/// Token kind for equal `=`.
|
|
Equal,
|
|
/// Token kind for dot `.`.
|
|
Dot,
|
|
/// Token kind for percent `%`.
|
|
Percent,
|
|
/// Token kind for left bracket `{`.
|
|
Lbrace,
|
|
/// Token kind for right bracket `}`.
|
|
Rbrace,
|
|
/// Token kind for double equal `==`.
|
|
EqEqual,
|
|
/// Token kind for not equal `!=`.
|
|
NotEqual,
|
|
/// Token kind for less than or equal `<=`.
|
|
LessEqual,
|
|
/// Token kind for greater than or equal `>=`.
|
|
GreaterEqual,
|
|
/// Token kind for tilde `~`.
|
|
Tilde,
|
|
/// Token kind for caret `^`.
|
|
CircumFlex,
|
|
/// Token kind for left shift `<<`.
|
|
LeftShift,
|
|
/// Token kind for right shift `>>`.
|
|
RightShift,
|
|
/// Token kind for double star `**`.
|
|
DoubleStar,
|
|
/// Token kind for double star equal `**=`.
|
|
DoubleStarEqual,
|
|
/// Token kind for plus equal `+=`.
|
|
PlusEqual,
|
|
/// Token kind for minus equal `-=`.
|
|
MinusEqual,
|
|
/// Token kind for star equal `*=`.
|
|
StarEqual,
|
|
/// Token kind for slash equal `/=`.
|
|
SlashEqual,
|
|
/// Token kind for percent equal `%=`.
|
|
PercentEqual,
|
|
/// Token kind for ampersand equal `&=`.
|
|
AmperEqual,
|
|
/// Token kind for vertical bar equal `|=`.
|
|
VbarEqual,
|
|
/// Token kind for caret equal `^=`.
|
|
CircumflexEqual,
|
|
/// Token kind for left shift equal `<<=`.
|
|
LeftShiftEqual,
|
|
/// Token kind for right shift equal `>>=`.
|
|
RightShiftEqual,
|
|
/// Token kind for double slash `//`.
|
|
DoubleSlash,
|
|
/// Token kind for double slash equal `//=`.
|
|
DoubleSlashEqual,
|
|
/// Token kind for colon equal `:=`.
|
|
ColonEqual,
|
|
/// Token kind for at `@`.
|
|
At,
|
|
/// Token kind for at equal `@=`.
|
|
AtEqual,
|
|
/// Token kind for arrow `->`.
|
|
Rarrow,
|
|
/// Token kind for ellipsis `...`.
|
|
Ellipsis,
|
|
|
|
// The keywords should be sorted in alphabetical order. If the boundary tokens for the
|
|
// "Keywords" and "Soft keywords" group change, update the related methods on `TokenKind`.
|
|
|
|
// Keywords
|
|
And,
|
|
As,
|
|
Assert,
|
|
Async,
|
|
Await,
|
|
Break,
|
|
Class,
|
|
Continue,
|
|
Def,
|
|
Del,
|
|
Elif,
|
|
Else,
|
|
Except,
|
|
False,
|
|
Finally,
|
|
For,
|
|
From,
|
|
Global,
|
|
If,
|
|
Import,
|
|
In,
|
|
Is,
|
|
Lambda,
|
|
None,
|
|
Nonlocal,
|
|
Not,
|
|
Or,
|
|
Pass,
|
|
Raise,
|
|
Return,
|
|
True,
|
|
Try,
|
|
While,
|
|
With,
|
|
Yield,
|
|
|
|
// Soft keywords
|
|
Case,
|
|
Match,
|
|
Type,
|
|
|
|
Unknown,
|
|
}
|
|
|
|
impl TokenKind {
|
|
/// Returns `true` if this is an end of file token.
|
|
#[inline]
|
|
pub const fn is_eof(self) -> bool {
|
|
matches!(self, TokenKind::EndOfFile)
|
|
}
|
|
|
|
/// Returns `true` if this is either a newline or non-logical newline token.
|
|
#[inline]
|
|
pub const fn is_any_newline(self) -> bool {
|
|
matches!(self, TokenKind::Newline | TokenKind::NonLogicalNewline)
|
|
}
|
|
|
|
/// Returns `true` if the token is a keyword (including soft keywords).
|
|
///
|
|
/// See also [`is_soft_keyword`], [`is_non_soft_keyword`].
|
|
///
|
|
/// [`is_soft_keyword`]: TokenKind::is_soft_keyword
|
|
/// [`is_non_soft_keyword`]: TokenKind::is_non_soft_keyword
|
|
#[inline]
|
|
pub fn is_keyword(self) -> bool {
|
|
TokenKind::And <= self && self <= TokenKind::Type
|
|
}
|
|
|
|
/// Returns `true` if the token is strictly a soft keyword.
|
|
///
|
|
/// See also [`is_keyword`], [`is_non_soft_keyword`].
|
|
///
|
|
/// [`is_keyword`]: TokenKind::is_keyword
|
|
/// [`is_non_soft_keyword`]: TokenKind::is_non_soft_keyword
|
|
#[inline]
|
|
pub fn is_soft_keyword(self) -> bool {
|
|
TokenKind::Case <= self && self <= TokenKind::Type
|
|
}
|
|
|
|
/// Returns `true` if the token is strictly a non-soft keyword.
|
|
///
|
|
/// See also [`is_keyword`], [`is_soft_keyword`].
|
|
///
|
|
/// [`is_keyword`]: TokenKind::is_keyword
|
|
/// [`is_soft_keyword`]: TokenKind::is_soft_keyword
|
|
#[inline]
|
|
pub fn is_non_soft_keyword(self) -> bool {
|
|
TokenKind::And <= self && self <= TokenKind::Yield
|
|
}
|
|
|
|
#[inline]
|
|
pub const fn is_operator(self) -> bool {
|
|
matches!(
|
|
self,
|
|
TokenKind::Lpar
|
|
| TokenKind::Rpar
|
|
| TokenKind::Lsqb
|
|
| TokenKind::Rsqb
|
|
| TokenKind::Comma
|
|
| TokenKind::Semi
|
|
| TokenKind::Plus
|
|
| TokenKind::Minus
|
|
| TokenKind::Star
|
|
| TokenKind::Slash
|
|
| TokenKind::Vbar
|
|
| TokenKind::Amper
|
|
| TokenKind::Less
|
|
| TokenKind::Greater
|
|
| TokenKind::Equal
|
|
| TokenKind::Dot
|
|
| TokenKind::Percent
|
|
| TokenKind::Lbrace
|
|
| TokenKind::Rbrace
|
|
| TokenKind::EqEqual
|
|
| TokenKind::NotEqual
|
|
| TokenKind::LessEqual
|
|
| TokenKind::GreaterEqual
|
|
| TokenKind::Tilde
|
|
| TokenKind::CircumFlex
|
|
| TokenKind::LeftShift
|
|
| TokenKind::RightShift
|
|
| TokenKind::DoubleStar
|
|
| TokenKind::PlusEqual
|
|
| TokenKind::MinusEqual
|
|
| TokenKind::StarEqual
|
|
| TokenKind::SlashEqual
|
|
| TokenKind::PercentEqual
|
|
| TokenKind::AmperEqual
|
|
| TokenKind::VbarEqual
|
|
| TokenKind::CircumflexEqual
|
|
| TokenKind::LeftShiftEqual
|
|
| TokenKind::RightShiftEqual
|
|
| TokenKind::DoubleStarEqual
|
|
| TokenKind::DoubleSlash
|
|
| TokenKind::DoubleSlashEqual
|
|
| TokenKind::At
|
|
| TokenKind::AtEqual
|
|
| TokenKind::Rarrow
|
|
| TokenKind::Ellipsis
|
|
| TokenKind::ColonEqual
|
|
| TokenKind::Colon
|
|
| TokenKind::And
|
|
| TokenKind::Or
|
|
| TokenKind::Not
|
|
| TokenKind::In
|
|
| TokenKind::Is
|
|
)
|
|
}
|
|
|
|
/// Returns `true` if this is a singleton token i.e., `True`, `False`, or `None`.
|
|
#[inline]
|
|
pub const fn is_singleton(self) -> bool {
|
|
matches!(self, TokenKind::False | TokenKind::True | TokenKind::None)
|
|
}
|
|
|
|
/// Returns `true` if this is a trivia token i.e., a comment or a non-logical newline.
|
|
#[inline]
|
|
pub const fn is_trivia(&self) -> bool {
|
|
matches!(self, TokenKind::Comment | TokenKind::NonLogicalNewline)
|
|
}
|
|
|
|
#[inline]
|
|
pub const fn is_arithmetic(self) -> bool {
|
|
matches!(
|
|
self,
|
|
TokenKind::DoubleStar
|
|
| TokenKind::Star
|
|
| TokenKind::Plus
|
|
| TokenKind::Minus
|
|
| TokenKind::Slash
|
|
| TokenKind::DoubleSlash
|
|
| TokenKind::At
|
|
)
|
|
}
|
|
|
|
#[inline]
|
|
pub const fn is_bitwise_or_shift(self) -> bool {
|
|
matches!(
|
|
self,
|
|
TokenKind::LeftShift
|
|
| TokenKind::LeftShiftEqual
|
|
| TokenKind::RightShift
|
|
| TokenKind::RightShiftEqual
|
|
| TokenKind::Amper
|
|
| TokenKind::AmperEqual
|
|
| TokenKind::Vbar
|
|
| TokenKind::VbarEqual
|
|
| TokenKind::CircumFlex
|
|
| TokenKind::CircumflexEqual
|
|
| TokenKind::Tilde
|
|
)
|
|
}
|
|
|
|
/// Returns `true` if the current token is a unary arithmetic operator.
|
|
#[inline]
|
|
pub const fn is_unary_arithmetic_operator(self) -> bool {
|
|
matches!(self, TokenKind::Plus | TokenKind::Minus)
|
|
}
|
|
|
|
/// Returns the [`UnaryOp`] that corresponds to this token kind, if it is a unary arithmetic
|
|
/// operator, otherwise return [None].
|
|
///
|
|
/// Use [`as_unary_operator`] to match against any unary operator.
|
|
///
|
|
/// [`as_unary_operator`]: TokenKind::as_unary_operator
|
|
#[inline]
|
|
pub(crate) const fn as_unary_arithmetic_operator(self) -> Option<UnaryOp> {
|
|
Some(match self {
|
|
TokenKind::Plus => UnaryOp::UAdd,
|
|
TokenKind::Minus => UnaryOp::USub,
|
|
_ => return None,
|
|
})
|
|
}
|
|
|
|
/// Returns the [`UnaryOp`] that corresponds to this token kind, if it is a unary operator,
|
|
/// otherwise return [None].
|
|
///
|
|
/// Use [`as_unary_arithmetic_operator`] to match against only an arithmetic unary operator.
|
|
///
|
|
/// [`as_unary_arithmetic_operator`]: TokenKind::as_unary_arithmetic_operator
|
|
#[inline]
|
|
pub(crate) const fn as_unary_operator(self) -> Option<UnaryOp> {
|
|
Some(match self {
|
|
TokenKind::Plus => UnaryOp::UAdd,
|
|
TokenKind::Minus => UnaryOp::USub,
|
|
TokenKind::Tilde => UnaryOp::Invert,
|
|
TokenKind::Not => UnaryOp::Not,
|
|
_ => return None,
|
|
})
|
|
}
|
|
|
|
/// Returns the [`BoolOp`] that corresponds to this token kind, if it is a boolean operator,
|
|
/// otherwise return [None].
|
|
#[inline]
|
|
pub(crate) const fn as_bool_operator(self) -> Option<BoolOp> {
|
|
Some(match self {
|
|
TokenKind::And => BoolOp::And,
|
|
TokenKind::Or => BoolOp::Or,
|
|
_ => return None,
|
|
})
|
|
}
|
|
|
|
/// Returns the binary [`Operator`] that corresponds to the current token, if it's a binary
|
|
/// operator, otherwise return [None].
|
|
///
|
|
/// Use [`as_augmented_assign_operator`] to match against an augmented assignment token.
|
|
///
|
|
/// [`as_augmented_assign_operator`]: TokenKind::as_augmented_assign_operator
|
|
pub(crate) const fn as_binary_operator(self) -> Option<Operator> {
|
|
Some(match self {
|
|
TokenKind::Plus => Operator::Add,
|
|
TokenKind::Minus => Operator::Sub,
|
|
TokenKind::Star => Operator::Mult,
|
|
TokenKind::At => Operator::MatMult,
|
|
TokenKind::DoubleStar => Operator::Pow,
|
|
TokenKind::Slash => Operator::Div,
|
|
TokenKind::DoubleSlash => Operator::FloorDiv,
|
|
TokenKind::Percent => Operator::Mod,
|
|
TokenKind::Amper => Operator::BitAnd,
|
|
TokenKind::Vbar => Operator::BitOr,
|
|
TokenKind::CircumFlex => Operator::BitXor,
|
|
TokenKind::LeftShift => Operator::LShift,
|
|
TokenKind::RightShift => Operator::RShift,
|
|
_ => return None,
|
|
})
|
|
}
|
|
|
|
/// Returns the [`Operator`] that corresponds to this token kind, if it is
|
|
/// an augmented assignment operator, or [`None`] otherwise.
|
|
#[inline]
|
|
pub(crate) const fn as_augmented_assign_operator(self) -> Option<Operator> {
|
|
Some(match self {
|
|
TokenKind::PlusEqual => Operator::Add,
|
|
TokenKind::MinusEqual => Operator::Sub,
|
|
TokenKind::StarEqual => Operator::Mult,
|
|
TokenKind::AtEqual => Operator::MatMult,
|
|
TokenKind::DoubleStarEqual => Operator::Pow,
|
|
TokenKind::SlashEqual => Operator::Div,
|
|
TokenKind::DoubleSlashEqual => Operator::FloorDiv,
|
|
TokenKind::PercentEqual => Operator::Mod,
|
|
TokenKind::AmperEqual => Operator::BitAnd,
|
|
TokenKind::VbarEqual => Operator::BitOr,
|
|
TokenKind::CircumflexEqual => Operator::BitXor,
|
|
TokenKind::LeftShiftEqual => Operator::LShift,
|
|
TokenKind::RightShiftEqual => Operator::RShift,
|
|
_ => return None,
|
|
})
|
|
}
|
|
}
|
|
|
|
impl From<BoolOp> for TokenKind {
|
|
#[inline]
|
|
fn from(op: BoolOp) -> Self {
|
|
match op {
|
|
BoolOp::And => TokenKind::And,
|
|
BoolOp::Or => TokenKind::Or,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl From<UnaryOp> for TokenKind {
|
|
#[inline]
|
|
fn from(op: UnaryOp) -> Self {
|
|
match op {
|
|
UnaryOp::Invert => TokenKind::Tilde,
|
|
UnaryOp::Not => TokenKind::Not,
|
|
UnaryOp::UAdd => TokenKind::Plus,
|
|
UnaryOp::USub => TokenKind::Minus,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl From<Operator> for TokenKind {
|
|
#[inline]
|
|
fn from(op: Operator) -> Self {
|
|
match op {
|
|
Operator::Add => TokenKind::Plus,
|
|
Operator::Sub => TokenKind::Minus,
|
|
Operator::Mult => TokenKind::Star,
|
|
Operator::MatMult => TokenKind::At,
|
|
Operator::Div => TokenKind::Slash,
|
|
Operator::Mod => TokenKind::Percent,
|
|
Operator::Pow => TokenKind::DoubleStar,
|
|
Operator::LShift => TokenKind::LeftShift,
|
|
Operator::RShift => TokenKind::RightShift,
|
|
Operator::BitOr => TokenKind::Vbar,
|
|
Operator::BitXor => TokenKind::CircumFlex,
|
|
Operator::BitAnd => TokenKind::Amper,
|
|
Operator::FloorDiv => TokenKind::DoubleSlash,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl fmt::Display for TokenKind {
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
let value = match self {
|
|
TokenKind::Unknown => "Unknown",
|
|
TokenKind::Newline => "newline",
|
|
TokenKind::NonLogicalNewline => "NonLogicalNewline",
|
|
TokenKind::Indent => "indent",
|
|
TokenKind::Dedent => "dedent",
|
|
TokenKind::EndOfFile => "end of file",
|
|
TokenKind::Name => "name",
|
|
TokenKind::Int => "int",
|
|
TokenKind::Float => "float",
|
|
TokenKind::Complex => "complex",
|
|
TokenKind::String => "string",
|
|
TokenKind::FStringStart => "FStringStart",
|
|
TokenKind::FStringMiddle => "FStringMiddle",
|
|
TokenKind::FStringEnd => "FStringEnd",
|
|
TokenKind::IpyEscapeCommand => "IPython escape command",
|
|
TokenKind::Comment => "comment",
|
|
TokenKind::Question => "'?'",
|
|
TokenKind::Exclamation => "'!'",
|
|
TokenKind::Lpar => "'('",
|
|
TokenKind::Rpar => "')'",
|
|
TokenKind::Lsqb => "'['",
|
|
TokenKind::Rsqb => "']'",
|
|
TokenKind::Lbrace => "'{'",
|
|
TokenKind::Rbrace => "'}'",
|
|
TokenKind::Equal => "'='",
|
|
TokenKind::ColonEqual => "':='",
|
|
TokenKind::Dot => "'.'",
|
|
TokenKind::Colon => "':'",
|
|
TokenKind::Semi => "';'",
|
|
TokenKind::Comma => "','",
|
|
TokenKind::Rarrow => "'->'",
|
|
TokenKind::Plus => "'+'",
|
|
TokenKind::Minus => "'-'",
|
|
TokenKind::Star => "'*'",
|
|
TokenKind::DoubleStar => "'**'",
|
|
TokenKind::Slash => "'/'",
|
|
TokenKind::DoubleSlash => "'//'",
|
|
TokenKind::Percent => "'%'",
|
|
TokenKind::Vbar => "'|'",
|
|
TokenKind::Amper => "'&'",
|
|
TokenKind::CircumFlex => "'^'",
|
|
TokenKind::LeftShift => "'<<'",
|
|
TokenKind::RightShift => "'>>'",
|
|
TokenKind::Tilde => "'~'",
|
|
TokenKind::At => "'@'",
|
|
TokenKind::Less => "'<'",
|
|
TokenKind::Greater => "'>'",
|
|
TokenKind::EqEqual => "'=='",
|
|
TokenKind::NotEqual => "'!='",
|
|
TokenKind::LessEqual => "'<='",
|
|
TokenKind::GreaterEqual => "'>='",
|
|
TokenKind::PlusEqual => "'+='",
|
|
TokenKind::MinusEqual => "'-='",
|
|
TokenKind::StarEqual => "'*='",
|
|
TokenKind::DoubleStarEqual => "'**='",
|
|
TokenKind::SlashEqual => "'/='",
|
|
TokenKind::DoubleSlashEqual => "'//='",
|
|
TokenKind::PercentEqual => "'%='",
|
|
TokenKind::VbarEqual => "'|='",
|
|
TokenKind::AmperEqual => "'&='",
|
|
TokenKind::CircumflexEqual => "'^='",
|
|
TokenKind::LeftShiftEqual => "'<<='",
|
|
TokenKind::RightShiftEqual => "'>>='",
|
|
TokenKind::AtEqual => "'@='",
|
|
TokenKind::Ellipsis => "'...'",
|
|
TokenKind::False => "'False'",
|
|
TokenKind::None => "'None'",
|
|
TokenKind::True => "'True'",
|
|
TokenKind::And => "'and'",
|
|
TokenKind::As => "'as'",
|
|
TokenKind::Assert => "'assert'",
|
|
TokenKind::Async => "'async'",
|
|
TokenKind::Await => "'await'",
|
|
TokenKind::Break => "'break'",
|
|
TokenKind::Class => "'class'",
|
|
TokenKind::Continue => "'continue'",
|
|
TokenKind::Def => "'def'",
|
|
TokenKind::Del => "'del'",
|
|
TokenKind::Elif => "'elif'",
|
|
TokenKind::Else => "'else'",
|
|
TokenKind::Except => "'except'",
|
|
TokenKind::Finally => "'finally'",
|
|
TokenKind::For => "'for'",
|
|
TokenKind::From => "'from'",
|
|
TokenKind::Global => "'global'",
|
|
TokenKind::If => "'if'",
|
|
TokenKind::Import => "'import'",
|
|
TokenKind::In => "'in'",
|
|
TokenKind::Is => "'is'",
|
|
TokenKind::Lambda => "'lambda'",
|
|
TokenKind::Nonlocal => "'nonlocal'",
|
|
TokenKind::Not => "'not'",
|
|
TokenKind::Or => "'or'",
|
|
TokenKind::Pass => "'pass'",
|
|
TokenKind::Raise => "'raise'",
|
|
TokenKind::Return => "'return'",
|
|
TokenKind::Try => "'try'",
|
|
TokenKind::While => "'while'",
|
|
TokenKind::Match => "'match'",
|
|
TokenKind::Type => "'type'",
|
|
TokenKind::Case => "'case'",
|
|
TokenKind::With => "'with'",
|
|
TokenKind::Yield => "'yield'",
|
|
};
|
|
f.write_str(value)
|
|
}
|
|
}
|
|
|
|
bitflags! {
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
|
|
pub(crate) struct TokenFlags: u8 {
|
|
/// The token is a string with double quotes (`"`).
|
|
const DOUBLE_QUOTES = 1 << 0;
|
|
/// The token is a triple-quoted string i.e., it starts and ends with three consecutive
|
|
/// quote characters (`"""` or `'''`).
|
|
const TRIPLE_QUOTED_STRING = 1 << 1;
|
|
|
|
/// The token is a unicode string i.e., prefixed with `u` or `U`
|
|
const UNICODE_STRING = 1 << 2;
|
|
/// The token is a byte string i.e., prefixed with `b` or `B`
|
|
const BYTE_STRING = 1 << 3;
|
|
/// The token is an f-string i.e., prefixed with `f` or `F`
|
|
const F_STRING = 1 << 4;
|
|
/// The token is a raw string and the prefix character is in lowercase.
|
|
const RAW_STRING_LOWERCASE = 1 << 5;
|
|
/// The token is a raw string and the prefix character is in uppercase.
|
|
const RAW_STRING_UPPERCASE = 1 << 6;
|
|
|
|
/// The token is a raw string i.e., prefixed with `r` or `R`
|
|
const RAW_STRING = Self::RAW_STRING_LOWERCASE.bits() | Self::RAW_STRING_UPPERCASE.bits();
|
|
}
|
|
}
|
|
|
|
impl StringFlags for TokenFlags {
|
|
fn quote_style(self) -> Quote {
|
|
if self.intersects(TokenFlags::DOUBLE_QUOTES) {
|
|
Quote::Double
|
|
} else {
|
|
Quote::Single
|
|
}
|
|
}
|
|
|
|
fn is_triple_quoted(self) -> bool {
|
|
self.intersects(TokenFlags::TRIPLE_QUOTED_STRING)
|
|
}
|
|
|
|
fn prefix(self) -> AnyStringPrefix {
|
|
if self.intersects(TokenFlags::F_STRING) {
|
|
if self.intersects(TokenFlags::RAW_STRING_LOWERCASE) {
|
|
AnyStringPrefix::Format(FStringPrefix::Raw { uppercase_r: false })
|
|
} else if self.intersects(TokenFlags::RAW_STRING_UPPERCASE) {
|
|
AnyStringPrefix::Format(FStringPrefix::Raw { uppercase_r: true })
|
|
} else {
|
|
AnyStringPrefix::Format(FStringPrefix::Regular)
|
|
}
|
|
} else if self.intersects(TokenFlags::BYTE_STRING) {
|
|
if self.intersects(TokenFlags::RAW_STRING_LOWERCASE) {
|
|
AnyStringPrefix::Bytes(ByteStringPrefix::Raw { uppercase_r: false })
|
|
} else if self.intersects(TokenFlags::RAW_STRING_UPPERCASE) {
|
|
AnyStringPrefix::Bytes(ByteStringPrefix::Raw { uppercase_r: true })
|
|
} else {
|
|
AnyStringPrefix::Bytes(ByteStringPrefix::Regular)
|
|
}
|
|
} else if self.intersects(TokenFlags::RAW_STRING_LOWERCASE) {
|
|
AnyStringPrefix::Regular(StringLiteralPrefix::Raw { uppercase: false })
|
|
} else if self.intersects(TokenFlags::RAW_STRING_UPPERCASE) {
|
|
AnyStringPrefix::Regular(StringLiteralPrefix::Raw { uppercase: true })
|
|
} else if self.intersects(TokenFlags::UNICODE_STRING) {
|
|
AnyStringPrefix::Regular(StringLiteralPrefix::Unicode)
|
|
} else {
|
|
AnyStringPrefix::Regular(StringLiteralPrefix::Empty)
|
|
}
|
|
}
|
|
}
|
|
|
|
impl TokenFlags {
|
|
/// Returns `true` if the token is an f-string.
|
|
pub(crate) const fn is_f_string(self) -> bool {
|
|
self.intersects(TokenFlags::F_STRING)
|
|
}
|
|
|
|
/// Returns `true` if the token is a triple-quoted f-string.
|
|
pub(crate) fn is_triple_quoted_fstring(self) -> bool {
|
|
self.contains(TokenFlags::F_STRING | TokenFlags::TRIPLE_QUOTED_STRING)
|
|
}
|
|
|
|
/// Returns `true` if the token is a raw string.
|
|
pub(crate) const fn is_raw_string(self) -> bool {
|
|
self.intersects(TokenFlags::RAW_STRING)
|
|
}
|
|
|
|
/// Converts this type to [`AnyStringFlags`], setting the equivalent flags.
|
|
pub(crate) fn as_any_string_flags(self) -> AnyStringFlags {
|
|
AnyStringFlags::new(self.prefix(), self.quote_style(), self.is_triple_quoted())
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Debug, Default)]
|
|
pub(crate) enum TokenValue {
|
|
#[default]
|
|
None,
|
|
/// Token value for a name, commonly known as an identifier.
|
|
///
|
|
/// Unicode names are NFKC-normalized by the lexer,
|
|
/// matching [the behaviour of Python's lexer](https://docs.python.org/3/reference/lexical_analysis.html#identifiers)
|
|
Name(Name),
|
|
/// Token value for an integer.
|
|
Int(Int),
|
|
/// Token value for a floating point number.
|
|
Float(f64),
|
|
/// Token value for a complex number.
|
|
Complex {
|
|
/// The real part of the complex number.
|
|
real: f64,
|
|
/// The imaginary part of the complex number.
|
|
imag: f64,
|
|
},
|
|
/// Token value for a string.
|
|
String(Box<str>),
|
|
/// Token value that includes the portion of text inside the f-string that's not
|
|
/// part of the expression part and isn't an opening or closing brace.
|
|
FStringMiddle(Box<str>),
|
|
/// Token value for IPython escape commands. These are recognized by the lexer
|
|
/// only when the mode is [`Mode::Ipython`].
|
|
IpyEscapeCommand {
|
|
/// The magic command value.
|
|
value: Box<str>,
|
|
/// The kind of magic command.
|
|
kind: IpyEscapeKind,
|
|
},
|
|
}
|