From 1e5b753bf46a4eb4fb32a062d6162063303f6cc7 Mon Sep 17 00:00:00 2001 From: =?utf8?q?Jo=C3=A3o=20T=C3=A1vora?= Date: Wed, 16 Aug 2017 02:41:17 +0100 Subject: [PATCH 1/1] Initial commit --- lisp/progmodes/eglot.el | 393 ++++++++++++++++++++++++++++++++++++++++ 1 file changed, 393 insertions(+) create mode 100644 lisp/progmodes/eglot.el diff --git a/lisp/progmodes/eglot.el b/lisp/progmodes/eglot.el new file mode 100644 index 00000000000..f667eca867d --- /dev/null +++ b/lisp/progmodes/eglot.el @@ -0,0 +1,393 @@ +;;; eglot.el --- A client for Language Server Protocol (LSP) servers -*- lexical-binding: t; -*- + +;; Copyright (C) 2017 João Távora + +;; Author: João Távora +;; Keywords: extensions + +;; This program is free software; you can redistribute it and/or modify +;; it under the terms of the GNU General Public License as published by +;; the Free Software Foundation, either version 3 of the License, or +;; (at your option) any later version. + +;; This program is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; GNU General Public License for more details. + +;; You should have received a copy of the GNU General Public License +;; along with this program. If not, see . + +;;; Commentary: + +;; + +;;; Code: + +(require 'json) +(require 'cl-lib) +(require 'project) + +(defgroup eglot nil + "Interaction with Language Server Protocol servers" + :prefix "eglot-" + :group 'applications) + +(defvar eglot-executables '((rust-mode . ("rls"))) + "Alist mapping major modes to server executables") + +;;; TODO: Soon to be per-project +(defvar eglot--processes-by-project (make-hash-table :test #'equal)) + +(defun eglot--current-process () + "The current logical EGLOT process" + (let ((cur (project-current))) + (unless cur + (eglot--error "No current project, so no process")) + (gethash cur eglot--processes-by-project))) + +(defmacro eglot--define-process-var (var-sym initval &optional doc) + (declare (indent 2)) + `(progn + (put ',var-sym 'function-documentation ,doc) + (defun ,var-sym (&optional process) + (let* ((proc (or process (eglot--current-process))) + (probe (process-get proc ',var-sym))) + (or probe + (let ((def ,initval)) + (process-put proc ',var-sym def) + def)))) + (gv-define-setter ,var-sym (to-store &optional process) + (let ((prop ',var-sym)) + `(let ((proc (or ,process (eglot--current-process)))) + (process-put proc ',prop ,to-store)))))) + +(eglot--define-process-var eglot--message-mark nil + "Point where next unread message starts") + +(eglot--define-process-var eglot--expected-bytes nil + "How many bytes declared by server") + +(eglot--define-process-var eglot--continuations (make-hash-table) + "A hash table of request ID to continuation lambdas") + +(eglot--define-process-var eglot--events-buffer nil + "A buffer pretty-printing the EGLOT RPC events") + +(cl-defmacro eglot--request (process + method + params + success-fn + &key + error-fn + timeout-fn + (async-p t)) + (append `(eglot--call-with-request + ,process + ,async-p + ,method + ,params + (cl-function ,success-fn)) + (and error-fn + `((cl-function ,error-fn))) + (and timeout-fn + `((cl-function ,timeout-fn))))) + +(defun eglot--command () + (cdr (assoc major-mode eglot-executables))) + +(defun eglot-new-process (&optional interactive) + "Starts a new EGLOT process and initializes it" + (interactive (list t)) + (let ((project (project-current)) + (command (eglot--command))) + (unless command (eglot--error "Cannot work without an LSP executable")) + (unless project (eglot--error "Cannot work without a current project!")) + (let ((current-process (eglot--current-process))) + (when (and current-process + (process-live-p current-process)) + (eglot-quit-server current-process 'sync))) + (let ((good-name + (format "EGLOT server (%s)" + (file-name-base + (directory-file-name + (car (project-roots (project-current)))))))) + (with-current-buffer (get-buffer-create + (format "*%s inferior*" good-name)) + (let* ((proc + (make-process :name good-name + :buffer (current-buffer) + :command command + :connection-type 'pipe + :filter 'eglot--process-filter + :sentinel 'eglot--process-sentinel + :stderr (get-buffer-create (format "*%s stderr*" + good-name)))) + (inhibit-read-only t)) + (puthash (project-current) proc eglot--processes-by-project) + (erase-buffer) + (let ((marker (point-marker))) + (set-marker-insertion-type marker nil) + (setf (eglot--message-mark proc) marker)) + (read-only-mode t) + (with-current-buffer (eglot-events-buffer proc) + (let ((inhibit-read-only t)) + (insert + (format "\n-----------------------------------\n")))) + (eglot--protocol-initialize proc) + (when interactive + (display-buffer (eglot-events-buffer proc)))))))) + +(defun eglot-quit-server (process &optional sync) + (interactive (list (eglot--current-process))) + (eglot--message "Asking server to terminate") + (eglot--request + process + :shutdown + nil + (lambda (&rest _anything) + (eglot--message "Now asking server to exit") + (process-put process 'eglot--moribund t) + (eglot--process-send process + `(:jsonrpc "2.0" + :method :exit))) + :async-p (not sync) + :timeout-fn (lambda () + (eglot--warn "Brutally deleting existing process %s" + process) + (process-put process 'eglot--moribund t) + (delete-process process)))) + +(defun eglot--process-sentinel (process change) + (with-current-buffer (process-buffer process) + (eglot--debug "Process state changed to %s" change) + (when (not (process-live-p process)) + (cond ((process-get process 'eglot--moribund) + (eglot--message "Process exited with status %s" + (process-exit-status process))) + (t + (eglot--warn "Process unexpectedly changed to %s" change)))))) + +(defun eglot--process-filter (proc string) + (when (buffer-live-p (process-buffer proc)) + (with-current-buffer (process-buffer proc) + (let ((moving (= (point) (process-mark proc))) + (inhibit-read-only t) + (pre-insertion-mark (copy-marker (process-mark proc))) + (expected-bytes (eglot--expected-bytes proc)) + (message-mark (eglot--message-mark proc))) + (save-excursion + ;; Insert the text, advancing the process marker. + (goto-char (process-mark proc)) + (insert string) + (set-marker (process-mark proc) (point))) + (if moving (goto-char (process-mark proc))) + + ;; check for new message header + ;; + (save-excursion + (goto-char pre-insertion-mark) + (let* ((match (search-forward-regexp + "\\(?:.*: .*\r\n\\)*Content-Length: \\([[:digit:]]+\\)\r\n\\(?:.*: .*\r\n\\)*\r\n" + (+ (point) 100) + t)) + (new-expected-bytes (and match + (string-to-number (match-string 1))))) + (when new-expected-bytes + (when expected-bytes + (eglot--warn + (concat "Unexpectedly starting new message but %s bytes" + "reportedly remaining from previous one") + expected-bytes)) + (set-marker message-mark (point)) + (setf (eglot--expected-bytes proc) new-expected-bytes) + (setq expected-bytes new-expected-bytes)))) + + ;; check for message body + ;; + (let ((available-bytes (- (position-bytes (process-mark proc)) + (position-bytes message-mark)))) + (cond ((not expected-bytes) + (eglot--warn + "Skipping %s bytes of unexpected garbage from process %s" + available-bytes + proc) + (set-marker message-mark (process-mark proc))) + ((>= available-bytes + expected-bytes) + (let* ((message-end (byte-to-position + (+ (position-bytes message-mark) + expected-bytes)))) + (save-excursion + (save-restriction + (goto-char message-mark) + (narrow-to-region message-mark + message-end) + (eglot--process-receive proc (let ((json-object-type 'plist)) + (json-read))))) + (set-marker message-mark message-end) + (setf (eglot--expected-bytes proc) nil))) + (t + ;; just adding some stuff to the end that doesn't yet + ;; complete the message + ))))))) + +(defun eglot-events-buffer (process &optional interactive) + (interactive (list (eglot--current-process) t)) + (let* ((probe (eglot--events-buffer process)) + (buffer (or (and (buffer-live-p probe) + probe) + (let ((buffer (get-buffer-create + (format "*%s events*" + (process-name process))))) + (with-current-buffer buffer + (buffer-disable-undo) + (read-only-mode t) + (setf (eglot--events-buffer process) + buffer)) + buffer)))) + (when interactive + (pop-to-buffer buffer)) + buffer)) + +(defun eglot--log-event (proc type message) + (with-current-buffer (eglot-events-buffer proc) + (let ((inhibit-read-only t)) + (goto-char (point-max)) + (insert (format "%s: \n%s\n" type (pp-to-string message)))))) + +(defun eglot--process-receive (proc message) + (let ((inhibit-read-only t)) + (insert (format "Server said:\n%s\n" message))) + (eglot--log-event proc 'server message) + ;; Maybe this is a responsee + ;; + (let* ((response-id (plist-get message :id)) + (err (plist-get message :error)) + (continuations (and response-id + (gethash response-id (eglot--continuations))))) + (cond ((and response-id + (not continuations)) + (eglot--warn "Ooops no continuation for id %s" response-id)) + (continuations + (cancel-timer (third continuations)) + (cond (err + (apply (second continuations) err)) + (t + (apply (first continuations) (plist-get message :result))))) + (t + (eglot--debug "No implemetation for notification %s yet" + (plist-get message :method)))))) + +;; (setq json-encoding-pretty-print nil) ; for debug +(defvar eglot--expect-carriage-return nil) + +(defun eglot--process-send (proc message) + (let* ((json (json-encode message)) + (to-send (format "Content-Length: %d\r\n\r\n%s" + (string-bytes json) + json))) + (process-send-string proc to-send) + (eglot--log-event proc 'client message))) + +(defvar eglot--next-request-id 0) + +(defun eglot--next-request-id () + (setq eglot--next-request-id (1+ eglot--next-request-id))) + +(defun eglot--call-with-request (process + async-p + method + params + success-fn + &optional error-fn timeout-fn) + (let* ((id (eglot--next-request-id)) + (timeout-fn (or timeout-fn + (lambda () + (eglot--warn "Tired of waiting for reply to %s" id) + (remhash id (eglot--continuations process))))) + (error-fn (or error-fn + (cl-function + (lambda (&key code message) + (eglot--warn "Request id=%s errored with code=%s: %s" + id code message))))) + (catch-tag (cl-gensym (format "eglot--tag-%d-" id)))) + (eglot--process-send process + `(:jsonrpc "2.0" + :id ,id + :method ,method + :params ,params)) + (catch catch-tag + (puthash id + (list (if async-p + success-fn + (lambda (&rest args) + (throw catch-tag (apply success-fn args)))) + (if async-p + error-fn + (lambda (&rest args) + (throw catch-tag (apply error-fn args)))) + (run-with-timer 5 nil + (if async-p + timeout-fn + (lambda () + (throw catch-tag (apply timeout-fn)))))) + (eglot--continuations process)) + (unless async-p + (while t + (unless (eq (process-status process) 'open) + (eglot--error "Process %s died unexpectedly" process)) + (accept-process-output nil 0.01)))))) + + +(defun eglot--protocol-initialize (process) + (eglot--request + process + :initialize + `(:processId ,(emacs-pid) + :rootPath ,(concat "" ;; FIXME RLS doesn't like "file://" + (expand-file-name (car (project-roots + (project-current))))) + :initializationOptions [] + :capabilities (:workspace (:executeCommand (:dynamicRegistration t)) + :textDocument (:synchronization (:didSave t))) + ) + (lambda (&key capabilities) + (cl-destructuring-bind + (&rest all + &key + _textDocumentSync + _hoverProvider + _completionProvider + _definitionProvider + _referencesProvider + _documentHighlightProvider + _documentSymbolProvider + _workspaceSymbolProvider + _codeActionProvider + _documentFormattingProvider + _documentRangeFormattingProvider + _renameProvider + _executeCommandProvider + ) + capabilities + (message "so yeah I got lots (%d) of capabilities" (length all)))))) + +(defun eglot--debug (format &rest args) + (display-warning 'eglot + (apply #'format format args) + :debug)) + +(defun eglot--error (format &rest args) + (error (apply #'format format args))) + +(defun eglot--message (format &rest args) + (message (concat "[eglot] " (apply #'format format args)))) + +(defun eglot--warn (format &rest args) + (display-warning 'eglot + (apply #'format format args) + :warning)) + +(provide 'eglot) +;;; eglot.el ends here -- 2.39.2