255 lines
8.1 KiB
Rust
Raw Normal View History

2018-09-01 20:21:11 +03:00
use std::{
2018-09-02 12:34:06 +03:00
cell::{Cell, RefCell},
fs,
path::{Path, PathBuf},
2018-09-02 14:46:15 +03:00
sync::Once,
time::Duration,
2018-09-01 20:21:11 +03:00
};
2018-12-06 21:03:39 +03:00
use crossbeam_channel::{after, select, Receiver};
2019-08-30 20:18:57 +03:00
use lsp_server::{Connection, Message, Notification, Request};
2019-01-14 13:55:56 +03:00
use lsp_types::{
2020-06-23 13:20:53 +02:00
notification::Exit, request::Shutdown, TextDocumentIdentifier, Url, WorkDoneProgress,
2018-09-01 20:21:11 +03:00
};
2020-05-10 19:25:37 +02:00
use lsp_types::{ProgressParams, ProgressParamsValue};
2018-09-01 20:21:11 +03:00
use serde::Serialize;
use serde_json::{to_string_pretty, Value};
2018-12-30 23:33:04 +03:00
use tempfile::TempDir;
2020-06-23 18:58:45 +02:00
use test_utils::{find_mismatch, Fixture};
2018-09-01 20:21:11 +03:00
use ra_db::AbsPathBuf;
use ra_project_model::ProjectManifest;
2020-04-01 18:46:26 +02:00
use rust_analyzer::{
config::{ClientCapsConfig, Config, FilesConfig, FilesWatcher, LinkedProject},
2020-05-10 19:25:37 +02:00
main_loop,
2020-04-01 18:46:26 +02:00
};
2018-09-01 20:21:11 +03:00
pub struct Project<'a> {
fixture: &'a str,
2019-08-19 15:41:18 +03:00
with_sysroot: bool,
tmp_dir: Option<TempDir>,
roots: Vec<PathBuf>,
2020-04-01 18:41:43 +02:00
config: Option<Box<dyn Fn(&mut Config)>>,
}
impl<'a> Project<'a> {
pub fn with_fixture(fixture: &str) -> Project {
Project { fixture, tmp_dir: None, roots: vec![], with_sysroot: false, config: None }
}
pub fn tmp_dir(mut self, tmp_dir: TempDir) -> Project<'a> {
self.tmp_dir = Some(tmp_dir);
self
}
pub(crate) fn root(mut self, path: &str) -> Project<'a> {
self.roots.push(path.into());
self
}
2019-08-19 15:41:18 +03:00
pub fn with_sysroot(mut self, sysroot: bool) -> Project<'a> {
self.with_sysroot = sysroot;
self
}
2020-04-01 18:41:43 +02:00
pub fn with_config(mut self, config: impl Fn(&mut Config) + 'static) -> Project<'a> {
self.config = Some(Box::new(config));
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(|| {
env_logger::builder().is_test(true).try_init().unwrap();
2020-04-25 14:52:23 +02:00
ra_prof::init_from(crate::PROFILE);
});
2018-09-01 20:21:11 +03:00
2020-06-23 18:58:45 +02:00
for entry in Fixture::parse(self.fixture) {
2020-06-23 18:34:50 +02:00
let path = tmp_dir.path().join(&entry.path['/'.len_utf8()..]);
fs::create_dir_all(path.parent().unwrap()).unwrap();
fs::write(path.as_path(), entry.text.as_bytes()).unwrap();
}
let tmp_dir_path = AbsPathBuf::assert(tmp_dir.path().to_path_buf());
let mut roots =
self.roots.into_iter().map(|root| tmp_dir_path.join(root)).collect::<Vec<_>>();
if roots.is_empty() {
roots.push(tmp_dir_path.clone());
}
let linked_projects = roots
.into_iter()
.map(|it| ProjectManifest::discover_single(&it).unwrap())
.map(LinkedProject::from)
.collect::<Vec<_>>();
2020-04-01 18:41:43 +02:00
let mut config = Config {
client_caps: ClientCapsConfig {
location_link: true,
code_action_literals: true,
work_done_progress: true,
..Default::default()
},
2020-04-01 18:41:43 +02:00
with_sysroot: self.with_sysroot,
linked_projects,
files: FilesConfig { watcher: FilesWatcher::Client, exclude: Vec::new() },
..Config::new(tmp_dir_path)
2020-04-01 18:41:43 +02:00
};
if let Some(f) = &self.config {
f(&mut config)
}
2020-06-23 13:20:53 +02:00
Server::new(tmp_dir, config)
2018-09-01 20:21:11 +03: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>,
messages: RefCell<Vec<Message>>,
2019-08-30 20:18:57 +03:00
_thread: jod_thread::JoinHandle<()>,
client: Connection,
2020-03-28 13:19:05 +01:00
/// XXX: remove the tempdir last
dir: TempDir,
2018-09-01 20:21:11 +03:00
}
impl Server {
2020-06-23 13:20:53 +02:00
fn new(dir: TempDir, config: Config) -> Server {
2019-08-30 20:18:57 +03:00
let (connection, client) = Connection::memory();
2019-08-30 20:18:57 +03:00
let _thread = jod_thread::Builder::new()
.name("test server".to_string())
.spawn(move || main_loop(config, connection).unwrap())
2019-08-30 20:18:57 +03:00
.expect("failed to spawn a thread");
2020-06-23 13:20:53 +02:00
Server { req_id: Cell::new(1), dir, messages: Default::default(), client, _thread }
2018-09-01 20:21:11 +03:00
}
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
}
2019-05-27 14:20:11 +03:00
pub fn notification<N>(&self, params: N::Params)
where
N: lsp_types::notification::Notification,
2019-05-27 14:20:11 +03:00
N::Params: Serialize,
{
let r = Notification::new(N::METHOD.to_string(), params);
2019-05-27 14:20:11 +03:00
self.send_notification(r)
}
pub fn request<R>(&self, params: R::Params, expected_resp: Value)
2018-09-01 20:21:11 +03:00
where
R: lsp_types::request::Request,
2018-09-01 20:21:11 +03:00
R::Params: Serialize,
{
2019-01-11 00:37:10 +03:00
let actual = self.send_request::<R>(params);
2019-06-03 10:01:10 -04: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 10:01:10 -04:00
);
}
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: lsp_types::request::Request,
2018-09-01 20:21:11 +03:00
R::Params: Serialize,
{
2019-01-11 00:37:10 +03:00
let id = self.req_id.get();
self.req_id.set(id + 1);
let r = Request::new(id.into(), R::METHOD.to_string(), params);
2018-09-02 16:36:03 +03:00
self.send_request_(r)
}
fn send_request_(&self, r: Request) -> Value {
let id = r.id.clone();
2019-08-30 20:18:57 +03:00
self.client.sender.send(r.into()).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 {
Message::Request(req) => {
if req.method != "window/workDoneProgress/create"
&& !(req.method == "client/registerCapability"
&& req.params.to_string().contains("workspace/didChangeWatchedFiles"))
{
panic!("unexpected request: {:?}", req)
}
}
Message::Notification(_) => (),
Message::Response(res) => {
2018-09-01 20:21:11 +03:00
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: &Message| match msg {
Message::Notification(n) if n.method == "$/progress" => {
match n.clone().extract::<ProgressParams>("$/progress").unwrap() {
ProgressParams {
2020-05-10 19:25:37 +02:00
token: lsp_types::ProgressToken::String(ref token),
value: ProgressParamsValue::WorkDone(WorkDoneProgress::End(_)),
2020-06-11 11:04:09 +02:00
} if token == "rustAnalyzer/roots scanned" => true,
_ => false,
}
}
_ => false,
})
}
fn wait_for_message_cond(&self, n: usize, cond: &dyn Fn(&Message) -> 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() {
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");
if cond(&msg) {
2018-09-04 00:49:21 +03:00
total += 1;
2018-09-02 14:46:15 +03:00
}
}
}
fn recv(&self) -> Option<Message> {
2019-08-30 20:18:57 +03:00
recv_timeout(&self.client.receiver).map(|msg| {
self.messages.borrow_mut().push(msg.clone());
msg
})
2018-09-02 12:34:06 +03:00
}
fn send_notification(&self, not: Notification) {
2019-08-30 20:18:57 +03:00
self.client.sender.send(Message::Notification(not)).unwrap();
2018-09-01 20:21:11 +03: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-08-30 20:18:57 +03:00
self.request::<Shutdown>((), Value::Null);
self.notification::<Exit>(());
2018-09-01 20:21:11 +03:00
}
}
2018-09-08 12:08:46 +03:00
fn recv_timeout(receiver: &Receiver<Message>) -> Option<Message> {
2019-05-29 22:14:06 +03:00
let timeout = Duration::from_secs(120);
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
}
}