2018-08-10 12:07:43 +00:00
|
|
|
#[macro_use]
|
|
|
|
extern crate failure;
|
|
|
|
#[macro_use]
|
|
|
|
extern crate serde_derive;
|
|
|
|
extern crate serde;
|
|
|
|
extern crate serde_json;
|
|
|
|
extern crate languageserver_types;
|
|
|
|
extern crate drop_bomb;
|
2018-08-10 14:49:45 +00:00
|
|
|
#[macro_use]
|
2018-08-10 12:07:43 +00:00
|
|
|
extern crate crossbeam_channel;
|
2018-08-10 14:49:45 +00:00
|
|
|
extern crate threadpool;
|
|
|
|
#[macro_use]
|
|
|
|
extern crate log;
|
2018-08-10 21:55:32 +00:00
|
|
|
extern crate url_serde;
|
2018-08-10 14:49:45 +00:00
|
|
|
extern crate flexi_logger;
|
2018-08-13 10:46:05 +00:00
|
|
|
extern crate walkdir;
|
2018-08-10 12:07:43 +00:00
|
|
|
extern crate libeditor;
|
|
|
|
extern crate libanalysis;
|
2018-08-11 11:44:12 +00:00
|
|
|
extern crate libsyntax2;
|
2018-08-15 14:24:20 +00:00
|
|
|
extern crate im;
|
2018-08-28 15:22:52 +00:00
|
|
|
extern crate relative_path;
|
2018-08-10 12:07:43 +00:00
|
|
|
|
|
|
|
mod io;
|
|
|
|
mod caps;
|
|
|
|
mod req;
|
|
|
|
mod dispatch;
|
2018-08-12 18:02:56 +00:00
|
|
|
mod conv;
|
2018-08-12 19:08:14 +00:00
|
|
|
mod main_loop;
|
2018-08-13 10:46:05 +00:00
|
|
|
mod vfs;
|
2018-08-15 14:24:20 +00:00
|
|
|
mod path_map;
|
2018-08-17 16:54:08 +00:00
|
|
|
mod server_world;
|
2018-08-13 10:46:05 +00:00
|
|
|
|
2018-08-10 14:49:45 +00:00
|
|
|
use threadpool::ThreadPool;
|
2018-08-12 19:08:14 +00:00
|
|
|
use crossbeam_channel::bounded;
|
2018-08-12 23:38:34 +00:00
|
|
|
use flexi_logger::{Logger, Duplicate};
|
2018-08-10 14:49:45 +00:00
|
|
|
|
|
|
|
use ::{
|
2018-08-15 14:24:20 +00:00
|
|
|
io::{Io, RawMsg, RawResponse, RawRequest, RawNotification},
|
2018-08-10 14:49:45 +00:00
|
|
|
};
|
2018-08-10 12:07:43 +00:00
|
|
|
|
|
|
|
pub type Result<T> = ::std::result::Result<T, ::failure::Error>;
|
|
|
|
|
|
|
|
fn main() -> Result<()> {
|
2018-08-27 21:42:13 +00:00
|
|
|
Logger::with_env_or_str("m=error")
|
2018-08-12 23:38:34 +00:00
|
|
|
.duplicate_to_stderr(Duplicate::All)
|
2018-08-10 14:49:45 +00:00
|
|
|
.log_to_file()
|
|
|
|
.directory("log")
|
|
|
|
.start()?;
|
2018-08-10 19:55:42 +00:00
|
|
|
info!("lifecycle: server started");
|
2018-08-10 14:49:45 +00:00
|
|
|
match ::std::panic::catch_unwind(|| main_inner()) {
|
|
|
|
Ok(res) => {
|
2018-08-10 19:55:42 +00:00
|
|
|
info!("lifecycle: terminating process with {:?}", res);
|
2018-08-10 14:49:45 +00:00
|
|
|
res
|
|
|
|
}
|
|
|
|
Err(_) => {
|
|
|
|
error!("server panicked");
|
|
|
|
bail!("server panicked")
|
2018-08-10 15:01:59 +00:00
|
|
|
}
|
2018-08-10 14:49:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main_inner() -> Result<()> {
|
2018-08-10 12:07:43 +00:00
|
|
|
let mut io = Io::from_stdio();
|
2018-08-10 14:49:45 +00:00
|
|
|
let res = initialize(&mut io);
|
|
|
|
info!("shutting down IO...");
|
|
|
|
let io_res = io.stop();
|
|
|
|
info!("... IO is down");
|
|
|
|
match (res, io_res) {
|
|
|
|
(Ok(()), Ok(())) => Ok(()),
|
|
|
|
(res, Ok(())) => res,
|
|
|
|
(Ok(()), io_res) => io_res,
|
|
|
|
(res, Err(io_err)) => {
|
|
|
|
error!("shutdown error: {:?}", io_err);
|
|
|
|
res
|
|
|
|
}
|
|
|
|
}
|
2018-08-10 12:07:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn initialize(io: &mut Io) -> Result<()> {
|
2018-08-12 19:23:44 +00:00
|
|
|
match io.recv()? {
|
|
|
|
RawMsg::Notification(n) =>
|
|
|
|
bail!("expected initialize request, got {:?}", n),
|
|
|
|
RawMsg::Response(res) =>
|
|
|
|
bail!("expected initialize request, got {:?}", res),
|
2018-08-12 18:47:27 +00:00
|
|
|
|
2018-08-12 19:23:44 +00:00
|
|
|
RawMsg::Request(req) => {
|
|
|
|
let mut req = Some(req);
|
|
|
|
dispatch::handle_request::<req::Initialize, _>(&mut req, |_params, resp| {
|
2018-08-12 23:38:34 +00:00
|
|
|
let res = req::InitializeResult { capabilities: caps::server_capabilities() };
|
2018-08-12 19:23:44 +00:00
|
|
|
let resp = resp.into_response(Ok(res))?;
|
|
|
|
io.send(RawMsg::Response(resp));
|
|
|
|
Ok(())
|
|
|
|
})?;
|
|
|
|
if let Some(req) = req {
|
|
|
|
bail!("expected initialize request, got {:?}", req)
|
|
|
|
}
|
|
|
|
match io.recv()? {
|
|
|
|
RawMsg::Notification(n) => {
|
|
|
|
if n.method != "initialized" {
|
|
|
|
bail!("expected initialized notification");
|
2018-08-10 12:07:43 +00:00
|
|
|
}
|
|
|
|
}
|
2018-08-12 19:23:44 +00:00
|
|
|
_ => bail!("expected initialized notification"),
|
2018-08-10 12:07:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-08-12 18:47:27 +00:00
|
|
|
initialized(io)
|
2018-08-10 12:07:43 +00:00
|
|
|
}
|
|
|
|
|
2018-08-12 18:34:17 +00:00
|
|
|
enum Task {
|
|
|
|
Respond(RawResponse),
|
2018-08-12 21:09:30 +00:00
|
|
|
Request(RawRequest),
|
2018-08-12 18:34:17 +00:00
|
|
|
Notify(RawNotification),
|
|
|
|
Die(::failure::Error),
|
|
|
|
}
|
2018-08-10 14:49:45 +00:00
|
|
|
|
2018-08-10 12:07:43 +00:00
|
|
|
fn initialized(io: &mut Io) -> Result<()> {
|
2018-08-10 19:55:42 +00:00
|
|
|
{
|
|
|
|
let mut pool = ThreadPool::new(4);
|
2018-08-13 10:46:05 +00:00
|
|
|
let (task_sender, task_receiver) = bounded::<Task>(16);
|
|
|
|
let (fs_events_receiver, watcher) = vfs::watch(vec![
|
2018-08-13 12:35:53 +00:00
|
|
|
::std::env::current_dir()?,
|
2018-08-13 10:46:05 +00:00
|
|
|
]);
|
2018-08-10 19:55:42 +00:00
|
|
|
info!("lifecycle: handshake finished, server ready to serve requests");
|
2018-08-13 10:46:05 +00:00
|
|
|
let res = main_loop::main_loop(
|
|
|
|
io,
|
|
|
|
&mut pool,
|
|
|
|
task_sender,
|
|
|
|
task_receiver.clone(),
|
|
|
|
fs_events_receiver,
|
|
|
|
);
|
|
|
|
|
2018-08-10 19:55:42 +00:00
|
|
|
info!("waiting for background jobs to finish...");
|
2018-08-13 10:46:05 +00:00
|
|
|
task_receiver.for_each(drop);
|
2018-08-10 19:55:42 +00:00
|
|
|
pool.join();
|
|
|
|
info!("...background jobs have finished");
|
2018-08-13 10:46:05 +00:00
|
|
|
|
|
|
|
info!("waiting for file watcher to finish...");
|
|
|
|
watcher.stop()?;
|
|
|
|
info!("...file watcher has finished");
|
|
|
|
|
2018-08-10 19:55:42 +00:00
|
|
|
res
|
|
|
|
}?;
|
|
|
|
|
|
|
|
match io.recv()? {
|
|
|
|
RawMsg::Notification(n) => {
|
|
|
|
if n.method == "exit" {
|
|
|
|
info!("lifecycle: shutdown complete");
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
bail!("unexpected notification during shutdown: {:?}", n)
|
|
|
|
}
|
|
|
|
m => {
|
|
|
|
bail!("unexpected message during shutdown: {:?}", m)
|
|
|
|
}
|
|
|
|
}
|
2018-08-10 14:49:45 +00:00
|
|
|
}
|