emacs-guix/elisp/guix-utils.el

569 lines
20 KiB
EmacsLisp
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

;;; guix-utils.el --- General utility functions -*- lexical-binding: t -*-
;; Copyright © 20142018 Alex Kost <alezost@gmail.com>
;; This file is part of Emacs-Guix.
;; Emacs-Guix 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.
;;
;; Emacs-Guix 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 Emacs-Guix. If not, see <http://www.gnu.org/licenses/>.
;;; Commentary:
;; This file provides auxiliary general code for Emacs-Guix package.
;;; Code:
(require 'cl-lib)
(require 'dash)
(require 'bui-utils)
(require 'guix nil t)
(defun guix-guixsd? ()
"Return non-nil, if current system is GuixSD."
(file-exists-p "/run/current-system"))
(defun guix-assert-build-farm ()
"Raise an error if `build-farm' package does not exist."
(unless (require 'build-farm nil t)
(error "This feature requires `build-farm' package, \
which is not installed")))
(defun guix-concat-strings (strings separator &optional location)
"Return new string by concatenating STRINGS with SEPARATOR.
If LOCATION is a symbol `head', add another SEPARATOR to the
beginning of the returned string; if `tail' - add SEPARATOR to
the end of the string; if nil, do not add SEPARATOR; otherwise
add both to the end and to the beginning."
(let ((str (mapconcat #'identity strings separator)))
(cond ((null location)
str)
((eq location 'head)
(concat separator str))
((eq location 'tail)
(concat str separator))
(t
(concat separator str separator)))))
(defun guix-list-maybe (object)
"If OBJECT is list, return it; otherwise return (list OBJECT)."
(if (listp object)
object
(list object)))
(defun guix-shell-quote-argument (argument)
"Quote shell command ARGUMENT.
This function is similar to `shell-quote-argument', but less strict."
(if (equal argument "")
"''"
(replace-regexp-in-string
"\n" "'\n'"
(replace-regexp-in-string
(rx (not (any alnum "-=,./\n"))) "\\\\\\&" argument))))
(defun guix-command-symbol (&optional args)
"Return symbol by concatenating 'guix' and ARGS (strings)."
(intern (guix-concat-strings (cons "guix" args) "-")))
(defun guix-command-string (&optional args)
"Return 'guix ARGS ...' string with quoted shell arguments."
(let ((args (mapcar #'guix-shell-quote-argument args)))
(guix-concat-strings (cons "guix" args) " ")))
(defun guix-copy-command-as-kill (args &optional no-message?)
"Put 'guix ARGS ...' string into `kill-ring'.
See also `guix-copy-as-kill'."
(bui-copy-as-kill (guix-command-string args) no-message?))
(defun guix-compose-buffer-name (base-name postfix)
"Return buffer name by appending BASE-NAME and POSTFIX.
In a simple case the result is:
BASE-NAME: POSTFIX
If BASE-NAME is wrapped by '*', then the result is:
*BASE-NAME: POSTFIX*"
(let ((re (rx string-start
(group (? "*"))
(group (*? any))
(group (? "*"))
string-end)))
(or (string-match re base-name)
(error "Unexpected error in defining buffer name"))
(let ((first* (match-string 1 base-name))
(name-body (match-string 2 base-name))
(last* (match-string 3 base-name)))
;; Handle the case when buffer name is wrapped by '*'.
(if (and (string= "*" first*)
(string= "*" last*))
(concat "*" name-body ": " postfix "*")
(concat base-name ": " postfix)))))
(defun guix-completing-read (prompt table &optional predicate
require-match initial-input
hist def inherit-input-method)
"Same as `completing-read' but return nil instead of an empty string."
(let ((res (completing-read prompt table predicate
require-match initial-input
hist def inherit-input-method)))
(unless (string= "" res) res)))
(defun guix-completing-read-multiple (prompt table &optional predicate
require-match initial-input
hist def inherit-input-method)
"Same as `completing-read-multiple' but remove duplicates in result."
(cl-remove-duplicates
(completing-read-multiple prompt table predicate
require-match initial-input
hist def inherit-input-method)
:test #'string=))
(declare-function org-read-date "org" t)
(defun guix-read-date (prompt)
"Prompt for a date or time using `org-read-date'.
Return time value."
(require 'org)
(org-read-date nil t nil prompt))
(declare-function pcmpl-unix-user-names "pcmpl-unix")
(defun guix-read-user-name (&optional prompt initial-input)
"Prompt for a user name using completions."
(require 'pcmpl-unix)
(guix-completing-read (or prompt "User name: ")
(pcmpl-unix-user-names)
nil nil initial-input))
(defun guix-switch-to-buffer-or-funcall (buffer-or-name function
&optional message)
"Switch to BUFFER-OR-NAME if it exists.
If BUFFER-OR-NAME does not exist, call FUNCTION without
arguments, also display a message if MESSAGE is specified (it can
be either nil, a string, or another value for a default
message)."
(let ((buffer (get-buffer buffer-or-name)))
(if buffer
(progn
(switch-to-buffer buffer)
(when message
(message (if (stringp message)
message
(substitute-command-keys "\
Press '\\[revert-buffer]' to update this buffer.")))))
(funcall function))))
(defun guix-display-buffer (buffer)
"Switch to BUFFER, preferably reusing a window displaying this buffer."
(pop-to-buffer buffer
'((display-buffer-reuse-window
display-buffer-same-window))))
(cl-defun guix-pretty-print-buffer
(&optional buffer-or-name
&key (modified-flag nil modified-flag-bound?))
"Pretty-print the contents of BUFFER-OR-NAME.
MODIFIED-FLAG defines if the buffer should marked as modified or
unmodified. If this flag is not set, the modification status
of the buffer stays unchanged (as it was before prettifying)."
(let ((modified? (buffer-modified-p))
(inhibit-read-only t))
(with-current-buffer (or buffer-or-name (current-buffer))
(goto-char (point-max))
(let (sexp-beg)
(while (setq sexp-beg (scan-sexps (point) -1))
(goto-char sexp-beg)
(delete-horizontal-space t)
(unless (= (point) (line-beginning-position))
(insert "\n"))
(indent-pp-sexp 'pp)))
(set-buffer-modified-p (if modified-flag-bound?
modified-flag
modified?)))))
(defun guix-pretty-print-file (file-name &optional mode)
"Show FILE-NAME contents in MODE and pretty-print it.
If MODE is nil, use `scheme-mode'.
Put the point in the beginning of buffer.
Return buffer with the prettified contents."
(let* ((base-name (file-name-nondirectory file-name))
(buffer (generate-new-buffer base-name)))
(with-current-buffer buffer
(insert-file-contents file-name)
(goto-char (point-min))
(funcall (or mode 'scheme-mode)))
(guix-pretty-print-buffer buffer)
buffer))
(defun guix-replace-match (regexp string &optional group)
"Replace all occurrences of REGEXP with STRING in the current buffer.
GROUP specifies a parenthesized expression used in REGEXP."
(save-excursion
(goto-char (point-min))
(while (re-search-forward regexp nil t)
(replace-match string nil nil nil group))))
(defmacro guix-while-search (regexp &rest body)
"Evaluate BODY after each search for REGEXP in the current buffer."
(declare (indent 1) (debug t))
`(save-excursion
(goto-char (point-min))
(while (re-search-forward ,regexp nil t)
,@body)))
(defmacro guix-while-null (&rest body)
"Evaluate BODY until its result becomes non-nil."
(declare (indent 0) (debug t))
(let ((result-var (make-symbol "result")))
`(let (,result-var)
(while (null ,result-var)
(setq ,result-var ,@body))
,result-var)))
(defun guix-modify (object &rest modifiers)
"Apply MODIFIERS to OBJECT.
OBJECT is passed as an argument to the first function from
MODIFIERS list, the returned result is passed to the second
function from the list and so on. Return result of the last
modifier call."
(if (null modifiers)
object
(apply #'guix-modify
(funcall (car modifiers) object)
(cdr modifiers))))
(defun guix-modify-objects (objects &rest modifiers)
"Apply MODIFIERS to each object from a list of OBJECTS.
See `guix-modify' for details."
(--map (apply #'guix-modify it modifiers)
objects))
(defun guix-make-symbol (&rest symbols)
"Return `guix-SYMBOLS-...' symbol."
(apply #'bui-make-symbol 'guix symbols))
(defmacro guix-define-groups (name &rest args)
"Define `guix-NAME' and `guix-NAME-faces' customization groups.
See `bui-define-groups' for details."
(declare (indent 1))
`(bui-define-groups ,(bui-make-symbol 'guix name)
:parent-group guix
:parent-faces-group guix-faces
,@args))
;;; Files and Dired
(defcustom guix-find-file-function #'find-file
"Function used to find a file.
This function is called by `guix-find-file' with a file name as a
single argument."
:type '(choice (function-item find-file)
(function-item org-open-file)
(function :tag "Other function"))
:group 'guix)
(defcustom guix-support-dired t
"Whether guix commands support `dired-mode' or not.
Some commands (like `guix-hash' or `guix-package-from-file') take
a file name as argument. If you are in `dired-mode', you may or
may not wish to use the file at point for these commands. This
variable allows you to control this behavior.
If non-nil, do not prompt for a file name in `dired-mode' and use
the file on the current line instead.
If nil, always prompt for a file name."
:type 'boolean
:group 'guix)
(defun guix-file-name (file-name)
"Expand FILE-NAME and remove trailing slash if needed."
(directory-file-name (expand-file-name file-name)))
(defun guix-read-file-name (&optional prompt dir default-filename
mustmatch initial predicate)
"Read file name.
This function is similar to `read-file-name' except it also
expands the file name."
(expand-file-name
(read-file-name (or prompt "File: ")
dir default-filename
mustmatch initial predicate)))
(declare-function dired-get-filename "dired" t)
(defun guix-read-file-name-maybe (&optional prompt dir default-filename
mustmatch initial predicate)
"Read file name or get it from `dired-mode'.
See `guix-support-dired' for details. See also `guix-read-file-name'."
(if (and guix-support-dired
(derived-mode-p 'dired-mode))
(dired-get-filename)
(guix-read-file-name prompt dir default-filename
mustmatch initial predicate)))
(defun guix-read-os-file-name ()
"Read file name with GuixSD 'operating-system' declaration."
(guix-read-file-name-maybe "System configuration file: "))
(defun guix-find-file (file)
"Find FILE (using `guix-find-file-function') if it exists."
(if (file-exists-p file)
(funcall guix-find-file-function file)
(message "File '%s' does not exist." file)))
(defvar url-handler-regexp)
(defun guix-find-file-or-url (file-or-url)
"Find FILE-OR-URL."
;; The code is taken from `browse-url-emacs'.
(require 'url-handlers)
(let ((file-name-handler-alist
(cons (cons url-handler-regexp 'url-file-handler)
file-name-handler-alist)))
(find-file file-or-url)))
(defun guix-assert-files-exist (&rest files)
"Raise an error if any of FILES does not exist."
(dolist (file files)
(unless (file-exists-p file)
(user-error "File does not exist: '%s'" file))))
(defun guix-guile-site-directory (&optional root compiled)
"Return default directory with Guile site files.
Return nil, if this directory does not exist.
ROOT is the parent directory where the default one is placed.
Example of ROOT: \"/usr/local\".
By default, the directory with Scheme files is returned, for
example:
ROOT/share/guile/site/2.2
However, if COMPILED is non-nil, the directory with
compiled (.go) files is returned, for example:
ROOT/lib/guile/2.2/site-ccache
"
(let* ((dir (expand-file-name (if compiled
"lib/guile"
"share/guile/site")
(or root "/")))
(dir (and (file-exists-p dir)
;; digit "[0-9]" is the part of file name (which is
;; "2.3" or alike). Is there a better way to find
;; the directory?
(car (directory-files dir t "[0-9]")))))
(when dir
(if compiled
(expand-file-name "site-ccache" dir)
dir))))
;;; Temporary file names
(defvar guix-temporary-directory nil
"Directory for writing temporary Guix files.
If nil, it will be set when it will be used the first time.
This directory will be deleted on Emacs exit.")
(defun guix-temporary-directory ()
"Return `guix-temporary-directory' (set it if needed)."
(or (and guix-temporary-directory
(file-exists-p guix-temporary-directory)
guix-temporary-directory)
(setq guix-temporary-directory
(make-temp-file "emacs-guix-" 'dir))))
(defun guix-temporary-file-name (name &optional suffix)
"Return file NAME from `guix-temporary-directory'.
If such file name already exists, or if SUFFIX string is
specified, make the returned name unique."
(let* ((file-name (expand-file-name name (guix-temporary-directory)))
(file-name (if suffix
(concat (make-temp-name file-name) suffix)
file-name)))
(if (file-exists-p file-name)
(guix-temporary-file-name name (or suffix ""))
file-name)))
(defun guix-delete-temporary-directory ()
"Delete `guix-temporary-directory' if it exists."
(when (and guix-temporary-directory
(file-exists-p guix-temporary-directory))
(condition-case nil
(delete-directory (guix-temporary-directory) 'recursive)
(error
(message "Failed to delete temporary Guix directory: %s"
guix-temporary-directory)))))
(add-hook 'kill-emacs-hook 'guix-delete-temporary-directory)
;;; Fontification
(defvar guix-font-lock-flush-function
(if (fboundp 'font-lock-flush)
#'font-lock-flush ; appeared in Emacs 25.1
#'jit-lock-refontify)
"Function used to refontify a buffer.
This function is called without arguments after
enabling/disabling `guix-prettify-mode',
`guix-build-log-minor-mode' and `guix-devel-mode'.
If nil, do not perform refontifying.")
(defun guix-font-lock-flush ()
"Refontify the current buffer using `guix-font-lock-flush-function'."
(when guix-font-lock-flush-function
(if (fboundp guix-font-lock-flush-function)
(funcall guix-font-lock-flush-function)
(message "Unknown function: %S" guix-font-lock-flush-function))))
;;; Diff
(defvar guix-diff-switches "-u"
"A string or list of strings specifying switches to be passed to diff.")
(defun guix-diff (old new &optional switches no-async)
"Same as `diff', but use `guix-diff-switches' as default."
(diff old new (or switches guix-diff-switches) no-async))
;;; Completing readers definers
(defmacro guix-define-reader (name read-fun completions prompt
&optional require-match default)
"Define NAME function to read from minibuffer.
READ-FUN may be `completing-read', `completing-read-multiple' or
another function with the same arguments."
(declare (indent 1))
`(defun ,name (&optional prompt initial-contents)
(,read-fun (or prompt ,prompt)
,completions nil ,require-match
initial-contents nil ,default)))
(defmacro guix-define-readers (&rest args)
"Define reader functions.
ARGS should have a form [KEYWORD VALUE] ... The following
keywords are available:
- `completions-var' - variable used to get completions.
- `completions-getter' - function used to get completions.
- `require-match' - if the match is required (see
`completing-read' for details); default is t.
- `default' - default value.
- `single-reader', `single-prompt' - name of a function to read
a single value, and a prompt for it.
- `multiple-reader', `multiple-prompt' - name of a function to
read multiple values, and a prompt for it.
- `multiple-separator' - if specified, another
`<multiple-reader-name>-string' function returning a string
of multiple values separated the specified separator will be
defined."
(bui-plist-let args
((completions-var :completions-var)
(completions-getter :completions-getter)
(require-match :require-match t)
(default :default)
(single-reader :single-reader)
(single-prompt :single-prompt)
(multiple-reader :multiple-reader)
(multiple-prompt :multiple-prompt)
(multiple-separator :multiple-separator))
(let ((completions
(cond ((and completions-var completions-getter)
`(or ,completions-var
(setq ,completions-var
(funcall ',completions-getter))))
(completions-var
completions-var)
(completions-getter
`(funcall ',completions-getter)))))
`(progn
,(when (and completions-var
(not (boundp completions-var)))
`(defvar ,completions-var nil))
,(when single-reader
`(guix-define-reader ,single-reader
guix-completing-read ,completions ,single-prompt
,require-match ,default))
,(when multiple-reader
`(guix-define-reader ,multiple-reader
completing-read-multiple ,completions ,multiple-prompt
,require-match ,default))
,(when (and multiple-reader multiple-separator)
(let ((name (intern (concat (symbol-name multiple-reader)
"-string"))))
`(defun ,name (&optional prompt initial-contents)
(guix-concat-strings
(,multiple-reader prompt initial-contents)
,multiple-separator))))))))
;;; Memoizing
(defun guix-memoize (function)
"Return a memoized version of FUNCTION."
(let ((cache (make-hash-table :test 'equal)))
(lambda (&rest args)
(let ((result (gethash args cache 'not-found)))
(if (eq result 'not-found)
(let ((result (apply function args)))
(puthash args result cache)
result)
result)))))
(defmacro guix-memoized-defun (name arglist docstring &rest body)
"Define a memoized function NAME.
See `defun' for the meaning of arguments."
(declare (doc-string 3) (indent 2))
`(defalias ',name
(guix-memoize (lambda ,arglist ,@body))
;; Add '(name args ...)' string with real arglist to the docstring,
;; because *Help* will display '(name &rest ARGS)' for a defined
;; function (since `guix-memoize' returns a lambda with '(&rest
;; args)').
,(format "(%S %s)\n\n%s"
name
(mapconcat #'symbol-name arglist " ")
docstring)))
(defmacro guix-memoized-defalias (symbol definition &optional docstring)
"Set SYMBOL's function definition to memoized version of DEFINITION."
(declare (doc-string 3) (indent 1))
`(defalias ',symbol
(guix-memoize #',definition)
,(or docstring
(format "Memoized version of `%S'." definition))))
(provide 'guix-utils)
;;; guix-utils.el ends here