diff --git a/crates/rome_js_semantic/src/semantic_model/model.rs b/crates/rome_js_semantic/src/semantic_model/model.rs new file mode 100644 index 000000000000..93dc1e593d12 --- /dev/null +++ b/crates/rome_js_semantic/src/semantic_model/model.rs @@ -0,0 +1,340 @@ +use super::*; +use rome_js_syntax::JsAnyRoot; + +#[derive(Copy, Clone, Debug)] +pub(crate) struct BindingIndex(usize); + +impl From for BindingIndex { + fn from(v: usize) -> Self { + BindingIndex(v) + } +} + +#[derive(Copy, Clone, Debug)] +pub(crate) struct ReferenceIndex(usize, usize); + +impl ReferenceIndex { + pub(crate) fn binding(&self) -> BindingIndex { + BindingIndex(self.0) + } +} + +impl From<(BindingIndex, usize)> for ReferenceIndex { + fn from((binding_index, index): (BindingIndex, usize)) -> Self { + ReferenceIndex(binding_index.0, index) + } +} + +/// Contains all the data of the [SemanticModel] and only lives behind an [Arc]. +/// +/// That allows any returned struct (like [Scope], [Binding]) +/// to outlive the [SemanticModel], and to not include lifetimes. +#[derive(Debug)] +pub(crate) struct SemanticModelData { + pub(crate) root: JsAnyRoot, + // All scopes of this model + pub(crate) scopes: Vec, + pub(crate) scope_by_range: rust_lapper::Lapper, + // Maps the start of a node range to a scope id + pub(crate) scope_hoisted_to_by_range: FxHashMap, + // Map to each by its range + pub(crate) node_by_range: FxHashMap, + // Maps any range in the code to its bindings (usize points to bindings vec) + pub(crate) declared_at_by_range: FxHashMap, + // List of all the declarations + pub(crate) bindings: Vec, + // Index bindings by range + pub(crate) bindings_by_range: FxHashMap, + // All bindings that were exported + pub(crate) exported: FxHashSet, + /// All references that could not be resolved + pub(crate) unresolved_references: Vec, + /// All globals references + pub(crate) globals: Vec, +} + +impl SemanticModelData { + pub(crate) fn binding(&self, index: BindingIndex) -> &SemanticModelBindingData { + &self.bindings[index.0] + } + + pub(crate) fn reference(&self, index: ReferenceIndex) -> &SemanticModelReference { + let binding = &self.bindings[index.0]; + &binding.references[index.1] + } + + pub(crate) fn next_reference(&self, index: ReferenceIndex) -> Option<&SemanticModelReference> { + let binding = &self.bindings[index.0]; + binding.references.get(index.1 + 1) + } + + pub(crate) fn scope(&self, range: &TextRange) -> usize { + let start = range.start().into(); + let end = range.end().into(); + let scopes = self + .scope_by_range + .find(start, end) + .filter(|x| !(start < x.start || end > x.stop)); + + // We always want the most tight scope + match scopes.map(|x| x.val).max() { + Some(val) => val, + // We always have at least one scope, the global one. + None => unreachable!("Expected global scope not present"), + } + } + + fn scope_hoisted_to(&self, range: &TextRange) -> Option { + self.scope_hoisted_to_by_range.get(&range.start()).cloned() + } + + pub fn is_exported(&self, range: TextRange) -> bool { + self.exported.contains(&range) + } +} + +impl PartialEq for SemanticModelData { + fn eq(&self, other: &Self) -> bool { + self.root == other.root + } +} + +impl Eq for SemanticModelData {} + +/// The façade for all semantic information. +/// - Scope: [scope] +/// - Declarations: [declaration] +/// +/// See [SemanticModelData] for more information about the internals. +#[derive(Clone)] +pub struct SemanticModel { + pub(crate) data: Arc, +} + +impl SemanticModel { + pub(crate) fn new(data: SemanticModelData) -> Self { + Self { + data: Arc::new(data), + } + } + + /// Iterate all scopes + pub fn scopes(&self) -> impl Iterator + '_ { + self.data.scopes.iter().enumerate().map(|(id, _)| Scope { + data: self.data.clone(), + id, + }) + } + + /// Returns the global scope of the model + pub fn global_scope(&self) -> Scope { + Scope { + data: self.data.clone(), + id: 0, + } + } + + /// Returns the [Scope] which the syntax is part of. + /// Can also be called from [AstNode]::scope extension method. + /// + /// ```rust + /// use rome_rowan::{AstNode, SyntaxNodeCast}; + /// use rome_js_syntax::{SourceType, JsReferenceIdentifier}; + /// use rome_js_semantic::{semantic_model, SemanticModelOptions, SemanticScopeExtensions}; + /// use rome_diagnostics::file::FileId; + /// + /// let r = rome_js_parser::parse("function f(){let a = arguments[0]; let b = a + 1;}", FileId::zero(), SourceType::js_module()); + /// let model = semantic_model(&r.tree(), SemanticModelOptions::default()); + /// + /// let arguments_reference = r + /// .syntax() + /// .descendants() + /// .filter_map(|x| x.cast::()) + /// .find(|x| x.text() == "arguments") + /// .unwrap(); + /// + /// let block_scope = model.scope(&arguments_reference.syntax()); + /// // or + /// let block_scope = arguments_reference.scope(&model); + /// ``` + pub fn scope(&self, node: &JsSyntaxNode) -> Scope { + let range = node.text_range(); + let id = self.data.scope(&range); + Scope { + data: self.data.clone(), + id, + } + } + + /// Returns the [Scope] which the specified syntax node was hoisted to, if any. + /// Can also be called from [AstNode]::scope_hoisted_to extension method. + pub fn scope_hoisted_to(&self, node: &JsSyntaxNode) -> Option { + let range = node.text_range(); + let id = self.data.scope_hoisted_to(&range)?; + Some(Scope { + data: self.data.clone(), + id, + }) + } + + pub fn all_bindings(&self) -> impl Iterator + '_ { + self.data.bindings.iter().map(|x| Binding { + data: self.data.clone(), + index: x.id, + }) + } + + /// Returns the [Binding] of a reference. + /// Can also be called from "binding" extension method. + /// + /// ```rust + /// use rome_rowan::{AstNode, SyntaxNodeCast}; + /// use rome_js_syntax::{SourceType, JsReferenceIdentifier}; + /// use rome_js_semantic::{semantic_model, BindingExtensions, SemanticModelOptions}; + /// use rome_diagnostics::file::FileId; + /// + /// let r = rome_js_parser::parse("function f(){let a = arguments[0]; let b = a + 1;}", FileId::zero(), SourceType::js_module()); + /// let model = semantic_model(&r.tree(), SemanticModelOptions::default()); + /// + /// let arguments_reference = r + /// .syntax() + /// .descendants() + /// .filter_map(|x| x.cast::()) + /// .find(|x| x.text() == "arguments") + /// .unwrap(); + /// + /// let arguments_binding = model.binding(&arguments_reference); + /// // or + /// let arguments_binding = arguments_reference.binding(&model); + /// ``` + pub fn binding(&self, reference: &impl HasDeclarationAstNode) -> Option { + let reference = reference.node(); + let range = reference.syntax().text_range(); + let id = *self.data.declared_at_by_range.get(&range)?; + Some(Binding { + data: self.data.clone(), + index: id.into(), + }) + } + + /// Returns an iterator of all the globals references in the program + pub fn all_global_references( + &self, + ) -> std::iter::Successors Option> + { + let first = self + .data + .globals + .get(0) + .and_then(|global| global.references.get(0)) + .map(|_| GlobalReference { + data: self.data.clone(), + global_id: 0, + id: 0, + }); + fn succ(current: &GlobalReference) -> Option { + let mut global_id = current.global_id; + let mut id = current.id + 1; + while global_id < current.data.globals.len() { + let reference = current + .data + .globals + .get(global_id) + .and_then(|global| global.references.get(id)) + .map(|_| GlobalReference { + data: current.data.clone(), + global_id, + id, + }); + + match reference { + Some(reference) => return Some(reference), + None => { + global_id += 1; + id = 0; + } + } + } + + None + } + std::iter::successors(first, succ) + } + + /// Returns an iterator of all the unresolved references in the program + pub fn all_unresolved_references( + &self, + ) -> std::iter::Successors< + UnresolvedReference, + fn(&UnresolvedReference) -> Option, + > { + let first = self + .data + .unresolved_references + .get(0) + .map(|_| UnresolvedReference { + data: self.data.clone(), + id: 0, + }); + fn succ(current: &UnresolvedReference) -> Option { + let id = current.id + 1; + current + .data + .unresolved_references + .get(id) + .map(|_| UnresolvedReference { + data: current.data.clone(), + id, + }) + } + std::iter::successors(first, succ) + } + + /// Returns if the node is exported or is a reference to a binding + /// that is exported. + /// + /// When a binding is specified this method returns a bool. + /// + /// When a reference is specified this method returns Option, + /// because there is no guarantee that the corresponding declaration exists. + pub fn is_exported(&self, node: &T) -> T::Result + where + T: CanBeImportedExported, + { + node.is_exported(self) + } + + /// Returns if the node is imported or is a reference to a binding + /// that is imported. + /// + /// When a binding is specified this method returns a bool. + /// + /// When a reference is specified this method returns Option, + /// because there is no guarantee that the corresponding declaration exists. + pub fn is_imported(&self, node: &T) -> T::Result + where + T: CanBeImportedExported, + { + node.is_imported(self) + } + + /// Returns the [Closure] associated with the node. + pub fn closure(&self, node: &impl HasClosureAstNode) -> Closure { + Closure::from_node(self.data.clone(), node) + } + + /// Returns true or false if the expression is constant, which + /// means it does not depend on any other variables. + pub fn is_constant(&self, expr: &JsAnyExpression) -> bool { + is_constant::is_constant(expr) + } + + pub fn as_binding(&self, binding: &impl IsBindingAstNode) -> Binding { + let range = binding.syntax().text_range(); + let id = &self.data.bindings_by_range[&range]; + Binding { + data: self.data.clone(), + index: (*id).into(), + } + } +}