diff --git a/Cargo.lock b/Cargo.lock index 2f3dbc7e801..1cc8bc4a8c5 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2001,6 +2001,7 @@ dependencies = [ "acvm", "async-lsp", "codespan-lsp", + "codespan-reporting", "lsp-types 0.94.0", "noirc_driver", "noirc_errors", diff --git a/crates/lsp/Cargo.toml b/crates/lsp/Cargo.toml index 1714ba9ee01..a1d4a295b9b 100644 --- a/crates/lsp/Cargo.toml +++ b/crates/lsp/Cargo.toml @@ -10,6 +10,7 @@ edition.workspace = true [dependencies] acvm.workspace = true codespan-lsp.workspace = true +codespan-reporting.workspace = true lsp-types.workspace = true noirc_driver.workspace = true noirc_errors.workspace = true diff --git a/crates/lsp/src/lib.rs b/crates/lsp/src/lib.rs index 15bff76321a..7b99e5ac02b 100644 --- a/crates/lsp/src/lib.rs +++ b/crates/lsp/src/lib.rs @@ -1,6 +1,6 @@ use std::{ future::Future, - ops::ControlFlow, + ops::{self, ControlFlow}, pin::Pin, task::{Context, Poll}, }; @@ -10,11 +10,13 @@ use async_lsp::{ router::Router, AnyEvent, AnyNotification, AnyRequest, ClientSocket, Error, LanguageClient, LspService, ResponseError, }; +use codespan_reporting::files; use lsp_types::{ - notification, request, Diagnostic, DiagnosticSeverity, DidChangeConfigurationParams, - DidChangeTextDocumentParams, DidCloseTextDocumentParams, DidOpenTextDocumentParams, - DidSaveTextDocumentParams, InitializeParams, InitializeResult, InitializedParams, - PublishDiagnosticsParams, Range, ServerCapabilities, TextDocumentSyncOptions, + notification, request, CodeLens, CodeLensOptions, CodeLensParams, Command, Diagnostic, + DiagnosticSeverity, DidChangeConfigurationParams, DidChangeTextDocumentParams, + DidCloseTextDocumentParams, DidOpenTextDocumentParams, DidSaveTextDocumentParams, + InitializeParams, InitializeResult, InitializedParams, Position, PublishDiagnosticsParams, + Range, ServerCapabilities, TextDocumentSyncOptions, }; use noirc_driver::Driver; use noirc_errors::{DiagnosticKind, FileDiagnostic}; @@ -22,6 +24,9 @@ use noirc_frontend::graph::CrateType; use serde_json::Value as JsonValue; use tower::Service; +const TEST_COMMAND: &str = "nargo.test"; +const TEST_CODELENS_TITLE: &str = "▶\u{fe0e} Run Test"; + // State for the LSP gets implemented on this struct and is internal to the implementation #[derive(Debug)] struct LspState { @@ -45,6 +50,7 @@ impl NargoLspService { router .request::(on_initialize) .request::(on_shutdown) + .request::(on_code_lens_request) .notification::(on_initialized) .notification::(on_did_change_configuration) .notification::(on_did_open_text_document) @@ -99,16 +105,17 @@ fn on_initialize( _params: InitializeParams, ) -> impl Future> { async { - let text_document_sync = TextDocumentSyncOptions { - save: Some(true.into()), - ..TextDocumentSyncOptions::default() - }; + let text_document_sync = + TextDocumentSyncOptions { save: Some(true.into()), ..Default::default() }; + + let code_lens = CodeLensOptions { resolve_provider: Some(false) }; Ok(InitializeResult { capabilities: ServerCapabilities { text_document_sync: Some(text_document_sync.into()), + code_lens_provider: Some(code_lens), // Add capabilities before this spread when adding support for one - ..ServerCapabilities::default() + ..Default::default() }, server_info: None, }) @@ -122,6 +129,60 @@ fn on_shutdown( async { Ok(()) } } +fn on_code_lens_request( + _state: &mut LspState, + params: CodeLensParams, +) -> impl Future>, ResponseError>> { + async move { + // TODO: Requiring `Language` and `is_opcode_supported` to construct a driver makes for some real stinky code + // The driver should not require knowledge of the backend; instead should be implemented as an independent pass (in nargo?) + let mut driver = Driver::new(&Language::R1CS, Box::new(|_op| false)); + + let file_path = ¶ms.text_document.uri.to_file_path().unwrap(); + + driver.create_local_crate(file_path, CrateType::Binary); + + // We ignore the warnings and errors produced by compilation for producing codelenses + // because we can still get the test functions even if compilation fails + let _ = driver.check_crate(false); + + let fm = driver.file_manager(); + let files = fm.as_simple_files(); + let tests = driver.get_all_test_functions_in_crate_matching(""); + + let mut lenses: Vec = vec![]; + for func_id in tests { + let location = driver.function_meta(&func_id).name.location; + let file_id = location.file; + // TODO(#1681): This file_id never be 0 because the "path" where it maps is the directory, not a file + if file_id.as_usize() != 0 { + continue; + } + + let func_name = driver.function_name(func_id); + + let range = byte_span_to_range(files, file_id.as_usize(), location.span.into()) + .unwrap_or_default(); + + let command = Command { + title: TEST_CODELENS_TITLE.into(), + command: TEST_COMMAND.into(), + arguments: Some(vec![func_name.into()]), + }; + + let lens = CodeLens { range, command: command.into(), data: None }; + + lenses.push(lens); + } + + if lenses.is_empty() { + Ok(None) + } else { + Ok(Some(lenses)) + } + } +} + fn on_initialized( _state: &mut LspState, _params: InitializedParams, @@ -181,7 +242,7 @@ fn on_did_save_text_document( let files = fm.as_simple_files(); for FileDiagnostic { file_id, diagnostic } in file_diagnostics { - // TODO: This file_id never be 0 because the "path" where it maps is the directory, not a file + // TODO(#1681): This file_id never be 0 because the "path" where it maps is the directory, not a file if file_id.as_usize() != 0 { continue; } @@ -190,15 +251,9 @@ fn on_did_save_text_document( // TODO: Should this be the first item in secondaries? Should we bail when we find a range? for sec in diagnostic.secondaries { - // TODO: Codespan ranges are often (always?) off by some amount of characters - if let Ok(codespan_range) = - codespan_lsp::byte_span_to_range(files, file_id.as_usize(), sec.span.into()) - { - // We have to manually attach each because the codespan_lsp restricts lsp-types to the wrong version range - range.start.line = codespan_range.start.line; - range.start.character = codespan_range.start.character; - range.end.line = codespan_range.end.line; - range.end.character = codespan_range.end.character; + // Not using `unwrap_or_default` here because we don't want to overwrite a valid range with a default range + if let Some(r) = byte_span_to_range(files, file_id.as_usize(), sec.span.into()) { + range = r } } let severity = match diagnostic.kind { @@ -227,6 +282,31 @@ fn on_exit(_state: &mut LspState, _params: ()) -> ControlFlow + ?Sized>( + files: &'a F, + file_id: F::FileId, + span: ops::Range, +) -> Option { + // TODO(#1683): Codespan ranges are often (always?) off by some amount of characters + if let Ok(codespan_range) = codespan_lsp::byte_span_to_range(files, file_id, span) { + // We have to manually construct a Range because the codespan_lsp restricts lsp-types to the wrong version range + // TODO: codespan is unmaintained and we should probably subsume it. Ref https://github.com/brendanzab/codespan/issues/345 + let range = Range { + start: Position { + line: codespan_range.start.line, + character: codespan_range.start.character, + }, + end: Position { + line: codespan_range.end.line, + character: codespan_range.end.character, + }, + }; + Some(range) + } else { + None + } +} + #[cfg(test)] mod lsp_tests { use lsp_types::TextDocumentSyncCapability; @@ -247,6 +327,7 @@ mod lsp_tests { text_document_sync: Some(TextDocumentSyncCapability::Options( TextDocumentSyncOptions { save: Some(_), .. } )), + code_lens_provider: Some(CodeLensOptions { resolve_provider: Some(false) }), .. } )); diff --git a/crates/noirc_driver/src/lib.rs b/crates/noirc_driver/src/lib.rs index 2df60fa75b5..f0fd3518546 100644 --- a/crates/noirc_driver/src/lib.rs +++ b/crates/noirc_driver/src/lib.rs @@ -13,6 +13,7 @@ use noirc_evaluator::{create_circuit, ssa_refactor::experimental_create_circuit} use noirc_frontend::graph::{CrateId, CrateName, CrateType, LOCAL_CRATE}; use noirc_frontend::hir::def_map::{Contract, CrateDefMap}; use noirc_frontend::hir::Context; +use noirc_frontend::hir_def::function::FuncMeta; use noirc_frontend::monomorphization::monomorphize; use noirc_frontend::node_interner::FuncId; use serde::{Deserialize, Serialize}; @@ -389,6 +390,10 @@ impl Driver { pub fn function_name(&self, id: FuncId) -> &str { self.context.def_interner.function_name(&id) } + + pub fn function_meta(&self, func_id: &FuncId) -> FuncMeta { + self.context.def_interner.function_meta(func_id) + } } impl Default for Driver {