You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

371 lines
14 KiB

;;; dash-defs.el --- Definitions for Dash examples -*- lexical-binding: t -*-
;; Copyright (C) 2021-2025 Free Software Foundation, Inc.
;; 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 <https://www.gnu.org/licenses/>.
;;; Code:
(require 'dash)
(require 'ert)
(require 'find-func)
;; Added in Emacs 24.4; wrap in `eval-when-compile' when support is dropped.
(require 'subr-x nil t)
(declare-function string-remove-prefix "subr-x" (prefix string))
(declare-function string-remove-suffix "subr-x" (suffix string))
(defvar dash--groups ()
"Alist of grouped examples.
Each element is of the form (NAME . DOC) or (FN . EXAMPLES)
corresponding to the eponymous arguments of `def-example-group'
and `defexamples', respectively. The only difference is that
EXAMPLES are partitioned into triples (ACTUAL OP EXPECTED), where
EXPECTED should be the result of evaluating ACTUAL, and OP is one
of the following comparison operators:
- `=>' ACTUAL should be `equal' to EXPECTED.
- `~>' ACTUAL should be `approx=' to EXPECTED.
- `!!>' ACTUAL should signal the EXPECTED error,
either an error symbol or an error object.")
(defvar dash--epsilon 1e-15
"Epsilon used in `approx='.")
(defun approx= (u v)
"Like `=', but compares floats within `dash--epsilon'.
This allows approximate comparison of floats to work around
differences in implementation between systems. Used in place of
`equal' when testing actual and expected values with `~>'."
(or (= u v)
(< (/ (abs (- u v))
(max (abs u) (abs v)))
dash--epsilon)))
(defun dash--example-to-test (example)
"Return an ERT assertion form based on EXAMPLE.
Signal an error if EXAMPLE is malformed."
(pcase example
(`(,actual => ,expected) `(should (equal ,actual ,expected)))
(`(,actual ~> ,expected) `(should (approx= ,actual ,expected)))
(`(,actual !!> ,(and (pred symbolp) expected))
`(should-error ,actual :type ',expected))
(`(,actual !!> ,expected)
`(should (equal (should-error ,actual) ',expected)))
(_ (error "Invalid test case: %S" example))))
(defmacro def-example-group (name doc &rest examples)
"Define a group with NAME and DOC of EXAMPLES of several functions.
See `dash--groups'."
(declare (indent defun))
`(progn
(push (cons ,name ,doc) dash--groups)
,@examples))
(defvar dash--find-example-regexp
;; Could use `rx' directly in Emacs 27+
;; (or not rely on `find-function-space-re').
(let ((re `(: "(defexamples" (regexp ,find-function-space-re)
symbol-start (regexp "%s") symbol-end)))
(rx-to-string re t))
"Regexp matching `defexamples' for `find-function-regexp-alist'.")
(defmacro defexamples (fn &rest examples)
"Define a set of EXAMPLES and corresponding ERT tests for FN.
See `dash--groups'."
(declare (indent defun))
(let (triples tests)
(while (let ((triple (-take 3 examples)))
(push (dash--example-to-test triple) tests)
(push triple triples)
(setq examples (nthcdr 3 examples))))
`(progn
(push (cons ',fn ',(nreverse triples)) dash--groups)
(when (fboundp 'find-function-update-type-alist)
;; Help `ert-describe-test' locate examples in Emacs 31+.
(find-function-update-type-alist
',fn 'ert--test 'dash--find-example-regexp))
(ert-deftest ,fn () ,@(nreverse tests)))))
;; Added in Emacs 25.1.
(defvar text-quoting-style)
(autoload 'help-fns--analyze-function "help-fns")
(defun dash--describe (fn)
"Return the (ARGLIST . DOCSTRING) of FN symbol.
Based on `describe-function-1'."
;; Gained last arg in Emacs 25.1.
(declare-function help-fns--signature "help-fns"
(function doc real-def real-function buffer))
(or (get fn 'dash-doc)
(with-temp-buffer
(pcase-let* ((text-quoting-style 'grave)
(`(,real-fn ,_def ,_alias ,real-def)
(help-fns--analyze-function fn))
(buf (current-buffer))
(doc-raw (documentation fn t))
(doc (help-fns--signature
fn doc-raw real-def real-fn buf)))
(goto-char (1+ (point-min)))
(delete-region (point) (progn (forward-sexp) (1+ (point))))
(downcase-region (point) (point-max))
(backward-char)
;; Memoize.
(put fn 'dash-doc (cons (read buf) doc))))))
(defun dash--replace-all (old new)
"Replace occurrences of OLD with NEW in current buffer."
(goto-char (point-min))
(while (search-forward old nil t)
(replace-match new t t)))
(defun dash--github-link (fn)
"Return a GitHub Flavored Markdown link to FN."
(or (get fn 'dash-link)
(let* ((sig (car (dash--describe fn)))
(id (string-remove-prefix "!" (format "%s%s" fn sig)))
(id (replace-regexp-in-string (rx (+ (not (in alnum ?-))))
"-" id t t))
(id (string-remove-suffix "-" id)))
;; Memoize.
(put fn 'dash-link (format "[`%s`](#%s)" fn id)))))
(defun dash--argnames-to-md ()
"Downcase and quote arg names in current buffer for Markdown."
(let ((beg (point-min)))
(while (setq beg (text-property-any beg (point-max)
'face 'help-argument-name))
(goto-char beg)
(insert ?`)
(goto-char (or (next-single-property-change (point) 'face)
(point-max)))
(downcase-region (1+ beg) (point))
(insert ?`)
(setq beg (point)))))
(defun dash--metavars-to-md ()
"Downcase and quote metavariables in current buffer for Markdown."
(goto-char (point-min))
(while (re-search-forward (rx bow (group (in upper) (* (in upper ?-)) (* num))
(| (group ?\() (: (group (? "th")) eow)))
nil t)
(unless (match-beginning 2)
(let* ((suf (match-string 3))
(var (format "`%s`%s" (downcase (match-string 1)) suf)))
(replace-match var t t)))))
(defun dash--hyperlinks-to-md ()
"Convert hyperlinks in current buffer from Elisp to Markdown."
(goto-char (point-min))
(while (re-search-forward (rx ?` (+? (not (in " `"))) ?\') nil t)
(let ((fn (intern (substring (match-string 0) 1 -1))))
(replace-match (if (assq fn dash--groups)
(save-match-data (dash--github-link fn))
(format "`%s`" fn))
t t))))
(defun dash--booleans-to-md ()
"Mark up booleans (nil/t) in current buffer as Markdown."
(goto-char (point-min))
(while (re-search-forward (rx bow (| "nil" "t") eow) nil t)
(unless (memql (char-before (match-beginning 0)) '(?\' ?`))
(replace-match "`\\&`" t))))
(defun dash--indent-md-blocks ()
"Indent example blocks in current buffer for Markdown."
(goto-char (point-min))
(while (re-search-forward (rx bol " ") nil t)
(replace-match " " t t)))
(defun dash--docstring-to-md (doc)
"Transcribe DOC to Markdown syntax."
(with-temp-buffer
(insert doc)
(dash--argnames-to-md)
(dash--metavars-to-md)
(dash--hyperlinks-to-md)
(dash--booleans-to-md)
(dash--indent-md-blocks)
(buffer-string)))
(defun dash--docstring-to-texi (doc)
"Transcribe DOC to Texinfo syntax."
(with-temp-buffer
(insert doc)
;; Escape literal ?@.
(dash--replace-all "@" "@@")
(goto-char (point-min))
;; TODO: Use `help-argument-name' like in `dash--argnames-to-md'?
(while (re-search-forward
(rx (| (group bow (in "A-Z") (* (in "A-Z" ?-)) (* num) eow)
(: ?` (group (+? (not (in ?\s)))) ?\')
(group bow (| "nil" "t") eow)
(: "..." (? (group eol)))))
nil t)
(cond ((match-beginning 1)
;; Downcase metavariable reference.
(downcase-region (match-beginning 1) (match-end 1))
(replace-match "@var{\\1}" t))
((match-beginning 2)
;; `quoted' symbol.
(replace-match (if (assq (intern (match-string 2)) dash--groups)
"@code{\\2} (@pxref{\\2})"
"@code{\\2}")
t))
;; nil/t.
((match-beginning 3)
(unless (= (char-before (match-beginning 3)) ?\')
(replace-match "@code{\\3}" t)))
;; Ellipses.
((match-beginning 4) (replace-match "@enddots{}" t t))
((replace-match "@dots{}" t t))))
(buffer-string)))
;; Added in Emacs 26.1.
(defvar print-escape-control-characters)
(defun dash--lisp-to-md (obj)
"Print Lisp OBJ suitably for Markdown."
(let ((print-quoted t)
(print-escape-control-characters t))
(save-excursion (prin1 obj)))
(while (re-search-forward (rx (| (group ?\' symbol-start "nil" symbol-end)
(group "\\00") "\\?"))
nil 'move)
(replace-match (cond ((match-beginning 1) "()") ; 'nil -> ().
((match-beginning 2) "\\") ; \00N -> \N.
("?")) ; `-any\?' -> `-any?'.
t t)))
(defun dash--lisp-to-texi (obj)
"Print Lisp OBJ suitably for Texinfo."
(save-excursion (dash--lisp-to-md obj))
(while (re-search-forward (rx (in "{}")) nil 'move)
(replace-match "@\\&" t))) ;; { -> @{.
(defun dash--expected (obj err)
"Prepare OBJ for printing as an expected evaluation result.
ERR non-nil means OBJ is either an error symbol or error object."
(cond ((and (eq (car-safe obj) 'quote)
(not (equal obj ''())))
;; Unquote expected result.
(cadr obj))
;; Print actual error message.
(err (error-message-string (-list obj)))
(obj)))
(defun dash--example-to-md (example)
"Return a Markdown string documenting EXAMPLE."
(pcase-let* ((`(,actual ,op ,expected) example)
(err (eq op '!!>)))
(setq expected (dash--expected expected err))
(with-output-to-string
(with-current-buffer standard-output
(dash--lisp-to-md actual)
(insert " ;; ")
(cond ((memq op '(=> ~>))
(princ op)
(insert ?\s)
(dash--lisp-to-md expected))
(err (princ expected))
((error "Invalid test case: %S" example)))))))
(defun dash--example-to-texi (example)
"Return a Texinfo string documenting EXAMPLE."
(pcase-let* ((`(,actual ,op ,expected) example)
(err (eq op '!!>)))
(setq expected (dash--expected expected err))
(with-output-to-string
(with-current-buffer standard-output
(insert "@group\n")
(dash--lisp-to-texi actual)
(insert "\n " (if err "@error{}" "@result{}") ?\s)
(funcall (if err #'princ #'dash--lisp-to-texi) expected)
(insert "\n@end group")))))
(defun dash--group-to-md (group)
"Return a Markdown string documenting GROUP."
(pcase group
(`(,(and (pred stringp) name) . ,doc)
(concat "## " name "\n\n" (dash--docstring-to-md doc) "\n"))
((and `(,fn . ,examples)
(let `(,sig . ,doc) (dash--describe fn)))
(format "#### %s `%s`\n\n%s\n\n```el\n%s\n```\n"
fn sig (dash--docstring-to-md doc)
(mapconcat #'dash--example-to-md (-take 3 examples) "\n")))))
(defun dash--group-to-texi (group)
"Return a Texinfo string documenting GROUP."
;; Added in Emacs 24.4.
(declare-function macrop "subr" (object))
(pcase group
(`(,(and (pred stringp) name) . ,doc)
(concat "@node " name "\n@section " name "\n\n"
(dash--docstring-to-texi doc) "\n"))
((and `(,fn . ,examples)
(let `(,sig . ,doc) (dash--describe fn))
(let type (if (macrop fn) "defmac" "defun")))
(format (concat "@anchor{%s}\n"
"@%s %s %s\n"
"%s\n\n"
"@example\n%s\n@end example\n"
"@end %s\n")
fn type fn sig (dash--docstring-to-texi doc)
(mapconcat #'dash--example-to-texi (-take 3 examples) "\n")
type))))
(defun dash--summary-to-md (group)
"Return a Markdown string summarizing GROUP."
(pcase group
(`(,(and (pred stringp) name) . ,doc)
(concat "\n### " name "\n\n" (dash--docstring-to-md doc) "\n"))
((and `(,fn . ,_) (let sig (car (dash--describe fn))))
(format "* %s `%s`" (dash--github-link fn) sig))))
(autoload 'lm-version "lisp-mnt")
(defun dash--make-md ()
"Generate Markdown README."
(with-temp-file "README.md"
(insert-file-contents "readme-template.md")
(dash--replace-all "[[ dash-version ]]" (lm-version "dash.el"))
(dash--replace-all "[[ function-list ]]"
(mapconcat #'dash--summary-to-md dash--groups "\n"))
(dash--replace-all "[[ function-docs ]]"
(mapconcat #'dash--group-to-md dash--groups "\n"))))
(defun dash--make-texi ()
"Generate Texinfo manual."
(with-temp-file "dash.texi"
(insert-file-contents "dash-template.texi")
(dash--replace-all "@c [[ dash-version ]]" (lm-version "dash.el"))
(dash--replace-all
"@c [[ function-list ]]"
(mapconcat (lambda (group) (concat "* " (car group) "::"))
(--filter (stringp (car it)) dash--groups)
"\n"))
(dash--replace-all "@c [[ function-docs ]]"
(mapconcat #'dash--group-to-texi dash--groups "\n"))))
(defun dash-make-docs ()
"Generate Dash Markdown README and Texinfo manual."
(let ((dash--groups (reverse dash--groups))
(case-fold-search nil))
(dash--make-md)
(dash--make-texi)))
(provide 'dash-defs)
;;; dash-defs.el ends here