Commit 6dd702a7 authored by Tassilo Horn's avatar Tassilo Horn


parent 9682aa2f
Pipeline #5863 passed with stage
in 56 minutes and 44 seconds
......@@ -139,12 +139,208 @@ The second subexpression should match the bug reference (usually a number)."
(when url
(browse-url url))))))
(defun bug-reference--maybe-setup-from-vc (url url-rx bug-rx bug-url-fmt)
(when (string-match url-rx url)
(setq-local bug-reference-bug-regexp bug-rx)
(setq-local bug-reference-url-format
(let (groups)
(dotimes (i (/ (length (match-data)) 2))
(push (match-string i url) groups))
(funcall bug-url-fmt (nreverse groups))))))
(defvar bug-reference-setup-from-vc-alist
`(;; GNU projects on savannah. FIXME: Only a fraction of
;; them uses debbugs.
"\\<\\([Bb]ug ?#?\\)\\([0-9]+\\(?:#[0-9]+\\)?\\)\\>"
,(lambda (_) ""))
;; GitHub projects. Here #17 may refer to either an issue
;; or a pull request but visiting the issue/17 web page
;; will automatically redirect to the pull/17 page if 17 is
;; a PR. Explicit user/project#17 links to possibly
;; different projects are also supported.
,(lambda (groups)
(let ((ns-project (nth 1 groups)))
(lambda ()
(concat ""
;; Explicit user/proj#18 link.
(match-string 1)
(match-string 2))))))
;; GitLab projects. Here #18 is an issue and !17 is a merge
;; request. Explicit namespace/project#18 references to possibly
;; different projects are also supported.
,(lambda (groups)
(let ((ns-project (nth 1 groups)))
(lambda ()
(concat ""
(or (match-string 1)
(if (string= (match-string 3) "#")
(match-string 2)))))))
"An alist for setting up `bug-reference-mode' based on VC URL.
Each element has the form (URL-REGEXP BUG-REGEXP URL-FORMAT-FN).
URL-REGEXP is matched against the version control URL of the
current buffer's file. If it matches, BUG-REGEXP is set as
`bug-reference-bug-regexp'. URL-FORMAT-FN is a function of one
argument that receives a list of the groups 0 to N of matching
URL-REGEXP against the VCS URL and return the value to be set as
(defun bug-reference-try-setup-from-vc ()
"Try setting up `bug-reference-mode' based on VCS information.
Tests each configuration from `bug-reference-setup-from-vc-alist'
and sets it if applicable."
(when buffer-file-name
(let* ((backend (vc-responsible-backend buffer-file-name t))
(or (ignore-errors
(vc-call-backend backend 'repository-url "upstream"))
(vc-call-backend backend 'repository-url)))))
(when url
(catch 'found
(dolist (config bug-reference-setup-from-vc-alist)
(when (apply #'bug-reference--maybe-setup-from-vc
url config)
(throw 'found t))))))))
(defvar bug-reference-setup-from-mail-alist
`((,(regexp-opt '("emacs" "auctex" "gnus") 'words)
,(regexp-opt '("" ""
;; List-Id of Gnus devel mailing list.
"\\([Bb]ug ?#?\\)\\([0-9]+\\(?:#[0-9]+\\)?\\)"
;; TODO: Adapt docstring!
"An alist for setting up `bug-reference-mode' based in mail modes.
This takes action if `bug-reference-mode' is enabled in group and
message buffers of Emacs mail clients. Currently, only Gnus is
Each element has the form
GROUP-REGEXP is a regexp matched against the current mail folder
name or newsgroup. HEADER-REGEXP is a regexp matched against the
From, To, Cc, and List-ID header values of the current mail or
newsgroup message. If any of those matches, BUG-REGEXP is set as
`bug-reference-bug-regexp' and URL-FORMAT is set as
Note: In Gnus, if a summary buffer has been set up based on
GROUP-REGEXP, all article buffers opened from there will get the
same `bug-reference-url-format' and `bug-reference-url-format'.")
(defvar gnus-newsgroup-name)
(defun bug-reference--maybe-setup-from-mail (group headers)
(catch 'setup-done
(dolist (config bug-reference-setup-from-mail-alist)
(when (or
(and group
(car config)
(string-match-p (car config) group))
(and headers
(nth 1 config)
(catch 'matching-header
(dolist (h headers)
(when (and h (string-match-p (nth 1 config) h))
(throw 'matching-header t))))))
(setq-local bug-reference-bug-regexp (nth 2 config))
(setq-local bug-reference-url-format (nth 3 config))
(throw 'setup-done t)))))
(defun bug-reference-try-setup-from-gnus ()
"Try setting up `bug-reference-mode' based on Gnus group or article.
Tests each configuration from `bug-reference-setup-from-mail-alist'
and sets it if applicable."
(when (and (derived-mode-p 'gnus-summary-mode)
(bound-and-true-p gnus-newsgroup-name))
;; Gnus reuses its article buffer so we have to check whenever the
;; article changes.
(add-hook 'gnus-article-prepare-hook
(bug-reference--maybe-setup-from-mail gnus-newsgroup-name nil)))
(defvar gnus-article-buffer)
(defvar gnus-summary-buffer)
(declare-function mail-header-extract "mailheader")
(declare-function mail-header "mailheader")
(defun bug-reference--try-setup-gnus-article ()
"Error in bug-reference--try-setup-gnus-article: %S"
(when (and bug-reference-mode ;; Only if enabled in article buffers.
;; Apparently, gnus-article-prepare-hook is run in the
;; summary buffer...
(buffer-live-p (get-buffer gnus-article-buffer)))
(with-current-buffer gnus-article-buffer
(catch 'setup-done
;; Copy over the values from the summary buffer.
(when (and gnus-summary-buffer
(buffer-live-p gnus-summary-buffer))
(setq-local bug-reference-bug-regexp
(with-current-buffer gnus-summary-buffer
(setq-local bug-reference-url-format
(with-current-buffer gnus-summary-buffer
(when (and bug-reference-bug-regexp
(throw 'setup-done t)))
;; If the summary had no values, try setting according to
;; the values of the From, To, and Cc headers.
(let ((headers (save-excursion
(goto-char (point-min))
(dolist (h '(list-id to from cc))
(let ((val (mail-header h headers)))
(when val
(push val header-values))))
nil header-values)))))))
(defun bug-reference--after-hook ()
(when (or bug-reference-mode
;; Automatic setup only if the variables aren't already set, e.g.,
;; by a local variables section in the file.
(unless (and bug-reference-bug-regexp
"Error during bug-reference auto-setup: %S"
(catch 'setup
(dolist (f (list #'bug-reference-try-setup-from-vc
(when (funcall f)
(throw 'setup t))))))))
(define-minor-mode bug-reference-mode
"Toggle hyperlinking bug references in the buffer (Bug Reference mode)."
:after-hook (bug-reference--after-hook)
(if bug-reference-mode
(jit-lock-register #'bug-reference-fontify)
(jit-lock-unregister #'bug-reference-fontify)
......@@ -158,6 +354,7 @@ The second subexpression should match the bug reference (usually a number)."
:after-hook (bug-reference--after-hook)
(if bug-reference-prog-mode
(jit-lock-register #'bug-reference-fontify)
(jit-lock-unregister #'bug-reference-fontify)
......@@ -964,7 +964,7 @@ use."
(throw 'found bk))))
(defun vc-responsible-backend (file)
(defun vc-responsible-backend (file &optional no-error)
"Return the name of a backend system that is responsible for FILE.
If FILE is already registered, return the
......@@ -974,7 +974,10 @@ responsible for FILE is returned.
Note that if FILE is a symbolic link, it will not be resolved --
the responsible backend system for the symbolic link itself will
be reported."
be reported.
If NO-ERROR is nil, signal an error that no VC backend is
responsible for the given file."
(or (and (not (file-directory-p file)) (vc-backend file))
(catch 'found
;; First try: find a responsible backend. If this is for registration,
......@@ -982,7 +985,8 @@ be reported."
(dolist (backend vc-handled-backends)
(and (vc-call-backend backend 'responsible-p file)
(throw 'found backend))))
(error "No VC backend is responsible for %s" file)))
(unless no-error
(error "No VC backend is responsible for %s" file))))
(defun vc-expand-dirs (file-or-dir-list backend)
"Expands directories in a file list specification.
Markdown is supported
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment