This repository has been archived on 2022-11-18. You can view files and clone it, but cannot push or open issues or pull requests.
emacs/emacs-racer/racer.el

863 lines
32 KiB
EmacsLisp

;;; racer.el --- code completion, goto-definition and docs browsing for Rust via racer -*- lexical-binding: t -*-
;; Copyright (c) 2014 Phil Dawes
;; Author: Phil Dawes
;; URL: https://github.com/racer-rust/emacs-racer
;; Version: 1.3
;; Package-Requires: ((emacs "24.3") (rust-mode "0.2.0") (dash "2.13.0") (s "1.10.0") (f "0.18.2"))
;; Keywords: abbrev, convenience, matching, rust, tools
;; This file is not part of GNU Emacs.
;; Permission is hereby granted, free of charge, to any
;; person obtaining a copy of this software and associated
;; documentation files (the "Software"), to deal in the
;; Software without restriction, including without
;; limitation the rights to use, copy, modify, merge,
;; publish, distribute, sublicense, and/or sell copies of
;; the Software, and to permit persons to whom the Software
;; is furnished to do so, subject to the following
;; conditions:
;; The above copyright notice and this permission notice
;; shall be included in all copies or substantial portions
;; of the Software.
;; THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF
;; ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED
;; TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
;; PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT
;; SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
;; CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
;; OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR
;; IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
;; DEALINGS IN THE SOFTWARE.
;;; Commentary:
;; Please see the readme for full documentation:
;; https://github.com/racer-rust/emacs-racer
;;; Quickstart:
;; You will need to configure Emacs to find racer:
;;
;; (setq racer-rust-src-path "<path-to-rust-srcdir>/src/")
;; (setq racer-cmd "<path-to-racer>/target/release/racer")
;;
;; To activate racer in Rust buffers, run:
;;
;; (add-hook 'rust-mode-hook #'racer-mode)
;;
;; You can also use racer to find definition at point via
;; `racer-find-definition', bound to `M-.' by default.
;;
;; Finally, you can also use Racer to show the signature of the
;; current function in the minibuffer:
;;
;; (add-hook 'racer-mode-hook #'eldoc-mode)
;;; Code:
(require 'dash)
(require 'etags)
(require 'rust-mode)
(require 's)
(require 'f)
(require 'thingatpt)
(require 'button)
(require 'help-mode)
(require 'deferred)
(defgroup racer nil
"Code completion, goto-definition and docs browsing for Rust via racer."
:link '(url-link "https://github.com/racer-rust/emacs-racer/")
:group 'rust-mode)
(defcustom racer-cmd
(or (executable-find "racer")
(f-expand "~/.cargo/bin/racer")
"/usr/local/bin/racer")
"Path to the racer binary."
:type 'file
:group 'racer)
(defcustom racer-rust-src-path
(or
(getenv "RUST_SRC_PATH")
(when (executable-find "rustc")
(let* ((sysroot (s-trim-right
(shell-command-to-string
(format "%s --print sysroot" (executable-find "rustc")))))
(src-path (f-join sysroot "lib/rustlib/src/rust/src")))
(when (file-exists-p src-path)
src-path)
src-path))
"/usr/local/src/rust/src")
"Path to the rust source tree.
If nil, we will query $RUST_SRC_PATH at runtime.
If $RUST_SRC_PATH is not set, look for rust source in rustup's install directory."
:type 'file
:group 'racer)
(defcustom racer-cargo-home
(or
(getenv "CARGO_HOME")
"~/.cargo")
"Path to your current cargo home. Usually `~/.cargo'.
If nil, we will query $CARGO_HOME at runtime."
:type 'file
:group 'racer)
(defcustom racer-use-company-backend nil
"Use the asynchronous company backend."
:type 'boolean
:group 'racer)
(defun racer--cargo-project-root ()
"Find the root of the current Cargo project."
(let ((root (locate-dominating-file (or (buffer-file-name (buffer-base-buffer)) default-directory)
"Cargo.toml")))
(and root (file-truename root))))
(defun racer--header (text)
"Helper function for adding text properties to TEXT."
(propertize text 'face 'racer-help-heading-face))
(defvar racer--prev-state nil)
(defun racer-debug ()
"Open a buffer describing the last racer command run.
Helps users find configuration issues, or file bugs on
racer or racer.el."
(interactive)
(unless racer--prev-state
(user-error "Must run a racer command before debugging"))
(let ((buf (get-buffer-create "*racer-debug*"))
(inhibit-read-only t))
(with-current-buffer buf
(erase-buffer)
(setq buffer-read-only t)
(let* ((process-environment
(plist-get racer--prev-state :process-environment))
(rust-src-path-used
(--first (s-prefix-p "RUST_SRC_PATH=" it) process-environment))
(cargo-home-used
(--first (s-prefix-p "CARGO_HOME=" it) process-environment))
(stdout (plist-get racer--prev-state :stdout))
(stderr (plist-get racer--prev-state :stderr)))
(insert
;; Summarise the actual command that we run.
(racer--header "The last racer command was:\n\n")
(format "$ cd %s\n"
(plist-get racer--prev-state :default-directory))
(format "$ export %s\n" cargo-home-used)
(format "$ export %s\n" rust-src-path-used)
(format "$ %s %s\n\n"
(plist-get racer--prev-state :program)
(s-join " " (plist-get racer--prev-state :args)))
;; Describe the exit code and outputs.
(racer--header
(format "This command terminated with exit code %s.\n\n"
(plist-get racer--prev-state :exit-code)))
(if (s-blank? stdout)
(racer--header "No output on stdout.\n\n")
(format "%s\n\n%s\n\n"
(racer--header "stdout:")
(s-trim-right stdout)))
(if (s-blank? stderr)
(racer--header "No output on stderr.\n\n")
(format "%s\n\n%s\n\n"
(racer--header "stderr:")
(s-trim-right stderr)))
;; Give copy-paste instructions for reproducing any errors
;; the user has seen.
(racer--header
(s-word-wrap 60 "The temporary file will have been deleted. You should be able to reproduce the same output from racer with the following command:\n\n"))
(format "$ %s %s %s %s\n\n" cargo-home-used rust-src-path-used
(plist-get racer--prev-state :program)
(s-join " "
(-drop-last 1 (plist-get racer--prev-state :args))))
;; Tell the user what to do next if they have problems.
(racer--header "Please report bugs ")
(racer--url-button "on GitHub" "https://github.com/racer-rust/emacs-racer/issues/new")
(racer--header "."))))
(switch-to-buffer buf)
(goto-char (point-min))))
(defun racer--process-shell-ouput (output)
"Process OUTPUT from the racer shell call. Return stdout if
command exited OK, and throw an error otherwise. The output is
expected to be a list of (exit-code stdout stderr)"
(-let [(exit-code stdout stderr) output]
;; Use `equal' instead of `zero' as exit-code can be a string
;; "Aborted" if racer crashes.
(unless (equal 0 exit-code)
(user-error "%s exited with %s. `M-x racer-debug' for more info"
racer-cmd exit-code))
stdout))
(defun racer--call (defer command &rest args)
"Call racer command COMMAND with args ARGS.
Return stdout if COMMAND exits normally, otherwise show an error.
If DEFER is non-nil, return deferred object instead"
(let ((rust-src-path (or racer-rust-src-path (getenv "RUST_SRC_PATH")))
(cargo-home (or racer-cargo-home (getenv "CARGO_HOME"))))
(when (null rust-src-path)
(user-error "You need to set `racer-rust-src-path' or `RUST_SRC_PATH'"))
(unless (file-exists-p rust-src-path)
(user-error "No such directory: %s. Please set `racer-rust-src-path' or `RUST_SRC_PATH'"
rust-src-path))
(let ((default-directory (or (racer--cargo-project-root) default-directory))
(process-environment (append (list
(format "RUST_SRC_PATH=%s" (expand-file-name rust-src-path))
(format "CARGO_HOME=%s" (expand-file-name cargo-home)))
process-environment)))
(if defer
(deferred:nextc
(racer--shell-command-defer racer-cmd (cons command args))
'racer--process-shell-ouput)
(racer--process-shell-ouput
(racer--shell-command racer-cmd (cons command args)))))))
(defmacro racer--with-deferred-temporary-file (path-sym &rest body)
"Create a temporary file, and bind its path to PATH-SYM.
Evaluate BODY, then delete the temporary file once deferred task
completes."
(declare (indent 1) (debug (symbolp body)))
`(let ((,path-sym (make-temp-file "racer")))
(deferred:nextc
(progn ,@body)
(lambda (output)
(delete-file ,path-sym)
output))))
(defmacro racer--with-temporary-file (path-sym &rest body)
"Create a temporary file, and bind its path to PATH-SYM.
Evaluate BODY, then delete the temporary file."
(declare (indent 1) (debug (symbolp body)))
`(let ((,path-sym (make-temp-file "racer")))
(unwind-protect
(progn ,@body)
(delete-file ,path-sym))))
(defun racer--slurp (file)
"Return the contents of FILE as a string."
(with-temp-buffer
(insert-file-contents-literally file)
(buffer-string)))
(defun racer--set-prev-state (racer-cmd args exit-code stdout stderr)
"Set the racer--prev-state previous variable."
(setq racer--prev-state
(list
:program racer-cmd
:args args
:exit-code exit-code
:stdout stdout
:stderr stderr
:default-directory default-directory
:process-environment process-environment)))
(defun racer--shell-command (program args)
"Execute PROGRAM with ARGS.
Return a list (exit-code stdout stderr)."
(racer--with-temporary-file tmp-file-for-stderr
(let (exit-code stdout stderr)
;; Create a temporary buffer for `call-process` to write stdout
;; into.
(with-temp-buffer
(setq exit-code
(apply #'call-process program nil
(list (current-buffer) tmp-file-for-stderr)
nil args))
(setq stdout (buffer-string)))
(setq stderr (racer--slurp tmp-file-for-stderr))
(racer--set-prev-state racer-cmd args exit-code stdout stderr)
(list exit-code stdout stderr))))
(defun racer--shell-command-defer (program args)
"Execute PROGRAM with ARGS.
Return a list (exit-code stdout stderr)."
(deferred:nextc
(apply #'deferred:process-ec program args)
(lambda (output)
;; deferred combines stdout and stderr
(let ((exit-code (nth 0 output))
(stdout (nth 1 output))
(stderr ""))
(racer--set-prev-state racer-cmd args exit-code stdout stderr)
(list exit-code stdout stderr)))))
(defun racer--call-at-point (command &optional no-defer)
"Call racer command COMMAND at point of current buffer.
Return a list of all the lines returned by the command.
Optionally force racer to not defer command if NO-DEFER is
non-nil."
(let ((line (number-to-string (line-number-at-pos)))
(column (number-to-string (racer--current-column)))
(filename (buffer-file-name (buffer-base-buffer))))
(if (and (string= command "complete")
(not no-defer))
(racer--with-deferred-temporary-file tmp-file
(write-region nil nil tmp-file nil 'silent)
(deferred:nextc
(racer--call 'defer command line column filename tmp-file)
(lambda (output)
(s-lines (s-trim-right output)))))
(racer--with-temporary-file tmp-file
(write-region nil nil tmp-file nil 'silent)
(s-lines
(s-trim-right
(racer--call nil command line column filename tmp-file)))))))
(defun racer--read-rust-string (string)
"Convert STRING, a rust string literal, to an elisp string."
(when string
(->> string
;; Remove outer double quotes.
(s-chop-prefix "\"")
(s-chop-suffix "\"")
;; Replace escaped characters.
(s-replace "\\n" "\n")
(s-replace "\\\"" "\"")
(s-replace "\\'" "'")
(s-replace "\\;" ";"))))
(defun racer--split-parts (raw-output)
"Given RAW-OUTPUT from racer, split on semicolons and doublequotes.
Unescape strings as necessary."
(let ((parts nil)
(current "")
(i 0))
(while (< i (length raw-output))
(let ((char (elt raw-output i))
(prev-char (and (> i 0) (elt raw-output (1- i)))))
(cond
;; A semicolon that wasn't escaped, start a new part.
((and (equal char ?\;) (not (equal prev-char ?\\)))
(push current parts)
(setq current ""))
(t
(setq current (concat current (string char))))))
(setq i (1+ i)))
(push current parts)
(mapcar #'racer--read-rust-string (nreverse parts))))
(defun racer--split-snippet-match (line)
"Given LINE, a string \"MATCH ...\" from complete-with-snippet,
split it into its constituent parts."
(let* ((match-parts (racer--split-parts line))
(docstring (nth 7 match-parts)))
(when (and match-parts (equal (length match-parts) 8))
(list :name (s-chop-prefix "MATCH " (nth 0 match-parts))
:line (string-to-number (nth 2 match-parts))
:column (string-to-number (nth 3 match-parts))
:path (nth 4 match-parts)
;; Struct or Function:
:kind (nth 5 match-parts)
:signature (nth 6 match-parts)
:docstring (if (> (length docstring) 0) docstring nil)))))
(defun racer--describe-at-point (name)
"Get a description of the symbol at point matching NAME.
If there are multiple possibilities with this NAME, prompt
the user to choose."
(let* ((output-lines (save-excursion
;; Move to the end of the current symbol, to
;; increase racer accuracy.
(skip-syntax-forward "w_")
(racer--call-at-point "complete-with-snippet")))
(all-matches (--map (when (s-starts-with-p "MATCH " it)
(racer--split-snippet-match it))
output-lines))
(relevant-matches (--filter (equal (plist-get it :name) name)
all-matches)))
(if (> (length relevant-matches) 1)
;; We might have multiple matches with the same name but
;; different types. E.g. Vec::from.
(let ((signature
(completing-read "Multiple matches: "
(--map (plist-get it :signature) relevant-matches))))
(--first (equal (plist-get it :signature) signature) relevant-matches))
(-first-item relevant-matches))))
(defun racer--help-buf (contents)
"Create a *Racer Help* buffer with CONTENTS."
(let ((buf (get-buffer-create "*Racer Help*"))
;; If the buffer already existed, we need to be able to
;; override `buffer-read-only'.
(inhibit-read-only t))
(with-current-buffer buf
(erase-buffer)
(insert contents)
(setq buffer-read-only t)
(goto-char (point-min))
(racer-help-mode))
buf))
(defface racer-help-heading-face
'((t :weight bold))
"Face for markdown headings in *Racer Help* buffers.")
(defun racer--url-p (target)
"Return t if TARGET looks like a fully qualified URL."
(not (null
(string-match-p (rx bol "http" (? "s") "://") target))))
(defun racer--propertize-links (markdown)
"Propertize links in MARKDOWN."
(replace-regexp-in-string
;; Text of the form [foo](http://example.com)
(rx "[" (group (+? (not (any "]")))) "](" (group (+? anything)) ")")
;; For every match:
(lambda (whole-match)
;; Extract link and target.
(let ((link-text (match-string 1 whole-match))
(link-target (match-string 2 whole-match)))
;; If it's a web URL, use a clickable link.
(if (racer--url-p link-target)
(racer--url-button link-text link-target)
;; Otherwise, just discard the target.
link-text)))
markdown))
(defun racer--propertize-all-inline-code (markdown)
"Given a single line MARKDOWN, replace all instances of `foo` or
\[`foo`\] with a propertized string."
(let ((highlight-group
(lambda (whole-match)
(racer--syntax-highlight (match-string 1 whole-match)))))
(->> markdown
(replace-regexp-in-string
(rx "[`" (group (+? anything)) "`]")
highlight-group)
(replace-regexp-in-string
(rx "`" (group (+? anything)) "`")
highlight-group))))
(defun racer--indent-block (str)
"Indent every line in STR."
(s-join "\n" (--map (concat " " it) (s-lines str))))
(defun racer--trim-newlines (str)
"Remove newlines from the start and end of STR."
(->> str
(s-chop-prefix "\n")
(s-chop-suffix "\n")))
(defun racer--remove-footnote-links (str)
"Remove footnote links from markdown STR."
(->> (s-lines str)
(--remove (string-match-p (rx bol "[`" (+? anything) "`]: ") it))
(s-join "\n")
;; Collapse consecutive blank lines caused by removing footnotes.
(s-replace "\n\n\n" "\n\n")))
(defun racer--docstring-sections (docstring)
"Split DOCSTRING into text, code and heading sections."
(let* ((sections nil)
(current-section-lines nil)
(section-type :text)
;; Helper function.
(finish-current-section
(lambda ()
(when current-section-lines
(let ((current-section
(s-join "\n" (nreverse current-section-lines))))
(unless (s-blank? current-section)
(push (list section-type current-section) sections))
(setq current-section-lines nil))))))
(dolist (line (s-lines docstring))
(cond
;; If this is a closing ```
((and (s-starts-with-p "```" line) (eq section-type :code))
(push line current-section-lines)
(funcall finish-current-section)
(setq section-type :text))
;; If this is an opening ```
((s-starts-with-p "```" line)
(funcall finish-current-section)
(push line current-section-lines)
(setq section-type :code))
;; Headings
((and (not (eq section-type :code)) (s-starts-with-p "# " line))
(funcall finish-current-section)
(push (list :heading line) sections))
;; Normal text.
(t
(push line current-section-lines))))
(funcall finish-current-section)
(nreverse sections)))
(defun racer--clean-code-section (section)
"Given a SECTION, a markdown code block, remove
fenced code delimiters and code annotations."
(->> (s-lines section)
(-drop 1)
(-drop-last 1)
;; Ignore annotations like # #[allow(dead_code)]
(--remove (s-starts-with-p "# " it))
(s-join "\n")))
(defun racer--propertize-docstring (docstring)
"Replace markdown syntax in DOCSTRING with text properties."
(let* ((sections (racer--docstring-sections docstring))
(propertized-sections
(--map (-let [(section-type section) it]
;; Remove trailing newlines, so we can ensure we
;; have consistent blank lines between sections.
(racer--trim-newlines
(pcase section-type
(:text
(racer--propertize-all-inline-code
(racer--propertize-links
(racer--remove-footnote-links
section))))
(:code
(racer--indent-block
(racer--syntax-highlight
(racer--clean-code-section section))))
(:heading
(racer--header
(s-chop-prefix "# " section))))))
sections)))
(s-join "\n\n" propertized-sections)))
(defun racer--find-file (path line column)
"Open PATH and move point to LINE and COLUMN."
(find-file path)
(goto-char (point-min))
(forward-line (1- line))
(forward-char column))
(defun racer--button-go-to-src (button)
(racer--find-file
(button-get button 'path)
(button-get button 'line)
(button-get button 'column)))
(define-button-type 'racer-src-button
'action 'racer--button-go-to-src
'follow-link t
'help-echo "Go to definition")
(defun racer--url-button (text url)
"Return a button that opens a browser at URL."
(with-temp-buffer
(insert-text-button
text
:type 'help-url
'help-args (list url))
(buffer-string)))
(defun racer--src-button (path line column)
"Return a button that navigates to PATH at LINE number and
COLUMN number."
;; Convert "/foo/bar/baz/foo.rs" to "baz/foo.rs"
(let* ((filename (f-filename path))
(parent-dir (f-filename (f-parent path)))
(short-path (f-join parent-dir filename)))
(with-temp-buffer
(insert-text-button
short-path
:type 'racer-src-button
'path path
'line line
'column column)
(buffer-string))))
(defun racer--kind-description (raw-kind)
"Human friendly description of a rust kind.
For example, 'EnumKind' -> 'an enum kind'."
(let* ((parts (s-split-words raw-kind))
(description (s-join " " (--map (downcase it) parts)))
(a (if (string-match-p (rx bos (or "a" "e" "i" "o" "u")) description)
"an" "a")))
(format "%s %s" a description)))
(defun racer--describe (name)
"Return a *Racer Help* buffer for the function or type at point.
If there are multiple candidates at point, use NAME to find the
correct value."
(let ((description (racer--describe-at-point name)))
(when description
(let* ((name (plist-get description :name))
(raw-docstring (plist-get description :docstring))
(docstring (if raw-docstring
(racer--propertize-docstring raw-docstring)
"Not documented."))
(kind (plist-get description :kind)))
(racer--help-buf
(format
"%s is %s defined in %s.\n\n%s%s"
name
(racer--kind-description kind)
(racer--src-button
(plist-get description :path)
(plist-get description :line)
(plist-get description :column))
(if (equal kind "Module")
;; No point showing the 'signature' of modules, which is
;; just their full path.
""
(format " %s\n\n" (racer--syntax-highlight (plist-get description :signature))))
docstring))))))
(defun racer-describe ()
"Show a *Racer Help* buffer for the function or type at point."
(interactive)
(let ((buf (racer--describe (thing-at-point 'symbol))))
(if buf
(temp-buffer-window-show buf)
(user-error "No function or type found at point"))))
(defvar racer-help-mode-map
(let ((map (make-sparse-keymap)))
(set-keymap-parent map (make-composed-keymap button-buffer-map
special-mode-map))
map)
"Keymap for racer help mode.")
(define-derived-mode racer-help-mode fundamental-mode
"Racer-Help"
"Major mode for *Racer Help* buffers.
Commands:
\\{racer-help-mode-map}")
(defcustom racer-complete-in-comments
nil
"If non-nil, query racer for completions inside comments too."
:type 'boolean
:group 'racer)
(defun racer-complete-at-point ()
"Complete the symbol at point."
(let* ((ppss (syntax-ppss))
(in-string (nth 3 ppss))
(in-comment (nth 4 ppss)))
(when (and
(not in-string)
(or (not in-comment) racer-complete-in-comments))
(let* ((bounds (bounds-of-thing-at-point 'symbol))
(beg (or (car bounds) (point)))
(end (or (cdr bounds) (point))))
(list beg end
(completion-table-dynamic #'racer-complete-sync)
:annotation-function #'racer-complete--annotation
:company-prefix-length (racer-complete--prefix-p beg end)
:company-docsig #'racer-complete--docsig
:company-doc-buffer #'racer--describe
:company-location #'racer-complete--location)))))
(defun racer--file-and-parent (path)
"Convert /foo/bar/baz/q.txt to baz/q.txt."
(let ((file (f-filename path))
(parent (f-filename (f-parent path))))
(f-join parent file)))
(defun racer--parse-complete-candidates (candidates)
"Parse the completion CANDIDATES returned by racer."
(->> candidates
(--filter (s-starts-with? "MATCH" it))
(--map (-let [(name line col file matchtype ctx)
(s-split-up-to "," (s-chop-prefix "MATCH " it) 5)]
(put-text-property 0 1 'line (string-to-number line) name)
(put-text-property 0 1 'col (string-to-number col) name)
(put-text-property 0 1 'file file name)
(put-text-property 0 1 'matchtype matchtype name)
(put-text-property 0 1 'ctx ctx name)
name))))
(defun racer-complete (&optional _ignore)
"Completion candidates at point."
(deferred:nextc
(racer--call-at-point "complete")
'racer--parse-complete-candidates))
(defun racer-complete-sync (&optional _ignore)
"Completion candidates at point. Return synchronously."
(racer--parse-complete-candidates
(racer--call-at-point "complete" 'no-defer)))
(defun racer--trim-up-to (needle s)
"Return content after the occurrence of NEEDLE in S."
(-if-let (idx (s-index-of needle s))
(substring s (+ idx (length needle)))
s))
(defun racer-complete--prefix-p (beg _end)
"Return t if a completion should be triggered for a prefix between BEG and END."
(save-excursion
(goto-char beg)
;; If we're at the beginning of the buffer, we can't look back 2
;; characters.
(ignore-errors
(looking-back "\\.\\|::" 2))))
(defun racer-complete--annotation (arg)
"Return an annotation for completion candidate ARG."
(let* ((ctx (get-text-property 0 'ctx arg))
(type (get-text-property 0 'matchtype arg))
(pretty-ctx
(pcase type
("Module"
(if (string= arg ctx)
""
(concat " " (racer--file-and-parent ctx))))
("StructField"
(concat " " ctx))
(_
(->> ctx
(racer--trim-up-to arg)
(s-chop-suffixes '(" {" "," ";")))))))
(format "%s : %s" pretty-ctx type)))
(defun racer-complete--docsig (arg)
"Return a signature for completion candidate ARG."
(racer--syntax-highlight (format "%s" (get-text-property 0 'ctx arg))))
(defun racer-complete--location (arg)
"Return location of completion candidate ARG."
(cons (get-text-property 0 'file arg)
(get-text-property 0 'line arg)))
(defun racer--current-column ()
"Get the current column based on underlying character representation."
(length (buffer-substring-no-properties
(line-beginning-position) (point))))
;;;###autoload
(defun racer-find-definition ()
"Run the racer find-definition command and process the results."
(interactive)
(-if-let (match (--first (s-starts-with? "MATCH" it)
(racer--call-at-point "find-definition")))
(-let [(_name line col file _matchtype _ctx)
(s-split-up-to "," (s-chop-prefix "MATCH " match) 5)]
(if (fboundp 'xref-push-marker-stack)
(xref-push-marker-stack)
(with-no-warnings
(ring-insert find-tag-marker-ring (point-marker))))
(racer--find-file file (string-to-number line) (string-to-number col)))
(error "No definition found")))
(defun racer--syntax-highlight (str)
"Apply font-lock properties to a string STR of Rust code."
(let (result)
;; Load all of STR in a rust-mode buffer, and use its
;; highlighting.
(with-temp-buffer
(insert str)
(delay-mode-hooks (rust-mode))
(if (fboundp 'font-lock-ensure)
(font-lock-ensure)
(with-no-warnings
(font-lock-fontify-buffer)))
(setq result (buffer-string)))
(when (and
;; If we haven't applied any properties yet,
(null (text-properties-at 0 result))
;; and if it's a standalone symbol, then assume it's a
;; variable.
(string-match-p (rx bos (+ (any lower "_")) eos) str))
(setq result (propertize str 'face 'font-lock-variable-name-face)))
result))
(defun racer--goto-func-name ()
"If point is inside a function call, move to the function name.
foo(bar, |baz); -> foo|(bar, baz);"
(let ((last-paren-pos (nth 1 (syntax-ppss)))
(start-pos (point)))
(when last-paren-pos
;; Move to just before the last paren.
(goto-char last-paren-pos)
;; If we're inside a round paren, we're inside a function call.
(unless (looking-at "(")
;; Otherwise, return to our start position, as point may have been on a
;; function already:
;; foo|(bar, baz);
(goto-char start-pos)))))
(defun racer--relative (path &optional directory)
"Return PATH relative to DIRECTORY (`default-directory' by default).
If PATH is not in DIRECTORY, just abbreviate it."
(unless directory
(setq directory default-directory))
(if (s-starts-with? directory path)
(concat "./" (f-relative path directory))
(f-abbrev path)))
(defun racer-eldoc ()
"Show eldoc for context at point."
(save-excursion
(racer--goto-func-name)
;; If there's a variable at point:
(-when-let* ((rust-sym (symbol-at-point))
(comp-possibilities (racer-complete-sync))
(matching-possibility
(--find (string= it (symbol-name rust-sym)) comp-possibilities))
(prototype (get-text-property 0 'ctx matching-possibility))
(matchtype (get-text-property 0 'matchtype matching-possibility)))
(if (equal matchtype "Module")
(racer--relative prototype)
;; Syntax highlight function signatures.
(racer--syntax-highlight prototype)))))
(defvar racer-mode-map
(let ((map (make-sparse-keymap)))
(define-key map (kbd "M-.") #'racer-find-definition)
(define-key map (kbd "M-,") #'pop-tag-mark)
map))
(defun racer--get-prefix ()
"Get a prefix from current position."
(company-grab-symbol-cons "\\.\\|::" 2))
(defun company-racer-candidates (callback)
"Return candidates for PREFIX with CALLBACK."
(deferred:nextc
(racer-complete)
(lambda (candidates)
(funcall callback candidates))))
(defun racer-company-backend (command &optional arg &rest ignored)
"`company-mode' completion back-end for racer.
Provide completion info according to COMMAND and ARG. IGNORED, not used."
(interactive (list 'interactive))
(cl-case command
(interactive (company-begin-backend 'racer-company-backend))
(prefix (and (derived-mode-p 'rust-mode)
(not (company-in-string-or-comment))
(or (racer--get-prefix) 'stop)))
(candidates (cons :async 'company-racer-candidates))
(annotation (racer-complete--annotation arg))
(location (racer-complete--location arg))
(meta (racer-complete--docsig arg))
(doc-buffer (racer--describe arg))))
;;;###autoload
(define-minor-mode racer-mode
"Minor mode for racer."
:lighter " racer"
:keymap racer-mode-map
(setq-local eldoc-documentation-function #'racer-eldoc)
(if racer-use-company-backend
(add-to-list (make-local-variable 'company-backends)
'racer-company-backend)
(set (make-local-variable 'completion-at-point-functions) nil)
(add-hook 'completion-at-point-functions #'racer-complete-at-point)))
(define-obsolete-function-alias 'racer-turn-on-eldoc 'eldoc-mode)
(define-obsolete-function-alias 'racer-activate 'racer-mode)
(provide 'racer)
;;; racer.el ends here