glua_ls 1.0.27

Language server for Garry's Mod Lua (GLua).
Documentation
#[cfg(feature = "cli")]
use clap::{Parser, ValueEnum};

use crate::context::ClientId;

#[allow(unused)]
#[derive(Debug, Clone)]
#[cfg_attr(feature = "cli", derive(Parser))]
#[cfg_attr(feature = "cli", command(version))]
pub struct CmdArgs {
    /// Communication method
    #[cfg_attr(feature = "cli", structopt(long, short, default_value = "stdio"))]
    pub communication: Communication,

    /// IP address to listen on (only valid when using TCP)
    #[cfg_attr(feature = "cli", structopt(long, default_value = "127.0.0.1"))]
    pub ip: String,

    /// Port number to listen on (only valid when using TCP)
    #[cfg_attr(feature = "cli", structopt(long, default_value = "5007"))]
    pub port: u16,

    /// Logging level
    #[cfg_attr(feature = "cli", structopt(long, default_value = "info"))]
    pub log_level: LogLevel,

    /// Path to the log file. Use 'none' to disable log file output.
    #[cfg_attr(feature = "cli", structopt(long, default_value = "none"))]
    pub log_path: NoneableString,

    /// Path to the resources and logs directory. Use 'none' to indicate that assets should not be output to the file system.
    #[cfg_attr(feature = "cli", structopt(long, default_value = ""))]
    pub resources_path: NoneableString,

    /// Whether to load the standard library.
    #[cfg_attr(feature = "cli", structopt(long, default_value = "true"))]
    pub load_stdlib: CmdBool,

    /// Force editor mode. Options: vscode, intellij, neovim
    #[cfg_attr(feature = "cli", structopt(long))]
    pub editor: Option<Editor>,

    /// Path to GMod annotations folder. Use 'none' to disable. When not provided, falls back to VSCode extension path if available.
    #[cfg_attr(feature = "cli", structopt(long))]
    pub gmod_annotations_path: Option<NoneableString>,
}

/// Logging level enum
#[derive(Debug, Clone, Copy)]
#[cfg_attr(feature = "cli", derive(ValueEnum))]
pub enum LogLevel {
    /// Error level
    Error,
    /// Warning level
    Warn,
    /// Info level
    Info,
    /// Debug level
    Debug,
}

impl std::str::FromStr for LogLevel {
    type Err = String;

    fn from_str(input: &str) -> Result<LogLevel, Self::Err> {
        match input.to_lowercase().as_str() {
            "error" => Ok(LogLevel::Error),
            "warn" => Ok(LogLevel::Warn),
            "info" => Ok(LogLevel::Info),
            "debug" => Ok(LogLevel::Debug),
            _ => Err(format!(
                "Invalid log level: '{}'. Please choose 'error', 'warn', 'info', 'debug'",
                input
            )),
        }
    }
}

#[derive(Debug, Clone, Copy)]
#[cfg_attr(feature = "cli", derive(ValueEnum))]
pub enum Communication {
    Stdio,
    Tcp,
}

impl std::str::FromStr for Communication {
    type Err = String;

    fn from_str(input: &str) -> Result<Communication, Self::Err> {
        match input.to_lowercase().as_str() {
            "stdio" => Ok(Communication::Stdio),
            "tcp" => Ok(Communication::Tcp),
            _ => Err(format!(
                "Invalid communication method: '{}'. Please choose 'stdio', 'tcp'",
                input
            )),
        }
    }
}

/// A string that can be "None" to represent an empty option
#[derive(Debug, Clone)]
pub struct NoneableString(pub Option<String>);

impl std::str::FromStr for NoneableString {
    type Err = String;

    fn from_str(s: &str) -> Result<Self, Self::Err> {
        if s.eq_ignore_ascii_case("none") {
            Ok(NoneableString(None))
        } else {
            Ok(NoneableString(Some(s.to_string())))
        }
    }
}

#[allow(unused)]
impl NoneableString {
    pub fn as_deref(&self) -> Option<&str> {
        self.0.as_deref()
    }
}

#[derive(Debug, Clone)]
pub struct CmdBool(pub bool);

impl std::str::FromStr for CmdBool {
    type Err = String;

    fn from_str(s: &str) -> Result<Self, Self::Err> {
        match s.to_lowercase().as_str() {
            "true" => Ok(CmdBool(true)),
            "false" => Ok(CmdBool(false)),
            _ => Err(format!(
                "Invalid boolean value: '{}'. Please choose 'true' or 'false'",
                s
            )),
        }
    }
}

#[derive(Debug, Clone)]
#[cfg_attr(feature = "cli", derive(ValueEnum))]
pub enum Editor {
    Vscode,
    Intellij,
    Neovim,
}

impl std::str::FromStr for Editor {
    type Err = String;

    fn from_str(input: &str) -> Result<Editor, Self::Err> {
        match input.to_lowercase().as_str() {
            "vscode" => Ok(Editor::Vscode),
            "intellij" => Ok(Editor::Intellij),
            "neovim" => Ok(Editor::Neovim),
            _ => Err(format!(
                "Invalid editor: '{}'. Please choose 'vscode', 'intellij', 'neovim'",
                input
            )),
        }
    }
}

impl From<Editor> for ClientId {
    fn from(editor: Editor) -> Self {
        match editor {
            Editor::Vscode => ClientId::VSCode,
            Editor::Intellij => ClientId::Intellij,
            Editor::Neovim => ClientId::Neovim,
        }
    }
}