tsl-devkit/lsp-server/src/utils/args_parser.cpp

167 lines
6.2 KiB
C++

#include <iostream>
#include <spdlog/sinks/stdout_sinks.h>
#include <spdlog/sinks/basic_file_sink.h>
#include "./args_parser.hpp"
namespace lsp::utils
{
ArgsParser& ArgsParser::Instance()
{
static ArgsParser instance;
return instance;
}
const ServerConfig& ArgsParser::Parse(int argc, char* argv[])
{
config_ = ServerConfig {};
bool use_stdio = false;
for (int i = 1; i < argc; ++i)
{
std::string arg = argv[i];
if (arg == "--help")
{
config_.show_help = true;
return config_;
}
if (arg == "--log=trace")
config_.log_level = spdlog::level::trace;
else if (arg == "--log=debug")
config_.log_level = spdlog::level::debug;
else if (arg == "--log=info")
config_.log_level = spdlog::level::info;
else if (arg == "--log=warn")
config_.log_level = spdlog::level::warn;
else if (arg == "--log=error")
config_.log_level = spdlog::level::err;
else if (arg == "--log=off")
config_.log_level = spdlog::level::off;
else if (arg.find("--log-file=") == 0)
config_.log_file = arg.substr(11);
else if (arg == "--log-stderr")
config_.use_stderr = true;
else if (arg.find("--interpreter=") == 0)
config_.interpreter_path = arg.substr(14);
else if (arg.find("--threads=") == 0)
{
std::string count_str = arg.substr(10);
try
{
size_t count = std::stoul(count_str);
if (count == 0 || count > 32)
{
std::cerr << "[TSL-LSP] Error: Thread count must be between 1 and 32" << std::endl;
std::exit(1);
}
config_.thread_count = count;
}
catch (const std::exception&)
{
std::cerr << "[TSL-LSP] Error: Invalid thread count: " << count_str << std::endl;
std::exit(1);
}
}
else if (arg == "--stdio")
{
use_stdio = true;
}
else
{
std::cerr << "[TSL-LSP] Error: Unknown argument: " << arg << std::endl;
std::cerr << "Use --help for usage information" << std::endl;
}
}
std::cerr << "TSL Language Server " << __DATE__ << std::endl;
if (use_stdio)
std::cerr << "[TSL-LSP] JosnRPc using stdio" << std::endl;
// config_.interpreter_path = "/mnt/c/Programs/Tinysoft/TSLGen2/";
return config_;
}
const ServerConfig& ArgsParser::GetConfig() const
{
return config_;
}
void ArgsParser::SetupLogger(const ServerConfig& config)
{
std::vector<spdlog::sink_ptr> sinks;
if (config.use_stderr)
sinks.push_back(std::make_shared<spdlog::sinks::stderr_sink_mt>());
if (!config.log_file.empty())
{
try
{
sinks.push_back(std::make_shared<spdlog::sinks::basic_file_sink_mt>(config.log_file, true));
}
catch (const std::exception& e)
{
std::cerr << "[TSL-LSP] Failed to create log file: " << e.what() << std::endl;
}
}
auto logger = std::make_shared<spdlog::logger>("tsl_lsp", sinks.begin(), sinks.end());
logger->set_level(config.log_level);
logger->set_pattern("[%Y-%m-%d %H:%M:%S] [%^%l%$] %v");
spdlog::set_default_logger(logger);
spdlog::flush_on(spdlog::level::warn);
}
void ArgsParser::PrintHelp(const std::string& program_name)
{
// 获取硬件线程数
size_t hardware_threads = std::thread::hardware_concurrency();
if (hardware_threads == 0)
hardware_threads = 0; // 表示无法检测
std::cout << "TSL Language Server Protocol (LSP) Server\n"
<< "Version: " << __DATE__ << " build\n"
<< "\n"
<< "Usage: " << program_name << " [options]\n"
<< "\n"
<< "Options:\n"
<< " --help Show this help message and exit\n"
<< "\n"
<< "Interpreter options:\n"
<< " --interpreter=PATH Set path to TSL interpreter executable\n"
<< " Required for some language features\n"
<< "\n"
<< "Logging options:\n"
<< " --log=LEVEL Set log level (trace, debug, info, warn, error, off)\n"
<< " Default: info\n"
<< " --log-file=PATH Write logs to file at PATH\n"
<< " --log-stderr Output logs to stderr\n"
<< "\n"
<< "Performance options:\n"
<< " --threads=N Set number of worker threads (1-32)\n"
<< " Default: 4\n";
// 显示硬件信息
if (hardware_threads > 0)
std::cout << " Hardware threads available: " << hardware_threads << "\n";
else
std::cout << " Hardware threads: unable to detect\n";
std::cout << "\n"
<< "Examples:\n"
<< " # Run with debug logging to stderr\n"
<< " " << program_name << " --log=debug --log-stderr\n"
<< "\n"
<< " # Run with 8 worker threads and log to file\n"
<< " " << program_name << " --threads=8 --log-file=server.log\n"
<< "\n"
<< " # Run with maximum hardware threads\n";
if (hardware_threads > 0)
std::cout << " " << program_name << " --threads=" << hardware_threads << "\n";
else
std::cout << " " << program_name << " --threads=16 # adjust based on your CPU\n";
}
}