glua_ls 1.0.27

Language server for Garry's Mod Lua (GLua).
Documentation
use glua_code_analysis::{DiagnosticCode, FileId, SemanticModel};
use lsp_types::{
    CodeAction, CodeActionKind, CodeActionOrCommand, CodeActionResponse, Diagnostic,
    NumberOrString, Range, WorkspaceEdit,
};

use super::actions::{
    build_add_doc_tag, build_disable_file_changes, build_disable_next_line_changes,
    build_gmod_null_check, build_need_check_nil, build_preferred_local_alias_fix,
};
use crate::handlers::command::{DisableAction, make_disable_code_command};

pub fn build_actions(
    semantic_model: &SemanticModel,
    diagnostics: Vec<Diagnostic>,
) -> Option<CodeActionResponse> {
    let mut actions = Vec::new();
    let file_id = semantic_model.get_file_id();
    for diagnostic in diagnostics {
        if diagnostic.source.is_none() {
            continue;
        }

        let source = diagnostic.source.unwrap();
        if source != "GLuaLS" {
            continue;
        }

        if let Some(code) = diagnostic.code
            && let NumberOrString::String(action_string) = code
        {
            let diagnostic_code = DiagnosticCode::from_name_or_legacy(&action_string);
            add_fix_code_action(
                semantic_model,
                &mut actions,
                diagnostic_code,
                file_id,
                diagnostic.range,
                &diagnostic.data,
            );
            add_disable_code_action(
                semantic_model,
                &mut actions,
                diagnostic_code,
                file_id,
                diagnostic.range,
            );
        }
    }

    if actions.is_empty() {
        return None;
    }

    Some(actions)
}

#[allow(unused_variables)]
fn add_fix_code_action(
    semantic_model: &SemanticModel,
    actions: &mut Vec<CodeActionOrCommand>,
    diagnostic_code: DiagnosticCode,
    file_id: FileId,
    range: Range,
    data: &Option<serde_json::Value>,
) -> Option<()> {
    match diagnostic_code {
        DiagnosticCode::NeedCheckNil | DiagnosticCode::UncheckedNilAccess => {
            build_need_check_nil(semantic_model, actions, range, data)
        }
        DiagnosticCode::UnknownDocTag => build_add_doc_tag(semantic_model, actions, range, data),
        DiagnosticCode::PreferredLocalAlias => {
            build_preferred_local_alias_fix(semantic_model, actions, range, data)
        }
        DiagnosticCode::GmodNullCheck => {
            build_gmod_null_check(semantic_model, actions, range, data)
        }
        _ => Some(()),
    }
}

fn add_disable_code_action(
    semantic_model: &SemanticModel,
    actions: &mut Vec<CodeActionOrCommand>,
    diagnostic_code: DiagnosticCode,
    file_id: FileId,
    range: Range,
) -> Option<()> {
    // LuaSyntaxError no need to disable
    if diagnostic_code == DiagnosticCode::SyntaxError {
        return Some(());
    }

    actions.push(CodeActionOrCommand::CodeAction(CodeAction {
        title: t!(
            "Disable current line diagnostic (%{name})",
            name = diagnostic_code.get_name()
        )
        .to_string(),
        kind: Some(CodeActionKind::QUICKFIX),
        edit: Some(WorkspaceEdit {
            changes: build_disable_next_line_changes(semantic_model, range.start, diagnostic_code),
            ..Default::default()
        }),
        ..Default::default()
    }));

    actions.push(CodeActionOrCommand::CodeAction(CodeAction {
        title: t!(
            "Disable all diagnostics in current file (%{name})",
            name = diagnostic_code.get_name()
        )
        .to_string(),
        kind: Some(CodeActionKind::QUICKFIX),
        edit: Some(WorkspaceEdit {
            changes: build_disable_file_changes(semantic_model, diagnostic_code),
            ..Default::default()
        }),
        ..Default::default()
    }));

    actions.push(CodeActionOrCommand::CodeAction(CodeAction {
        title: t!(
            "Disable all diagnostics in current project (%{name})",
            name = diagnostic_code.get_name()
        )
        .to_string(),
        kind: Some(CodeActionKind::QUICKFIX),
        command: Some(make_disable_code_command(
            t!(
                "Disable all diagnostics in current project (%{name})",
                name = diagnostic_code.get_name()
            )
            .as_ref(),
            DisableAction::Project,
            diagnostic_code,
            file_id,
            range,
        )),

        ..Default::default()
    }));

    Some(())
}