use std::fmt::Debug; use std::marker::PhantomData; use ruff_db::parsed::ParsedModuleRef; use ruff_python_ast::{AnyNodeRef, NodeIndex}; use ruff_python_ast::{AnyRootNodeRef, HasNodeIndex}; use ruff_text_size::Ranged; /// Reference to an AST node. /// /// This type acts as a reference to an AST node within a given module that remains /// stable regardless of whether the AST is garbage collected. As such, accessing a /// node through the [`AstNodeRef`] requires a reference to the current [`ParsedModuleRef`] /// for the module containing the node. /// /// ## Usage in salsa tracked structs /// It's important that [`AstNodeRef`] fields in salsa tracked structs are tracked fields /// (attributed with `#[tracked`]). It prevents that the tracked struct gets a new ID /// every time the AST changes, which in turn, invalidates the result of any query /// that takes said tracked struct as a query argument or returns the tracked struct as part of its result. /// /// For example, marking the [`AstNodeRef`] as tracked on `Expression` /// has the effect that salsa will consider the expression as "unchanged" for as long as it: /// /// * belongs to the same file /// * belongs to the same scope /// * has the same kind /// * was created in the same order /// /// This means that changes to expressions in other scopes don't invalidate the expression's id, giving /// us some form of scope-stable identity for expressions. Only queries accessing the node field /// run on every AST change. All other queries only run when the expression's identity changes. #[derive(Clone)] pub struct AstNodeRef { /// The index of the node in the AST. index: NodeIndex, /// Debug information. #[cfg(debug_assertions)] kind: ruff_python_ast::NodeKind, #[cfg(debug_assertions)] range: ruff_text_size::TextRange, // Note that because the module address is not stored in release builds, `AstNodeRef` // cannot implement `Eq`, as indices are only unique within a given instance of the // AST. #[cfg(debug_assertions)] module_addr: usize, _node: PhantomData, } impl AstNodeRef { pub(crate) fn index(&self) -> NodeIndex { self.index } } impl AstNodeRef where T: HasNodeIndex + Ranged + PartialEq + Debug, for<'ast> AnyNodeRef<'ast>: From<&'ast T>, for<'ast> &'ast T: TryFrom>, { /// Creates a new `AstNodeRef` that references `node`. /// /// This method may panic or produce unspecified results if the provided module is from a /// different file or Salsa revision than the module to which the node belongs. pub(super) fn new(module_ref: &ParsedModuleRef, node: &T) -> Self { let index = node.node_index().load(); debug_assert_eq!(module_ref.get_by_index(index).try_into().ok(), Some(node)); Self { index, #[cfg(debug_assertions)] module_addr: module_ref.module().addr(), #[cfg(debug_assertions)] kind: AnyNodeRef::from(node).kind(), #[cfg(debug_assertions)] range: node.range(), _node: PhantomData, } } /// Returns a reference to the wrapped node. /// /// This method may panic or produce unspecified results if the provided module is from a /// different file or Salsa revision than the module to which the node belongs. pub fn node<'ast>(&self, module_ref: &'ast ParsedModuleRef) -> &'ast T { #[cfg(debug_assertions)] assert_eq!(module_ref.module().addr(), self.module_addr); // The user guarantees that the module is from the same file and Salsa // revision, so the file contents cannot have changed. module_ref .get_by_index(self.index) .try_into() .ok() .expect("AST indices should never change within the same revision") } } #[expect(unsafe_code)] unsafe impl salsa::Update for AstNodeRef { unsafe fn maybe_update(old_pointer: *mut Self, new_value: Self) -> bool { let old_ref = unsafe { &mut (*old_pointer) }; // The equality of an `AstNodeRef` depends on both the module address and the node index, // but the former is not stored in release builds to save memory. As such, AST nodes // are always considered change when the AST is reparsed, which is acceptable because // any change to the AST is likely to invalidate most node indices anyways. *old_ref = new_value; true } } impl get_size2::GetSize for AstNodeRef {} #[allow(clippy::missing_fields_in_debug)] impl Debug for AstNodeRef where T: Debug, for<'ast> &'ast T: TryFrom>, { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { #[cfg(debug_assertions)] { f.debug_struct("AstNodeRef") .field("kind", &self.kind) .field("range", &self.range) .finish() } #[cfg(not(debug_assertions))] { // Unfortunately we have no access to the AST here. f.debug_tuple("AstNodeRef").finish_non_exhaustive() } } }