2018-09-01 20:21:11 +03:00
|
|
|
use std::{
|
2018-09-02 12:34:06 +03:00
|
|
|
cell::{Cell, RefCell},
|
2018-10-15 17:44:23 -04:00
|
|
|
fs,
|
2019-04-13 19:45:21 +02:00
|
|
|
path::{Path, PathBuf},
|
2018-09-02 14:46:15 +03:00
|
|
|
sync::Once,
|
2018-10-15 17:44:23 -04:00
|
|
|
time::Duration,
|
2018-09-01 20:21:11 +03:00
|
|
|
};
|
|
|
|
|
2018-12-06 21:03:39 +03:00
|
|
|
use crossbeam_channel::{after, select, Receiver};
|
2018-09-01 20:21:11 +03:00
|
|
|
use flexi_logger::Logger;
|
2018-10-15 17:44:23 -04:00
|
|
|
use gen_lsp_server::{RawMessage, RawNotification, RawRequest};
|
2019-01-14 13:55:56 +03:00
|
|
|
use lsp_types::{
|
2018-09-03 23:32:42 +03:00
|
|
|
notification::DidOpenTextDocument,
|
2018-10-15 17:44:23 -04:00
|
|
|
request::{Request, Shutdown},
|
|
|
|
DidOpenTextDocumentParams, TextDocumentIdentifier, TextDocumentItem, Url,
|
2019-03-05 21:59:01 +02:00
|
|
|
notification::{Notification, ShowMessage},
|
2018-09-01 20:21:11 +03:00
|
|
|
};
|
|
|
|
use serde::Serialize;
|
2018-12-06 21:07:31 +01:00
|
|
|
use serde_json::{to_string_pretty, Value};
|
2018-12-30 23:33:04 +03:00
|
|
|
use tempfile::TempDir;
|
2019-02-14 20:43:45 +03:00
|
|
|
use thread_worker::Worker;
|
2018-12-06 21:07:31 +01:00
|
|
|
use test_utils::{parse_fixture, find_mismatch};
|
2018-09-01 20:21:11 +03:00
|
|
|
|
2018-10-15 17:44:23 -04:00
|
|
|
use ra_lsp_server::{
|
|
|
|
main_loop, req,
|
2019-03-06 11:34:38 +02:00
|
|
|
InitializationOptions,
|
2018-10-15 17:44:23 -04:00
|
|
|
};
|
2018-09-01 20:21:11 +03:00
|
|
|
|
2019-04-13 20:33:49 +02:00
|
|
|
pub struct Project<'a> {
|
|
|
|
fixture: &'a str,
|
|
|
|
tmp_dir: Option<TempDir>,
|
|
|
|
roots: Vec<PathBuf>,
|
2019-03-05 22:29:23 +01:00
|
|
|
}
|
|
|
|
|
2019-04-13 20:33:49 +02:00
|
|
|
impl<'a> Project<'a> {
|
|
|
|
pub fn with_fixture(fixture: &str) -> Project {
|
|
|
|
Project { fixture, tmp_dir: None, roots: vec![] }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn tmp_dir(mut self, tmp_dir: TempDir) -> Project<'a> {
|
|
|
|
self.tmp_dir = Some(tmp_dir);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn root(mut self, path: &str) -> Project<'a> {
|
|
|
|
self.roots.push(path.into());
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn server(self) -> Server {
|
|
|
|
let tmp_dir = self.tmp_dir.unwrap_or_else(|| TempDir::new().unwrap());
|
|
|
|
static INIT: Once = Once::new();
|
|
|
|
INIT.call_once(|| {
|
|
|
|
let _ = Logger::with_env_or_str(crate::LOG).start().unwrap();
|
|
|
|
});
|
2018-09-01 20:21:11 +03:00
|
|
|
|
2019-04-13 20:33:49 +02:00
|
|
|
let mut paths = vec![];
|
2018-10-31 21:37:32 +03:00
|
|
|
|
2019-04-13 20:33:49 +02:00
|
|
|
for entry in parse_fixture(self.fixture) {
|
|
|
|
let path = tmp_dir.path().join(entry.meta);
|
|
|
|
fs::create_dir_all(path.parent().unwrap()).unwrap();
|
|
|
|
fs::write(path.as_path(), entry.text.as_bytes()).unwrap();
|
|
|
|
paths.push((path, entry.text));
|
|
|
|
}
|
|
|
|
|
|
|
|
let roots = self.roots.into_iter().map(|root| tmp_dir.path().join(root)).collect();
|
|
|
|
|
|
|
|
Server::new(tmp_dir, roots, paths)
|
2018-09-01 20:21:11 +03:00
|
|
|
}
|
2019-04-13 20:33:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn project(fixture: &str) -> Server {
|
|
|
|
Project::with_fixture(fixture).server()
|
2018-09-01 20:21:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Server {
|
|
|
|
req_id: Cell<u64>,
|
2018-09-02 12:34:06 +03:00
|
|
|
messages: RefCell<Vec<RawMessage>>,
|
2018-09-01 20:21:11 +03:00
|
|
|
dir: TempDir,
|
2018-09-08 13:15:01 +03:00
|
|
|
worker: Option<Worker<RawMessage, RawMessage>>,
|
2018-09-01 20:21:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Server {
|
2019-04-13 20:33:49 +02:00
|
|
|
fn new(dir: TempDir, roots: Vec<PathBuf>, files: Vec<(PathBuf, String)>) -> Server {
|
2018-09-01 20:21:11 +03:00
|
|
|
let path = dir.path().to_path_buf();
|
2019-04-13 20:33:49 +02:00
|
|
|
|
|
|
|
let roots = if roots.is_empty() { vec![path] } else { roots };
|
|
|
|
|
2019-02-14 20:43:45 +03:00
|
|
|
let worker = Worker::<RawMessage, RawMessage>::spawn(
|
2018-09-08 13:15:01 +03:00
|
|
|
"test server",
|
|
|
|
128,
|
|
|
|
move |mut msg_receiver, mut msg_sender| {
|
2019-03-06 11:34:38 +02:00
|
|
|
main_loop(
|
2019-04-13 20:33:49 +02:00
|
|
|
roots,
|
2019-03-06 11:34:38 +02:00
|
|
|
InitializationOptions::default(),
|
|
|
|
&mut msg_receiver,
|
|
|
|
&mut msg_sender,
|
|
|
|
)
|
|
|
|
.unwrap()
|
2018-10-15 17:44:23 -04:00
|
|
|
},
|
2018-09-08 13:15:01 +03:00
|
|
|
);
|
2018-09-01 20:21:11 +03:00
|
|
|
let res = Server {
|
|
|
|
req_id: Cell::new(1),
|
|
|
|
dir,
|
2018-09-02 12:34:06 +03:00
|
|
|
messages: Default::default(),
|
2018-09-08 13:15:01 +03:00
|
|
|
worker: Some(worker),
|
2018-09-01 20:21:11 +03:00
|
|
|
};
|
2018-09-08 12:08:46 +03:00
|
|
|
|
2018-09-01 20:21:11 +03:00
|
|
|
for (path, text) in files {
|
|
|
|
res.send_notification(RawNotification::new::<DidOpenTextDocument>(
|
2018-09-02 15:18:43 +03:00
|
|
|
&DidOpenTextDocumentParams {
|
2018-09-01 20:21:11 +03:00
|
|
|
text_document: TextDocumentItem {
|
|
|
|
uri: Url::from_file_path(path).unwrap(),
|
|
|
|
language_id: "rust".to_string(),
|
|
|
|
version: 0,
|
|
|
|
text,
|
2018-10-15 17:44:23 -04:00
|
|
|
},
|
|
|
|
},
|
2018-09-01 20:21:11 +03:00
|
|
|
))
|
|
|
|
}
|
|
|
|
res
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn doc_id(&self, rel_path: &str) -> TextDocumentIdentifier {
|
|
|
|
let path = self.dir.path().join(rel_path);
|
2019-02-08 14:49:43 +03:00
|
|
|
TextDocumentIdentifier { uri: Url::from_file_path(path).unwrap() }
|
2018-09-01 20:21:11 +03:00
|
|
|
}
|
|
|
|
|
2018-12-06 21:07:31 +01:00
|
|
|
pub fn request<R>(&self, params: R::Params, expected_resp: Value)
|
2018-09-01 20:21:11 +03:00
|
|
|
where
|
|
|
|
R: Request,
|
|
|
|
R::Params: Serialize,
|
|
|
|
{
|
2019-01-11 00:37:10 +03:00
|
|
|
let actual = self.send_request::<R>(params);
|
2018-12-06 21:07:31 +01:00
|
|
|
match find_mismatch(&expected_resp, &actual) {
|
|
|
|
Some((expected_part, actual_part)) => panic!(
|
|
|
|
"JSON mismatch\nExpected:\n{}\nWas:\n{}\nExpected part:\n{}\nActual part:\n{}\n",
|
|
|
|
to_string_pretty(&expected_resp).unwrap(),
|
|
|
|
to_string_pretty(&actual).unwrap(),
|
|
|
|
to_string_pretty(expected_part).unwrap(),
|
|
|
|
to_string_pretty(actual_part).unwrap(),
|
|
|
|
),
|
|
|
|
None => {}
|
|
|
|
}
|
2018-09-01 20:21:11 +03:00
|
|
|
}
|
|
|
|
|
2019-01-11 00:37:10 +03:00
|
|
|
pub fn send_request<R>(&self, params: R::Params) -> Value
|
2018-09-01 20:21:11 +03:00
|
|
|
where
|
|
|
|
R: Request,
|
|
|
|
R::Params: Serialize,
|
|
|
|
{
|
2019-01-11 00:37:10 +03:00
|
|
|
let id = self.req_id.get();
|
|
|
|
self.req_id.set(id + 1);
|
|
|
|
|
2018-09-02 15:18:43 +03:00
|
|
|
let r = RawRequest::new::<R>(id, ¶ms);
|
2018-09-02 16:36:03 +03:00
|
|
|
self.send_request_(r)
|
|
|
|
}
|
2018-10-15 17:44:23 -04:00
|
|
|
fn send_request_(&self, r: RawRequest) -> Value {
|
2018-09-02 16:36:03 +03:00
|
|
|
let id = r.id;
|
2019-02-14 20:43:45 +03:00
|
|
|
self.worker.as_ref().unwrap().sender().send(RawMessage::Request(r)).unwrap();
|
2018-09-02 12:34:06 +03:00
|
|
|
while let Some(msg) = self.recv() {
|
2018-09-01 20:21:11 +03:00
|
|
|
match msg {
|
|
|
|
RawMessage::Request(req) => panic!("unexpected request: {:?}", req),
|
|
|
|
RawMessage::Notification(_) => (),
|
|
|
|
RawMessage::Response(res) => {
|
|
|
|
assert_eq!(res.id, id);
|
|
|
|
if let Some(err) = res.error {
|
|
|
|
panic!("error response: {:#?}", err);
|
|
|
|
}
|
|
|
|
return res.result.unwrap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
panic!("no response");
|
|
|
|
}
|
2019-03-07 17:46:17 +03:00
|
|
|
pub fn wait_until_workspace_is_loaded(&self) {
|
|
|
|
self.wait_for_message_cond(1, &|msg: &RawMessage| match msg {
|
2019-03-05 21:59:01 +02:00
|
|
|
RawMessage::Notification(n) if n.method == ShowMessage::METHOD => {
|
2019-03-05 22:25:24 +02:00
|
|
|
let msg = n.clone().cast::<req::ShowMessage>().unwrap();
|
2019-03-07 17:46:17 +03:00
|
|
|
msg.message.starts_with("workspace loaded")
|
2018-10-15 17:44:23 -04:00
|
|
|
}
|
|
|
|
_ => false,
|
2019-03-07 17:46:17 +03:00
|
|
|
})
|
|
|
|
}
|
|
|
|
fn wait_for_message_cond(&self, n: usize, cond: &dyn Fn(&RawMessage) -> bool) {
|
2018-09-04 00:49:21 +03:00
|
|
|
let mut total = 0;
|
2018-09-02 14:46:15 +03:00
|
|
|
for msg in self.messages.borrow().iter() {
|
2019-03-07 17:46:17 +03:00
|
|
|
if cond(msg) {
|
2018-09-04 00:49:21 +03:00
|
|
|
total += 1
|
2018-09-02 14:46:15 +03:00
|
|
|
}
|
|
|
|
}
|
2018-09-04 00:49:21 +03:00
|
|
|
while total < n {
|
|
|
|
let msg = self.recv().expect("no response");
|
2019-03-07 17:46:17 +03:00
|
|
|
if cond(&msg) {
|
2018-09-04 00:49:21 +03:00
|
|
|
total += 1;
|
2018-09-02 14:46:15 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-09-02 12:34:06 +03:00
|
|
|
fn recv(&self) -> Option<RawMessage> {
|
2019-02-14 20:43:45 +03:00
|
|
|
recv_timeout(&self.worker.as_ref().unwrap().receiver()).map(|msg| {
|
2018-10-15 17:44:23 -04:00
|
|
|
self.messages.borrow_mut().push(msg.clone());
|
|
|
|
msg
|
|
|
|
})
|
2018-09-02 12:34:06 +03:00
|
|
|
}
|
2018-09-01 20:21:11 +03:00
|
|
|
fn send_notification(&self, not: RawNotification) {
|
2019-02-14 20:43:45 +03:00
|
|
|
self.worker.as_ref().unwrap().sender().send(RawMessage::Notification(not)).unwrap();
|
2018-09-01 20:21:11 +03:00
|
|
|
}
|
2019-04-13 19:45:21 +02:00
|
|
|
|
|
|
|
pub fn path(&self) -> &Path {
|
|
|
|
self.dir.path()
|
|
|
|
}
|
2018-09-01 20:21:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for Server {
|
|
|
|
fn drop(&mut self) {
|
2019-01-11 00:37:10 +03:00
|
|
|
self.send_request::<Shutdown>(());
|
2018-09-01 20:21:11 +03:00
|
|
|
}
|
|
|
|
}
|
2018-09-08 12:08:46 +03:00
|
|
|
|
|
|
|
fn recv_timeout(receiver: &Receiver<RawMessage>) -> Option<RawMessage> {
|
2019-01-11 00:37:10 +03:00
|
|
|
let timeout = Duration::from_secs(50);
|
2018-09-08 12:08:46 +03:00
|
|
|
select! {
|
2018-12-30 23:23:31 +03:00
|
|
|
recv(receiver) -> msg => msg.ok(),
|
|
|
|
recv(after(timeout)) -> _ => panic!("timed out"),
|
2018-09-08 12:08:46 +03:00
|
|
|
}
|
|
|
|
}
|