mirror of
https://github.com/astral-sh/ruff.git
synced 2025-09-29 05:15:12 +00:00
184 lines
5.6 KiB
Rust
184 lines
5.6 KiB
Rust
use crate::{nodes, Expr};
|
|
use smallvec::{smallvec, SmallVec};
|
|
|
|
/// A representation of a qualified name, like `typing.List`.
|
|
pub type CallPath<'a> = SmallVec<[&'a str; 8]>;
|
|
|
|
/// Convert an `Expr` to its [`CallPath`] segments (like `["typing", "List"]`).
|
|
pub fn collect_call_path(expr: &Expr) -> Option<CallPath> {
|
|
// Unroll the loop up to eight times, to match the maximum number of expected attributes.
|
|
// In practice, unrolling appears to give about a 4x speed-up on this hot path.
|
|
let attr1 = match expr {
|
|
Expr::Attribute(attr1) => attr1,
|
|
// Ex) `foo`
|
|
Expr::Name(nodes::ExprName { id, .. }) => {
|
|
return Some(CallPath::from_slice(&[id.as_str()]))
|
|
}
|
|
_ => return None,
|
|
};
|
|
|
|
let attr2 = match attr1.value.as_ref() {
|
|
Expr::Attribute(attr2) => attr2,
|
|
// Ex) `foo.bar`
|
|
Expr::Name(nodes::ExprName { id, .. }) => {
|
|
return Some(CallPath::from_slice(&[id.as_str(), attr1.attr.as_str()]))
|
|
}
|
|
_ => return None,
|
|
};
|
|
|
|
let attr3 = match attr2.value.as_ref() {
|
|
Expr::Attribute(attr3) => attr3,
|
|
// Ex) `foo.bar.baz`
|
|
Expr::Name(nodes::ExprName { id, .. }) => {
|
|
return Some(CallPath::from_slice(&[
|
|
id.as_str(),
|
|
attr2.attr.as_str(),
|
|
attr1.attr.as_str(),
|
|
]));
|
|
}
|
|
_ => return None,
|
|
};
|
|
|
|
let attr4 = match attr3.value.as_ref() {
|
|
Expr::Attribute(attr4) => attr4,
|
|
// Ex) `foo.bar.baz.bop`
|
|
Expr::Name(nodes::ExprName { id, .. }) => {
|
|
return Some(CallPath::from_slice(&[
|
|
id.as_str(),
|
|
attr3.attr.as_str(),
|
|
attr2.attr.as_str(),
|
|
attr1.attr.as_str(),
|
|
]));
|
|
}
|
|
_ => return None,
|
|
};
|
|
|
|
let attr5 = match attr4.value.as_ref() {
|
|
Expr::Attribute(attr5) => attr5,
|
|
// Ex) `foo.bar.baz.bop.bap`
|
|
Expr::Name(nodes::ExprName { id, .. }) => {
|
|
return Some(CallPath::from_slice(&[
|
|
id.as_str(),
|
|
attr4.attr.as_str(),
|
|
attr3.attr.as_str(),
|
|
attr2.attr.as_str(),
|
|
attr1.attr.as_str(),
|
|
]));
|
|
}
|
|
_ => return None,
|
|
};
|
|
|
|
let attr6 = match attr5.value.as_ref() {
|
|
Expr::Attribute(attr6) => attr6,
|
|
// Ex) `foo.bar.baz.bop.bap.bab`
|
|
Expr::Name(nodes::ExprName { id, .. }) => {
|
|
return Some(CallPath::from_slice(&[
|
|
id.as_str(),
|
|
attr5.attr.as_str(),
|
|
attr4.attr.as_str(),
|
|
attr3.attr.as_str(),
|
|
attr2.attr.as_str(),
|
|
attr1.attr.as_str(),
|
|
]));
|
|
}
|
|
_ => return None,
|
|
};
|
|
|
|
let attr7 = match attr6.value.as_ref() {
|
|
Expr::Attribute(attr7) => attr7,
|
|
// Ex) `foo.bar.baz.bop.bap.bab.bob`
|
|
Expr::Name(nodes::ExprName { id, .. }) => {
|
|
return Some(CallPath::from_slice(&[
|
|
id.as_str(),
|
|
attr6.attr.as_str(),
|
|
attr5.attr.as_str(),
|
|
attr4.attr.as_str(),
|
|
attr3.attr.as_str(),
|
|
attr2.attr.as_str(),
|
|
attr1.attr.as_str(),
|
|
]));
|
|
}
|
|
_ => return None,
|
|
};
|
|
|
|
let attr8 = match attr7.value.as_ref() {
|
|
Expr::Attribute(attr8) => attr8,
|
|
// Ex) `foo.bar.baz.bop.bap.bab.bob.bib`
|
|
Expr::Name(nodes::ExprName { id, .. }) => {
|
|
return Some(CallPath::from_slice(&[
|
|
id.as_str(),
|
|
attr7.attr.as_str(),
|
|
attr6.attr.as_str(),
|
|
attr5.attr.as_str(),
|
|
attr4.attr.as_str(),
|
|
attr3.attr.as_str(),
|
|
attr2.attr.as_str(),
|
|
attr1.attr.as_str(),
|
|
]));
|
|
}
|
|
_ => return None,
|
|
};
|
|
|
|
collect_call_path(&attr8.value).map(|mut segments| {
|
|
segments.extend([
|
|
attr8.attr.as_str(),
|
|
attr7.attr.as_str(),
|
|
attr6.attr.as_str(),
|
|
attr5.attr.as_str(),
|
|
attr4.attr.as_str(),
|
|
attr3.attr.as_str(),
|
|
attr2.attr.as_str(),
|
|
attr1.attr.as_str(),
|
|
]);
|
|
segments
|
|
})
|
|
}
|
|
|
|
/// Convert an `Expr` to its call path (like `List`, or `typing.List`).
|
|
pub fn compose_call_path(expr: &Expr) -> Option<String> {
|
|
collect_call_path(expr).map(|call_path| format_call_path(&call_path))
|
|
}
|
|
|
|
/// Format a call path for display.
|
|
pub fn format_call_path(call_path: &[&str]) -> String {
|
|
if call_path
|
|
.first()
|
|
.expect("Unable to format empty call path")
|
|
.is_empty()
|
|
{
|
|
call_path[1..].join(".")
|
|
} else {
|
|
call_path.join(".")
|
|
}
|
|
}
|
|
|
|
/// Create a [`CallPath`] from an unqualified name.
|
|
///
|
|
/// ```rust
|
|
/// # use smallvec::smallvec;
|
|
/// # use ruff_python_ast::call_path::from_unqualified_name;
|
|
///
|
|
/// assert_eq!(from_unqualified_name("typing.List").as_slice(), ["typing", "List"]);
|
|
/// assert_eq!(from_unqualified_name("list").as_slice(), ["list"]);
|
|
/// ```
|
|
pub fn from_unqualified_name(name: &str) -> CallPath {
|
|
name.split('.').collect()
|
|
}
|
|
|
|
/// Create a [`CallPath`] from a fully-qualified name.
|
|
///
|
|
/// ```rust
|
|
/// # use smallvec::smallvec;
|
|
/// # use ruff_python_ast::call_path::from_qualified_name;
|
|
///
|
|
/// assert_eq!(from_qualified_name("typing.List").as_slice(), ["typing", "List"]);
|
|
/// assert_eq!(from_qualified_name("list").as_slice(), ["", "list"]);
|
|
/// ```
|
|
pub fn from_qualified_name(name: &str) -> CallPath {
|
|
if name.contains('.') {
|
|
name.split('.').collect()
|
|
} else {
|
|
// Special-case: for builtins, return `["", "int"]` instead of `["int"]`.
|
|
smallvec!["", name]
|
|
}
|
|
}
|