1use std::sync::Arc;
2
3use crate::lsp_log::LogMenuItem;
4
5use super::*;
6use futures::StreamExt;
7use gpui::{AppContext as _, SemanticVersion, TestAppContext, VisualTestContext};
8use language::{tree_sitter_rust, FakeLspAdapter, Language, LanguageConfig, LanguageMatcher};
9use lsp::LanguageServerName;
10use lsp_log::LogKind;
11use project::{FakeFs, Project};
12use serde_json::json;
13use settings::SettingsStore;
14
15#[gpui::test]
16async fn test_lsp_logs(cx: &mut TestAppContext) {
17 if std::env::var("RUST_LOG").is_ok() {
18 env_logger::init();
19 }
20
21 init_test(cx);
22
23 let fs = FakeFs::new(cx.background_executor.clone());
24 fs.insert_tree(
25 "/the-root",
26 json!({
27 "test.rs": "",
28 "package.json": "",
29 }),
30 )
31 .await;
32
33 let project = Project::test(fs.clone(), ["/the-root".as_ref()], cx).await;
34
35 let language_registry = project.read_with(cx, |project, _| project.languages().clone());
36 language_registry.add(Arc::new(Language::new(
37 LanguageConfig {
38 name: "Rust".into(),
39 matcher: LanguageMatcher {
40 path_suffixes: vec!["rs".to_string()],
41 ..Default::default()
42 },
43 ..Default::default()
44 },
45 Some(tree_sitter_rust::LANGUAGE.into()),
46 )));
47 let mut fake_rust_servers = language_registry.register_fake_lsp(
48 "Rust",
49 FakeLspAdapter {
50 name: "the-rust-language-server",
51 ..Default::default()
52 },
53 );
54
55 let log_store = cx.new(LogStore::new);
56 log_store.update(cx, |store, cx| store.add_project(&project, cx));
57
58 let _rust_buffer = project
59 .update(cx, |project, cx| {
60 project.open_local_buffer_with_lsp("/the-root/test.rs", cx)
61 })
62 .await
63 .unwrap();
64
65 let mut language_server = fake_rust_servers.next().await.unwrap();
66 language_server
67 .receive_notification::<lsp::notification::DidOpenTextDocument>()
68 .await;
69
70 let window =
71 cx.add_window(|window, cx| LspLogView::new(project.clone(), log_store.clone(), window, cx));
72 let log_view = window.root(cx).unwrap();
73 let mut cx = VisualTestContext::from_window(*window, cx);
74
75 language_server.notify::<lsp::notification::LogMessage>(&lsp::LogMessageParams {
76 message: "hello from the server".into(),
77 typ: lsp::MessageType::INFO,
78 });
79 cx.executor().run_until_parked();
80
81 log_view.update(&mut cx, |view, cx| {
82 assert_eq!(
83 view.menu_items(cx).unwrap(),
84 &[LogMenuItem {
85 server_id: language_server.server.server_id(),
86 server_name: LanguageServerName("the-rust-language-server".into()),
87 worktree_root_name: project
88 .read(cx)
89 .worktrees(cx)
90 .next()
91 .unwrap()
92 .read(cx)
93 .root_name()
94 .to_string(),
95 rpc_trace_enabled: false,
96 selected_entry: LogKind::Logs,
97 trace_level: lsp::TraceValue::Off,
98 server_kind: lsp_log::LanguageServerKind::Local {
99 project: project.downgrade()
100 }
101 }]
102 );
103 assert_eq!(view.editor.read(cx).text(cx), "hello from the server\n");
104 });
105}
106
107fn init_test(cx: &mut gpui::TestAppContext) {
108 cx.update(|cx| {
109 let settings_store = SettingsStore::test(cx);
110 cx.set_global(settings_store);
111 theme::init(theme::LoadThemes::JustBase, cx);
112 release_channel::init(SemanticVersion::default(), cx);
113 language::init(cx);
114 client::init_settings(cx);
115 Project::init_settings(cx);
116 editor::init_settings(cx);
117 });
118}