glua_ls 1.0.27

Language server for Garry's Mod Lua (GLua).
Documentation
use std::collections::HashMap;

use glua_code_analysis::{LuaCompilation, LuaDeclId, SemanticModel};
use glua_parser::{
    LuaAst, LuaAstNode, LuaAstToken, LuaClosureExpr, LuaCommentOwner, LuaDocTagParam, LuaStat,
    LuaTableField,
};
use lsp_types::Uri;

#[allow(clippy::mutable_key_type)]
pub fn rename_decl_references(
    semantic_model: &SemanticModel,
    compilation: &LuaCompilation,
    decl_id: LuaDeclId,
    new_name: String,
    result: &mut HashMap<Uri, HashMap<lsp_types::Range, String>>,
) -> Option<()> {
    let decl = semantic_model
        .get_db()
        .get_decl_index()
        .get_decl(&decl_id)?;
    if decl.is_local() {
        let local_references = semantic_model
            .get_db()
            .get_reference_index()
            .get_decl_references(&decl_id.file_id, &decl_id);
        let document = semantic_model.get_document();
        let uri = document.get_uri();
        if let Some(decl_refs) = local_references {
            for decl_ref in &decl_refs.cells {
                let range = document.to_lsp_range(decl_ref.range)?;
                result
                    .entry(uri.clone())
                    .or_default()
                    .insert(range, new_name.clone());
            }
        }

        let decl_range = get_decl_name_token_lsp_range(semantic_model, decl_id)?;
        result
            .entry(uri)
            .or_default()
            .insert(decl_range, new_name.clone());

        if decl.is_param() {
            rename_doc_param(semantic_model, decl_id, new_name, result);
        }

        return Some(());
    } else {
        let name = decl.get_name();
        let global_references = semantic_model
            .get_db()
            .get_reference_index()
            .get_global_references(name)?;

        let mut semantic_cache = HashMap::new();
        for in_filed_syntax_id in global_references {
            let semantic_model = if let Some(semantic_model) =
                semantic_cache.get_mut(&in_filed_syntax_id.file_id)
            {
                semantic_model
            } else {
                let semantic_model = compilation.get_semantic_model(in_filed_syntax_id.file_id)?;
                semantic_cache.insert(in_filed_syntax_id.file_id, semantic_model);
                semantic_cache.get_mut(&in_filed_syntax_id.file_id)?
            };
            let document = semantic_model.get_document();
            let uri = document.get_uri();
            let range = document.to_lsp_range(in_filed_syntax_id.value.get_range())?;
            result
                .entry(uri)
                .or_default()
                .insert(range, new_name.clone());
        }
    }

    Some(())
}

fn get_decl_name_token_lsp_range(
    semantic_model: &SemanticModel,
    decl_id: LuaDeclId,
) -> Option<lsp_types::Range> {
    let decl = semantic_model
        .get_db()
        .get_decl_index()
        .get_decl(&decl_id)?;
    let document = semantic_model.get_document_by_file_id(decl_id.file_id)?;
    document.to_lsp_range(decl.get_range())
}

#[allow(clippy::mutable_key_type)]
fn rename_doc_param(
    semantic_model: &SemanticModel,
    decl_id: LuaDeclId,
    new_name: String,
    result: &mut HashMap<Uri, HashMap<lsp_types::Range, String>>,
) -> Option<()> {
    let decl = semantic_model
        .get_db()
        .get_decl_index()
        .get_decl(&decl_id)?;
    let name = decl.get_name();
    let syntax_id = decl.get_syntax_id();
    let root = semantic_model.get_root();
    let param_node = LuaAst::cast(syntax_id.to_node_from_root(root.syntax())?)?;
    let closure_expr = param_node.ancestors::<LuaClosureExpr>().next()?;
    let comments = if let Some(table_field) = closure_expr.get_parent::<LuaTableField>() {
        table_field.get_comments()
    } else if let Some(stat) = closure_expr.ancestors::<LuaStat>().next() {
        stat.get_comments()
    } else {
        return None;
    };

    let document = semantic_model.get_document();
    let uri = document.get_uri();
    for comment in comments {
        for tag_doc in comment.get_doc_tags() {
            if let Some(doc_param) = LuaDocTagParam::cast(tag_doc.syntax().clone())
                && let Some(name_token) = doc_param.get_name_token()
            {
                if name_token.get_text() != name {
                    continue;
                }

                let range = document.to_lsp_range(name_token.get_range())?;
                result
                    .entry(uri.clone())
                    .or_default()
                    .insert(range, new_name.clone());
            }
        }
    }

    Some(())
}