rust/crates/ra_lsp_server/tests/heavy_tests/support.rs

235 lines
7.0 KiB
Rust
Raw Normal View History

2018-09-01 12:21:11 -05:00
use std::{
2018-09-02 04:34:06 -05:00
cell::{Cell, RefCell},
fs,
path::{Path, PathBuf},
2018-09-02 06:46:15 -05:00
sync::Once,
time::Duration,
2018-09-01 12:21:11 -05:00
};
2018-12-06 12:03:39 -06:00
use crossbeam_channel::{after, select, Receiver};
2018-09-01 12:21:11 -05:00
use flexi_logger::Logger;
use gen_lsp_server::{RawMessage, RawNotification, RawRequest};
2019-01-14 04:55:56 -06:00
use lsp_types::{
2018-09-03 15:32:42 -05:00
notification::DidOpenTextDocument,
request::{Request, Shutdown},
DidOpenTextDocumentParams, TextDocumentIdentifier, TextDocumentItem, Url,
notification::{Notification, ShowMessage},
2018-09-01 12:21:11 -05:00
};
use serde::Serialize;
use serde_json::{to_string_pretty, Value};
2018-12-30 14:33:04 -06:00
use tempfile::TempDir;
use thread_worker::Worker;
use test_utils::{parse_fixture, find_mismatch};
2018-09-01 12:21:11 -05:00
use ra_lsp_server::{
main_loop, req,
InitializationOptions,
};
2018-09-01 12:21:11 -05:00
pub struct Project<'a> {
fixture: &'a str,
tmp_dir: Option<TempDir>,
roots: Vec<PathBuf>,
}
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();
2019-05-27 06:20:11 -05:00
ra_prof::set_filter(if crate::PROFILE.is_empty() {
ra_prof::Filter::disabled()
} else {
ra_prof::Filter::from_spec(&crate::PROFILE)
});
});
2018-09-01 12:21:11 -05:00
let mut paths = vec![];
2018-10-31 13:37:32 -05: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 12:21:11 -05:00
}
}
pub fn project(fixture: &str) -> Server {
Project::with_fixture(fixture).server()
2018-09-01 12:21:11 -05:00
}
pub struct Server {
req_id: Cell<u64>,
2018-09-02 04:34:06 -05:00
messages: RefCell<Vec<RawMessage>>,
2018-09-01 12:21:11 -05:00
dir: TempDir,
worker: Worker<RawMessage, RawMessage>,
2018-09-01 12:21:11 -05:00
}
impl Server {
fn new(dir: TempDir, roots: Vec<PathBuf>, files: Vec<(PathBuf, String)>) -> Server {
2018-09-01 12:21:11 -05:00
let path = dir.path().to_path_buf();
let roots = if roots.is_empty() { vec![path] } else { roots };
let worker = Worker::<RawMessage, RawMessage>::spawn(
2018-09-08 05:15:01 -05:00
"test server",
128,
move |mut msg_receiver, mut msg_sender| {
main_loop(
roots,
InitializationOptions::default(),
&mut msg_receiver,
&mut msg_sender,
)
.unwrap()
},
2018-09-08 05:15:01 -05:00
);
let res = Server { req_id: Cell::new(1), dir, messages: Default::default(), worker };
2018-09-08 04:08:46 -05:00
2018-09-01 12:21:11 -05:00
for (path, text) in files {
res.send_notification(RawNotification::new::<DidOpenTextDocument>(
2018-09-02 07:18:43 -05:00
&DidOpenTextDocumentParams {
2018-09-01 12:21:11 -05:00
text_document: TextDocumentItem {
uri: Url::from_file_path(path).unwrap(),
language_id: "rust".to_string(),
version: 0,
text,
},
},
2018-09-01 12:21:11 -05:00
))
}
res
}
pub fn doc_id(&self, rel_path: &str) -> TextDocumentIdentifier {
let path = self.dir.path().join(rel_path);
2019-02-08 05:49:43 -06:00
TextDocumentIdentifier { uri: Url::from_file_path(path).unwrap() }
2018-09-01 12:21:11 -05:00
}
2019-05-27 06:20:11 -05:00
pub fn notification<N>(&self, params: N::Params)
where
N: Notification,
N::Params: Serialize,
{
let r = RawNotification::new::<N>(&params);
self.send_notification(r)
}
pub fn request<R>(&self, params: R::Params, expected_resp: Value)
2018-09-01 12:21:11 -05:00
where
R: Request,
R::Params: Serialize,
{
2019-01-10 15:37:10 -06:00
let actual = self.send_request::<R>(params);
2019-06-03 09:01:10 -05:00
if let Some((expected_part, actual_part)) = find_mismatch(&expected_resp, &actual) {
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(),
2019-06-03 09:01:10 -05:00
);
}
2018-09-01 12:21:11 -05:00
}
2019-01-10 15:37:10 -06:00
pub fn send_request<R>(&self, params: R::Params) -> Value
2018-09-01 12:21:11 -05:00
where
R: Request,
R::Params: Serialize,
{
2019-01-10 15:37:10 -06:00
let id = self.req_id.get();
self.req_id.set(id + 1);
2018-09-02 07:18:43 -05:00
let r = RawRequest::new::<R>(id, &params);
2018-09-02 08:36:03 -05:00
self.send_request_(r)
}
fn send_request_(&self, r: RawRequest) -> Value {
2018-09-02 08:36:03 -05:00
let id = r.id;
self.worker.sender().send(RawMessage::Request(r)).unwrap();
2018-09-02 04:34:06 -05:00
while let Some(msg) = self.recv() {
2018-09-01 12:21:11 -05: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");
}
pub fn wait_until_workspace_is_loaded(&self) {
self.wait_for_message_cond(1, &|msg: &RawMessage| match msg {
RawMessage::Notification(n) if n.method == ShowMessage::METHOD => {
2019-03-05 14:25:24 -06:00
let msg = n.clone().cast::<req::ShowMessage>().unwrap();
msg.message.starts_with("workspace loaded")
}
_ => false,
})
}
fn wait_for_message_cond(&self, n: usize, cond: &dyn Fn(&RawMessage) -> bool) {
2018-09-03 16:49:21 -05:00
let mut total = 0;
2018-09-02 06:46:15 -05:00
for msg in self.messages.borrow().iter() {
if cond(msg) {
2018-09-03 16:49:21 -05:00
total += 1
2018-09-02 06:46:15 -05:00
}
}
2018-09-03 16:49:21 -05:00
while total < n {
let msg = self.recv().expect("no response");
if cond(&msg) {
2018-09-03 16:49:21 -05:00
total += 1;
2018-09-02 06:46:15 -05:00
}
}
}
2018-09-02 04:34:06 -05:00
fn recv(&self) -> Option<RawMessage> {
recv_timeout(&self.worker.receiver()).map(|msg| {
self.messages.borrow_mut().push(msg.clone());
msg
})
2018-09-02 04:34:06 -05:00
}
2018-09-01 12:21:11 -05:00
fn send_notification(&self, not: RawNotification) {
self.worker.sender().send(RawMessage::Notification(not)).unwrap();
2018-09-01 12:21:11 -05:00
}
pub fn path(&self) -> &Path {
self.dir.path()
}
2018-09-01 12:21:11 -05:00
}
impl Drop for Server {
fn drop(&mut self) {
2019-01-10 15:37:10 -06:00
self.send_request::<Shutdown>(());
2018-09-01 12:21:11 -05:00
}
}
2018-09-08 04:08:46 -05:00
fn recv_timeout(receiver: &Receiver<RawMessage>) -> Option<RawMessage> {
2019-05-29 14:14:06 -05:00
let timeout = Duration::from_secs(120);
2018-09-08 04:08:46 -05:00
select! {
2018-12-30 14:23:31 -06:00
recv(receiver) -> msg => msg.ok(),
recv(after(timeout)) -> _ => panic!("timed out"),
2018-09-08 04:08:46 -05:00
}
}