;;; magithub-core.el --- core functions for magithub -*- lexical-binding: t; -*- ;; Copyright (C) 2016 Sean Allred ;; Author: Sean Allred ;; 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 . ;;; 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"))) (when 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