use std::panic::RefUnwindSafe;
use std::path::PathBuf;
use std::process::ExitCode;
use std::time::SystemTime;
use std::{io, panic};
use anyhow::Result;
use cairo_lang_filesystem::db::FilesGroup;
use cairo_lang_filesystem::ids::FileLongId;
use cairo_lang_semantic::plugin::PluginSuite;
use crossbeam::channel::{Receiver, select_biased};
use lsp_server::Message;
use lsp_types::RegistrationParams;
use salsa::{Database, Durability};
use tracing::{debug, error, info};
use crate::lang::lsp::LsProtoGroup;
use crate::lang::proc_macros::controller::ProcMacroChannelsReceivers;
use crate::lsp::capabilities::server::{
collect_dynamic_registrations, collect_server_capabilities,
};
use crate::lsp::result::LSPResult;
use crate::project::{ProjectController, ProjectUpdate};
use crate::server::client::{Notifier, Requester, Responder};
use crate::server::connection::{Connection, ConnectionInitializer};
use crate::server::panic::is_cancelled;
use crate::server::schedule::thread::JoinHandle;
use crate::server::schedule::{Scheduler, Task, event_loop_thread};
use crate::state::State;
mod config;
mod env_config;
mod ide;
mod lang;
pub mod lsp;
mod project;
mod server;
mod state;
mod toolchain;
#[non_exhaustive]
#[derive(Default, Clone)]
pub struct Tricks {
pub extra_plugin_suites:
Option<&'static (dyn Fn() -> Vec<PluginSuite> + Send + Sync + RefUnwindSafe)>,
}
pub fn start() -> ExitCode {
start_with_tricks(Tricks::default())
}
pub fn start_with_tricks(tricks: Tricks) -> ExitCode {
let _log_guard = init_logging();
set_panic_hook();
info!("language server starting");
env_config::report_to_logs();
let exit_code = match Backend::new(tricks) {
Ok(backend) => {
if let Err(err) = backend.run().map(|handle| handle.join()) {
error!("language server encountered an unrecoverable error: {err}");
ExitCode::from(1)
} else {
ExitCode::from(0)
}
}
Err(err) => {
error!("language server failed during initialization: {err}");
ExitCode::from(1)
}
};
info!("language server stopped");
exit_code
}
#[cfg(feature = "testing")]
pub fn build_service_for_e2e_tests()
-> (Box<dyn FnOnce() -> BackendForTesting + Send>, lsp_server::Connection) {
BackendForTesting::new_for_testing(Default::default())
}
fn init_logging() -> Option<impl Drop> {
use std::fs;
use std::io::IsTerminal;
use tracing_chrome::ChromeLayerBuilder;
use tracing_subscriber::filter::{EnvFilter, LevelFilter, Targets};
use tracing_subscriber::fmt::Layer;
use tracing_subscriber::fmt::time::Uptime;
use tracing_subscriber::prelude::*;
let mut guard = None;
let fmt_layer = Layer::new()
.with_writer(io::stderr)
.with_timer(Uptime::default())
.with_ansi(io::stderr().is_terminal())
.with_filter(
EnvFilter::builder()
.with_default_directive(LevelFilter::WARN.into())
.with_env_var(env_config::CAIRO_LS_LOG)
.from_env_lossy(),
);
let profile_layer = if env_config::tracing_profile() {
let mut path = PathBuf::from(format!(
"./cairols-profile-{}.json",
SystemTime::UNIX_EPOCH.elapsed().unwrap().as_micros()
));
let profile_file = fs::File::create(&path).expect("Failed to create profile file.");
if let Ok(canonical) = fs::canonicalize(&path) {
path = canonical;
}
eprintln!("this LS run will output tracing profile to: {}", path.display());
eprintln!(
"open that file with https://ui.perfetto.dev (or chrome://tracing) to analyze it"
);
let (profile_layer, profile_layer_guard) =
ChromeLayerBuilder::new().writer(profile_file).include_args(true).build();
let profile_layer = profile_layer.with_filter(
Targets::new().with_default(LevelFilter::TRACE).with_target("salsa", LevelFilter::WARN),
);
guard = Some(profile_layer_guard);
Some(profile_layer)
} else {
None
};
tracing::subscriber::set_global_default(
tracing_subscriber::registry().with(fmt_layer).with(profile_layer),
)
.expect("Could not set up global logger.");
guard
}
fn set_panic_hook() {
let previous_hook = panic::take_hook();
panic::set_hook(Box::new(move |info| {
if !is_cancelled(info.payload()) {
previous_hook(info);
}
}))
}
struct Backend {
connection: Connection,
state: State,
}
#[cfg(feature = "testing")]
pub struct BackendForTesting(Backend);
#[cfg(feature = "testing")]
impl BackendForTesting {
fn new_for_testing(
tricks: Tricks,
) -> (Box<dyn FnOnce() -> BackendForTesting + Send>, lsp_server::Connection) {
let (connection_initializer, client) = ConnectionInitializer::memory();
let init = Box::new(|| {
BackendForTesting(Backend::initialize(tricks, connection_initializer).unwrap())
});
(init, client)
}
pub fn run_for_tests(self) -> Result<JoinHandle<Result<()>>> {
self.0.run()
}
}
impl Backend {
fn new(tricks: Tricks) -> Result<Self> {
let connection_initializer = ConnectionInitializer::stdio();
Self::initialize(tricks, connection_initializer)
}
fn initialize(tricks: Tricks, connection_initializer: ConnectionInitializer) -> Result<Self> {
let (id, init_params) = connection_initializer.initialize_start()?;
let client_capabilities = init_params.capabilities;
let server_capabilities = collect_server_capabilities(&client_capabilities);
let connection = connection_initializer.initialize_finish(id, server_capabilities)?;
let state = State::new(connection.make_sender(), client_capabilities, tricks);
Ok(Self { connection, state })
}
fn run(self) -> Result<JoinHandle<Result<()>>> {
event_loop_thread(move || {
let Self { mut state, connection } = self;
let proc_macro_channels = state.proc_macro_controller.init_channels();
let project_updates_receiver = state.project_controller.init_channel();
let mut scheduler = Scheduler::new(&mut state, connection.make_sender());
Self::dispatch_setup_tasks(&mut scheduler);
scheduler.on_sync_task(Self::maybe_swap_database);
scheduler.on_sync_task(Self::refresh_diagnostics);
let result = Self::event_loop(
&connection,
proc_macro_channels,
project_updates_receiver,
scheduler,
);
state.db.salsa_runtime_mut().synthetic_write(Durability::LOW);
if let Err(err) = connection.close() {
error!("failed to close connection to the language server: {err:?}");
}
result
})
}
fn dispatch_setup_tasks(scheduler: &mut Scheduler<'_>) {
scheduler.local(Self::register_dynamic_capabilities);
scheduler.local(|state, _notifier, requester, _responder| {
let _ = state.config.reload(requester, &state.client_capabilities);
});
}
fn register_dynamic_capabilities(
state: &mut State,
_notifier: Notifier,
requester: &mut Requester<'_>,
_responder: Responder,
) {
let registrations = collect_dynamic_registrations(&state.client_capabilities);
let _ = requester
.request::<lsp_types::request::RegisterCapability>(
RegistrationParams { registrations },
|()| {
debug!("configuration file watcher successfully registered");
Task::nothing()
},
)
.inspect_err(|e| {
error!(
"failed to register dynamic capabilities, some features may not work \
properly: {e:?}"
)
});
}
fn event_loop(
connection: &Connection,
proc_macro_channels: ProcMacroChannelsReceivers,
project_updates_receiver: Receiver<ProjectUpdate>,
mut scheduler: Scheduler<'_>,
) -> Result<()> {
let incoming = connection.incoming();
loop {
select_biased! {
recv(project_updates_receiver) -> project_update => {
let Ok(project_update) = project_update else { break };
scheduler.local(move |state, notifier, _, _| ProjectController::handle_update(state, notifier, project_update));
}
recv(incoming) -> msg => {
let Ok(msg) = msg else { break };
if connection.handle_shutdown(&msg)? {
break;
}
let task = match msg {
Message::Request(req) => server::request(req),
Message::Notification(notification) => server::notification(notification),
Message::Response(response) => scheduler.response(response),
};
scheduler.dispatch(task);
}
recv(proc_macro_channels.response) -> response => {
let Ok(()) = response else { break };
scheduler.local(Self::on_proc_macro_response);
}
recv(proc_macro_channels.error) -> error => {
let Ok(()) = error else { break };
scheduler.local(Self::on_proc_macro_error);
}
}
}
Ok(())
}
fn on_proc_macro_error(state: &mut State, _: Notifier, _: &mut Requester<'_>, _: Responder) {
state.proc_macro_controller.handle_error(&mut state.db, &state.config);
}
fn on_proc_macro_response(state: &mut State, _: Notifier, _: &mut Requester<'_>, _: Responder) {
state.proc_macro_controller.on_response(&mut state.db, &state.config);
}
fn maybe_swap_database(state: &mut State, _notifier: Notifier) {
state.db_swapper.maybe_swap(
&mut state.db,
&state.open_files,
&state.tricks,
&mut state.project_controller,
);
}
fn refresh_diagnostics(state: &mut State, _notifier: Notifier) {
state.diagnostics_controller.refresh(state);
}
fn reload(state: &mut State, requester: &mut Requester<'_>) -> LSPResult<()> {
state.project_controller.clear_loaded_workspaces();
state.config.reload(requester, &state.client_capabilities)?;
for uri in state.open_files.iter() {
let Some(file_id) = state.db.file_for_url(uri) else { continue };
if let FileLongId::OnDisk(file_path) = state.db.lookup_intern_file(file_id) {
state.project_controller.update_project_for_file(&file_path);
}
}
Ok(())
}
}