From 2c73b919ce052b074e7dd758c17b4ff29a7939a2 Mon Sep 17 00:00:00 2001 From: Emilio Jesus Gallego Arias Date: Wed, 15 Feb 2023 17:48:50 +0100 Subject: [PATCH] [protocol] New request `coq/getDocument` to get serialized document. cc: https://github.com/ejgallego/coq-serapi/pull/318 --- CHANGES.md | 3 +++ controller/lsp_core.ml | 3 +++ controller/rq_document.ml | 22 ++++++++++++++++++++ controller/rq_document.mli | 8 ++++++++ editor/code/lib/types.ts | 23 +++++++++++++++++++++ editor/code/package.json | 4 ++++ editor/code/src/client.ts | 21 ++++++++++++++++++- etc/doc/PROTOCOL.md | 42 ++++++++++++++++++++++++++++++++++++++ lsp/jCoq.ml | 7 +++++++ lsp/jFleche.ml | 25 +++++++++++++++++++++++ lsp/jFleche.mli | 25 +++++++++++++++++++++++ 11 files changed, 182 insertions(+), 1 deletion(-) create mode 100644 controller/rq_document.ml create mode 100644 controller/rq_document.mli diff --git a/CHANGES.md b/CHANGES.md index dfcc1a38f..55ad2aa2e 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -8,6 +8,9 @@ fixes #321) - Trigger completion on quote char "'" (@ejgallego, #350) - Fix typo on keybinding config for show goals (@tomtomjhj, #357) + - New request `coq/getDocument` to get serialized full document + contents. Thanks to Clément Pit-Claudel for feedback and ideas. + (@ejgallego, #350) # coq-lsp 0.1.5: Form --------------------- diff --git a/controller/lsp_core.ml b/controller/lsp_core.ml index 0ff70ab62..b0518dd7f 100644 --- a/controller/lsp_core.ml +++ b/controller/lsp_core.ml @@ -253,6 +253,7 @@ let do_document_request ~params ~handler = Postpone (PendingRequest.DocRequest { uri; handler }) let do_symbols = do_document_request ~handler:Rq_symbols.symbols +let do_document = do_document_request ~handler:Rq_document.request let do_trace params = let trace = string_field "value" params in @@ -341,6 +342,8 @@ let dispatch_request ~method_ ~params : RAction.t = | "textDocument/hover" -> do_hover ~params (* Proof-specific stuff *) | "proof/goals" -> do_goals ~params + (* Coq specific stuff *) + | "coq/getDocument" -> do_document ~params (* Generic handler *) | msg -> LIO.trace "no_handler" msg; diff --git a/controller/rq_document.ml b/controller/rq_document.ml new file mode 100644 index 000000000..77aed6c80 --- /dev/null +++ b/controller/rq_document.ml @@ -0,0 +1,22 @@ +(************************************************************************) +(* Coq Language Server Protocol -- Document *) +(* Copyright 2019 MINES ParisTech -- Dual License LGPL 2.1 / GPL3+ *) +(* Copyright 2019-2023 Inria -- Dual License LGPL 2.1 / GPL3+ *) +(* Written by: Emilio J. Gallego Arias *) +(************************************************************************) + +let to_span { Fleche.Doc.Node.range; ast; _ } = + let span = Option.map (fun { Fleche.Doc.Node.Ast.v; _ } -> v) ast in + Lsp.JFleche.RangedSpan.{ range; span } + +let to_completed = function + | Fleche.Doc.Completion.Yes range -> + { Lsp.JFleche.CompletionStatus.status = `Yes; range } + | Stopped range -> { status = `Stopped; range } + | Failed range | FailedPermanent range -> { status = `Failed; range } + +let request ~doc = + let { Fleche.Doc.nodes; completed; _ } = doc in + let spans = List.map to_span nodes in + let completed = to_completed completed in + Lsp.JFleche.FlecheDocument.({ spans; completed } |> to_yojson) diff --git a/controller/rq_document.mli b/controller/rq_document.mli new file mode 100644 index 000000000..033a09783 --- /dev/null +++ b/controller/rq_document.mli @@ -0,0 +1,8 @@ +(************************************************************************) +(* Coq Language Server Protocol -- Document *) +(* Copyright 2019 MINES ParisTech -- Dual License LGPL 2.1 / GPL3+ *) +(* Copyright 2019-2023 Inria -- Dual License LGPL 2.1 / GPL3+ *) +(* Written by: Emilio J. Gallego Arias *) +(************************************************************************) + +val request : Requests.document_request diff --git a/editor/code/lib/types.ts b/editor/code/lib/types.ts index 6958c68b2..86f74eec6 100644 --- a/editor/code/lib/types.ts +++ b/editor/code/lib/types.ts @@ -53,3 +53,26 @@ export type Pp = | ["Pp_comment", string[]]; export type PpString = Pp | string; + +export interface FlecheDocumentParams { + textDocument: VersionedTextDocumentIdentifier; +} + +// Status of the document, Yes if fully checked, range contains the last seen lexical token +interface CompletionStatus { + status: ["Yes" | "Stopped" | "Failed"]; + range: Range; +} + +// Implementation-specific span information, for now the serialized Ast if present. +type SpanInfo = any; + +interface RangedSpan { + range: Range; + span?: SpanInfo; +} + +export interface FlecheDocument { + spans: RangedSpan[]; + completed: CompletionStatus; +} diff --git a/editor/code/package.json b/editor/code/package.json index 3c06f703f..ad90682ac 100644 --- a/editor/code/package.json +++ b/editor/code/package.json @@ -88,6 +88,10 @@ { "command": "coq-lsp.goals", "title": "Coq LSP: Show Goals at Point" + }, + { + "command": "coq-lsp.document", + "title": "Coq LSP: Get document in serialized format, print in console" } ], "keybindings": [ diff --git a/editor/code/src/client.ts b/editor/code/src/client.ts index 6689c6e4b..e8b7a1b7f 100644 --- a/editor/code/src/client.ts +++ b/editor/code/src/client.ts @@ -14,7 +14,11 @@ import { LanguageClientOptions, RevealOutputChannelOn, } from "vscode-languageclient/node"; - +import { + RequestType, + VersionedTextDocumentIdentifier, +} from "vscode-languageclient"; +import { FlecheDocumentParams, FlecheDocument } from "../lib/types"; import { CoqLspClientConfig, CoqLspServerConfig } from "./config"; import { InfoPanel } from "./goals"; import { FileProgressManager } from "./progress"; @@ -138,8 +142,23 @@ export function activate(context: ExtensionContext): void { context.subscriptions.push(goalsHook); + const docReq = new RequestType( + "coq/getDocument" + ); + + let getDocument = (editor: TextEditor) => { + let uri = editor.document.uri; + let version = editor.document.version; + let textDocument = VersionedTextDocumentIdentifier.create( + uri.toString(), + version + ); + let params: FlecheDocumentParams = { textDocument }; + client.sendRequest(docReq, params).then((fd) => console.log(fd)); + }; coqCommand("restart", restart); coqEditorCommand("goals", goals); + coqEditorCommand("document", getDocument); restart(); } diff --git a/etc/doc/PROTOCOL.md b/etc/doc/PROTOCOL.md index 6609419f9..0c6927b30 100644 --- a/etc/doc/PROTOCOL.md +++ b/etc/doc/PROTOCOL.md @@ -201,3 +201,45 @@ interface CoqFileProgressParams { #### Changelog - v0.1.1: exact copy from Lean protocol (spec under Apache License) + +### Document Ast Request + +The `coq/getDocument` request returns a serialized version of Fleche's +document. It is modelled after LSP's standard +`textDocument/documentSymbol`, but returns instead the full document +contents as understood by Flèche. + +Caveats: Flèche notion of document is evolving, in particular you +should not assume that the document will remain a list, but it will +become a tree at some point. + +```typescript +interface FlecheDocumentParams { + textDocument: VersionedTextDocumentIdentifier; +} +``` + +```typescript +// Status of the document, Yes if fully checked, range contains the last seen lexical token +interface CompletionStatus { + status : ['Yes' | 'Stopped' | 'Failed'] + range : Range +}; + +// Implementation-specific span information, for now the serialized Ast if present. +type SpanInfo = any; + +interface RangedSpan { + range : Range; + span?: SpanInfo +}; + +interface FlecheDocument { + spans: RangedSpan[]; + completed : CompletionStatus +}; +``` + +#### Changelog + +- v0.1.6: initial version diff --git a/lsp/jCoq.ml b/lsp/jCoq.ml index 8e7897d15..d18965ce8 100644 --- a/lsp/jCoq.ml +++ b/lsp/jCoq.ml @@ -56,3 +56,10 @@ module Goals = struct type 'pp reified_pp = ('pp reified_goal, 'pp) goals [@@deriving to_yojson] end + +module Ast = struct + type t = Coq.Ast.t + + let to_yojson x = + Serlib.Ser_vernacexpr.vernac_control_to_yojson (Coq.Ast.to_coq x) +end diff --git a/lsp/jFleche.ml b/lsp/jFleche.ml index a113418b3..838df3457 100644 --- a/lsp/jFleche.ml +++ b/lsp/jFleche.ml @@ -16,6 +16,7 @@ (************************************************************************) module Pp = JCoq.Pp +module Ast = JCoq.Ast module Lang = JLang module Config = struct @@ -172,3 +173,27 @@ module CompletionData = struct } [@@deriving yojson] end + +module CompletionStatus = struct + type t = + { status : [ `Yes | `Stopped | `Failed ] + ; range : Lang.Range.t + } + [@@deriving yojson] +end + +module RangedSpan = struct + type t = + { range : Lang.Range.t + ; span : Ast.t option [@default None] + } + [@@deriving to_yojson] +end + +module FlecheDocument = struct + type t = + { spans : RangedSpan.t list + ; completed : CompletionStatus.t + } + [@@deriving to_yojson] +end diff --git a/lsp/jFleche.mli b/lsp/jFleche.mli index fb0230007..35a8569ba 100644 --- a/lsp/jFleche.mli +++ b/lsp/jFleche.mli @@ -132,3 +132,28 @@ module CompletionData : sig } [@@deriving yojson] end + +(** Coq-lsp-specific *) +module CompletionStatus : sig + type t = + { status : [ `Yes | `Stopped | `Failed ] + ; range : Lang.Range.t + } + [@@deriving yojson] +end + +module RangedSpan : sig + type t = + { range : Lang.Range.t + ; span : Coq.Ast.t option [@default None] + } + [@@deriving to_yojson] +end + +module FlecheDocument : sig + type t = + { spans : RangedSpan.t list + ; completed : CompletionStatus.t + } + [@@deriving to_yojson] +end