Gergely Polonkai 449ebf466d Update packages
2016-10-03 13:57:29 +02:00

184 lines
6.0 KiB
EmacsLisp

;;; magithub-core.el --- core functions for magithub -*- lexical-binding: t; -*-
;; Copyright (C) 2016 Sean Allred
;; Author: Sean Allred <code@seanallred.com>
;; Keywords: tools
;; 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 <http://www.gnu.org/licenses/>.
;;; Commentary:
;;
;;; Code:
(require 'magit)
(defun magithub-github-repository-p ()
"Non-nil if \"origin\" points to GitHub or a whitelisted domain."
(let ((url (magit-get "remote" "origin" "url")))
(cl-some (lambda (domain) (s-contains? domain url))
(cons "github.com" (magit-get-all "hub" "host")))))
(defun magithub-repo-id ()
"Returns an identifying value for this repository."
(magit-get "remote" "origin" "url"))
(defun magithub--api-available-p ()
"Non-nil if the API is available."
(let ((magit-git-executable "ping")
(magit-pre-call-git-hook nil)
(magit-git-global-arguments nil))
(= 0 (magit-git-exit-code "-c 1" "api.github.com"))))
(defun magithub--completing-read-multiple (prompt collection)
"Using PROMPT, get a list of elements in COLLECTION.
This function continues until all candidates have been entered or
until the user enters a value of \"\". Duplicate entries are not
allowed."
(let (label-list this-label done)
(while (not done)
(setq collection (remove this-label collection)
this-label "")
(when collection
;; @todo it would be nice to detect whether or not we are
;; allowed to create labels -- if not, we can require-match
(setq this-label (completing-read prompt collection)))
(unless (setq done (s-blank? this-label))
(push this-label label-list)))
label-list))
(defconst magithub-hash-regexp
(rx bow (= 40 (| digit (any (?A . ?F) (?a . ?f)))) eow)
"Regexp for matching commit hashes.")
(defcustom magithub-hub-executable "hub"
"The hub executable used by Magithub."
:group 'magithub
:package-version '(magithub . "0.1")
:type 'string)
(defvar magithub-debug-mode nil
"When non-nil, echo hub commands before they're executed.")
(defvar magithub-hub-error nil
"When non-nil, this is a message for when hub fails.")
(defmacro magithub-with-hub (&rest body)
`(let ((magit-git-executable magithub-hub-executable)
(magit-pre-call-git-hook nil)
(magit-git-global-arguments nil))
,@body))
(defun magithub--hub-command (magit-function command args)
(unless (executable-find magithub-hub-executable)
(user-error "Hub (hub.github.com) not installed; aborting"))
(unless (file-exists-p "~/.config/hub")
(user-error "Hub hasn't been initialized yet; aborting"))
(when magithub-debug-mode
(message "Calling hub with args: %S %S" command args))
(with-timeout (5 (error "Took too long! %s%S" command args))
(magithub-with-hub (funcall magit-function command args))))
(defun magithub--command (command &optional args)
"Run COMMAND synchronously using `magithub-hub-executable'."
(magithub--hub-command #'magit-run-git command args))
(defun magithub--command-with-editor (command &optional args)
"Run COMMAND asynchronously using `magithub-hub-executable'.
Ensure GIT_EDITOR is set up appropriately."
(magithub--hub-command #'magit-run-git-with-editor command args))
(defun magithub--command-output (command &optional args)
"Run COMMAND synchronously using `magithub-hub-executable'
and returns its output as a list of lines."
(magithub--hub-command #'magit-git-lines command args))
(defun magithub--command-quick (command &optional args)
"Quickly execute COMMAND with ARGS."
(ignore (magithub--command-output command args)))
(defun magithub--meta-new-issue ()
"Open a new Magithub issue.
See /.github/ISSUE_TEMPLATE.md in this repository."
(interactive)
(browse-url "https://github.com/vermiculus/magithub/issues/new"))
(defun magithub--meta-help ()
"Opens Magithub help."
(interactive)
(browse-url "https://gitter.im/vermiculus/magithub"))
(defun magithub-enable ()
"Enable Magithub for this repository."
(interactive)
(magit-set "yes" "magithub" "enabled")
(when (derived-mode-p 'magit-status-mode)
(magit-refresh)))
(defun magithub-disable ()
"Disable Magithub for this repository."
(interactive)
(magit-set "no" "magithub" "enabled")
(when (derived-mode-p 'magit-status-mode)
(magit-refresh)))
(defun magithub-enabled-p ()
"Returns non-nil when Magithub is enabled for this repository."
(when (member (magit-get "magithub" "enabled") '("yes" nil)) t))
(defun magithub-enabled-toggle ()
"Toggle Magithub"
(interactive)
(if (magithub-enabled-p)
(magithub-disable)
(magithub-enable)))
(defun magithub-usable-p ()
"Non-nil if Magithub should do its thing."
(and (executable-find magithub-hub-executable)
(magithub-enabled-p)
(magithub-github-repository-p)))
(defun magithub-error (err-message tag &optional trace)
"Report a Magithub error."
(setq trace (or trace (with-output-to-string (backtrace))))
(when (y-or-n-p (concat tag " Report? (A bug report will be placed in your clipboard.)"))
(with-current-buffer-window
(get-buffer-create "*magithub issue*")
#'display-buffer-pop-up-window nil
(when (fboundp 'markdown-mode) (markdown-mode))
(insert
(kill-new
(format
"## Automated error report
### Description
%s
### Backtrace
```
%s```
"
(read-string "Briefly describe what you were doing: ")
trace))))
(magithub--meta-new-issue))
(error err-message))
(provide 'magithub-core)
;;; magithub-core.el ends here