use std::{
    collections::HashMap,
    io::Write,
    str::FromStr,
    sync::{Arc, Mutex},
};
use tokio::sync::RwLock;
pub mod custom_notifications;
use anyhow::Result;
#[cfg(feature = "cli")]
use clap::Parser;
use dashmap::DashMap;
use sha2::Digest;
use tower_lsp::{
    jsonrpc::Result as RpcResult,
    lsp_types::{
        CompletionItem, CompletionItemKind, CompletionOptions, CompletionParams, CompletionResponse, CreateFilesParams,
        DeleteFilesParams, Diagnostic, DiagnosticOptions, DiagnosticServerCapabilities, DiagnosticSeverity,
        DidChangeConfigurationParams, DidChangeTextDocumentParams, DidChangeWatchedFilesParams,
        DidChangeWorkspaceFoldersParams, DidCloseTextDocumentParams, DidOpenTextDocumentParams,
        DidSaveTextDocumentParams, DocumentDiagnosticParams, DocumentDiagnosticReport, DocumentDiagnosticReportResult,
        DocumentFilter, DocumentFormattingParams, DocumentSymbol, DocumentSymbolParams, DocumentSymbolResponse,
        Documentation, FoldingRange, FoldingRangeParams, FoldingRangeProviderCapability, FullDocumentDiagnosticReport,
        Hover, HoverContents, HoverParams, HoverProviderCapability, InitializeParams, InitializeResult,
        InitializedParams, InlayHint, InlayHintParams, InsertTextFormat, MarkupContent, MarkupKind, MessageType, OneOf,
        Position, RelatedFullDocumentDiagnosticReport, RenameFilesParams, RenameParams, SemanticToken,
        SemanticTokenModifier, SemanticTokenType, SemanticTokens, SemanticTokensFullOptions, SemanticTokensLegend,
        SemanticTokensOptions, SemanticTokensParams, SemanticTokensRegistrationOptions, SemanticTokensResult,
        SemanticTokensServerCapabilities, ServerCapabilities, SignatureHelp, SignatureHelpOptions, SignatureHelpParams,
        StaticRegistrationOptions, TextDocumentItem, TextDocumentRegistrationOptions, TextDocumentSyncCapability,
        TextDocumentSyncKind, TextDocumentSyncOptions, TextEdit, WorkDoneProgressOptions, WorkspaceEdit,
        WorkspaceFolder, WorkspaceFoldersServerCapabilities, WorkspaceServerCapabilities,
    },
    Client, LanguageServer,
};
use crate::{
    ast::types::{Expr, VariableKind},
    executor::SourceRange,
    lsp::{backend::Backend as _, util::IntoDiagnostic},
    parser::PIPE_OPERATOR,
    token::TokenType,
};
lazy_static::lazy_static! {
    pub static ref SEMANTIC_TOKEN_TYPES: Vec<SemanticTokenType> = {
        let mut gen = TokenType::all_semantic_token_types().unwrap();
        gen.extend(vec![
            SemanticTokenType::PARAMETER,
            SemanticTokenType::PROPERTY,
        ]);
        gen
    };
    pub static ref SEMANTIC_TOKEN_MODIFIERS: Vec<SemanticTokenModifier> = {
        vec![
            SemanticTokenModifier::DECLARATION,
            SemanticTokenModifier::DEFINITION,
            SemanticTokenModifier::DEFAULT_LIBRARY,
            SemanticTokenModifier::READONLY,
            SemanticTokenModifier::STATIC,
        ]
    };
}
#[derive(Clone, Debug)]
#[cfg_attr(feature = "cli", derive(Parser))]
pub struct Server {
    #[cfg_attr(feature = "cli", clap(long, default_value = "8080"))]
    pub socket: i32,
    #[cfg_attr(feature = "cli", clap(short, long, default_value = "false"))]
    pub stdio: bool,
}
#[derive(Clone)]
pub struct Backend {
    pub client: Client,
    pub fs: Arc<crate::fs::FileManager>,
    pub workspace_folders: DashMap<String, WorkspaceFolder>,
    pub stdlib_completions: HashMap<String, CompletionItem>,
    pub stdlib_signatures: HashMap<String, SignatureHelp>,
    pub token_map: DashMap<String, Vec<crate::token::Token>>,
    pub ast_map: DashMap<String, crate::ast::types::Program>,
    pub memory_map: DashMap<String, crate::executor::ProgramMemory>,
    pub code_map: DashMap<String, Vec<u8>>,
    pub diagnostics_map: DashMap<String, Vec<Diagnostic>>,
    pub symbols_map: DashMap<String, Vec<DocumentSymbol>>,
    pub semantic_tokens_map: DashMap<String, Vec<SemanticToken>>,
    pub zoo_client: kittycad::Client,
    pub can_send_telemetry: bool,
    pub executor_ctx: Arc<RwLock<Option<crate::executor::ExecutorContext>>>,
    pub can_execute: Arc<RwLock<bool>>,
    pub is_initialized: Arc<RwLock<bool>>,
}
#[async_trait::async_trait]
impl crate::lsp::backend::Backend for Backend {
    fn client(&self) -> &Client {
        &self.client
    }
    fn fs(&self) -> &Arc<crate::fs::FileManager> {
        &self.fs
    }
    async fn is_initialized(&self) -> bool {
        *self.is_initialized.read().await
    }
    async fn set_is_initialized(&self, is_initialized: bool) {
        *self.is_initialized.write().await = is_initialized;
    }
    async fn workspace_folders(&self) -> Vec<WorkspaceFolder> {
        self.workspace_folders.iter().map(|i| i.clone()).collect()
    }
    async fn add_workspace_folders(&self, folders: Vec<WorkspaceFolder>) {
        for folder in folders {
            self.workspace_folders.insert(folder.name.to_string(), folder);
        }
    }
    async fn remove_workspace_folders(&self, folders: Vec<WorkspaceFolder>) {
        for folder in folders {
            self.workspace_folders.remove(&folder.name);
        }
    }
    fn code_map(&self) -> &DashMap<String, Vec<u8>> {
        &self.code_map
    }
    async fn insert_code_map(&self, uri: String, text: Vec<u8>) {
        self.code_map.insert(uri, text);
    }
    async fn remove_from_code_map(&self, uri: String) -> Option<Vec<u8>> {
        self.code_map.remove(&uri).map(|x| x.1)
    }
    async fn clear_code_state(&self) {
        self.code_map.clear();
        self.token_map.clear();
        self.ast_map.clear();
        self.diagnostics_map.clear();
        self.symbols_map.clear();
        self.semantic_tokens_map.clear();
    }
    fn current_diagnostics_map(&self) -> &DashMap<String, Vec<Diagnostic>> {
        &self.diagnostics_map
    }
    async fn inner_on_change(&self, params: TextDocumentItem, force: bool) {
        let filename = params.uri.to_string();
        let tokens = match crate::token::lexer(¶ms.text) {
            Ok(tokens) => tokens,
            Err(err) => {
                self.add_to_diagnostics(¶ms, &[err], true).await;
                self.token_map.remove(&filename);
                self.ast_map.remove(&filename);
                self.symbols_map.remove(&filename);
                self.semantic_tokens_map.remove(&filename);
                self.memory_map.remove(&filename);
                return;
            }
        };
        let has_memory = if let Some(memory) = self.memory_map.get(&filename) {
            *memory != crate::executor::ProgramMemory::default()
        } else {
            false
        };
        let tokens_changed = if let Some(previous_tokens) = self.token_map.get(&filename) {
            *previous_tokens != tokens
        } else {
            true
        };
        if !tokens_changed && !force && has_memory && !self.has_diagnostics(params.uri.as_ref()).await {
            return;
        }
        if tokens_changed {
            self.token_map.insert(params.uri.to_string(), tokens.clone());
            self.update_semantic_tokens(&tokens, ¶ms).await;
        }
        let parser = crate::parser::Parser::new(tokens.clone());
        let result = parser.ast();
        let mut ast = match result {
            Ok(ast) => ast,
            Err(err) => {
                self.add_to_diagnostics(¶ms, &[err], true).await;
                self.ast_map.remove(&filename);
                self.symbols_map.remove(&filename);
                self.memory_map.remove(&filename);
                return;
            }
        };
        ast.compute_digest();
        let ast_changed = match self.ast_map.get(&filename) {
            Some(old_ast) => {
                *old_ast != ast
            }
            None => true,
        };
        if !ast_changed && !force && has_memory && !self.has_diagnostics(params.uri.as_ref()).await {
            return;
        }
        if ast_changed {
            self.ast_map.insert(params.uri.to_string(), ast.clone());
            self.symbols_map.insert(
                params.uri.to_string(),
                ast.get_lsp_symbols(¶ms.text).unwrap_or_default(),
            );
            self.update_semantic_tokens(&tokens, ¶ms).await;
            let discovered_findings = ast.lint_all().into_iter().flatten().collect::<Vec<_>>();
            self.add_to_diagnostics(¶ms, &discovered_findings, false).await;
        }
        if self.can_execute().await || self.executor_ctx().await.is_none() {
            self.client
                .send_notification::<custom_notifications::AstUpdated>(ast.clone())
                .await;
        }
        if self.execute(¶ms, &ast).await.is_err() {
            return;
        }
        self.clear_diagnostics_map(¶ms.uri, Some(DiagnosticSeverity::ERROR))
            .await;
    }
}
impl Backend {
    pub async fn can_execute(&self) -> bool {
        *self.can_execute.read().await
    }
    pub async fn executor_ctx(&self) -> tokio::sync::RwLockReadGuard<'_, Option<crate::executor::ExecutorContext>> {
        self.executor_ctx.read().await
    }
    async fn update_semantic_tokens(&self, tokens: &[crate::token::Token], params: &TextDocumentItem) {
        let mut semantic_tokens = vec![];
        let mut last_position = Position::new(0, 0);
        for token in tokens {
            let Ok(token_type) = SemanticTokenType::try_from(token.token_type) else {
                continue;
            };
            let mut token_type_index = match self.get_semantic_token_type_index(&token_type) {
                Some(index) => index,
                None => {
                    self.client
                        .log_message(
                            MessageType::ERROR,
                            format!("token type `{:?}` not accounted for", token_type),
                        )
                        .await;
                    continue;
                }
            };
            let source_range: SourceRange = token.into();
            let position = source_range.start_to_lsp_position(¶ms.text);
            let token_modifiers_bitset = if let Some(ast) = self.ast_map.get(params.uri.as_str()) {
                let token_index = Arc::new(Mutex::new(token_type_index));
                let modifier_index: Arc<Mutex<u32>> = Arc::new(Mutex::new(0));
                crate::walk::walk(&ast, &|node: crate::walk::Node| {
                    let node_range: SourceRange = (&node).into();
                    if !node_range.contains(source_range.start()) {
                        return Ok(true);
                    }
                    let get_modifier = |modifier: Vec<SemanticTokenModifier>| -> Result<bool> {
                        let mut mods = modifier_index.lock().map_err(|_| anyhow::anyhow!("mutex"))?;
                        let Some(token_modifier_index) = self.get_semantic_token_modifier_index(modifier) else {
                            return Ok(true);
                        };
                        if *mods == 0 {
                            *mods = token_modifier_index;
                        } else {
                            *mods |= token_modifier_index;
                        }
                        Ok(false)
                    };
                    match node {
                        crate::walk::Node::TagDeclarator(_) => {
                            return get_modifier(vec![
                                SemanticTokenModifier::DEFINITION,
                                SemanticTokenModifier::STATIC,
                            ]);
                        }
                        crate::walk::Node::VariableDeclarator(variable) => {
                            let sr: SourceRange = (&variable.id).into();
                            if sr.contains(source_range.start()) {
                                if let Expr::FunctionExpression(_) = &variable.init {
                                    let mut ti = token_index.lock().map_err(|_| anyhow::anyhow!("mutex"))?;
                                    *ti = match self.get_semantic_token_type_index(&SemanticTokenType::FUNCTION) {
                                        Some(index) => index,
                                        None => token_type_index,
                                    };
                                }
                                return get_modifier(vec![
                                    SemanticTokenModifier::DECLARATION,
                                    SemanticTokenModifier::READONLY,
                                ]);
                            }
                        }
                        crate::walk::Node::Parameter(_) => {
                            let mut ti = token_index.lock().map_err(|_| anyhow::anyhow!("mutex"))?;
                            *ti = match self.get_semantic_token_type_index(&SemanticTokenType::PARAMETER) {
                                Some(index) => index,
                                None => token_type_index,
                            };
                            return Ok(false);
                        }
                        crate::walk::Node::MemberExpression(member_expression) => {
                            let sr: SourceRange = (&member_expression.property).into();
                            if sr.contains(source_range.start()) {
                                let mut ti = token_index.lock().map_err(|_| anyhow::anyhow!("mutex"))?;
                                *ti = match self.get_semantic_token_type_index(&SemanticTokenType::PROPERTY) {
                                    Some(index) => index,
                                    None => token_type_index,
                                };
                                return Ok(false);
                            }
                        }
                        crate::walk::Node::ObjectProperty(object_property) => {
                            let sr: SourceRange = (&object_property.key).into();
                            if sr.contains(source_range.start()) {
                                let mut ti = token_index.lock().map_err(|_| anyhow::anyhow!("mutex"))?;
                                *ti = match self.get_semantic_token_type_index(&SemanticTokenType::PROPERTY) {
                                    Some(index) => index,
                                    None => token_type_index,
                                };
                            }
                            return get_modifier(vec![SemanticTokenModifier::DECLARATION]);
                        }
                        crate::walk::Node::CallExpression(call_expr) => {
                            let sr: SourceRange = (&call_expr.callee).into();
                            if sr.contains(source_range.start()) {
                                let mut ti = token_index.lock().map_err(|_| anyhow::anyhow!("mutex"))?;
                                *ti = match self.get_semantic_token_type_index(&SemanticTokenType::FUNCTION) {
                                    Some(index) => index,
                                    None => token_type_index,
                                };
                                if self.stdlib_completions.contains_key(&call_expr.callee.name) {
                                    return get_modifier(vec![SemanticTokenModifier::DEFAULT_LIBRARY]);
                                }
                                return Ok(false);
                            }
                        }
                        _ => {}
                    }
                    Ok(true)
                })
                .unwrap_or_default();
                let t = if let Ok(guard) = token_index.lock() { *guard } else { 0 };
                token_type_index = t;
                let m = if let Ok(guard) = modifier_index.lock() {
                    *guard
                } else {
                    0
                };
                m
            } else {
                0
            };
            if let Some(line) = params.text.lines().nth(position.line as usize) {
                if line.len() == position.character as usize {
                    let semantic_token = SemanticToken {
                        delta_line: position.line - last_position.line + 1,
                        delta_start: 0,
                        length: token.value.len() as u32,
                        token_type: token_type_index,
                        token_modifiers_bitset,
                    };
                    semantic_tokens.push(semantic_token);
                    last_position = Position::new(position.line + 1, 0);
                    continue;
                }
            }
            let semantic_token = SemanticToken {
                delta_line: position.line - last_position.line,
                delta_start: if position.line != last_position.line {
                    position.character
                } else {
                    position.character - last_position.character
                },
                length: token.value.len() as u32,
                token_type: token_type_index,
                token_modifiers_bitset,
            };
            semantic_tokens.push(semantic_token);
            last_position = position;
        }
        self.semantic_tokens_map.insert(params.uri.to_string(), semantic_tokens);
    }
    async fn clear_diagnostics_map(&self, uri: &url::Url, severity: Option<DiagnosticSeverity>) {
        let Some(mut items) = self.diagnostics_map.get_mut(uri.as_str()) else {
            return;
        };
        if let Some(severity) = severity {
            items.retain(|x| x.severity != Some(severity));
        } else {
            items.clear();
        }
        if items.is_empty() {
            #[cfg(not(target_arch = "wasm32"))]
            {
                self.client.publish_diagnostics(uri.clone(), items.clone(), None).await;
            }
            drop(items);
            self.diagnostics_map.remove(uri.as_str());
        } else {
            #[cfg(not(target_arch = "wasm32"))]
            {
                self.client.publish_diagnostics(uri.clone(), items.clone(), None).await;
            }
        }
    }
    async fn add_to_diagnostics<DiagT: IntoDiagnostic + std::fmt::Debug>(
        &self,
        params: &TextDocumentItem,
        diagnostics: &[DiagT],
        clear_all_before_add: bool,
    ) {
        if diagnostics.is_empty() {
            return;
        }
        if clear_all_before_add {
            self.clear_diagnostics_map(¶ms.uri, None).await;
        } else if diagnostics.iter().all(|x| x.severity() == DiagnosticSeverity::ERROR) {
            self.clear_diagnostics_map(¶ms.uri, Some(DiagnosticSeverity::ERROR))
                .await;
        } else if diagnostics
            .iter()
            .all(|x| x.severity() == DiagnosticSeverity::INFORMATION)
        {
            self.clear_diagnostics_map(¶ms.uri, Some(DiagnosticSeverity::INFORMATION))
                .await;
        }
        let mut items = if let Some(items) = self.diagnostics_map.get(params.uri.as_str()) {
            items.clone()
        } else {
            vec![]
        };
        for diagnostic in diagnostics {
            let d = diagnostic.to_lsp_diagnostic(¶ms.text);
            if !items.iter().any(|x| x == &d) {
                items.push(d);
            }
        }
        self.diagnostics_map.insert(params.uri.to_string(), items.clone());
        self.client.publish_diagnostics(params.uri.clone(), items, None).await;
    }
    async fn execute(&self, params: &TextDocumentItem, ast: &crate::ast::types::Program) -> Result<()> {
        if !self.can_execute().await {
            return Ok(());
        }
        let ctx = self.executor_ctx().await;
        let Some(ref executor_ctx) = *ctx else {
            return Ok(());
        };
        if !self.is_initialized().await {
            return Ok(());
        }
        executor_ctx.engine.clear_scene(SourceRange::default()).await?;
        let memory = match executor_ctx.run(ast, None).await {
            Ok(memory) => memory,
            Err(err) => {
                self.memory_map.remove(params.uri.as_str());
                self.add_to_diagnostics(params, &[err], false).await;
                return Err(anyhow::anyhow!("failed to execute code"));
            }
        };
        self.memory_map.insert(params.uri.to_string(), memory.clone());
        self.client
            .send_notification::<custom_notifications::MemoryUpdated>(memory)
            .await;
        Ok(())
    }
    pub fn get_semantic_token_type_index(&self, token_type: &SemanticTokenType) -> Option<u32> {
        SEMANTIC_TOKEN_TYPES
            .iter()
            .position(|x| *x == *token_type)
            .map(|y| y as u32)
    }
    pub fn get_semantic_token_modifier_index(&self, token_types: Vec<SemanticTokenModifier>) -> Option<u32> {
        if token_types.is_empty() {
            return None;
        }
        let mut modifier = None;
        for token_type in token_types {
            if let Some(index) = SEMANTIC_TOKEN_MODIFIERS
                .iter()
                .position(|x| *x == token_type)
                .map(|y| y as u32)
            {
                modifier = match modifier {
                    Some(modifier) => Some(modifier | index),
                    None => Some(index),
                };
            }
        }
        modifier
    }
    pub async fn create_zip(&self) -> Result<Vec<u8>> {
        let mut buf = vec![];
        let mut zip = zip::ZipWriter::new(std::io::Cursor::new(&mut buf));
        for code in self.code_map.iter() {
            let entry = code.key();
            let value = code.value();
            let file_name = entry.replace("file://", "").to_string();
            let options = zip::write::SimpleFileOptions::default().compression_method(zip::CompressionMethod::Stored);
            zip.start_file(file_name, options)?;
            zip.write_all(value)?;
        }
        zip.finish()?;
        Ok(buf)
    }
    pub async fn send_telemetry(&self) -> Result<()> {
        let user = self
            .zoo_client
            .users()
            .get_self()
            .await
            .map_err(|e| anyhow::anyhow!(e.to_string()))?;
        let mut hasher = sha2::Sha256::new();
        hasher.update(user.id);
        let result = hasher.finalize();
        let user_id_hash = format!("{:x}", result);
        let workspace_folders = self.workspace_folders().await;
        let project_names: Vec<&str> = workspace_folders.iter().map(|v| v.name.as_str()).collect::<Vec<_>>();
        let project_name = project_names
            .first()
            .ok_or_else(|| anyhow::anyhow!("no project names"))?
            .to_string();
        self.zoo_client
            .meta()
            .create_event(
                vec![kittycad::types::multipart::Attachment {
                    name: "attachment".to_string(),
                    filename: Some("attachment.zip".to_string()),
                    content_type: Some("application/x-zip".to_string()),
                    data: self.create_zip().await?,
                }],
                &kittycad::types::Event {
                    attachment_uri: None,
                    created_at: chrono::Utc::now(),
                    event_type: kittycad::types::ModelingAppEventType::SuccessfulCompileBeforeClose,
                    last_compiled_at: Some(chrono::Utc::now()),
                    project_description: None,
                    project_name,
                    source_id: uuid::Uuid::from_str("70178592-dfca-47b3-bd2d-6fce2bcaee04").unwrap(),
                    type_: kittycad::types::Type::ModelingAppEvent,
                    user_id: user_id_hash,
                },
            )
            .await
            .map_err(|e| anyhow::anyhow!(e.to_string()))?;
        Ok(())
    }
    pub async fn update_units(
        &self,
        params: custom_notifications::UpdateUnitsParams,
    ) -> RpcResult<Option<custom_notifications::UpdateUnitsResponse>> {
        let filename = params.text_document.uri.to_string();
        {
            let mut ctx = self.executor_ctx.write().await;
            let Some(ref mut executor_ctx) = *ctx else {
                self.client
                    .log_message(MessageType::ERROR, "no executor context set to update units for")
                    .await;
                return Ok(None);
            };
            self.client
                .log_message(MessageType::INFO, format!("update units: {:?}", params))
                .await;
            let has_memory = if let Some(memory) = self.memory_map.get(&filename) {
                *memory != crate::executor::ProgramMemory::default()
            } else {
                false
            };
            if executor_ctx.settings.units == params.units
                && !self.has_diagnostics(params.text_document.uri.as_ref()).await
                && has_memory
            {
                return Ok(None);
            }
            executor_ctx.update_units(params.units);
        }
        let new_params = TextDocumentItem {
            uri: params.text_document.uri.clone(),
            text: std::mem::take(&mut params.text.to_string()),
            version: Default::default(),
            language_id: Default::default(),
        };
        self.inner_on_change(new_params, true).await;
        if self.has_diagnostics(params.text_document.uri.as_ref()).await {
            return Ok(None);
        }
        Ok(Some(custom_notifications::UpdateUnitsResponse {}))
    }
    pub async fn update_can_execute(
        &self,
        params: custom_notifications::UpdateCanExecuteParams,
    ) -> RpcResult<custom_notifications::UpdateCanExecuteResponse> {
        let mut can_execute = self.can_execute.write().await;
        if *can_execute == params.can_execute {
            return Ok(custom_notifications::UpdateCanExecuteResponse {});
        }
        *can_execute = params.can_execute;
        Ok(custom_notifications::UpdateCanExecuteResponse {})
    }
}
#[tower_lsp::async_trait]
impl LanguageServer for Backend {
    async fn initialize(&self, params: InitializeParams) -> RpcResult<InitializeResult> {
        self.client
            .log_message(MessageType::INFO, format!("initialize: {:?}", params))
            .await;
        Ok(InitializeResult {
            capabilities: ServerCapabilities {
                completion_provider: Some(CompletionOptions {
                    resolve_provider: Some(false),
                    trigger_characters: Some(vec![".".to_string()]),
                    work_done_progress_options: Default::default(),
                    all_commit_characters: None,
                    ..Default::default()
                }),
                diagnostic_provider: Some(DiagnosticServerCapabilities::Options(DiagnosticOptions {
                    ..Default::default()
                })),
                document_formatting_provider: Some(OneOf::Left(true)),
                folding_range_provider: Some(FoldingRangeProviderCapability::Simple(true)),
                hover_provider: Some(HoverProviderCapability::Simple(true)),
                inlay_hint_provider: Some(OneOf::Left(true)),
                rename_provider: Some(OneOf::Left(true)),
                semantic_tokens_provider: Some(SemanticTokensServerCapabilities::SemanticTokensRegistrationOptions(
                    SemanticTokensRegistrationOptions {
                        text_document_registration_options: {
                            TextDocumentRegistrationOptions {
                                document_selector: Some(vec![DocumentFilter {
                                    language: Some("kcl".to_string()),
                                    scheme: Some("file".to_string()),
                                    pattern: None,
                                }]),
                            }
                        },
                        semantic_tokens_options: SemanticTokensOptions {
                            work_done_progress_options: WorkDoneProgressOptions::default(),
                            legend: SemanticTokensLegend {
                                token_types: SEMANTIC_TOKEN_TYPES.clone(),
                                token_modifiers: SEMANTIC_TOKEN_MODIFIERS.clone(),
                            },
                            range: Some(false),
                            full: Some(SemanticTokensFullOptions::Bool(true)),
                        },
                        static_registration_options: StaticRegistrationOptions::default(),
                    },
                )),
                signature_help_provider: Some(SignatureHelpOptions {
                    trigger_characters: None,
                    retrigger_characters: None,
                    ..Default::default()
                }),
                text_document_sync: Some(TextDocumentSyncCapability::Options(TextDocumentSyncOptions {
                    open_close: Some(true),
                    change: Some(TextDocumentSyncKind::FULL),
                    ..Default::default()
                })),
                workspace: Some(WorkspaceServerCapabilities {
                    workspace_folders: Some(WorkspaceFoldersServerCapabilities {
                        supported: Some(true),
                        change_notifications: Some(OneOf::Left(true)),
                    }),
                    file_operations: None,
                }),
                ..Default::default()
            },
            ..Default::default()
        })
    }
    async fn initialized(&self, params: InitializedParams) {
        self.do_initialized(params).await
    }
    async fn shutdown(&self) -> RpcResult<()> {
        self.do_shutdown().await
    }
    async fn did_change_workspace_folders(&self, params: DidChangeWorkspaceFoldersParams) {
        self.do_did_change_workspace_folders(params).await
    }
    async fn did_change_configuration(&self, params: DidChangeConfigurationParams) {
        self.do_did_change_configuration(params).await
    }
    async fn did_change_watched_files(&self, params: DidChangeWatchedFilesParams) {
        self.do_did_change_watched_files(params).await
    }
    async fn did_create_files(&self, params: CreateFilesParams) {
        self.do_did_create_files(params).await
    }
    async fn did_rename_files(&self, params: RenameFilesParams) {
        self.do_did_rename_files(params).await
    }
    async fn did_delete_files(&self, params: DeleteFilesParams) {
        self.do_did_delete_files(params).await
    }
    async fn did_open(&self, params: DidOpenTextDocumentParams) {
        self.do_did_open(params).await
    }
    async fn did_change(&self, params: DidChangeTextDocumentParams) {
        self.do_did_change(params).await;
    }
    async fn did_save(&self, params: DidSaveTextDocumentParams) {
        self.do_did_save(params).await
    }
    async fn did_close(&self, params: DidCloseTextDocumentParams) {
        self.do_did_close(params).await;
        if !self.can_send_telemetry {
            return;
        }
        #[cfg(target_arch = "wasm32")]
        {
            let be = self.clone();
            wasm_bindgen_futures::spawn_local(async move {
                if let Err(err) = be.send_telemetry().await {
                    be.client
                        .log_message(MessageType::WARNING, format!("failed to send telemetry: {}", err))
                        .await;
                }
            });
        }
        #[cfg(not(target_arch = "wasm32"))]
        if let Err(err) = self.send_telemetry().await {
            self.client
                .log_message(MessageType::WARNING, format!("failed to send telemetry: {}", err))
                .await;
        }
    }
    async fn hover(&self, params: HoverParams) -> RpcResult<Option<Hover>> {
        let filename = params.text_document_position_params.text_document.uri.to_string();
        let Some(current_code) = self.code_map.get(&filename) else {
            return Ok(None);
        };
        let Ok(current_code) = std::str::from_utf8(¤t_code) else {
            return Ok(None);
        };
        let pos = position_to_char_index(params.text_document_position_params.position, current_code);
        let Some(ast) = self.ast_map.get(&filename) else {
            return Ok(None);
        };
        let Some(hover) = ast.get_hover_value_for_position(pos, current_code) else {
            return Ok(None);
        };
        match hover {
            crate::ast::types::Hover::Function { name, range } => {
                let Some(completion) = self.stdlib_completions.get(&name) else {
                    return Ok(None);
                };
                let Some(docs) = &completion.documentation else {
                    return Ok(None);
                };
                let docs = match docs {
                    Documentation::String(docs) => docs,
                    Documentation::MarkupContent(MarkupContent { value, .. }) => value,
                };
                let Some(label_details) = &completion.label_details else {
                    return Ok(None);
                };
                Ok(Some(Hover {
                    contents: HoverContents::Markup(MarkupContent {
                        kind: MarkupKind::Markdown,
                        value: format!(
                            "```{}{}```\n{}",
                            name,
                            if let Some(detail) = &label_details.detail {
                                detail
                            } else {
                                ""
                            },
                            docs
                        ),
                    }),
                    range: Some(range),
                }))
            }
            crate::ast::types::Hover::Signature { .. } => Ok(None),
            crate::ast::types::Hover::Comment { value, range } => Ok(Some(Hover {
                contents: HoverContents::Markup(MarkupContent {
                    kind: MarkupKind::Markdown,
                    value,
                }),
                range: Some(range),
            })),
        }
    }
    async fn completion(&self, params: CompletionParams) -> RpcResult<Option<CompletionResponse>> {
        let mut completions = vec![CompletionItem {
            label: PIPE_OPERATOR.to_string(),
            label_details: None,
            kind: Some(CompletionItemKind::OPERATOR),
            detail: Some("A pipe operator.".to_string()),
            documentation: Some(Documentation::MarkupContent(MarkupContent {
                kind: MarkupKind::Markdown,
                value: "A pipe operator.".to_string(),
            })),
            deprecated: Some(false),
            preselect: None,
            sort_text: None,
            filter_text: None,
            insert_text: Some("|> ".to_string()),
            insert_text_format: Some(InsertTextFormat::PLAIN_TEXT),
            insert_text_mode: None,
            text_edit: None,
            additional_text_edits: None,
            command: None,
            commit_characters: None,
            data: None,
            tags: None,
        }];
        completions.extend(self.stdlib_completions.values().cloned());
        let Some(ast) = self
            .ast_map
            .get(params.text_document_position.text_document.uri.as_ref())
        else {
            return Ok(Some(CompletionResponse::Array(completions)));
        };
        let Some(current_code) = self
            .code_map
            .get(params.text_document_position.text_document.uri.as_ref())
        else {
            return Ok(Some(CompletionResponse::Array(completions)));
        };
        let Ok(current_code) = std::str::from_utf8(¤t_code) else {
            return Ok(Some(CompletionResponse::Array(completions)));
        };
        let position = position_to_char_index(params.text_document_position.position, current_code);
        if ast.get_non_code_meta_for_position(position).is_some() {
            return Ok(None);
        }
        let Ok(variables) = ast.completion_items() else {
            return Ok(Some(CompletionResponse::Array(completions)));
        };
        completions.extend(variables);
        Ok(Some(CompletionResponse::Array(completions)))
    }
    async fn diagnostic(&self, params: DocumentDiagnosticParams) -> RpcResult<DocumentDiagnosticReportResult> {
        let filename = params.text_document.uri.to_string();
        let Some(items) = self.diagnostics_map.get(&filename) else {
            return Ok(DocumentDiagnosticReportResult::Report(DocumentDiagnosticReport::Full(
                RelatedFullDocumentDiagnosticReport {
                    related_documents: None,
                    full_document_diagnostic_report: FullDocumentDiagnosticReport {
                        result_id: None,
                        items: vec![],
                    },
                },
            )));
        };
        Ok(DocumentDiagnosticReportResult::Report(DocumentDiagnosticReport::Full(
            RelatedFullDocumentDiagnosticReport {
                related_documents: None,
                full_document_diagnostic_report: FullDocumentDiagnosticReport {
                    result_id: None,
                    items: items.clone(),
                },
            },
        )))
    }
    async fn signature_help(&self, params: SignatureHelpParams) -> RpcResult<Option<SignatureHelp>> {
        let filename = params.text_document_position_params.text_document.uri.to_string();
        let Some(current_code) = self.code_map.get(&filename) else {
            return Ok(None);
        };
        let Ok(current_code) = std::str::from_utf8(¤t_code) else {
            return Ok(None);
        };
        let pos = position_to_char_index(params.text_document_position_params.position, current_code);
        let Some(ast) = self.ast_map.get(&filename) else {
            return Ok(None);
        };
        let Some(value) = ast.get_value_for_position(pos) else {
            return Ok(None);
        };
        let Some(hover) = value.get_hover_value_for_position(pos, current_code) else {
            return Ok(None);
        };
        match hover {
            crate::ast::types::Hover::Function { name, range: _ } => {
                let Some(signature) = self.stdlib_signatures.get(&name) else {
                    return Ok(None);
                };
                Ok(Some(signature.clone()))
            }
            crate::ast::types::Hover::Signature {
                name,
                parameter_index,
                range: _,
            } => {
                let Some(signature) = self.stdlib_signatures.get(&name) else {
                    return Ok(None);
                };
                let mut signature = signature.clone();
                signature.active_parameter = Some(parameter_index);
                Ok(Some(signature))
            }
            crate::ast::types::Hover::Comment { value: _, range: _ } => {
                return Ok(None);
            }
        }
    }
    async fn inlay_hint(&self, _params: InlayHintParams) -> RpcResult<Option<Vec<InlayHint>>> {
        Ok(None)
    }
    async fn semantic_tokens_full(&self, params: SemanticTokensParams) -> RpcResult<Option<SemanticTokensResult>> {
        let filename = params.text_document.uri.to_string();
        let Some(semantic_tokens) = self.semantic_tokens_map.get(&filename) else {
            return Ok(None);
        };
        Ok(Some(SemanticTokensResult::Tokens(SemanticTokens {
            result_id: None,
            data: semantic_tokens.clone(),
        })))
    }
    async fn document_symbol(&self, params: DocumentSymbolParams) -> RpcResult<Option<DocumentSymbolResponse>> {
        let filename = params.text_document.uri.to_string();
        let Some(symbols) = self.symbols_map.get(&filename) else {
            return Ok(None);
        };
        Ok(Some(DocumentSymbolResponse::Nested(symbols.clone())))
    }
    async fn formatting(&self, params: DocumentFormattingParams) -> RpcResult<Option<Vec<TextEdit>>> {
        let filename = params.text_document.uri.to_string();
        let Some(current_code) = self.code_map.get(&filename) else {
            return Ok(None);
        };
        let Ok(current_code) = std::str::from_utf8(¤t_code) else {
            return Ok(None);
        };
        let Ok(tokens) = crate::token::lexer(current_code) else {
            return Ok(None);
        };
        let parser = crate::parser::Parser::new(tokens);
        let Ok(ast) = parser.ast() else {
            return Ok(None);
        };
        let recast = ast.recast(
            &crate::ast::types::FormatOptions {
                tab_size: params.options.tab_size as usize,
                insert_final_newline: params.options.insert_final_newline.unwrap_or(false),
                use_tabs: !params.options.insert_spaces,
            },
            0,
        );
        let source_range = SourceRange([0, current_code.len()]);
        let range = source_range.to_lsp_range(current_code);
        Ok(Some(vec![TextEdit {
            new_text: recast,
            range,
        }]))
    }
    async fn rename(&self, params: RenameParams) -> RpcResult<Option<WorkspaceEdit>> {
        let filename = params.text_document_position.text_document.uri.to_string();
        let Some(current_code) = self.code_map.get(&filename) else {
            return Ok(None);
        };
        let Ok(current_code) = std::str::from_utf8(¤t_code) else {
            return Ok(None);
        };
        let Ok(tokens) = crate::token::lexer(current_code) else {
            return Ok(None);
        };
        let parser = crate::parser::Parser::new(tokens);
        let Ok(mut ast) = parser.ast() else {
            return Ok(None);
        };
        let pos = position_to_char_index(params.text_document_position.position, current_code);
        ast.rename_symbol(¶ms.new_name, pos);
        let recast = ast.recast(&Default::default(), 0);
        let source_range = SourceRange([0, current_code.len() - 1]);
        let range = source_range.to_lsp_range(current_code);
        Ok(Some(WorkspaceEdit {
            changes: Some(HashMap::from([(
                params.text_document_position.text_document.uri,
                vec![TextEdit {
                    new_text: recast,
                    range,
                }],
            )])),
            document_changes: None,
            change_annotations: None,
        }))
    }
    async fn folding_range(&self, params: FoldingRangeParams) -> RpcResult<Option<Vec<FoldingRange>>> {
        let filename = params.text_document.uri.to_string();
        let Some(ast) = self.ast_map.get(&filename) else {
            return Ok(None);
        };
        let folding_ranges = ast.get_lsp_folding_ranges();
        if folding_ranges.is_empty() {
            return Ok(None);
        }
        Ok(Some(folding_ranges))
    }
}
pub fn get_completions_from_stdlib(stdlib: &crate::std::StdLib) -> Result<HashMap<String, CompletionItem>> {
    let mut completions = HashMap::new();
    let combined = stdlib.combined();
    for internal_fn in combined.values() {
        completions.insert(internal_fn.name(), internal_fn.to_completion_item()?);
    }
    let variable_kinds = VariableKind::to_completion_items()?;
    for variable_kind in variable_kinds {
        completions.insert(variable_kind.label.clone(), variable_kind);
    }
    Ok(completions)
}
pub fn get_signatures_from_stdlib(stdlib: &crate::std::StdLib) -> Result<HashMap<String, SignatureHelp>> {
    let mut signatures = HashMap::new();
    let combined = stdlib.combined();
    for internal_fn in combined.values() {
        signatures.insert(internal_fn.name(), internal_fn.to_signature_help());
    }
    Ok(signatures)
}
fn position_to_char_index(position: Position, code: &str) -> usize {
    let mut char_position = 0;
    for (index, line) in code.lines().enumerate() {
        if index == position.line as usize {
            char_position += position.character as usize;
            break;
        } else {
            char_position += line.len() + 1;
        }
    }
    char_position
}