use std::ops::ControlFlow;
use std::time::Duration;
use async_lsp::client_monitor::ClientProcessMonitorLayer;
use async_lsp::concurrency::ConcurrencyLayer;
use async_lsp::panic::CatchUnwindLayer;
use async_lsp::router::Router;
use async_lsp::server::LifecycleLayer;
use async_lsp::tracing::TracingLayer;
use async_lsp::ClientSocket;
use lsp_types::{
notification, request, Hover, HoverContents, HoverProviderCapability, InitializeResult,
MarkedString, MessageType, OneOf, ServerCapabilities, ShowMessageParams,
};
use tower::ServiceBuilder;
use tracing::{info, Level};
struct ServerState {
client: ClientSocket,
counter: i32,
}
struct TickEvent;
#[tokio::main(flavor = "current_thread")]
async fn main() {
let (server, _) = async_lsp::MainLoop::new_server(|client| {
tokio::spawn({
let client = client.clone();
async move {
let mut interval = tokio::time::interval(Duration::from_secs(1));
loop {
interval.tick().await;
if client.emit(TickEvent).is_err() {
break;
}
}
}
});
let mut router = Router::new(ServerState {
client: client.clone(),
counter: 0,
});
router
.request::<request::Initialize, _>(|_, params| async move {
eprintln!("Initialize with {params:?}");
Ok(InitializeResult {
capabilities: ServerCapabilities {
hover_provider: Some(HoverProviderCapability::Simple(true)),
definition_provider: Some(OneOf::Left(true)),
..ServerCapabilities::default()
},
server_info: None,
})
})
.request::<request::HoverRequest, _>(|st, _| {
let client = st.client.clone();
let counter = st.counter;
async move {
tokio::time::sleep(Duration::from_secs(1)).await;
client
.notify::<notification::ShowMessage>(ShowMessageParams {
typ: MessageType::INFO,
message: "Hello LSP".into(),
})
.unwrap();
Ok(Some(Hover {
contents: HoverContents::Scalar(MarkedString::String(format!(
"I am a hover text {counter}!"
))),
range: None,
}))
}
})
.request::<request::GotoDefinition, _>(|_, _| async move {
unimplemented!("Not yet implemented!")
})
.notification::<notification::Initialized>(|_, _| ControlFlow::Continue(()))
.notification::<notification::DidChangeConfiguration>(|_, _| ControlFlow::Continue(()))
.notification::<notification::DidOpenTextDocument>(|_, _| ControlFlow::Continue(()))
.notification::<notification::DidChangeTextDocument>(|_, _| ControlFlow::Continue(()))
.notification::<notification::DidCloseTextDocument>(|_, _| ControlFlow::Continue(()))
.event::<TickEvent>(|st, _| {
info!("tick");
st.counter += 1;
ControlFlow::Continue(())
});
ServiceBuilder::new()
.layer(TracingLayer::default())
.layer(LifecycleLayer::default())
.layer(CatchUnwindLayer::default())
.layer(ConcurrencyLayer::default())
.layer(ClientProcessMonitorLayer::new(client))
.service(router)
});
tracing_subscriber::fmt()
.with_max_level(Level::INFO)
.with_ansi(false)
.with_writer(std::io::stderr)
.init();
#[cfg(unix)]
let (stdin, stdout) = (
async_lsp::stdio::PipeStdin::lock_tokio().unwrap(),
async_lsp::stdio::PipeStdout::lock_tokio().unwrap(),
);
#[cfg(not(unix))]
let (stdin, stdout) = (
tokio_util::compat::TokioAsyncReadCompatExt::compat(tokio::io::stdin()),
tokio_util::compat::TokioAsyncWriteCompatExt::compat_write(tokio::io::stdout()),
);
server.run_buffered(stdin, stdout).await.unwrap();
}