glua_ls 1.0.27

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

use log::info;
use serde_json::Value;

use crate::{context::ServerContextSnapshot, util::time_cancel_token};
use glua_code_analysis::file_path_to_uri;

use super::ClientConfig;

pub async fn get_client_config_default(
    context: &ServerContextSnapshot,
    config: &mut ClientConfig,
    scopes: Option<&[&str]>,
) -> Option<()> {
    let workspace_folders = context
        .workspace_manager()
        .read()
        .await
        .workspace_folders
        .clone();
    let client = context.client();
    let mut scope_uris = workspace_folders
        .iter()
        .filter_map(|workspace| file_path_to_uri(&workspace.root).map(Some))
        .collect::<Vec<_>>();
    if scope_uris.is_empty() {
        scope_uris.push(None);
    }

    let mut configs = Vec::new();
    let mut used_scope = None;
    for scope in scopes.unwrap_or(&["gluals"]) {
        let params = lsp_types::ConfigurationParams {
            items: scope_uris
                .iter()
                .map(|scope_uri| lsp_types::ConfigurationItem {
                    scope_uri: scope_uri.clone(),
                    section: Some(scope.to_string()),
                })
                .collect(),
        };
        log::info!("fetching client config for scope {scope:?}");
        let cancel_token = time_cancel_token(Duration::from_secs(5));
        let fetched_configs: Vec<_> = match client
            .get_configuration::<Value>(params, cancel_token)
            .await
        {
            Some(configs) => configs,
            None => {
                log::warn!("failed to fetch client config for scope {scope:?}");
                continue;
            }
        };
        let fetched_configs: Vec<_> = fetched_configs
            .into_iter()
            .filter(|config| !config.is_null())
            .collect();
        if !fetched_configs.is_empty() {
            info!("found client config in scope {scope:?}");
            configs = fetched_configs;
            used_scope = Some(scope.to_string());
        }
    }

    if let Some(used_scope) = used_scope {
        info!(
            "using client config from scope {used_scope:?}: {}",
            serde_json::to_string_pretty(&configs)
                .as_deref()
                .unwrap_or("<failed to serialize json>")
        );
    } else {
        info!("no client config found");
    }

    for config in &mut configs {
        // VSCode always sends default values for all options, even those that weren't
        // explicitly configured by user. This results in `null`s being sent for
        // every option. Naturally, serde chokes on these nulls when applying partial
        // configuration.
        //
        // Because of this, we have to ignore them here.
        skip_nulls(config);
    }

    config.partial_emmyrcs = Some(configs);

    Some(())
}

fn skip_nulls(v: &mut Value) {
    if let Value::Object(obj) = v {
        obj.retain(|_, v| !v.is_null());
        for (_, v) in obj {
            skip_nulls(v);
        }
    }
}