2018-09-01 17:21:11 +00:00
|
|
|
use std::{
|
2018-09-02 09:34:06 +00:00
|
|
|
cell::{Cell, RefCell},
|
2018-10-15 21:44:23 +00:00
|
|
|
fs,
|
2019-04-13 17:45:21 +00:00
|
|
|
path::{Path, PathBuf},
|
2018-09-02 11:46:15 +00:00
|
|
|
sync::Once,
|
2018-10-15 21:44:23 +00:00
|
|
|
time::Duration,
|
2018-09-01 17:21:11 +00:00
|
|
|
};
|
|
|
|
|
2018-12-06 18:03:39 +00:00
|
|
|
use crossbeam_channel::{after, select, Receiver};
|
2019-08-30 17:18:57 +00:00
|
|
|
use lsp_server::{Connection, Message, Notification, Request};
|
2019-01-14 10:55:56 +00:00
|
|
|
use lsp_types::{
|
2020-06-23 11:20:53 +00:00
|
|
|
notification::Exit, request::Shutdown, TextDocumentIdentifier, Url, WorkDoneProgress,
|
2018-09-01 17:21:11 +00:00
|
|
|
};
|
2020-05-10 17:25:37 +00:00
|
|
|
use lsp_types::{ProgressParams, ProgressParamsValue};
|
2020-11-13 16:38:26 +00:00
|
|
|
use project_model::{CargoConfig, ProjectManifest};
|
2020-04-01 16:46:26 +00:00
|
|
|
use rust_analyzer::{
|
2021-01-05 13:57:05 +00:00
|
|
|
config::{Config, FilesConfig, FilesWatcher, LinkedProject},
|
2020-05-10 17:25:37 +00:00
|
|
|
main_loop,
|
2020-04-01 16:46:26 +00:00
|
|
|
};
|
2020-07-08 16:22:57 +00:00
|
|
|
use serde::Serialize;
|
|
|
|
use serde_json::{to_string_pretty, Value};
|
|
|
|
use test_utils::{find_mismatch, Fixture};
|
|
|
|
use vfs::AbsPathBuf;
|
2018-09-01 17:21:11 +00:00
|
|
|
|
2020-07-23 20:26:25 +00:00
|
|
|
use crate::testdir::TestDir;
|
|
|
|
|
2020-11-02 15:31:38 +00:00
|
|
|
pub(crate) struct Project<'a> {
|
2019-04-13 18:33:49 +00:00
|
|
|
fixture: &'a str,
|
2019-08-19 12:41:18 +00:00
|
|
|
with_sysroot: bool,
|
2020-07-23 20:26:25 +00:00
|
|
|
tmp_dir: Option<TestDir>,
|
2019-04-13 18:33:49 +00:00
|
|
|
roots: Vec<PathBuf>,
|
2020-04-01 16:41:43 +00:00
|
|
|
config: Option<Box<dyn Fn(&mut Config)>>,
|
2019-03-05 21:29:23 +00:00
|
|
|
}
|
|
|
|
|
2019-04-13 18:33:49 +00:00
|
|
|
impl<'a> Project<'a> {
|
2020-11-02 15:31:38 +00:00
|
|
|
pub(crate) fn with_fixture(fixture: &str) -> Project {
|
2020-03-31 17:16:25 +00:00
|
|
|
Project { fixture, tmp_dir: None, roots: vec![], with_sysroot: false, config: None }
|
2019-04-13 18:33:49 +00:00
|
|
|
}
|
|
|
|
|
2020-11-02 15:31:38 +00:00
|
|
|
pub(crate) fn tmp_dir(mut self, tmp_dir: TestDir) -> Project<'a> {
|
2019-04-13 18:33:49 +00:00
|
|
|
self.tmp_dir = Some(tmp_dir);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2020-06-03 10:22:01 +00:00
|
|
|
pub(crate) fn root(mut self, path: &str) -> Project<'a> {
|
2019-04-13 18:33:49 +00:00
|
|
|
self.roots.push(path.into());
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2020-11-13 16:38:26 +00:00
|
|
|
pub(crate) fn with_sysroot(mut self, yes: bool) -> Project<'a> {
|
|
|
|
self.with_sysroot = yes;
|
2019-08-19 12:41:18 +00:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2020-11-02 15:31:38 +00:00
|
|
|
pub(crate) fn with_config(mut self, config: impl Fn(&mut Config) + 'static) -> Project<'a> {
|
2020-03-31 17:16:25 +00:00
|
|
|
self.config = Some(Box::new(config));
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2020-11-02 15:31:38 +00:00
|
|
|
pub(crate) fn server(self) -> Server {
|
2020-07-23 20:26:25 +00:00
|
|
|
let tmp_dir = self.tmp_dir.unwrap_or_else(|| TestDir::new());
|
2019-04-13 18:33:49 +00:00
|
|
|
static INIT: Once = Once::new();
|
|
|
|
INIT.call_once(|| {
|
2020-02-18 12:53:02 +00:00
|
|
|
env_logger::builder().is_test(true).try_init().unwrap();
|
2020-08-12 14:32:36 +00:00
|
|
|
profile::init_from(crate::PROFILE);
|
2019-04-13 18:33:49 +00:00
|
|
|
});
|
2018-09-01 17:21:11 +00:00
|
|
|
|
2020-06-23 16:58:45 +00:00
|
|
|
for entry in Fixture::parse(self.fixture) {
|
2020-06-23 16:34:50 +00:00
|
|
|
let path = tmp_dir.path().join(&entry.path['/'.len_utf8()..]);
|
2019-04-13 18:33:49 +00:00
|
|
|
fs::create_dir_all(path.parent().unwrap()).unwrap();
|
|
|
|
fs::write(path.as_path(), entry.text.as_bytes()).unwrap();
|
|
|
|
}
|
|
|
|
|
2020-06-24 11:34:24 +00:00
|
|
|
let tmp_dir_path = AbsPathBuf::assert(tmp_dir.path().to_path_buf());
|
2020-06-03 10:22:01 +00:00
|
|
|
let mut roots =
|
2020-06-24 11:34:24 +00:00
|
|
|
self.roots.into_iter().map(|root| tmp_dir_path.join(root)).collect::<Vec<_>>();
|
2020-06-03 10:22:01 +00:00
|
|
|
if roots.is_empty() {
|
2020-06-24 11:34:24 +00:00
|
|
|
roots.push(tmp_dir_path.clone());
|
2020-06-03 10:22:01 +00:00
|
|
|
}
|
|
|
|
let linked_projects = roots
|
|
|
|
.into_iter()
|
|
|
|
.map(|it| ProjectManifest::discover_single(&it).unwrap())
|
|
|
|
.map(LinkedProject::from)
|
|
|
|
.collect::<Vec<_>>();
|
2019-04-13 18:33:49 +00:00
|
|
|
|
2020-04-01 16:41:43 +00:00
|
|
|
let mut config = Config {
|
2021-01-05 13:57:05 +00:00
|
|
|
caps: lsp_types::ClientCapabilities {
|
|
|
|
text_document: Some(lsp_types::TextDocumentClientCapabilities {
|
|
|
|
definition: Some(lsp_types::GotoCapability {
|
|
|
|
link_support: Some(true),
|
|
|
|
..Default::default()
|
|
|
|
}),
|
|
|
|
code_action: Some(lsp_types::CodeActionClientCapabilities {
|
|
|
|
code_action_literal_support: Some(
|
|
|
|
lsp_types::CodeActionLiteralSupport::default(),
|
|
|
|
),
|
|
|
|
..Default::default()
|
|
|
|
}),
|
|
|
|
..Default::default()
|
|
|
|
}),
|
|
|
|
window: Some(lsp_types::WindowClientCapabilities {
|
|
|
|
work_done_progress: Some(true),
|
|
|
|
..Default::default()
|
|
|
|
}),
|
2020-04-26 22:54:05 +00:00
|
|
|
..Default::default()
|
|
|
|
},
|
2020-11-13 16:38:26 +00:00
|
|
|
cargo: CargoConfig { no_sysroot: !self.with_sysroot, ..Default::default() },
|
2020-06-03 10:22:01 +00:00
|
|
|
linked_projects,
|
2020-06-24 10:27:13 +00:00
|
|
|
files: FilesConfig { watcher: FilesWatcher::Client, exclude: Vec::new() },
|
2020-06-24 11:34:24 +00:00
|
|
|
..Config::new(tmp_dir_path)
|
2020-04-01 16:41:43 +00:00
|
|
|
};
|
2020-03-31 17:16:25 +00:00
|
|
|
if let Some(f) = &self.config {
|
|
|
|
f(&mut config)
|
|
|
|
}
|
|
|
|
|
2020-06-23 11:20:53 +00:00
|
|
|
Server::new(tmp_dir, config)
|
2018-09-01 17:21:11 +00:00
|
|
|
}
|
2019-04-13 18:33:49 +00:00
|
|
|
}
|
|
|
|
|
2020-11-02 15:31:38 +00:00
|
|
|
pub(crate) fn project(fixture: &str) -> Server {
|
2019-04-13 18:33:49 +00:00
|
|
|
Project::with_fixture(fixture).server()
|
2018-09-01 17:21:11 +00:00
|
|
|
}
|
|
|
|
|
2020-11-02 15:31:38 +00:00
|
|
|
pub(crate) struct Server {
|
2020-11-16 20:10:13 +00:00
|
|
|
req_id: Cell<i32>,
|
2019-08-30 14:24:11 +00:00
|
|
|
messages: RefCell<Vec<Message>>,
|
2019-08-30 17:18:57 +00:00
|
|
|
_thread: jod_thread::JoinHandle<()>,
|
|
|
|
client: Connection,
|
2020-03-28 12:19:05 +00:00
|
|
|
/// XXX: remove the tempdir last
|
2020-07-23 20:26:25 +00:00
|
|
|
dir: TestDir,
|
2018-09-01 17:21:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Server {
|
2020-07-23 20:26:25 +00:00
|
|
|
fn new(dir: TestDir, config: Config) -> Server {
|
2019-08-30 17:18:57 +00:00
|
|
|
let (connection, client) = Connection::memory();
|
2019-04-13 18:33:49 +00:00
|
|
|
|
2019-08-30 17:18:57 +00:00
|
|
|
let _thread = jod_thread::Builder::new()
|
|
|
|
.name("test server".to_string())
|
2020-06-03 10:22:01 +00:00
|
|
|
.spawn(move || main_loop(config, connection).unwrap())
|
2019-08-30 17:18:57 +00:00
|
|
|
.expect("failed to spawn a thread");
|
|
|
|
|
2020-06-23 11:20:53 +00:00
|
|
|
Server { req_id: Cell::new(1), dir, messages: Default::default(), client, _thread }
|
2018-09-01 17:21:11 +00:00
|
|
|
}
|
|
|
|
|
2020-11-02 15:31:38 +00:00
|
|
|
pub(crate) fn doc_id(&self, rel_path: &str) -> TextDocumentIdentifier {
|
2018-09-01 17:21:11 +00:00
|
|
|
let path = self.dir.path().join(rel_path);
|
2019-02-08 11:49:43 +00:00
|
|
|
TextDocumentIdentifier { uri: Url::from_file_path(path).unwrap() }
|
2018-09-01 17:21:11 +00:00
|
|
|
}
|
|
|
|
|
2020-11-02 15:31:38 +00:00
|
|
|
pub(crate) fn notification<N>(&self, params: N::Params)
|
2019-05-27 11:20:11 +00:00
|
|
|
where
|
2019-08-30 14:24:11 +00:00
|
|
|
N: lsp_types::notification::Notification,
|
2019-05-27 11:20:11 +00:00
|
|
|
N::Params: Serialize,
|
|
|
|
{
|
2019-08-30 14:24:11 +00:00
|
|
|
let r = Notification::new(N::METHOD.to_string(), params);
|
2019-05-27 11:20:11 +00:00
|
|
|
self.send_notification(r)
|
|
|
|
}
|
|
|
|
|
2020-11-02 15:31:38 +00:00
|
|
|
pub(crate) fn request<R>(&self, params: R::Params, expected_resp: Value)
|
2018-09-01 17:21:11 +00:00
|
|
|
where
|
2019-08-30 14:24:11 +00:00
|
|
|
R: lsp_types::request::Request,
|
2018-09-01 17:21:11 +00:00
|
|
|
R::Params: Serialize,
|
|
|
|
{
|
2019-01-10 21:37:10 +00:00
|
|
|
let actual = self.send_request::<R>(params);
|
2019-06-03 14:01:10 +00:00
|
|
|
if let Some((expected_part, actual_part)) = find_mismatch(&expected_resp, &actual) {
|
|
|
|
panic!(
|
2018-12-06 20:07:31 +00:00
|
|
|
"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 14:01:10 +00:00
|
|
|
);
|
2018-12-06 20:07:31 +00:00
|
|
|
}
|
2018-09-01 17:21:11 +00:00
|
|
|
}
|
|
|
|
|
2020-11-02 15:31:38 +00:00
|
|
|
pub(crate) fn send_request<R>(&self, params: R::Params) -> Value
|
2018-09-01 17:21:11 +00:00
|
|
|
where
|
2019-08-30 14:24:11 +00:00
|
|
|
R: lsp_types::request::Request,
|
2018-09-01 17:21:11 +00:00
|
|
|
R::Params: Serialize,
|
|
|
|
{
|
2019-01-10 21:37:10 +00:00
|
|
|
let id = self.req_id.get();
|
2020-11-16 20:10:13 +00:00
|
|
|
self.req_id.set(id.wrapping_add(1));
|
2019-01-10 21:37:10 +00:00
|
|
|
|
2019-08-30 14:24:11 +00:00
|
|
|
let r = Request::new(id.into(), R::METHOD.to_string(), params);
|
2018-09-02 13:36:03 +00:00
|
|
|
self.send_request_(r)
|
|
|
|
}
|
2019-08-30 14:24:11 +00:00
|
|
|
fn send_request_(&self, r: Request) -> Value {
|
|
|
|
let id = r.id.clone();
|
2019-08-30 17:18:57 +00:00
|
|
|
self.client.sender.send(r.into()).unwrap();
|
2018-09-02 09:34:06 +00:00
|
|
|
while let Some(msg) = self.recv() {
|
2018-09-01 17:21:11 +00:00
|
|
|
match msg {
|
2020-06-24 10:27:13 +00:00
|
|
|
Message::Request(req) => {
|
2020-07-02 13:32:56 +00:00
|
|
|
if req.method == "window/workDoneProgress/create" {
|
|
|
|
continue;
|
2020-06-24 10:27:13 +00:00
|
|
|
}
|
2020-07-02 13:32:56 +00:00
|
|
|
if req.method == "client/registerCapability" {
|
|
|
|
let params = req.params.to_string();
|
|
|
|
if ["workspace/didChangeWatchedFiles", "textDocument/didSave"]
|
|
|
|
.iter()
|
|
|
|
.any(|&it| params.contains(it))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
panic!("unexpected request: {:?}", req)
|
2020-06-24 10:27:13 +00:00
|
|
|
}
|
2019-08-30 14:24:11 +00:00
|
|
|
Message::Notification(_) => (),
|
|
|
|
Message::Response(res) => {
|
2018-09-01 17:21:11 +00:00
|
|
|
assert_eq!(res.id, id);
|
|
|
|
if let Some(err) = res.error {
|
|
|
|
panic!("error response: {:#?}", err);
|
|
|
|
}
|
|
|
|
return res.result.unwrap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
panic!("no response");
|
|
|
|
}
|
2020-11-02 15:31:38 +00:00
|
|
|
pub(crate) fn wait_until_workspace_is_loaded(self) -> Server {
|
2019-08-30 14:24:11 +00:00
|
|
|
self.wait_for_message_cond(1, &|msg: &Message| match msg {
|
2020-03-14 05:06:49 +00:00
|
|
|
Message::Notification(n) if n.method == "$/progress" => {
|
|
|
|
match n.clone().extract::<ProgressParams>("$/progress").unwrap() {
|
|
|
|
ProgressParams {
|
2020-05-10 17:25:37 +00:00
|
|
|
token: lsp_types::ProgressToken::String(ref token),
|
2020-03-14 05:06:49 +00:00
|
|
|
value: ProgressParamsValue::WorkDone(WorkDoneProgress::End(_)),
|
2020-06-11 09:04:09 +00:00
|
|
|
} if token == "rustAnalyzer/roots scanned" => true,
|
2020-03-14 05:06:49 +00:00
|
|
|
_ => false,
|
|
|
|
}
|
2018-10-15 21:44:23 +00:00
|
|
|
}
|
|
|
|
_ => false,
|
2020-08-25 17:02:28 +00:00
|
|
|
});
|
|
|
|
self
|
2019-03-07 14:46:17 +00:00
|
|
|
}
|
2019-08-30 14:24:11 +00:00
|
|
|
fn wait_for_message_cond(&self, n: usize, cond: &dyn Fn(&Message) -> bool) {
|
2018-09-03 21:49:21 +00:00
|
|
|
let mut total = 0;
|
2018-09-02 11:46:15 +00:00
|
|
|
for msg in self.messages.borrow().iter() {
|
2019-03-07 14:46:17 +00:00
|
|
|
if cond(msg) {
|
2018-09-03 21:49:21 +00:00
|
|
|
total += 1
|
2018-09-02 11:46:15 +00:00
|
|
|
}
|
|
|
|
}
|
2018-09-03 21:49:21 +00:00
|
|
|
while total < n {
|
|
|
|
let msg = self.recv().expect("no response");
|
2019-03-07 14:46:17 +00:00
|
|
|
if cond(&msg) {
|
2018-09-03 21:49:21 +00:00
|
|
|
total += 1;
|
2018-09-02 11:46:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-08-30 14:24:11 +00:00
|
|
|
fn recv(&self) -> Option<Message> {
|
2019-08-30 17:18:57 +00:00
|
|
|
recv_timeout(&self.client.receiver).map(|msg| {
|
2018-10-15 21:44:23 +00:00
|
|
|
self.messages.borrow_mut().push(msg.clone());
|
|
|
|
msg
|
|
|
|
})
|
2018-09-02 09:34:06 +00:00
|
|
|
}
|
2019-08-30 14:24:11 +00:00
|
|
|
fn send_notification(&self, not: Notification) {
|
2019-08-30 17:18:57 +00:00
|
|
|
self.client.sender.send(Message::Notification(not)).unwrap();
|
2018-09-01 17:21:11 +00:00
|
|
|
}
|
2019-04-13 17:45:21 +00:00
|
|
|
|
2020-11-02 15:31:38 +00:00
|
|
|
pub(crate) fn path(&self) -> &Path {
|
2019-04-13 17:45:21 +00:00
|
|
|
self.dir.path()
|
|
|
|
}
|
2018-09-01 17:21:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for Server {
|
|
|
|
fn drop(&mut self) {
|
2019-08-30 17:18:57 +00:00
|
|
|
self.request::<Shutdown>((), Value::Null);
|
|
|
|
self.notification::<Exit>(());
|
2018-09-01 17:21:11 +00:00
|
|
|
}
|
|
|
|
}
|
2018-09-08 09:08:46 +00:00
|
|
|
|
2019-08-30 14:24:11 +00:00
|
|
|
fn recv_timeout(receiver: &Receiver<Message>) -> Option<Message> {
|
2020-07-28 08:24:33 +00:00
|
|
|
let timeout =
|
|
|
|
if cfg!(target_os = "macos") { Duration::from_secs(300) } else { Duration::from_secs(120) };
|
2018-09-08 09:08:46 +00:00
|
|
|
select! {
|
2018-12-30 20:23:31 +00:00
|
|
|
recv(receiver) -> msg => msg.ok(),
|
|
|
|
recv(after(timeout)) -> _ => panic!("timed out"),
|
2018-09-08 09:08:46 +00:00
|
|
|
}
|
|
|
|
}
|