1 ;;; git.el --- A user interface for git
3 ;; Copyright (C) 2005, 2006, 2007, 2008, 2009 Alexandre Julliard <julliard@winehq.org>
7 ;; This program is free software; you can redistribute it and/or
8 ;; modify it under the terms of the GNU General Public License as
9 ;; published by the Free Software Foundation; either version 2 of
10 ;; the License, or (at your option) any later version.
12 ;; This program is distributed in the hope that it will be
13 ;; useful, but WITHOUT ANY WARRANTY; without even the implied
14 ;; warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
15 ;; PURPOSE. See the GNU General Public License for more details.
17 ;; You should have received a copy of the GNU General Public
18 ;; License along with this program; if not, write to the Free
19 ;; Software Foundation, Inc., 59 Temple Place, Suite 330, Boston,
24 ;; This file contains an interface for the git version control
25 ;; system. It provides easy access to the most frequently used git
26 ;; commands. The user interface is as far as possible identical to
27 ;; that of the PCL-CVS mode.
29 ;; To install: put this file on the load-path and place the following
30 ;; in your .emacs file:
34 ;; To start: `M-x git-status'
37 ;; - diff against other branch
38 ;; - renaming files from the status buffer
42 ;; - git-show-branch browser
47 ;; This file works on GNU Emacs 21 or later. It may work on older
48 ;; versions but this is not guaranteed.
50 ;; It may work on XEmacs 21, provided that you first install the ewoc
51 ;; and log-edit packages.
54 (eval-when-compile (require 'cl
))
61 ;;;; ------------------------------------------------------------
64 "A user interface for the git versioning system."
67 (defcustom git-committer-name nil
68 "User name to use for commits.
69 The default is to fall back to the repository config,
70 then to `add-log-full-name' and then to `user-full-name'."
72 :type
'(choice (const :tag
"Default" nil
)
73 (string :tag
"Name")))
75 (defcustom git-committer-email nil
76 "Email address to use for commits.
77 The default is to fall back to the git repository config,
78 then to `add-log-mailing-address' and then to `user-mail-address'."
80 :type
'(choice (const :tag
"Default" nil
)
81 (string :tag
"Email")))
83 (defcustom git-commits-coding-system nil
84 "Default coding system for the log message of git commits."
86 :type
'(choice (const :tag
"From repository config" nil
)
89 (defcustom git-append-signed-off-by nil
90 "Whether to append a Signed-off-by line to the commit message before editing."
94 (defcustom git-reuse-status-buffer t
95 "Whether `git-status' should try to reuse an existing buffer
96 if there is already one that displays the same directory."
100 (defcustom git-per-dir-ignore-file
".gitignore"
101 "Name of the per-directory ignore file."
105 (defcustom git-show-uptodate nil
106 "Whether to display up-to-date files."
110 (defcustom git-show-ignored nil
111 "Whether to display ignored files."
115 (defcustom git-show-unknown t
116 "Whether to display unknown files."
121 (defface git-status-face
122 '((((class color
) (background light
)) (:foreground
"purple"))
123 (((class color
) (background dark
)) (:foreground
"salmon")))
124 "Git mode face used to highlight added and modified files."
127 (defface git-unmerged-face
128 '((((class color
) (background light
)) (:foreground
"red" :bold t
))
129 (((class color
) (background dark
)) (:foreground
"red" :bold t
)))
130 "Git mode face used to highlight unmerged files."
133 (defface git-unknown-face
134 '((((class color
) (background light
)) (:foreground
"goldenrod" :bold t
))
135 (((class color
) (background dark
)) (:foreground
"goldenrod" :bold t
)))
136 "Git mode face used to highlight unknown files."
139 (defface git-uptodate-face
140 '((((class color
) (background light
)) (:foreground
"grey60"))
141 (((class color
) (background dark
)) (:foreground
"grey40")))
142 "Git mode face used to highlight up-to-date files."
145 (defface git-ignored-face
146 '((((class color
) (background light
)) (:foreground
"grey60"))
147 (((class color
) (background dark
)) (:foreground
"grey40")))
148 "Git mode face used to highlight ignored files."
151 (defface git-mark-face
152 '((((class color
) (background light
)) (:foreground
"red" :bold t
))
153 (((class color
) (background dark
)) (:foreground
"tomato" :bold t
)))
154 "Git mode face used for the file marks."
157 (defface git-header-face
158 '((((class color
) (background light
)) (:foreground
"blue"))
159 (((class color
) (background dark
)) (:foreground
"blue")))
160 "Git mode face used for commit headers."
163 (defface git-separator-face
164 '((((class color
) (background light
)) (:foreground
"brown"))
165 (((class color
) (background dark
)) (:foreground
"brown")))
166 "Git mode face used for commit separator."
169 (defface git-permission-face
170 '((((class color
) (background light
)) (:foreground
"green" :bold t
))
171 (((class color
) (background dark
)) (:foreground
"green" :bold t
)))
172 "Git mode face used for permission changes."
177 ;;;; ------------------------------------------------------------
179 (defconst git-log-msg-separator
"--- log message follows this line ---")
181 (defvar git-log-edit-font-lock-keywords
182 `(("^\\(Author:\\|Date:\\|Merge:\\|Signed-off-by:\\)\\(.*\\)$"
183 (1 font-lock-keyword-face
)
184 (2 font-lock-function-name-face
))
185 (,(concat "^\\(" (regexp-quote git-log-msg-separator
) "\\)$")
186 (1 font-lock-comment-face
))))
188 (defun git-get-env-strings (env)
189 "Build a list of NAME=VALUE strings from a list of environment strings."
190 (mapcar (lambda (entry) (concat (car entry
) "=" (cdr entry
))) env
))
192 (defun git-call-process (buffer &rest args
)
193 "Wrapper for call-process that sets environment strings."
194 (apply #'call-process
"git" nil buffer nil args
))
196 (defun git-call-process-display-error (&rest args
)
197 "Wrapper for call-process that displays error messages."
198 (let* ((dir default-directory
)
199 (buffer (get-buffer-create "*Git Command Output*"))
200 (ok (with-current-buffer buffer
201 (let ((default-directory dir
)
202 (buffer-read-only nil
))
204 (eq 0 (apply #'git-call-process
(list buffer t
) args
))))))
205 (unless ok
(display-message-or-buffer buffer
))
208 (defun git-call-process-string (&rest args
)
209 "Wrapper for call-process that returns the process output as a string,
210 or nil if the git command failed."
212 (and (eq 0 (apply #'git-call-process t args
))
215 (defun git-call-process-string-display-error (&rest args
)
216 "Wrapper for call-process that displays error message and returns
217 the process output as a string, or nil if the git command failed."
219 (if (eq 0 (apply #'git-call-process
(list t t
) args
))
221 (display-message-or-buffer (current-buffer))
224 (defun git-run-process-region (buffer start end program args
)
225 "Run a git process with a buffer region as input."
226 (let ((output-buffer (current-buffer))
227 (dir default-directory
))
228 (with-current-buffer buffer
230 (apply #'call-process-region start end program
231 nil
(list output-buffer t
) nil args
))))
233 (defun git-run-command-buffer (buffer-name &rest args
)
234 "Run a git command, sending the output to a buffer named BUFFER-NAME."
235 (let ((dir default-directory
)
236 (buffer (get-buffer-create buffer-name
)))
237 (message "Running git %s..." (car args
))
238 (with-current-buffer buffer
239 (let ((default-directory dir
)
240 (buffer-read-only nil
))
242 (apply #'git-call-process buffer args
)))
243 (message "Running git %s...done" (car args
))
246 (defun git-run-command-region (buffer start end env
&rest args
)
247 "Run a git command with specified buffer region as input."
250 (git-run-process-region
251 buffer start end
"env"
252 (append (git-get-env-strings env
) (list "git") args
))
253 (git-run-process-region buffer start end
"git" args
)))
255 (display-message-or-buffer (current-buffer))
258 (defun git-run-hook (hook env
&rest args
)
259 "Run a git hook and display its output if any."
260 (let ((dir default-directory
)
261 (hook-name (expand-file-name (concat ".git/hooks/" hook
))))
262 (or (not (file-executable-p hook-name
))
263 (let (status (buffer (get-buffer-create "*Git Hook Output*")))
264 (with-current-buffer buffer
269 (apply #'call-process
"env" nil
(list buffer t
) nil
270 (append (git-get-env-strings env
) (list hook-name
) args
))
271 (apply #'call-process hook-name nil
(list buffer t
) nil args
))))
272 (display-message-or-buffer buffer
)
275 (defun git-get-string-sha1 (string)
276 "Read a SHA1 from the specified string."
278 (string-match "[0-9a-f]\\{40\\}" string
)
279 (match-string 0 string
)))
281 (defun git-get-committer-name ()
282 "Return the name to use as GIT_COMMITTER_NAME."
283 ; copied from log-edit
284 (or git-committer-name
285 (git-config "user.name")
286 (and (boundp 'add-log-full-name
) add-log-full-name
)
287 (and (fboundp 'user-full-name
) (user-full-name))
288 (and (boundp 'user-full-name
) user-full-name
)))
290 (defun git-get-committer-email ()
291 "Return the email address to use as GIT_COMMITTER_EMAIL."
292 ; copied from log-edit
293 (or git-committer-email
294 (git-config "user.email")
295 (and (boundp 'add-log-mailing-address
) add-log-mailing-address
)
296 (and (fboundp 'user-mail-address
) (user-mail-address))
297 (and (boundp 'user-mail-address
) user-mail-address
)))
299 (defun git-get-commits-coding-system ()
300 "Return the coding system to use for commits."
301 (let ((repo-config (git-config "i18n.commitencoding")))
302 (or git-commits-coding-system
304 (fboundp 'locale-charset-to-coding-system
)
305 (locale-charset-to-coding-system repo-config
))
308 (defun git-get-logoutput-coding-system ()
309 "Return the coding system used for git-log output."
310 (let ((repo-config (or (git-config "i18n.logoutputencoding")
311 (git-config "i18n.commitencoding"))))
312 (or git-commits-coding-system
314 (fboundp 'locale-charset-to-coding-system
)
315 (locale-charset-to-coding-system repo-config
))
318 (defun git-escape-file-name (name)
319 "Escape a file name if necessary."
320 (if (string-match "[\n\t\"\\]" name
)
322 (mapconcat (lambda (c)
328 (t (char-to-string c
))))
333 (defun git-success-message (text files
)
334 "Print a success message after having handled FILES."
335 (let ((n (length files
)))
337 (message "%s %s" text
(car files
))
338 (message "%s %d files" text n
))))
340 (defun git-get-top-dir (dir)
341 "Retrieve the top-level directory of a git tree."
342 (let ((cdup (with-output-to-string
343 (with-current-buffer standard-output
345 (unless (eq 0 (git-call-process t
"rev-parse" "--show-cdup"))
346 (error "cannot find top-level git tree for %s." dir
))))))
347 (expand-file-name (concat (file-name-as-directory dir
)
348 (car (split-string cdup
"\n"))))))
351 (defun git-append-to-ignore (file)
352 "Add a file name to the ignore file in its directory."
353 (let* ((fullname (expand-file-name file
))
354 (dir (file-name-directory fullname
))
355 (name (file-name-nondirectory fullname
))
356 (ignore-name (expand-file-name git-per-dir-ignore-file dir
))
357 (created (not (file-exists-p ignore-name
))))
358 (save-window-excursion
359 (set-buffer (find-file-noselect ignore-name
))
360 (goto-char (point-max))
361 (unless (zerop (current-column)) (insert "\n"))
362 (insert "/" name
"\n")
363 (sort-lines nil
(point-min) (point-max))
366 (git-call-process nil
"update-index" "--add" "--" (file-relative-name ignore-name
)))
367 (git-update-status-files (list (file-relative-name ignore-name
)))))
369 ; propertize definition for XEmacs, stolen from erc-compat
371 (unless (fboundp 'propertize
)
372 (defun propertize (string &rest props
)
373 (let ((string (copy-sequence string
)))
375 (put-text-property 0 (length string
) (nth 0 props
) (nth 1 props
) string
)
376 (setq props
(cddr props
)))
379 ;;;; Wrappers for basic git commands
380 ;;;; ------------------------------------------------------------
382 (defun git-rev-parse (rev)
383 "Parse a revision name and return its SHA1."
385 (git-call-process-string "rev-parse" rev
)))
387 (defun git-config (key)
388 "Retrieve the value associated to KEY in the git repository config file."
389 (let ((str (git-call-process-string "config" key
)))
390 (and str
(car (split-string str
"\n")))))
392 (defun git-symbolic-ref (ref)
393 "Wrapper for the git-symbolic-ref command."
394 (let ((str (git-call-process-string "symbolic-ref" ref
)))
395 (and str
(car (split-string str
"\n")))))
397 (defun git-update-ref (ref newval
&optional oldval reason
)
398 "Update a reference by calling git-update-ref."
399 (let ((args (and oldval
(list oldval
))))
400 (when newval
(push newval args
))
405 (unless newval
(push "-d" args
))
406 (apply 'git-call-process-display-error
"update-ref" args
)))
408 (defun git-for-each-ref (&rest specs
)
409 "Return a list of refs using git-for-each-ref.
410 Each entry is a cons of (SHORT-NAME . FULL-NAME)."
413 (apply #'git-call-process t
"for-each-ref" "--format=%(refname)" specs
)
414 (goto-char (point-min))
415 (while (re-search-forward "^[^/\n]+/[^/\n]+/\\(.+\\)$" nil t
)
416 (push (cons (match-string 1) (match-string 0)) refs
)))
419 (defun git-read-tree (tree &optional index-file
)
420 "Read a tree into the index file."
421 (let ((process-environment
422 (append (and index-file
(list (concat "GIT_INDEX_FILE=" index-file
))) process-environment
)))
423 (apply 'git-call-process-display-error
"read-tree" (if tree
(list tree
)))))
425 (defun git-write-tree (&optional index-file
)
426 "Call git-write-tree and return the resulting tree SHA1 as a string."
427 (let ((process-environment
428 (append (and index-file
(list (concat "GIT_INDEX_FILE=" index-file
))) process-environment
)))
430 (git-call-process-string-display-error "write-tree"))))
432 (defun git-commit-tree (buffer tree parent
)
433 "Create a commit and possibly update HEAD.
434 Create a commit with the message in BUFFER using the tree with hash TREE.
435 Use PARENT as the parent of the new commit. If PARENT is the current \"HEAD\",
436 update the \"HEAD\" reference to the new commit."
437 (let ((author-name (git-get-committer-name))
438 (author-email (git-get-committer-email))
439 (subject "commit (initial): ")
440 author-date log-start log-end args coding-system-for-write
)
442 (setq subject
"commit: ")
445 (with-current-buffer buffer
446 (goto-char (point-min))
448 (setq log-start
(re-search-forward (concat "^" (regexp-quote git-log-msg-separator
) "\n") nil t
))
450 (narrow-to-region (point-min) log-start
)
451 (goto-char (point-min))
452 (when (re-search-forward "^Author: +\\(.*?\\) *<\\(.*\\)> *$" nil t
)
453 (setq author-name
(match-string 1)
454 author-email
(match-string 2)))
455 (goto-char (point-min))
456 (when (re-search-forward "^Date: +\\(.*\\)$" nil t
)
457 (setq author-date
(match-string 1)))
458 (goto-char (point-min))
459 (when (re-search-forward "^Merge: +\\(.*\\)" nil t
)
460 (setq subject
"commit (merge): ")
461 (dolist (parent (split-string (match-string 1) " +" t
))
463 (push parent args
))))
464 (setq log-start
(point-min)))
465 (setq log-end
(point-max))
466 (goto-char log-start
)
467 (when (re-search-forward ".*$" nil t
)
468 (setq subject
(concat subject
(match-string 0))))
469 (setq coding-system-for-write buffer-file-coding-system
))
472 (let ((env `(("GIT_AUTHOR_NAME" .
,author-name
)
473 ("GIT_AUTHOR_EMAIL" .
,author-email
)
474 ("GIT_COMMITTER_NAME" .
,(git-get-committer-name))
475 ("GIT_COMMITTER_EMAIL" .
,(git-get-committer-email)))))
476 (when author-date
(push `("GIT_AUTHOR_DATE" .
,author-date
) env
))
477 (apply #'git-run-command-region
478 buffer log-start log-end env
479 "commit-tree" tree
(nreverse args
))))))
480 (when commit
(git-update-ref "HEAD" commit parent subject
))
483 (defun git-empty-db-p ()
484 "Check if the git db is empty (no commit done yet)."
485 (not (eq 0 (git-call-process nil
"rev-parse" "--verify" "HEAD"))))
487 (defun git-get-merge-heads ()
488 "Retrieve the merge heads from the MERGE_HEAD file if present."
490 (when (file-readable-p ".git/MERGE_HEAD")
492 (insert-file-contents ".git/MERGE_HEAD" nil nil nil t
)
493 (goto-char (point-min))
494 (while (re-search-forward "[0-9a-f]\\{40\\}" nil t
)
495 (push (match-string 0) heads
))))
498 (defun git-get-commit-description (commit)
499 "Get a one-line description of COMMIT."
500 (let ((coding-system-for-read (git-get-logoutput-coding-system)))
501 (let ((descr (git-call-process-string "log" "--max-count=1" "--pretty=oneline" commit
)))
502 (if (and descr
(string-match "\\`\\([0-9a-f]\\{40\\}\\) *\\(.*\\)$" descr
))
503 (concat (substring (match-string 1 descr
) 0 10) " - " (match-string 2 descr
))
506 ;;;; File info structure
507 ;;;; ------------------------------------------------------------
509 ; fileinfo structure stolen from pcl-cvs
510 (defstruct (git-fileinfo
512 (:constructor git-create-fileinfo
(state name
&optional old-perm new-perm rename-state orig-name marked
))
513 (:conc-name git-fileinfo-
>))
515 state
;; current state
517 old-perm new-perm
;; permission flags
518 rename-state
;; rename or copy state
519 orig-name
;; original name for renames or copies
520 needs-update
;; whether file needs to be updated
521 needs-refresh
) ;; whether file needs to be refreshed
523 (defvar git-status nil
)
525 (defun git-set-fileinfo-state (info state
)
526 "Set the state of a file info."
527 (unless (eq (git-fileinfo->state info
) state
)
528 (setf (git-fileinfo->state info
) state
529 (git-fileinfo->new-perm info
) (git-fileinfo->old-perm info
)
530 (git-fileinfo->rename-state info
) nil
531 (git-fileinfo->orig-name info
) nil
532 (git-fileinfo->needs-update info
) nil
533 (git-fileinfo->needs-refresh info
) t
)))
535 (defun git-status-filenames-map (status func files
&rest args
)
536 "Apply FUNC to the status files names in the FILES list.
537 The list must be sorted."
539 (let ((file (pop files
))
540 (node (ewoc-nth status
0)))
541 (while (and file node
)
542 (let* ((info (ewoc-data node
))
543 (name (git-fileinfo->name info
)))
544 (if (string-lessp name file
)
545 (setq node
(ewoc-next status node
))
546 (if (string-equal name file
)
547 (apply func info args
))
548 (setq file
(pop files
))))))))
550 (defun git-set-filenames-state (status files state
)
551 "Set the state of a list of named files. The list must be sorted"
553 (git-status-filenames-map status
#'git-set-fileinfo-state files state
)
554 (unless state
;; delete files whose state has been set to nil
555 (ewoc-filter status
(lambda (info) (git-fileinfo->state info
))))))
557 (defun git-state-code (code)
558 "Convert from a string to a added/deleted/modified state."
559 (case (string-to-char code
)
568 (defun git-status-code-as-string (code)
569 "Format a git status code as string."
571 ('modified
(propertize "Modified" 'face
'git-status-face
))
572 ('unknown
(propertize "Unknown " 'face
'git-unknown-face
))
573 ('added
(propertize "Added " 'face
'git-status-face
))
574 ('deleted
(propertize "Deleted " 'face
'git-status-face
))
575 ('unmerged
(propertize "Unmerged" 'face
'git-unmerged-face
))
576 ('uptodate
(propertize "Uptodate" 'face
'git-uptodate-face
))
577 ('ignored
(propertize "Ignored " 'face
'git-ignored-face
))
580 (defun git-file-type-as-string (old-perm new-perm
)
581 "Return a string describing the file type based on its permissions."
582 (let* ((old-type (lsh (or old-perm
0) -
9))
583 (new-type (lsh (or new-perm
0) -
9))
588 (80 " (type change symlink -> file)")
589 (112 " (type change subproject -> file)")))
592 (64 " (type change file -> symlink)")
593 (112 " (type change subproject -> symlink)")
597 (64 " (type change file -> subproject)")
598 (80 " (type change symlink -> subproject)")
599 (t " (subproject)")))
600 (72 nil
) ;; directory (internal, not a real git state)
601 (0 ;; deleted or unknown
604 (112 " (subproject)")))
605 (t (format " (unknown type %o)" new-type
)))))
606 (cond (str (propertize str
'face
'git-status-face
))
607 ((eq new-type
72) "/")
610 (defun git-rename-as-string (info)
611 "Return a string describing the copy or rename associated with INFO, or an empty string if none."
612 (let ((state (git-fileinfo->rename-state info
)))
616 (if (eq state
'copy
) "copied from "
617 (if (eq (git-fileinfo->state info
) 'added
) "renamed from "
619 (git-escape-file-name (git-fileinfo->orig-name info
))
620 ")") 'face
'git-status-face
)
623 (defun git-permissions-as-string (old-perm new-perm
)
624 "Format a permission change as string."
626 (if (or (not old-perm
)
628 (eq 0 (logand ?
\111 (logxor old-perm new-perm
))))
630 (if (eq 0 (logand ?
\111 old-perm
)) "+x" "-x"))
631 'face
'git-permission-face
))
633 (defun git-fileinfo-prettyprint (info)
634 "Pretty-printer for the git-fileinfo structure."
635 (let ((old-perm (git-fileinfo->old-perm info
))
636 (new-perm (git-fileinfo->new-perm info
)))
637 (insert (concat " " (if (git-fileinfo->marked info
) (propertize "*" 'face
'git-mark-face
) " ")
638 " " (git-status-code-as-string (git-fileinfo->state info
))
639 " " (git-permissions-as-string old-perm new-perm
)
640 " " (git-escape-file-name (git-fileinfo->name info
))
641 (git-file-type-as-string old-perm new-perm
)
642 (git-rename-as-string info
)))))
644 (defun git-update-node-fileinfo (node info
)
645 "Update the fileinfo of the specified node. The names are assumed to match already."
646 (let ((data (ewoc-data node
)))
648 ;; preserve the marked flag
649 (git-fileinfo->marked info
) (git-fileinfo->marked data
)
650 (git-fileinfo->needs-update data
) nil
)
651 (when (not (equal info data
))
652 (setf (git-fileinfo->needs-refresh info
) t
653 (ewoc-data node
) info
))))
655 (defun git-insert-info-list (status infolist files
)
656 "Insert a sorted list of file infos in the status buffer, replacing existing ones if any."
657 (let* ((info (pop infolist
))
658 (node (ewoc-nth status
0))
659 (name (and info
(git-fileinfo->name info
)))
662 (let ((nodename (and node
(git-fileinfo->name
(ewoc-data node
)))))
663 (while (and files
(string-lessp (car files
) name
))
664 (push (pop files
) remaining
))
665 (when (and files
(string-equal (car files
) name
))
666 (setq files
(cdr files
)))
667 (cond ((not nodename
)
668 (setq node
(ewoc-enter-last status info
))
669 (setq info
(pop infolist
))
670 (setq name
(and info
(git-fileinfo->name info
))))
671 ((string-lessp nodename name
)
672 (setq node
(ewoc-next status node
)))
673 ((string-equal nodename name
)
674 ;; preserve the marked flag
675 (git-update-node-fileinfo node info
)
676 (setq info
(pop infolist
))
677 (setq name
(and info
(git-fileinfo->name info
))))
679 (setq node
(ewoc-enter-before status node info
))
680 (setq info
(pop infolist
))
681 (setq name
(and info
(git-fileinfo->name info
)))))))
682 (nconc (nreverse remaining
) files
)))
684 (defun git-run-diff-index (status files
)
685 "Run git-diff-index on FILES and parse the results into STATUS.
686 Return the list of files that haven't been handled."
689 (apply #'git-call-process t
"diff-index" "-z" "-M" "HEAD" "--" files
)
690 (goto-char (point-min))
691 (while (re-search-forward
692 ":\\([0-7]\\{6\\}\\) \\([0-7]\\{6\\}\\) [0-9a-f]\\{40\\} [0-9a-f]\\{40\\} \\(\\([ADMUT]\\)\0\\([^\0]+\\)\\|\\([CR]\\)[0-9]*\0\\([^\0]+\\)\0\\([^\0]+\\)\\)\0"
694 (let ((old-perm (string-to-number (match-string 1) 8))
695 (new-perm (string-to-number (match-string 2) 8))
696 (state (or (match-string 4) (match-string 6)))
697 (name (or (match-string 5) (match-string 7)))
698 (new-name (match-string 8)))
699 (if new-name
; copy or rename
700 (if (eq ?C
(string-to-char state
))
701 (push (git-create-fileinfo 'added new-name old-perm new-perm
'copy name
) infolist
)
702 (push (git-create-fileinfo 'deleted name
0 0 'rename new-name
) infolist
)
703 (push (git-create-fileinfo 'added new-name old-perm new-perm
'rename name
) infolist
))
704 (push (git-create-fileinfo (git-state-code state
) name old-perm new-perm
) infolist
)))))
705 (setq infolist
(sort (nreverse infolist
)
706 (lambda (info1 info2
)
707 (string-lessp (git-fileinfo->name info1
)
708 (git-fileinfo->name info2
)))))
709 (git-insert-info-list status infolist files
)))
711 (defun git-find-status-file (status file
)
712 "Find a given file in the status ewoc and return its node."
713 (let ((node (ewoc-nth status
0)))
714 (while (and node
(not (string= file
(git-fileinfo->name
(ewoc-data node
)))))
715 (setq node
(ewoc-next status node
)))
718 (defun git-run-ls-files (status files default-state
&rest options
)
719 "Run git-ls-files on FILES and parse the results into STATUS.
720 Return the list of files that haven't been handled."
723 (apply #'git-call-process t
"ls-files" "-z" (append options
(list "--") files
))
724 (goto-char (point-min))
725 (while (re-search-forward "\\([^\0]*?\\)\\(/?\\)\0" nil t
1)
726 (let ((name (match-string 1)))
727 (push (git-create-fileinfo default-state name
0
728 (if (string-equal "/" (match-string 2)) (lsh ?
\110 9) 0))
730 (setq infolist
(nreverse infolist
)) ;; assume it is sorted already
731 (git-insert-info-list status infolist files
)))
733 (defun git-run-ls-files-cached (status files default-state
)
734 "Run git-ls-files -c on FILES and parse the results into STATUS.
735 Return the list of files that haven't been handled."
738 (apply #'git-call-process t
"ls-files" "-z" "-s" "-c" "--" files
)
739 (goto-char (point-min))
740 (while (re-search-forward "\\([0-7]\\{6\\}\\) [0-9a-f]\\{40\\} 0\t\\([^\0]+\\)\0" nil t
)
741 (let* ((new-perm (string-to-number (match-string 1) 8))
742 (old-perm (if (eq default-state
'added
) 0 new-perm
))
743 (name (match-string 2)))
744 (push (git-create-fileinfo default-state name old-perm new-perm
) infolist
))))
745 (setq infolist
(nreverse infolist
)) ;; assume it is sorted already
746 (git-insert-info-list status infolist files
)))
748 (defun git-run-ls-unmerged (status files
)
749 "Run git-ls-files -u on FILES and parse the results into STATUS."
751 (apply #'git-call-process t
"ls-files" "-z" "-u" "--" files
)
752 (goto-char (point-min))
753 (let (unmerged-files)
754 (while (re-search-forward "[0-7]\\{6\\} [0-9a-f]\\{40\\} [123]\t\\([^\0]+\\)\0" nil t
)
755 (push (match-string 1) unmerged-files
))
756 (setq unmerged-files
(nreverse unmerged-files
)) ;; assume it is sorted already
757 (git-set-filenames-state status unmerged-files
'unmerged
))))
759 (defun git-get-exclude-files ()
760 "Get the list of exclude files to pass to git-ls-files."
762 (config (git-config "core.excludesfile")))
763 (when (file-readable-p ".git/info/exclude")
764 (push ".git/info/exclude" files
))
765 (when (and config
(file-readable-p config
))
769 (defun git-run-ls-files-with-excludes (status files default-state
&rest options
)
770 "Run git-ls-files on FILES with appropriate --exclude-from options."
771 (let ((exclude-files (git-get-exclude-files)))
772 (apply #'git-run-ls-files status files default-state
"--directory" "--no-empty-directory"
773 (concat "--exclude-per-directory=" git-per-dir-ignore-file
)
774 (append options
(mapcar (lambda (f) (concat "--exclude-from=" f
)) exclude-files
)))))
776 (defun git-update-status-files (&optional files mark-files
)
777 "Update the status of FILES from the index.
778 The FILES list must be sorted."
779 (unless git-status
(error "Not in git-status buffer."))
780 ;; set the needs-update flag on existing files
782 (git-status-filenames-map
783 git-status
(lambda (info) (setf (git-fileinfo->needs-update info
) t
)) files
)
784 (ewoc-map (lambda (info) (setf (git-fileinfo->needs-update info
) t
) nil
) git-status
)
785 (git-call-process nil
"update-index" "--refresh")
786 (when git-show-uptodate
787 (git-run-ls-files-cached git-status nil
'uptodate
)))
788 (let ((remaining-files
789 (if (git-empty-db-p) ; we need some special handling for an empty db
790 (git-run-ls-files-cached git-status files
'added
)
791 (git-run-diff-index git-status files
))))
792 (git-run-ls-unmerged git-status files
)
793 (when (or remaining-files
(and git-show-unknown
(not files
)))
794 (setq remaining-files
(git-run-ls-files-with-excludes git-status remaining-files
'unknown
"-o")))
795 (when (or remaining-files
(and git-show-ignored
(not files
)))
796 (setq remaining-files
(git-run-ls-files-with-excludes git-status remaining-files
'ignored
"-o" "-i")))
798 (setq remaining-files
(git-get-filenames (ewoc-collect git-status
#'git-fileinfo-
>needs-update
))))
799 (when remaining-files
800 (setq remaining-files
(git-run-ls-files-cached git-status remaining-files
'uptodate
)))
801 (git-set-filenames-state git-status remaining-files nil
)
802 (when mark-files
(git-mark-files git-status files
))
804 (git-refresh-ewoc-hf git-status
)))
806 (defun git-mark-files (status files
)
807 "Mark all the specified FILES, and unmark the others."
808 (let ((file (and files
(pop files
)))
809 (node (ewoc-nth status
0)))
811 (let ((info (ewoc-data node
)))
812 (if (and file
(string-equal (git-fileinfo->name info
) file
))
814 (unless (git-fileinfo->marked info
)
815 (setf (git-fileinfo->marked info
) t
)
816 (setf (git-fileinfo->needs-refresh info
) t
))
817 (setq file
(pop files
))
818 (setq node
(ewoc-next status node
)))
819 (when (git-fileinfo->marked info
)
820 (setf (git-fileinfo->marked info
) nil
)
821 (setf (git-fileinfo->needs-refresh info
) t
))
822 (if (and file
(string-lessp file
(git-fileinfo->name info
)))
823 (setq file
(pop files
))
824 (setq node
(ewoc-next status node
))))))))
826 (defun git-marked-files ()
827 "Return a list of all marked files, or if none a list containing just the file at cursor position."
828 (unless git-status
(error "Not in git-status buffer."))
829 (or (ewoc-collect git-status
(lambda (info) (git-fileinfo->marked info
)))
830 (list (ewoc-data (ewoc-locate git-status
)))))
832 (defun git-marked-files-state (&rest states
)
833 "Return a sorted list of marked files that are in the specified states."
834 (let ((files (git-marked-files))
837 (when (memq (git-fileinfo->state info
) states
)
841 (defun git-refresh-files ()
842 "Refresh all files that need it and clear the needs-refresh flag."
843 (unless git-status
(error "Not in git-status buffer."))
846 (let ((refresh (git-fileinfo->needs-refresh info
)))
847 (setf (git-fileinfo->needs-refresh info
) nil
)
850 ; move back to goal column
851 (when goal-column
(move-to-column goal-column
)))
853 (defun git-refresh-ewoc-hf (status)
854 "Refresh the ewoc header and footer."
855 (let ((branch (git-symbolic-ref "HEAD"))
856 (head (if (git-empty-db-p) "Nothing committed yet"
857 (git-get-commit-description "HEAD")))
858 (merge-heads (git-get-merge-heads)))
860 (format "Directory: %s\nBranch: %s\nHead: %s%s\n"
863 (if (string-match "^refs/heads/" branch
)
864 (substring branch
(match-end 0))
866 "none (detached HEAD)")
869 (concat "\nMerging: "
870 (mapconcat (lambda (str) (git-get-commit-description str
)) merge-heads
"\n "))
872 (if (ewoc-nth status
0) "" " No changes."))))
874 (defun git-get-filenames (files)
875 (mapcar (lambda (info) (git-fileinfo->name info
)) files
))
877 (defun git-update-index (index-file files
)
878 "Run git-update-index on a list of files."
879 (let ((process-environment (append (and index-file
(list (concat "GIT_INDEX_FILE=" index-file
)))
880 process-environment
))
881 added deleted modified
)
883 (case (git-fileinfo->state info
)
884 ('added
(push info added
))
885 ('deleted
(push info deleted
))
886 ('modified
(push info modified
))))
888 (or (not added
) (apply #'git-call-process-display-error
"update-index" "--add" "--" (git-get-filenames added
)))
889 (or (not deleted
) (apply #'git-call-process-display-error
"update-index" "--remove" "--" (git-get-filenames deleted
)))
890 (or (not modified
) (apply #'git-call-process-display-error
"update-index" "--" (git-get-filenames modified
))))))
892 (defun git-run-pre-commit-hook ()
893 "Run the pre-commit hook if any."
894 (unless git-status
(error "Not in git-status buffer."))
895 (let ((files (git-marked-files-state 'added
'deleted
'modified
)))
897 (not (file-executable-p ".git/hooks/pre-commit"))
898 (let ((index-file (make-temp-file "gitidx")))
900 (let ((head-tree (unless (git-empty-db-p) (git-rev-parse "HEAD^{tree}"))))
901 (git-read-tree head-tree index-file
)
902 (git-update-index index-file files
)
903 (git-run-hook "pre-commit" `(("GIT_INDEX_FILE" .
,index-file
))))
904 (delete-file index-file
))))))
906 (defun git-do-commit ()
907 "Perform the actual commit using the current buffer as log message."
909 (let ((buffer (current-buffer))
910 (index-file (make-temp-file "gitidx")))
911 (with-current-buffer log-edit-parent-buffer
912 (if (git-marked-files-state 'unmerged
)
913 (message "You cannot commit unmerged files, resolve them first.")
915 (let ((files (git-marked-files-state 'added
'deleted
'modified
))
917 (unless (git-empty-db-p)
918 (setq head
(git-rev-parse "HEAD")
919 head-tree
(git-rev-parse "HEAD^{tree}")))
920 (message "Running git commit...")
923 (git-read-tree head-tree index-file
)
924 (git-update-index nil files
) ;update both the default index
925 (git-update-index index-file files
) ;and the temporary one
926 (setq tree
(git-write-tree index-file
)))
927 (if (or (not (string-equal tree head-tree
))
928 (yes-or-no-p "The tree was not modified, do you really want to perform an empty commit? "))
929 (let ((commit (git-commit-tree buffer tree head
)))
931 (condition-case nil
(delete-file ".git/MERGE_HEAD") (error nil
))
932 (condition-case nil
(delete-file ".git/MERGE_MSG") (error nil
))
933 (with-current-buffer buffer
(erase-buffer))
934 (git-update-status-files (git-get-filenames files
))
935 (git-call-process nil
"rerere")
936 (git-call-process nil
"gc" "--auto")
937 (message "Committed %s." commit
)
938 (git-run-hook "post-commit" nil
)))
939 (message "Commit aborted."))))
940 (delete-file index-file
))))))
943 ;;;; Interactive functions
944 ;;;; ------------------------------------------------------------
946 (defun git-mark-file ()
947 "Mark the file that the cursor is on and move to the next one."
949 (unless git-status
(error "Not in git-status buffer."))
950 (let* ((pos (ewoc-locate git-status
))
951 (info (ewoc-data pos
)))
952 (setf (git-fileinfo->marked info
) t
)
953 (ewoc-invalidate git-status pos
)
954 (ewoc-goto-next git-status
1)))
956 (defun git-unmark-file ()
957 "Unmark the file that the cursor is on and move to the next one."
959 (unless git-status
(error "Not in git-status buffer."))
960 (let* ((pos (ewoc-locate git-status
))
961 (info (ewoc-data pos
)))
962 (setf (git-fileinfo->marked info
) nil
)
963 (ewoc-invalidate git-status pos
)
964 (ewoc-goto-next git-status
1)))
966 (defun git-unmark-file-up ()
967 "Unmark the file that the cursor is on and move to the previous one."
969 (unless git-status
(error "Not in git-status buffer."))
970 (let* ((pos (ewoc-locate git-status
))
971 (info (ewoc-data pos
)))
972 (setf (git-fileinfo->marked info
) nil
)
973 (ewoc-invalidate git-status pos
)
974 (ewoc-goto-prev git-status
1)))
976 (defun git-mark-all ()
979 (unless git-status
(error "Not in git-status buffer."))
980 (ewoc-map (lambda (info) (unless (git-fileinfo->marked info
)
981 (setf (git-fileinfo->marked info
) t
))) git-status
)
982 ; move back to goal column after invalidate
983 (when goal-column
(move-to-column goal-column
)))
985 (defun git-unmark-all ()
988 (unless git-status
(error "Not in git-status buffer."))
989 (ewoc-map (lambda (info) (when (git-fileinfo->marked info
)
990 (setf (git-fileinfo->marked info
) nil
)
992 ; move back to goal column after invalidate
993 (when goal-column
(move-to-column goal-column
)))
995 (defun git-toggle-all-marks ()
996 "Toggle all file marks."
998 (unless git-status
(error "Not in git-status buffer."))
999 (ewoc-map (lambda (info) (setf (git-fileinfo->marked info
) (not (git-fileinfo->marked info
))) t
) git-status
)
1000 ; move back to goal column after invalidate
1001 (when goal-column
(move-to-column goal-column
)))
1003 (defun git-next-file (&optional n
)
1004 "Move the selection down N files."
1006 (unless git-status
(error "Not in git-status buffer."))
1007 (ewoc-goto-next git-status n
))
1009 (defun git-prev-file (&optional n
)
1010 "Move the selection up N files."
1012 (unless git-status
(error "Not in git-status buffer."))
1013 (ewoc-goto-prev git-status n
))
1015 (defun git-next-unmerged-file (&optional n
)
1016 "Move the selection down N unmerged files."
1018 (unless git-status
(error "Not in git-status buffer."))
1019 (let* ((last (ewoc-locate git-status
))
1020 (node (ewoc-next git-status last
)))
1021 (while (and node
(> n
0))
1022 (when (eq 'unmerged
(git-fileinfo->state
(ewoc-data node
)))
1025 (setq node
(ewoc-next git-status node
)))
1026 (ewoc-goto-node git-status last
)))
1028 (defun git-prev-unmerged-file (&optional n
)
1029 "Move the selection up N unmerged files."
1031 (unless git-status
(error "Not in git-status buffer."))
1032 (let* ((last (ewoc-locate git-status
))
1033 (node (ewoc-prev git-status last
)))
1034 (while (and node
(> n
0))
1035 (when (eq 'unmerged
(git-fileinfo->state
(ewoc-data node
)))
1038 (setq node
(ewoc-prev git-status node
)))
1039 (ewoc-goto-node git-status last
)))
1041 (defun git-insert-file (file)
1042 "Insert file(s) into the git-status buffer."
1043 (interactive "fInsert file: ")
1044 (git-update-status-files (list (file-relative-name file
))))
1046 (defun git-add-file ()
1047 "Add marked file(s) to the index cache."
1049 (let ((files (git-get-filenames (git-marked-files-state 'unknown
'ignored
'unmerged
))))
1050 ;; FIXME: add support for directories
1052 (push (file-relative-name (read-file-name "File to add: " nil nil t
)) files
))
1053 (when (apply 'git-call-process-display-error
"update-index" "--add" "--" files
)
1054 (git-update-status-files files
)
1055 (git-success-message "Added" files
))))
1057 (defun git-ignore-file ()
1058 "Add marked file(s) to the ignore list."
1060 (let ((files (git-get-filenames (git-marked-files-state 'unknown
))))
1062 (push (file-relative-name (read-file-name "File to ignore: " nil nil t
)) files
))
1063 (dolist (f files
) (git-append-to-ignore f
))
1064 (git-update-status-files files
)
1065 (git-success-message "Ignored" files
)))
1067 (defun git-remove-file ()
1068 "Remove the marked file(s)."
1070 (let ((files (git-get-filenames (git-marked-files-state 'added
'modified
'unknown
'uptodate
'ignored
))))
1072 (push (file-relative-name (read-file-name "File to remove: " nil nil t
)) files
))
1075 (format "Remove %d files? " (length files
))
1076 (format "Remove %s? " (car files
))))
1078 (dolist (name files
)
1080 (if (file-directory-p name
)
1081 (delete-directory name
)
1082 (delete-file name
))))
1083 (when (apply 'git-call-process-display-error
"update-index" "--remove" "--" files
)
1084 (git-update-status-files files
)
1085 (git-success-message "Removed" files
)))
1086 (message "Aborting"))))
1088 (defun git-revert-file ()
1089 "Revert changes to the marked file(s)."
1091 (let ((files (git-marked-files-state 'added
'deleted
'modified
'unmerged
))
1096 (format "Revert %d files? " (length files
))
1097 (format "Revert %s? " (git-fileinfo->name
(car files
))))))
1098 (dolist (info files
)
1099 (case (git-fileinfo->state info
)
1100 ('added
(push (git-fileinfo->name info
) added
))
1101 ('deleted
(push (git-fileinfo->name info
) modified
))
1102 ('unmerged
(push (git-fileinfo->name info
) modified
))
1103 ('modified
(push (git-fileinfo->name info
) modified
))))
1104 ;; check if a buffer contains one of the files and isn't saved
1105 (dolist (file modified
)
1106 (let ((buffer (get-file-buffer file
)))
1107 (when (and buffer
(buffer-modified-p buffer
))
1108 (error "Buffer %s is modified. Please kill or save modified buffers before reverting." (buffer-name buffer
)))))
1111 (apply 'git-call-process-display-error
"update-index" "--force-remove" "--" added
))
1113 (apply 'git-call-process-display-error
"checkout" "HEAD" modified
))))
1114 (names (git-get-filenames files
)))
1115 (git-update-status-files names
)
1117 (dolist (file modified
)
1118 (let ((buffer (get-file-buffer file
)))
1119 (when buffer
(with-current-buffer buffer
(revert-buffer t t t
)))))
1120 (git-success-message "Reverted" names
))))))
1122 (defun git-remove-handled ()
1123 "Remove handled files from the status list."
1125 (ewoc-filter git-status
1127 (case (git-fileinfo->state info
)
1128 ('ignored git-show-ignored
)
1129 ('uptodate git-show-uptodate
)
1130 ('unknown git-show-unknown
)
1132 (unless (ewoc-nth git-status
0) ; refresh header if list is empty
1133 (git-refresh-ewoc-hf git-status
)))
1135 (defun git-toggle-show-uptodate ()
1136 "Toogle the option for showing up-to-date files."
1138 (if (setq git-show-uptodate
(not git-show-uptodate
))
1139 (git-refresh-status)
1140 (git-remove-handled)))
1142 (defun git-toggle-show-ignored ()
1143 "Toogle the option for showing ignored files."
1145 (if (setq git-show-ignored
(not git-show-ignored
))
1147 (message "Inserting ignored files...")
1148 (git-run-ls-files-with-excludes git-status nil
'ignored
"-o" "-i")
1150 (git-refresh-ewoc-hf git-status
)
1151 (message "Inserting ignored files...done"))
1152 (git-remove-handled)))
1154 (defun git-toggle-show-unknown ()
1155 "Toogle the option for showing unknown files."
1157 (if (setq git-show-unknown
(not git-show-unknown
))
1159 (message "Inserting unknown files...")
1160 (git-run-ls-files-with-excludes git-status nil
'unknown
"-o")
1162 (git-refresh-ewoc-hf git-status
)
1163 (message "Inserting unknown files...done"))
1164 (git-remove-handled)))
1166 (defun git-expand-directory (info)
1167 "Expand the directory represented by INFO to list its files."
1168 (when (eq (lsh (git-fileinfo->new-perm info
) -
9) ?
\110)
1169 (let ((dir (git-fileinfo->name info
)))
1170 (git-set-filenames-state git-status
(list dir
) nil
)
1171 (git-run-ls-files-with-excludes git-status
(list (concat dir
"/")) 'unknown
"-o")
1173 (git-refresh-ewoc-hf git-status
)
1176 (defun git-setup-diff-buffer (buffer)
1177 "Setup a buffer for displaying a diff."
1178 (let ((dir default-directory
))
1179 (with-current-buffer buffer
1181 (goto-char (point-min))
1182 (setq default-directory dir
)
1183 (setq buffer-read-only t
)))
1184 (display-buffer buffer
)
1185 ; shrink window only if it displays the status buffer
1186 (when (eq (window-buffer) (current-buffer))
1187 (shrink-window-if-larger-than-buffer)))
1189 (defun git-diff-file ()
1190 "Diff the marked file(s) against HEAD."
1192 (let ((files (git-marked-files)))
1193 (git-setup-diff-buffer
1194 (apply #'git-run-command-buffer
"*git-diff*" "diff-index" "-p" "-M" "HEAD" "--" (git-get-filenames files
)))))
1196 (defun git-diff-file-merge-head (arg)
1197 "Diff the marked file(s) against the first merge head (or the nth one with a numeric prefix)."
1199 (let ((files (git-marked-files))
1200 (merge-heads (git-get-merge-heads)))
1201 (unless merge-heads
(error "No merge in progress"))
1202 (git-setup-diff-buffer
1203 (apply #'git-run-command-buffer
"*git-diff*" "diff-index" "-p" "-M"
1204 (or (nth (1- arg
) merge-heads
) "HEAD") "--" (git-get-filenames files
)))))
1206 (defun git-diff-unmerged-file (stage)
1207 "Diff the marked unmerged file(s) against the specified stage."
1208 (let ((files (git-marked-files)))
1209 (git-setup-diff-buffer
1210 (apply #'git-run-command-buffer
"*git-diff*" "diff-files" "-p" stage
"--" (git-get-filenames files
)))))
1212 (defun git-diff-file-base ()
1213 "Diff the marked unmerged file(s) against the common base file."
1215 (git-diff-unmerged-file "-1"))
1217 (defun git-diff-file-mine ()
1218 "Diff the marked unmerged file(s) against my pre-merge version."
1220 (git-diff-unmerged-file "-2"))
1222 (defun git-diff-file-other ()
1223 "Diff the marked unmerged file(s) against the other's pre-merge version."
1225 (git-diff-unmerged-file "-3"))
1227 (defun git-diff-file-combined ()
1228 "Do a combined diff of the marked unmerged file(s)."
1230 (git-diff-unmerged-file "-c"))
1232 (defun git-diff-file-idiff ()
1233 "Perform an interactive diff on the current file."
1235 (let ((files (git-marked-files-state 'added
'deleted
'modified
)))
1236 (unless (eq 1 (length files
))
1237 (error "Cannot perform an interactive diff on multiple files."))
1238 (let* ((filename (car (git-get-filenames files
)))
1239 (buff1 (find-file-noselect filename
))
1240 (buff2 (git-run-command-buffer (concat filename
".~HEAD~") "cat-file" "blob" (concat "HEAD:" filename
))))
1241 (ediff-buffers buff1 buff2
))))
1243 (defun git-log-file ()
1244 "Display a log of changes to the marked file(s)."
1246 (let* ((files (git-marked-files))
1247 (coding-system-for-read git-commits-coding-system
)
1248 (buffer (apply #'git-run-command-buffer
"*git-log*" "rev-list" "--pretty" "HEAD" "--" (git-get-filenames files
))))
1249 (with-current-buffer buffer
1250 ; (git-log-mode) FIXME: implement log mode
1251 (goto-char (point-min))
1252 (setq buffer-read-only t
))
1253 (display-buffer buffer
)))
1255 (defun git-log-edit-files ()
1256 "Return a list of marked files for use in the log-edit buffer."
1257 (with-current-buffer log-edit-parent-buffer
1258 (git-get-filenames (git-marked-files-state 'added
'deleted
'modified
))))
1260 (defun git-log-edit-diff ()
1261 "Run a diff of the current files being committed from a log-edit buffer."
1262 (with-current-buffer log-edit-parent-buffer
1265 (defun git-append-sign-off (name email
)
1266 "Append a Signed-off-by entry to the current buffer, avoiding duplicates."
1267 (let ((sign-off (format "Signed-off-by: %s <%s>" name email
))
1268 (case-fold-search t
))
1269 (goto-char (point-min))
1270 (unless (re-search-forward (concat "^" (regexp-quote sign-off
)) nil t
)
1271 (goto-char (point-min))
1272 (unless (re-search-forward "^Signed-off-by: " nil t
)
1273 (setq sign-off
(concat "\n" sign-off
)))
1274 (goto-char (point-max))
1275 (insert sign-off
"\n"))))
1277 (defun git-setup-log-buffer (buffer &optional merge-heads author-name author-email subject date msg
)
1278 "Setup the log buffer for a commit."
1279 (unless git-status
(error "Not in git-status buffer."))
1280 (let ((dir default-directory
)
1281 (committer-name (git-get-committer-name))
1282 (committer-email (git-get-committer-email))
1283 (sign-off git-append-signed-off-by
))
1284 (with-current-buffer buffer
1289 (format "Author: %s <%s>\n%s%s"
1290 (or author-name committer-name
)
1291 (or author-email committer-email
)
1292 (if date
(format "Date: %s\n" date
) "")
1294 (format "Merge: %s\n"
1295 (mapconcat 'identity merge-heads
" "))
1297 'face
'git-header-face
)
1298 (propertize git-log-msg-separator
'face
'git-separator-face
)
1300 (when subject
(insert subject
"\n\n"))
1301 (cond (msg (insert msg
"\n"))
1302 ((file-readable-p ".git/rebase-apply/msg")
1303 (insert-file-contents ".git/rebase-apply/msg"))
1304 ((file-readable-p ".git/MERGE_MSG")
1305 (insert-file-contents ".git/MERGE_MSG")))
1306 ; delete empty lines at end
1307 (goto-char (point-min))
1308 (when (re-search-forward "\n+\\'" nil t
)
1309 (replace-match "\n" t t
))
1310 (when sign-off
(git-append-sign-off committer-name committer-email
)))
1313 (defun git-commit-file ()
1314 "Commit the marked file(s), asking for a commit message."
1316 (unless git-status
(error "Not in git-status buffer."))
1317 (when (git-run-pre-commit-hook)
1318 (let ((buffer (get-buffer-create "*git-commit*"))
1319 (coding-system (git-get-commits-coding-system))
1320 author-name author-email subject date
)
1321 (when (eq 0 (buffer-size buffer
))
1322 (when (file-readable-p ".git/rebase-apply/info")
1324 (insert-file-contents ".git/rebase-apply/info")
1325 (goto-char (point-min))
1326 (when (re-search-forward "^Author: \\(.*\\)\nEmail: \\(.*\\)$" nil t
)
1327 (setq author-name
(match-string 1))
1328 (setq author-email
(match-string 2)))
1329 (goto-char (point-min))
1330 (when (re-search-forward "^Subject: \\(.*\\)$" nil t
)
1331 (setq subject
(match-string 1)))
1332 (goto-char (point-min))
1333 (when (re-search-forward "^Date: \\(.*\\)$" nil t
)
1334 (setq date
(match-string 1)))))
1335 (git-setup-log-buffer buffer
(git-get-merge-heads) author-name author-email subject date
))
1336 (if (boundp 'log-edit-diff-function
)
1337 (log-edit 'git-do-commit nil
'((log-edit-listfun . git-log-edit-files
)
1338 (log-edit-diff-function . git-log-edit-diff
)) buffer
)
1339 (log-edit 'git-do-commit nil
'git-log-edit-files buffer
))
1340 (setq font-lock-keywords
(font-lock-compile-keywords git-log-edit-font-lock-keywords
))
1341 (setq paragraph-separate
(concat (regexp-quote git-log-msg-separator
) "$\\|Author: \\|Date: \\|Merge: \\|Signed-off-by: \\|\f\\|[ ]*$"))
1342 (setq buffer-file-coding-system coding-system
)
1343 (re-search-forward (regexp-quote (concat git-log-msg-separator
"\n")) nil t
))))
1345 (defun git-setup-commit-buffer (commit)
1346 "Setup the commit buffer with the contents of COMMIT."
1347 (let (parents author-name author-email subject date msg
)
1349 (let ((coding-system (git-get-logoutput-coding-system)))
1350 (git-call-process t
"log" "-1" "--pretty=medium" "--abbrev=40" commit
)
1351 (goto-char (point-min))
1352 (when (re-search-forward "^Merge: *\\(.*\\)$" nil t
)
1353 (setq parents
(cdr (split-string (match-string 1) " +"))))
1354 (when (re-search-forward "^Author: *\\(.*\\) <\\(.*\\)>$" nil t
)
1355 (setq author-name
(match-string 1))
1356 (setq author-email
(match-string 2)))
1357 (when (re-search-forward "^Date: *\\(.*\\)$" nil t
)
1358 (setq date
(match-string 1)))
1359 (while (re-search-forward "^ \\(.*\\)$" nil t
)
1360 (push (match-string 1) msg
))
1361 (setq msg
(nreverse msg
))
1362 (setq subject
(pop msg
))
1363 (while (and msg
(zerop (length (car msg
))) (pop msg
)))))
1364 (git-setup-log-buffer (get-buffer-create "*git-commit*")
1365 parents author-name author-email subject date
1366 (mapconcat #'identity msg
"\n"))))
1368 (defun git-get-commit-files (commit)
1369 "Retrieve a sorted list of files modified by COMMIT."
1372 (git-call-process t
"diff-tree" "-m" "-r" "-z" "--name-only" "--no-commit-id" "--root" commit
)
1373 (goto-char (point-min))
1374 (while (re-search-forward "\\([^\0]*\\)\0" nil t
1)
1375 (push (match-string 1) files
)))
1376 (sort files
#'string-lessp
)))
1378 (defun git-read-commit-name (prompt &optional default
)
1379 "Ask for a commit name, with completion for local branch, remote branch and tag."
1380 (completing-read prompt
1381 (list* "HEAD" "ORIG_HEAD" "FETCH_HEAD" (mapcar #'car
(git-for-each-ref)))
1382 nil nil nil nil default
))
1384 (defun git-checkout (branch &optional merge
)
1385 "Checkout a branch, tag, or any commit.
1386 Use a prefix arg if git should merge while checking out."
1388 (list (git-read-commit-name "Checkout: ")
1389 current-prefix-arg
))
1390 (unless git-status
(error "Not in git-status buffer."))
1391 (let ((args (list branch
"--")))
1392 (when merge
(push "-m" args
))
1393 (when (apply #'git-call-process-display-error
"checkout" args
)
1394 (git-update-status-files))))
1396 (defun git-branch (branch)
1397 "Create a branch from the current HEAD and switch to it."
1398 (interactive (list (git-read-commit-name "Branch: ")))
1399 (unless git-status
(error "Not in git-status buffer."))
1400 (if (git-rev-parse (concat "refs/heads/" branch
))
1401 (if (yes-or-no-p (format "Branch %s already exists, replace it? " branch
))
1402 (and (git-call-process-display-error "branch" "-f" branch
)
1403 (git-call-process-display-error "checkout" branch
))
1404 (message "Canceled."))
1405 (git-call-process-display-error "checkout" "-b" branch
))
1406 (git-refresh-ewoc-hf git-status
))
1408 (defun git-amend-commit ()
1409 "Undo the last commit on HEAD, and set things up to commit an
1410 amended version of it."
1412 (unless git-status
(error "Not in git-status buffer."))
1413 (when (git-empty-db-p) (error "No commit to amend."))
1414 (let* ((commit (git-rev-parse "HEAD"))
1415 (files (git-get-commit-files commit
)))
1416 (when (if (git-rev-parse "HEAD^")
1417 (git-call-process-display-error "reset" "--soft" "HEAD^")
1418 (and (git-update-ref "ORIG_HEAD" commit
)
1419 (git-update-ref "HEAD" nil commit
)))
1420 (git-update-status-files files t
)
1421 (git-setup-commit-buffer commit
)
1422 (git-commit-file))))
1424 (defun git-cherry-pick-commit (arg)
1425 "Cherry-pick a commit."
1426 (interactive (list (git-read-commit-name "Cherry-pick commit: ")))
1427 (unless git-status
(error "Not in git-status buffer."))
1428 (let ((commit (git-rev-parse (concat arg
"^0"))))
1429 (unless commit
(error "Not a valid commit '%s'." arg
))
1430 (when (git-rev-parse (concat commit
"^2"))
1431 (error "Cannot cherry-pick a merge commit."))
1432 (let ((files (git-get-commit-files commit
))
1433 (ok (git-call-process-display-error "cherry-pick" "-n" commit
)))
1434 (git-update-status-files files ok
)
1435 (with-current-buffer (git-setup-commit-buffer commit
)
1436 (goto-char (point-min))
1437 (if (re-search-forward "^\n*Signed-off-by:" nil t
1)
1438 (goto-char (match-beginning 0))
1439 (goto-char (point-max)))
1440 (insert "(cherry picked from commit " commit
")\n"))
1441 (when ok
(git-commit-file)))))
1443 (defun git-revert-commit (arg)
1445 (interactive (list (git-read-commit-name "Revert commit: ")))
1446 (unless git-status
(error "Not in git-status buffer."))
1447 (let ((commit (git-rev-parse (concat arg
"^0"))))
1448 (unless commit
(error "Not a valid commit '%s'." arg
))
1449 (when (git-rev-parse (concat commit
"^2"))
1450 (error "Cannot revert a merge commit."))
1451 (let ((files (git-get-commit-files commit
))
1452 (subject (git-get-commit-description commit
))
1453 (ok (git-call-process-display-error "revert" "-n" commit
)))
1454 (git-update-status-files files ok
)
1455 (when (string-match "^[0-9a-f]+ - \\(.*\\)$" subject
)
1456 (setq subject
(match-string 1 subject
)))
1457 (git-setup-log-buffer (get-buffer-create "*git-commit*")
1458 (git-get-merge-heads) nil nil
(format "Revert \"%s\"" subject
) nil
1459 (format "This reverts commit %s.\n" commit
))
1460 (when ok
(git-commit-file)))))
1462 (defun git-find-file ()
1463 "Visit the current file in its own buffer."
1465 (unless git-status
(error "Not in git-status buffer."))
1466 (let ((info (ewoc-data (ewoc-locate git-status
))))
1467 (unless (git-expand-directory info
)
1468 (find-file (git-fileinfo->name info
))
1469 (when (eq 'unmerged
(git-fileinfo->state info
))
1472 (defun git-find-file-other-window ()
1473 "Visit the current file in its own buffer in another window."
1475 (unless git-status
(error "Not in git-status buffer."))
1476 (let ((info (ewoc-data (ewoc-locate git-status
))))
1477 (find-file-other-window (git-fileinfo->name info
))
1478 (when (eq 'unmerged
(git-fileinfo->state info
))
1481 (defun git-find-file-imerge ()
1482 "Visit the current file in interactive merge mode."
1484 (unless git-status
(error "Not in git-status buffer."))
1485 (let ((info (ewoc-data (ewoc-locate git-status
))))
1486 (find-file (git-fileinfo->name info
))
1489 (defun git-view-file ()
1490 "View the current file in its own buffer."
1492 (unless git-status
(error "Not in git-status buffer."))
1493 (let ((info (ewoc-data (ewoc-locate git-status
))))
1494 (view-file (git-fileinfo->name info
))))
1496 (defun git-refresh-status ()
1497 "Refresh the git status buffer."
1499 (unless git-status
(error "Not in git-status buffer."))
1500 (message "Refreshing git status...")
1501 (git-update-status-files)
1502 (message "Refreshing git status...done"))
1504 (defun git-status-quit ()
1505 "Quit git-status mode."
1510 ;;;; ------------------------------------------------------------
1512 (defvar git-status-mode-hook nil
1513 "Run after `git-status-mode' is setup.")
1515 (defvar git-status-mode-map nil
1516 "Keymap for git major mode.")
1518 (defvar git-status nil
1519 "List of all files managed by the git-status mode.")
1521 (unless git-status-mode-map
1522 (let ((map (make-keymap))
1523 (commit-map (make-sparse-keymap))
1524 (diff-map (make-sparse-keymap))
1525 (toggle-map (make-sparse-keymap)))
1526 (suppress-keymap map
)
1527 (define-key map
"?" 'git-help
)
1528 (define-key map
"h" 'git-help
)
1529 (define-key map
" " 'git-next-file
)
1530 (define-key map
"a" 'git-add-file
)
1531 (define-key map
"c" 'git-commit-file
)
1532 (define-key map
"\C-c" commit-map
)
1533 (define-key map
"d" diff-map
)
1534 (define-key map
"=" 'git-diff-file
)
1535 (define-key map
"f" 'git-find-file
)
1536 (define-key map
"\r" 'git-find-file
)
1537 (define-key map
"g" 'git-refresh-status
)
1538 (define-key map
"i" 'git-ignore-file
)
1539 (define-key map
"I" 'git-insert-file
)
1540 (define-key map
"l" 'git-log-file
)
1541 (define-key map
"m" 'git-mark-file
)
1542 (define-key map
"M" 'git-mark-all
)
1543 (define-key map
"n" 'git-next-file
)
1544 (define-key map
"N" 'git-next-unmerged-file
)
1545 (define-key map
"o" 'git-find-file-other-window
)
1546 (define-key map
"p" 'git-prev-file
)
1547 (define-key map
"P" 'git-prev-unmerged-file
)
1548 (define-key map
"q" 'git-status-quit
)
1549 (define-key map
"r" 'git-remove-file
)
1550 (define-key map
"t" toggle-map
)
1551 (define-key map
"T" 'git-toggle-all-marks
)
1552 (define-key map
"u" 'git-unmark-file
)
1553 (define-key map
"U" 'git-revert-file
)
1554 (define-key map
"v" 'git-view-file
)
1555 (define-key map
"x" 'git-remove-handled
)
1556 (define-key map
"\C-?" 'git-unmark-file-up
)
1557 (define-key map
"\M-\C-?" 'git-unmark-all
)
1559 (define-key commit-map
"\C-a" 'git-amend-commit
)
1560 (define-key commit-map
"\C-b" 'git-branch
)
1561 (define-key commit-map
"\C-o" 'git-checkout
)
1562 (define-key commit-map
"\C-p" 'git-cherry-pick-commit
)
1563 (define-key commit-map
"\C-v" 'git-revert-commit
)
1565 (define-key diff-map
"b" 'git-diff-file-base
)
1566 (define-key diff-map
"c" 'git-diff-file-combined
)
1567 (define-key diff-map
"=" 'git-diff-file
)
1568 (define-key diff-map
"e" 'git-diff-file-idiff
)
1569 (define-key diff-map
"E" 'git-find-file-imerge
)
1570 (define-key diff-map
"h" 'git-diff-file-merge-head
)
1571 (define-key diff-map
"m" 'git-diff-file-mine
)
1572 (define-key diff-map
"o" 'git-diff-file-other
)
1574 (define-key toggle-map
"u" 'git-toggle-show-uptodate
)
1575 (define-key toggle-map
"i" 'git-toggle-show-ignored
)
1576 (define-key toggle-map
"k" 'git-toggle-show-unknown
)
1577 (define-key toggle-map
"m" 'git-toggle-all-marks
)
1578 (setq git-status-mode-map map
))
1579 (easy-menu-define git-menu git-status-mode-map
1582 ["Refresh" git-refresh-status t
]
1583 ["Commit" git-commit-file t
]
1584 ["Checkout..." git-checkout t
]
1585 ["New Branch..." git-branch t
]
1586 ["Cherry-pick Commit..." git-cherry-pick-commit t
]
1587 ["Revert Commit..." git-revert-commit t
]
1589 ["Next Unmerged File" git-next-unmerged-file t
]
1590 ["Prev Unmerged File" git-prev-unmerged-file t
]
1591 ["Interactive Merge File" git-find-file-imerge t
]
1592 ["Diff Against Common Base File" git-diff-file-base t
]
1593 ["Diff Combined" git-diff-file-combined t
]
1594 ["Diff Against Merge Head" git-diff-file-merge-head t
]
1595 ["Diff Against Mine" git-diff-file-mine t
]
1596 ["Diff Against Other" git-diff-file-other t
])
1598 ["Add File" git-add-file t
]
1599 ["Revert File" git-revert-file t
]
1600 ["Ignore File" git-ignore-file t
]
1601 ["Remove File" git-remove-file t
]
1602 ["Insert File" git-insert-file t
]
1604 ["Find File" git-find-file t
]
1605 ["View File" git-view-file t
]
1606 ["Diff File" git-diff-file t
]
1607 ["Interactive Diff File" git-diff-file-idiff t
]
1608 ["Log" git-log-file t
]
1610 ["Mark" git-mark-file t
]
1611 ["Mark All" git-mark-all t
]
1612 ["Unmark" git-unmark-file t
]
1613 ["Unmark All" git-unmark-all t
]
1614 ["Toggle All Marks" git-toggle-all-marks t
]
1615 ["Hide Handled Files" git-remove-handled t
]
1617 ["Show Uptodate Files" git-toggle-show-uptodate
:style toggle
:selected git-show-uptodate
]
1618 ["Show Ignored Files" git-toggle-show-ignored
:style toggle
:selected git-show-ignored
]
1619 ["Show Unknown Files" git-toggle-show-unknown
:style toggle
:selected git-show-unknown
]
1621 ["Quit" git-status-quit t
])))
1624 ;; git mode should only run in the *git status* buffer
1625 (put 'git-status-mode
'mode-class
'special
)
1627 (defun git-status-mode ()
1628 "Major mode for interacting with Git.
1630 \\{git-status-mode-map}"
1631 (kill-all-local-variables)
1632 (buffer-disable-undo)
1633 (setq mode-name
"git status"
1634 major-mode
'git-status-mode
1637 (use-local-map git-status-mode-map
)
1638 (let ((buffer-read-only nil
))
1640 (let ((status (ewoc-create 'git-fileinfo-prettyprint
"" "")))
1641 (set (make-local-variable 'git-status
) status
))
1642 (set (make-local-variable 'list-buffers-directory
) default-directory
)
1643 (make-local-variable 'git-show-uptodate
)
1644 (make-local-variable 'git-show-ignored
)
1645 (make-local-variable 'git-show-unknown
)
1646 (run-hooks 'git-status-mode-hook
)))
1648 (defun git-find-status-buffer (dir)
1649 "Find the git status buffer handling a specified directory."
1650 (let ((list (buffer-list))
1651 (fulldir (expand-file-name dir
))
1653 (while (and list
(not found
))
1654 (let ((buffer (car list
)))
1655 (with-current-buffer buffer
1656 (when (and list-buffers-directory
1657 (string-equal fulldir
(expand-file-name list-buffers-directory
))
1658 (eq major-mode
'git-status-mode
))
1659 (setq found buffer
))))
1660 (setq list
(cdr list
)))
1663 (defun git-status (dir)
1664 "Entry point into git-status mode."
1665 (interactive "DSelect directory: ")
1666 (setq dir
(git-get-top-dir dir
))
1667 (if (file-directory-p (concat (file-name-as-directory dir
) ".git"))
1668 (let ((buffer (or (and git-reuse-status-buffer
(git-find-status-buffer dir
))
1669 (create-file-buffer (expand-file-name "*git-status*" dir
)))))
1670 (switch-to-buffer buffer
)
1673 (git-refresh-status)
1674 (goto-char (point-min))
1675 (add-hook 'after-save-hook
'git-update-saved-file
))
1676 (message "%s is not a git working tree." dir
)))
1678 (defun git-update-saved-file ()
1679 "Update the corresponding git-status buffer when a file is saved.
1680 Meant to be used in `after-save-hook'."
1681 (let* ((file (expand-file-name buffer-file-name
))
1682 (dir (condition-case nil
(git-get-top-dir (file-name-directory file
)) (error nil
)))
1683 (buffer (and dir
(git-find-status-buffer dir
))))
1685 (with-current-buffer buffer
1686 (let ((filename (file-relative-name file dir
)))
1687 ; skip files located inside the .git directory
1688 (unless (string-match "^\\.git/" filename
)
1689 (git-call-process nil
"add" "--refresh" "--" filename
)
1690 (git-update-status-files (list filename
))))))))
1693 "Display help for Git mode."
1695 (describe-function 'git-status-mode
))
1698 ;;; git.el ends here